displayablegraph.class.inc.php 44 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340
  1. <?php
  2. // Copyright (C) 2015 Combodo SARL
  3. //
  4. // This file is part of iTop.
  5. //
  6. // iTop is free software; you can redistribute it and/or modify
  7. // it under the terms of the GNU Affero General Public License as published by
  8. // the Free Software Foundation, either version 3 of the License, or
  9. // (at your option) any later version.
  10. //
  11. // iTop is distributed in the hope that it will be useful,
  12. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. // GNU Affero General Public License for more details.
  15. //
  16. // You should have received a copy of the GNU Affero General Public License
  17. // along with iTop. If not, see <http://www.gnu.org/licenses/>
  18. /**
  19. * Special kind of Graph for producing some nice output
  20. *
  21. * @copyright Copyright (C) 2015 Combodo SARL
  22. * @license http://opensource.org/licenses/AGPL-3.0
  23. */
  24. class DisplayableNode extends GraphNode
  25. {
  26. public $x;
  27. public $y;
  28. /**
  29. * Create a new node inside a graph
  30. * @param SimpleGraph $oGraph
  31. * @param string $sId The unique identifier of this node inside the graph
  32. * @param number $x Horizontal position
  33. * @param number $y Vertical position
  34. */
  35. public function __construct(SimpleGraph $oGraph, $sId, $x = 0, $y = 0)
  36. {
  37. parent::__construct($oGraph, $sId);
  38. $this->x = $x;
  39. $this->y = $y;
  40. $this->bFiltered = false;
  41. }
  42. public function GetIconURL()
  43. {
  44. return $this->GetProperty('icon_url', '');
  45. }
  46. public function GetLabel()
  47. {
  48. return $this->GetProperty('label', $this->sId);
  49. }
  50. public function GetWidth()
  51. {
  52. return max(32, 5*strlen($this->GetProperty('label'))); // approximation of the text's bounding box
  53. }
  54. public function GetHeight()
  55. {
  56. return 32;
  57. }
  58. public function Distance2(DisplayableNode $oNode)
  59. {
  60. $dx = $this->x - $oNode->x;
  61. $dy = $this->y - $oNode->y;
  62. $d2 = $dx*$dx + $dy*$dy - $this->GetHeight()*$this->GetHeight();
  63. if ($d2 < 40)
  64. {
  65. $d2 = 40;
  66. }
  67. return $d2;
  68. }
  69. public function Distance(DisplayableNode $oNode)
  70. {
  71. return sqrt($this->Distance2($oNode));
  72. }
  73. public function GetForRaphael($aContextDefs)
  74. {
  75. $aNode = array();
  76. $aNode['shape'] = 'icon';
  77. $aNode['icon_url'] = $this->GetIconURL();
  78. $aNode['width'] = 32;
  79. $aNode['source'] = ($this->GetProperty('source') == true);
  80. $aNode['obj_class'] = get_class($this->GetProperty('object'));
  81. $aNode['obj_key'] = $this->GetProperty('object')->GetKey();
  82. $aNode['sink'] = ($this->GetProperty('sink') == true);
  83. $aNode['x'] = $this->x;
  84. $aNode['y']= $this->y;
  85. $aNode['label'] = $this->GetLabel();
  86. $aNode['id'] = $this->GetId();
  87. $fOpacity = ($this->GetProperty('is_reached') ? 1 : 0.4);
  88. $aNode['icon_attr'] = array('opacity' => $fOpacity);
  89. $aNode['text_attr'] = array('opacity' => $fOpacity);
  90. $aNode['tooltip'] = $this->GetTooltip($aContextDefs);
  91. $aNode['context_icons'] = array();
  92. $aContextRootCauses = $this->GetProperty('context_root_causes');
  93. if (!is_null($aContextRootCauses))
  94. {
  95. foreach($aContextRootCauses as $key => $aObjects)
  96. {
  97. $aNode['context_icons'][] = utils::GetAbsoluteUrlModulesRoot().$aContextDefs[$key]['icon'];
  98. }
  99. }
  100. return $aNode;
  101. }
  102. public function RenderAsPDF(TCPDF $oPdf, DisplayableGraph $oGraph, $fScale, $aContextDefs)
  103. {
  104. $Alpha = 1.0;
  105. $oPdf->SetFillColor(200, 200, 200);
  106. $oPdf->setAlpha(1);
  107. $sIconUrl = $this->GetProperty('icon_url');
  108. $sIconPath = str_replace(utils::GetAbsoluteUrlModulesRoot(), APPROOT.'env-'.utils::GetCurrentEnvironment().'/', $sIconUrl);
  109. if ($this->GetProperty('source'))
  110. {
  111. $oPdf->SetLineStyle(array('width' => 2*$fScale, 'cap' => 'round', 'join' => 'miter', 'dash' => 0, 'color' => array(204, 51, 51)));
  112. $oPdf->Circle($this->x * $fScale, $this->y * $fScale, 16 * 1.25 * $fScale, 0, 360, 'D');
  113. }
  114. else if ($this->GetProperty('sink'))
  115. {
  116. $oPdf->SetLineStyle(array('width' => 2*$fScale, 'cap' => 'round', 'join' => 'miter', 'dash' => 0, 'color' => array(51, 51, 204)));
  117. $oPdf->Circle($this->x * $fScale, $this->y * $fScale, 16 * 1.25 * $fScale, 0, 360, 'D');
  118. }
  119. if (!$this->GetProperty('is_reached'))
  120. {
  121. $sTempImageName = $this->CreateWhiteIcon($oGraph, $sIconPath);
  122. if ($sTempImageName != null)
  123. {
  124. $oPdf->Image($sTempImageName, ($this->x - 16)*$fScale, ($this->y - 16)*$fScale, 32*$fScale, 32*$fScale, 'PNG');
  125. }
  126. $Alpha = 0.4;
  127. $oPdf->setAlpha($Alpha);
  128. }
  129. $oPdf->Image($sIconPath, ($this->x - 16)*$fScale, ($this->y - 16)*$fScale, 32*$fScale, 32*$fScale);
  130. $aContextRootCauses = $this->GetProperty('context_root_causes');
  131. if (!is_null($aContextRootCauses))
  132. {
  133. $idx = 0;
  134. foreach($aContextRootCauses as $key => $aObjects)
  135. {
  136. $sgn = 2*($idx %2) -1;
  137. $coef = floor((1+$idx)/2) * $sgn;
  138. $alpha = $coef*pi()/4 - pi()/2;
  139. $x = $this->x * $fScale + cos($alpha) * 16*1.25 * $fScale;
  140. $y = $this->y * $fScale + sin($alpha) * 16*1.25 * $fScale;
  141. $l = 32 * $fScale / 3;
  142. $sIconPath = APPROOT.'env-'.utils::GetCurrentEnvironment().'/'.$aContextDefs[$key]['icon'];
  143. $oPdf->Image($sIconPath, $x - $l/2, $y - $l/2, $l, $l);
  144. $idx++;
  145. }
  146. }
  147. $oPdf->SetFont('dejavusans', '', 24 * $fScale, '', true);
  148. $width = $oPdf->GetStringWidth($this->GetProperty('label'));
  149. $height = $oPdf->GetStringHeight(1000, $this->GetProperty('label'));
  150. $oPdf->setAlpha(0.6 * $Alpha);
  151. $oPdf->SetFillColor(255, 255, 255);
  152. $oPdf->SetDrawColor(255, 255, 255);
  153. $oPdf->Rect($this->x*$fScale - $width/2, ($this->y + 18)*$fScale, $width, $height, 'DF');
  154. $oPdf->setAlpha($Alpha);
  155. $oPdf->SetTextColor(0, 0, 0);
  156. $oPdf->Text($this->x*$fScale - $width/2, ($this->y + 18)*$fScale, $this->GetProperty('label'));
  157. }
  158. /**
  159. * Create a "whitened" version of the icon (retaining the transparency) to be used a background for masking the underlying lines
  160. * @param string $sIconFile The path to the file containing the icon
  161. * @return NULL|string The path to a temporary file containing the white version of the icon
  162. */
  163. protected function CreateWhiteIcon(DisplayableGraph $oGraph, $sIconFile)
  164. {
  165. $aInfo = getimagesize($sIconFile);
  166. $im = null;
  167. switch($aInfo['mime'])
  168. {
  169. case 'image/png':
  170. if (function_exists('imagecreatefrompng'))
  171. {
  172. $im = imagecreatefrompng($sIconFile);
  173. }
  174. break;
  175. case 'image/gif':
  176. if (function_exists('imagecreatefromgif'))
  177. {
  178. $im = imagecreatefromgif($sIconFile);
  179. }
  180. break;
  181. case 'image/jpeg':
  182. case 'image/jpg':
  183. if (function_exists('imagecreatefromjpeg'))
  184. {
  185. $im = imagecreatefromjpeg($sIconFile);
  186. }
  187. break;
  188. default:
  189. return null;
  190. }
  191. if($im && imagefilter($im, IMG_FILTER_COLORIZE, 255, 255, 255))
  192. {
  193. $sTempImageName = $oGraph->GetTempImageName();
  194. imagesavealpha($im, true);
  195. imagepng($im, $sTempImageName);
  196. imagedestroy($im);
  197. return $sTempImageName;
  198. }
  199. else
  200. {
  201. return null;
  202. }
  203. }
  204. /**
  205. * Group together (as a special kind of nodes) all the similar neighbours of the current node
  206. * @param DisplayableGraph $oGraph
  207. * @param int $iThresholdCount
  208. * @param boolean $bDirectionUp
  209. * @param boolean $bDirectionDown
  210. */
  211. public function GroupSimilarNeighbours(DisplayableGraph $oGraph, $iThresholdCount, $bDirectionUp = false, $bDirectionDown = true)
  212. {
  213. if ($this->GetProperty('grouped') === true) return;
  214. $this->SetProperty('grouped', true);
  215. if ($bDirectionDown)
  216. {
  217. $aNodesPerClass = array();
  218. foreach($this->GetOutgoingEdges() as $oEdge)
  219. {
  220. $oNode = $oEdge->GetSinkNode();
  221. if ($oNode->GetProperty('class') !== null)
  222. {
  223. $sClass = $oNode->GetProperty('class');
  224. if (($sClass!== null) && (!array_key_exists($sClass, $aNodesPerClass)))
  225. {
  226. $aNodesPerClass[$sClass] = array(
  227. 'reached' => array(
  228. 'count' => 0,
  229. 'nodes' => array(),
  230. 'icon_url' => $oNode->GetProperty('icon_url'),
  231. ),
  232. 'not_reached' => array(
  233. 'count' => 0,
  234. 'nodes' => array(),
  235. 'icon_url' => $oNode->GetProperty('icon_url'),
  236. )
  237. );
  238. }
  239. $sKey = $oNode->GetProperty('is_reached') ? 'reached' : 'not_reached';
  240. if (!array_key_exists($oNode->GetId(), $aNodesPerClass[$sClass][$sKey]['nodes']))
  241. {
  242. $aNodesPerClass[$sClass][$sKey]['nodes'][$oNode->GetId()] = $oNode;
  243. $aNodesPerClass[$sClass][$sKey]['count'] += (int)$oNode->GetProperty('count', 1);
  244. }
  245. }
  246. else
  247. {
  248. $oNode->GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  249. }
  250. }
  251. foreach($aNodesPerClass as $sClass => $aDefs)
  252. {
  253. foreach($aDefs as $sStatus => $aGroupProps)
  254. {
  255. if (count($aGroupProps['nodes']) >= $iThresholdCount)
  256. {
  257. $sNewId = $this->GetId().'::'.(($sStatus == 'reached') ? '_reached': '');
  258. $oNewNode = $oGraph->GetNode($sNewId);
  259. if ($oNewNode == null)
  260. {
  261. $oNewNode = new DisplayableGroupNode($oGraph, $sNewId);
  262. $oNewNode->SetProperty('label', 'x'.$aGroupProps['count']);
  263. $oNewNode->SetProperty('icon_url', $aGroupProps['icon_url']);
  264. $oNewNode->SetProperty('class', $sClass);
  265. $oNewNode->SetProperty('is_reached', ($sStatus == 'reached'));
  266. $oNewNode->SetProperty('count', $aGroupProps['count']);
  267. }
  268. else
  269. {
  270. $oNewNode->SetProperty('count', $oNewNode->GetProperty('count')+$aGroupProps['count']);
  271. }
  272. try
  273. {
  274. $oIncomingEdge = new DisplayableEdge($oGraph, $this->GetId().'-'.$oNewNode->GetId(), $this, $oNewNode);
  275. }
  276. catch(Exception $e)
  277. {
  278. // Ignore this redundant egde
  279. }
  280. foreach($aGroupProps['nodes'] as $oNode)
  281. {
  282. foreach($oNode->GetIncomingEdges() as $oEdge)
  283. {
  284. if ($oEdge->GetSourceNode()->GetId() !== $this->GetId())
  285. {
  286. try
  287. {
  288. $oNewEdge = new DisplayableEdge($oGraph, $oEdge->GetId().'::'.$sClass, $oEdge->GetSourceNode(), $oNewNode);
  289. }
  290. catch(Exception $e)
  291. {
  292. // ignore this edge
  293. }
  294. }
  295. }
  296. foreach($oNode->GetOutgoingEdges() as $oEdge)
  297. {
  298. $aOutgoing[] = $oEdge->GetSinkNode();
  299. try
  300. {
  301. $oNewEdge = new DisplayableEdge($oGraph, $oEdge->GetId().'::'.$sClass, $oNewNode, $oEdge->GetSinkNode());
  302. }
  303. catch(Exception $e)
  304. {
  305. // ignore this edge
  306. }
  307. }
  308. if ($oGraph->GetNode($oNode->GetId()))
  309. {
  310. $oGraph->_RemoveNode($oNode);
  311. $oNewNode->AddObject($oNode->GetProperty('object'));
  312. }
  313. }
  314. $oNewNode->GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  315. }
  316. else
  317. {
  318. foreach($aGroupProps['nodes'] as $oNode)
  319. {
  320. $oNode->GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  321. }
  322. }
  323. }
  324. }
  325. }
  326. }
  327. public function GetTooltip($aContextDefs)
  328. {
  329. $sHtml = '';
  330. $oCurrObj = $this->GetProperty('object');
  331. $sSubClass = get_class($oCurrObj);
  332. $sHtml .= $oCurrObj->GetHyperlink()."<hr/>";
  333. $aContextRootCauses = $this->GetProperty('context_root_causes');
  334. if (!is_null($aContextRootCauses))
  335. {
  336. foreach($aContextRootCauses as $key => $aObjects)
  337. {
  338. $aContext = $aContextDefs[$key];
  339. $aRootCauses = array();
  340. foreach($aObjects as $oRootCause)
  341. {
  342. $aRootCauses[] = $oRootCause->GetHyperlink();
  343. }
  344. $sHtml .= '<p><img style="max-height: 24px; vertical-align:bottom;" src="'.utils::GetAbsoluteUrlModulesRoot().$aContext['icon'].'" title="'.htmlentities(Dict::S($aContext['dict'])).'">&nbsp;'.implode(', ', $aRootCauses).'</p>';
  345. }
  346. $sHtml .= '<hr/>';
  347. }
  348. $sHtml .= '<table><tbody>';
  349. foreach(MetaModel::GetZListItems($sSubClass, 'list') as $sAttCode)
  350. {
  351. $oAttDef = MetaModel::GetAttributeDef($sSubClass, $sAttCode);
  352. $sHtml .= '<tr><td>'.$oAttDef->GetLabel().':&nbsp;</td><td>'.$oCurrObj->GetAsHtml($sAttCode).'</td></tr>';
  353. }
  354. $sHtml .= '</tbody></table>';
  355. return $sHtml;
  356. }
  357. }
  358. class DisplayableRedundancyNode extends DisplayableNode
  359. {
  360. public function GetWidth()
  361. {
  362. return 24;
  363. }
  364. public function GetForRaphael($aContextDefs)
  365. {
  366. $aNode = array();
  367. $aNode['shape'] = 'disc';
  368. $aNode['icon_url'] = $this->GetIconURL();
  369. $aNode['source'] = ($this->GetProperty('source') == true);
  370. $aNode['width'] = $this->GetWidth();
  371. $aNode['x'] = $this->x;
  372. $aNode['y']= $this->y;
  373. $aNode['label'] = $this->GetLabel();
  374. $aNode['id'] = $this->GetId();
  375. $fDiscOpacity = ($this->GetProperty('is_reached') ? 1 : 0.2);
  376. $sColor = ($this->GetProperty('is_reached_count') > $this->GetProperty('threshold')) ? '#c33' : '#999';
  377. $aNode['disc_attr'] = array('stroke-width' => 3, 'stroke' => '#000', 'fill' => $sColor, 'opacity' => $fDiscOpacity);
  378. $fTextOpacity = ($this->GetProperty('is_reached') ? 1 : 0.4);
  379. $aNode['text_attr'] = array('fill' => '#fff', 'opacity' => $fTextOpacity);
  380. $aNode['tooltip'] = $this->GetTooltip($aContextDefs);
  381. return $aNode;
  382. }
  383. public function RenderAsPDF(TCPDF $oPdf, DisplayableGraph $oGraph, $fScale, $aContextDefs)
  384. {
  385. $oPdf->SetAlpha(1);
  386. if($this->GetProperty('is_reached_count') > $this->GetProperty('threshold'))
  387. {
  388. $oPdf->SetFillColor(200, 0, 0);
  389. }
  390. else
  391. {
  392. $oPdf->SetFillColor(144, 144, 144);
  393. }
  394. $oPdf->SetDrawColor(0, 0, 0);
  395. $oPdf->Circle($this->x*$fScale, $this->y*$fScale, 16*$fScale, 0, 360, 'DF');
  396. $oPdf->SetTextColor(255, 255, 255);
  397. $oPdf->SetFont('dejavusans', '', 28 * $fScale, '', true);
  398. $sLabel = (string)$this->GetProperty('label');
  399. $width = $oPdf->GetStringWidth($sLabel, 'dejavusans', 'B', 24*$fScale);
  400. $height = $oPdf->GetStringHeight(1000, $sLabel);
  401. $xPos = (float)$this->x*$fScale - $width/2;
  402. $yPos = (float)$this->y*$fScale - $height/2;
  403. $oPdf->SetXY(($this->x - 16)*$fScale, ($this->y - 16)*$fScale);
  404. $oPdf->Cell(32*$fScale, 32*$fScale, $sLabel, 0, 0, 'C', 0, '', 0, false, 'T', 'C');
  405. }
  406. /**
  407. * @see DisplayableNode::GroupSimilarNeighbours()
  408. */
  409. public function GroupSimilarNeighbours(DisplayableGraph $oGraph, $iThresholdCount, $bDirectionUp = false, $bDirectionDown = true)
  410. {
  411. parent::GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  412. if ($bDirectionUp)
  413. {
  414. $aNodesPerClass = array();
  415. foreach($this->GetIncomingEdges() as $oEdge)
  416. {
  417. $oNode = $oEdge->GetSourceNode();
  418. if (($oNode->GetProperty('class') !== null) && (!$oNode->GetProperty('is_reached')))
  419. {
  420. $sClass = $oNode->GetProperty('class');
  421. if (!array_key_exists($sClass, $aNodesPerClass))
  422. {
  423. $aNodesPerClass[$sClass] = array('reached' => array(), 'not_reached' => array());
  424. }
  425. $aNodesPerClass[$sClass][$oNode->GetProperty('is_reached') ? 'reached' : 'not_reached'][] = $oNode;
  426. }
  427. else
  428. {
  429. //$oNode->GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  430. }
  431. }
  432. foreach($aNodesPerClass as $sClass => $aDefs)
  433. {
  434. foreach($aDefs as $sStatus => $aNodes)
  435. {
  436. if (count($aNodes) >= $iThresholdCount)
  437. {
  438. $oNewNode = new DisplayableGroupNode($oGraph, '-'.$this->GetId().'::'.$sClass.'/'.$sStatus);
  439. $oNewNode->SetProperty('label', 'x'.count($aNodes));
  440. $oNewNode->SetProperty('icon_url', $aNodes[0]->GetProperty('icon_url'));
  441. $oNewNode->SetProperty('is_reached', $aNodes[0]->GetProperty('is_reached'));
  442. $oOutgoingEdge = new DisplayableEdge($oGraph, '-'.$this->GetId().'-'.$oNewNode->GetId().'/'.$sStatus, $oNewNode, $this);
  443. foreach($aNodes as $oNode)
  444. {
  445. foreach($oNode->GetIncomingEdges() as $oEdge)
  446. {
  447. $oNewEdge = new DisplayableEdge($oGraph, '-'.$oEdge->GetId().'::'.$sClass, $oEdge->GetSourceNode(), $oNewNode);
  448. }
  449. foreach($oNode->GetOutgoingEdges() as $oEdge)
  450. {
  451. if ($oEdge->GetSinkNode()->GetId() !== $this->GetId())
  452. {
  453. $aOutgoing[] = $oEdge->GetSinkNode();
  454. $oNewEdge = new DisplayableEdge($oGraph, '-'.$oEdge->GetId().'::'.$sClass.'/'.$sStatus, $oNewNode, $oEdge->GetSinkNode());
  455. }
  456. }
  457. $oGraph->_RemoveNode($oNode);
  458. $oNewNode->AddObject($oNode->GetProperty('object'));
  459. }
  460. //$oNewNode->GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  461. }
  462. else
  463. {
  464. foreach($aNodes as $oNode)
  465. {
  466. //$oNode->GroupSimilarNeighbours($oGraph, $iThresholdCount, $bDirectionUp, $bDirectionDown);
  467. }
  468. }
  469. }
  470. }
  471. }
  472. }
  473. public function GetTooltip($aContextDefs)
  474. {
  475. $sHtml = '';
  476. $sHtml .= Dict::S('UI:RelationTooltip:Redundancy')."<hr>";
  477. $sHtml .= '<table><tbody>';
  478. $sHtml .= "<tr><td>".Dict::Format('UI:RelationTooltip:ImpactedItems_N_of_M' , $this->GetProperty('is_reached_count'), $this->GetProperty('min_up') + $this->GetProperty('threshold'))."</td></tr>";
  479. $sHtml .= "<tr><td>".Dict::Format('UI:RelationTooltip:CriticalThreshold_N_of_M' , $this->GetProperty('threshold'), $this->GetProperty('min_up') + $this->GetProperty('threshold'))."</td></tr>";
  480. $sHtml .= '</tbody></table>';
  481. return $sHtml;
  482. }
  483. }
  484. class DisplayableEdge extends GraphEdge
  485. {
  486. public function RenderAsPDF(TCPDF $oPdf, DisplayableGraph $oGraph, $fScale, $aContextDefs)
  487. {
  488. $xStart = $this->GetSourceNode()->x * $fScale;
  489. $yStart = $this->GetSourceNode()->y * $fScale;
  490. $xEnd = $this->GetSinkNode()->x * $fScale;
  491. $yEnd = $this->GetSinkNode()->y * $fScale;
  492. $bReached = ($this->GetSourceNode()->GetProperty('is_reached') && $this->GetSinkNode()->GetProperty('is_reached'));
  493. $oPdf->setAlpha(1);
  494. if ($bReached)
  495. {
  496. $aColor = array(100, 100, 100);
  497. }
  498. else
  499. {
  500. $aColor = array(200, 200, 200);
  501. }
  502. $oPdf->SetLineStyle(array('width' => 2*$fScale, 'cap' => 'round', 'join' => 'miter', 'dash' => 0, 'color' => $aColor));
  503. $oPdf->Line($xStart, $yStart, $xEnd, $yEnd);
  504. $vx = $xEnd - $xStart;
  505. $vy = $yEnd - $yStart;
  506. $l = sqrt($vx*$vx + $vy*$vy);
  507. $vx = $vx / $l;
  508. $vy = $vy / $l;
  509. $ux = -$vy;
  510. $uy = $vx;
  511. $lPos = max($l/2, $l - 40*$fScale);
  512. $iArrowSize = 5*$fScale;
  513. $x = $xStart + $lPos * $vx;
  514. $y = $yStart + $lPos * $vy;
  515. $oPdf->Line($x, $y, $x + $iArrowSize * ($ux-$vx), $y + $iArrowSize * ($uy-$vy));
  516. $oPdf->Line($x, $y, $x - $iArrowSize * ($ux+$vx), $y - $iArrowSize * ($uy+$vy));
  517. }
  518. }
  519. class DisplayableGroupNode extends DisplayableNode
  520. {
  521. protected $aObjects;
  522. public function __construct(SimpleGraph $oGraph, $sId, $x = 0, $y = 0)
  523. {
  524. parent::__construct($oGraph, $sId, $x, $y);
  525. $this->aObjects = array();
  526. }
  527. public function AddObject(DBObject $oObj)
  528. {
  529. $this->aObjects[$oObj->GetKey()] = $oObj;
  530. }
  531. public function GetObjects()
  532. {
  533. return $this->aObjects;
  534. }
  535. public function GetWidth()
  536. {
  537. return 50;
  538. }
  539. public function GetForRaphael($aContextDefs)
  540. {
  541. $aNode = array();
  542. $aNode['shape'] = 'group';
  543. $aNode['icon_url'] = $this->GetIconURL();
  544. $aNode['source'] = ($this->GetProperty('source') == true);
  545. $aNode['width'] = $this->GetWidth();
  546. $aNode['x'] = $this->x;
  547. $aNode['y']= $this->y;
  548. $aNode['label'] = $this->GetLabel();
  549. $aNode['id'] = $this->GetId();
  550. $aNode['group_index'] = $this->GetProperty('group_index'); // if supplied
  551. $fDiscOpacity = ($this->GetProperty('is_reached') ? 1 : 0.2);
  552. $fTextOpacity = ($this->GetProperty('is_reached') ? 1 : 0.4);
  553. $aNode['icon_attr'] = array('opacity' => $fTextOpacity);
  554. $aNode['disc_attr'] = array('stroke-width' => 3, 'stroke' => '#000', 'fill' => '#fff', 'opacity' => $fDiscOpacity);
  555. $aNode['text_attr'] = array('fill' => '#000', 'opacity' => $fTextOpacity);
  556. $aNode['tooltip'] = $this->GetTooltip($aContextDefs);
  557. return $aNode;
  558. }
  559. public function RenderAsPDF(TCPDF $oPdf, DisplayableGraph $oGraph, $fScale, $aContextDefs)
  560. {
  561. $bReached = $this->GetProperty('is_reached');
  562. $oPdf->SetFillColor(255, 255, 255);
  563. if ($bReached)
  564. {
  565. $aBorderColor = array(100, 100, 100);
  566. }
  567. else
  568. {
  569. $aBorderColor = array(200, 200, 200);
  570. }
  571. $oPdf->SetLineStyle(array('width' => 2*$fScale, 'cap' => 'round', 'join' => 'miter', 'dash' => 0, 'color' => $aBorderColor));
  572. $sIconUrl = $this->GetProperty('icon_url');
  573. $sIconPath = str_replace(utils::GetAbsoluteUrlModulesRoot(), APPROOT.'env-'.utils::GetCurrentEnvironment().'/', $sIconUrl);
  574. $oPdf->SetAlpha(1);
  575. $oPdf->Circle($this->x*$fScale, $this->y*$fScale, $this->GetWidth() / 2 * $fScale, 0, 360, 'DF');
  576. if ($bReached)
  577. {
  578. $oPdf->SetAlpha(1);
  579. }
  580. else
  581. {
  582. $oPdf->SetAlpha(0.4);
  583. }
  584. $oPdf->Image($sIconPath, ($this->x - 17)*$fScale, ($this->y - 17)*$fScale, 16*$fScale, 16*$fScale);
  585. $oPdf->Image($sIconPath, ($this->x + 1)*$fScale, ($this->y - 17)*$fScale, 16*$fScale, 16*$fScale);
  586. $oPdf->Image($sIconPath, ($this->x -8)*$fScale, ($this->y +1)*$fScale, 16*$fScale, 16*$fScale);
  587. $oPdf->SetFont('dejavusans', '', 24 * $fScale, '', true);
  588. $width = $oPdf->GetStringWidth($this->GetProperty('label'));
  589. $oPdf->SetTextColor(0, 0, 0);
  590. $oPdf->Text($this->x*$fScale - $width/2, ($this->y + 25)*$fScale, $this->GetProperty('label'));
  591. }
  592. public function GetTooltip($aContextDefs)
  593. {
  594. $sHtml = '';
  595. $iGroupIdx = $this->GetProperty('group_index');
  596. $sHtml .= Dict::Format('UI:RelationGroupNumber_N', (1+$iGroupIdx));
  597. return $sHtml;
  598. }
  599. }
  600. /**
  601. * A Graph that can be displayed interactively using Raphael JS or saved as a PDF document
  602. */
  603. class DisplayableGraph extends SimpleGraph
  604. {
  605. protected $bDirectionDown;
  606. protected $aTempImages;
  607. protected $aSourceObjects;
  608. protected $aSinkObjects;
  609. public function __construct()
  610. {
  611. parent::__construct();
  612. $this->aTempImages = array();
  613. $this->aSourceObjects = array();
  614. $this->aSinkObjects = array();
  615. }
  616. public function GetTempImageName()
  617. {
  618. $sNewTempName = tempnam(APPROOT.'data', 'img-');
  619. $this->aTempImages[] = $sNewTempName;
  620. return $sNewTempName;
  621. }
  622. public function __destruct()
  623. {
  624. foreach($this->aTempImages as $sTempFile)
  625. {
  626. @unlink($sTempFile);
  627. }
  628. }
  629. /**
  630. * Build a DisplayableGraph from a RelationGraph
  631. * @param RelationGraph $oGraph
  632. * @param number $iGroupingThreshold
  633. * @param string $bDirectionDown
  634. * @return DisplayableGraph
  635. */
  636. public static function FromRelationGraph(RelationGraph $oGraph, $iGroupingThreshold = 20, $bDirectionDown = true)
  637. {
  638. $oNewGraph = new DisplayableGraph();
  639. $oNewGraph->bDirectionDown = $bDirectionDown;
  640. $oNodesIter = new RelationTypeIterator($oGraph, 'Node');
  641. foreach($oNodesIter as $oNode)
  642. {
  643. switch(get_class($oNode))
  644. {
  645. case 'RelationObjectNode':
  646. $oNewNode = new DisplayableNode($oNewGraph, $oNode->GetId(), 0, 0);
  647. $oObj = $oNode->GetProperty('object');
  648. $sClass = get_class($oObj);
  649. if ($oNode->GetProperty('source'))
  650. {
  651. if (!array_key_exists($sClass, $oNewGraph->aSourceObjects))
  652. {
  653. $oNewGraph->aSourceObjects[$sClass] = array();
  654. }
  655. $oNewGraph->aSourceObjects[$sClass][] = $oObj->GetKey();
  656. $oNewNode->SetProperty('source', true);
  657. }
  658. if ($oNode->GetProperty('sink'))
  659. {
  660. if (!array_key_exists($sClass, $oNewGraph->aSinkObjects))
  661. {
  662. $oNewGraph->aSinkObjects[$sClass] = array();
  663. }
  664. $oNewGraph->aSinkObjects[$sClass][] = $oObj->GetKey();
  665. $oNewNode->SetProperty('sink', true);
  666. }
  667. $oNewNode->SetProperty('class', $sClass);
  668. $oNewNode->SetProperty('object', $oObj);
  669. $oNewNode->SetProperty('icon_url', $oObj->GetIcon(false));
  670. $oNewNode->SetProperty('label', $oObj->GetRawName());
  671. $oNewNode->SetProperty('is_reached', $bDirectionDown ? $oNode->GetProperty('is_reached') : true); // When going "up" is_reached does not matter
  672. $oNewNode->SetProperty('is_reached_allowed', $oNode->GetProperty('is_reached_allowed'));
  673. $oNewNode->SetProperty('context_root_causes', $oNode->GetProperty('context_root_causes'));
  674. break;
  675. default:
  676. $oNewNode = new DisplayableRedundancyNode($oNewGraph, $oNode->GetId(), 0, 0);
  677. $iNbReached = (is_null($oNode->GetProperty('is_reached_count'))) ? 0 : $oNode->GetProperty('is_reached_count');
  678. $oNewNode->SetProperty('label', $iNbReached."/".($oNode->GetProperty('min_up') + $oNode->GetProperty('threshold')));
  679. $oNewNode->SetProperty('min_up', $oNode->GetProperty('min_up'));
  680. $oNewNode->SetProperty('threshold', $oNode->GetProperty('threshold'));
  681. $oNewNode->SetProperty('is_reached_count', $iNbReached);
  682. $oNewNode->SetProperty('is_reached', true);
  683. }
  684. }
  685. $oEdgesIter = new RelationTypeIterator($oGraph, 'Edge');
  686. foreach($oEdgesIter as $oEdge)
  687. {
  688. $oSourceNode = $oNewGraph->GetNode($oEdge->GetSourceNode()->GetId());
  689. $oSinkNode = $oNewGraph->GetNode($oEdge->GetSinkNode()->GetId());
  690. $oNewEdge = new DisplayableEdge($oNewGraph, $oEdge->GetId(), $oSourceNode, $oSinkNode);
  691. }
  692. // Remove duplicate edges between two nodes
  693. $oEdgesIter = new RelationTypeIterator($oNewGraph, 'Edge');
  694. $aEdgeKeys = array();
  695. foreach($oEdgesIter as $oEdge)
  696. {
  697. $sSourceId = $oEdge->GetSourceNode()->GetId();
  698. $sSinkId = $oEdge->GetSinkNode()->GetId();
  699. if ($sSourceId == $sSinkId)
  700. {
  701. // Remove self referring edges
  702. $oNewGraph->_RemoveEdge($oEdge);
  703. }
  704. else
  705. {
  706. $sKey = $sSourceId.'//'.$sSinkId;
  707. if (array_key_exists($sKey, $aEdgeKeys))
  708. {
  709. // Remove duplicate edges
  710. $oNewGraph->_RemoveEdge($oEdge);
  711. }
  712. else
  713. {
  714. $aEdgeKeys[$sKey] = true;
  715. }
  716. }
  717. }
  718. $oNodesIter = new RelationTypeIterator($oNewGraph, 'Node');
  719. foreach($oNodesIter as $oNode)
  720. {
  721. if ($oNode->GetProperty('source'))
  722. {
  723. $oNode->GroupSimilarNeighbours($oNewGraph, $iGroupingThreshold, true, true);
  724. }
  725. }
  726. // Groups numbering
  727. $oIterator = new RelationTypeIterator($oNewGraph, 'Node');
  728. $iGroupIdx = 0;
  729. foreach($oIterator as $oNode)
  730. {
  731. if ($oNode instanceof DisplayableGroupNode)
  732. {
  733. $aGroups[] = $oNode->GetObjects();
  734. $oNode->SetProperty('group_index', $iGroupIdx);
  735. $iGroupIdx++;
  736. }
  737. }
  738. // Remove duplicate edges between two nodes
  739. $oEdgesIter = new RelationTypeIterator($oNewGraph, 'Edge');
  740. $aEdgeKeys = array();
  741. foreach($oEdgesIter as $oEdge)
  742. {
  743. $sSourceId = $oEdge->GetSourceNode()->GetId();
  744. $sSinkId = $oEdge->GetSinkNode()->GetId();
  745. if ($sSourceId == $sSinkId)
  746. {
  747. // Remove self referring edges
  748. $oNewGraph->_RemoveEdge($oEdge);
  749. }
  750. else
  751. {
  752. $sKey = $sSourceId.'//'.$sSinkId;
  753. if (array_key_exists($sKey, $aEdgeKeys))
  754. {
  755. // Remove duplicate edges
  756. $oNewGraph->_RemoveEdge($oEdge);
  757. }
  758. else
  759. {
  760. $aEdgeKeys[$sKey] = true;
  761. }
  762. }
  763. }
  764. return $oNewGraph;
  765. }
  766. /**
  767. * Initializes the positions by rendering using Graphviz in xdot format
  768. * and parsing the output.
  769. * @throws Exception
  770. */
  771. public function InitFromGraphviz()
  772. {
  773. $sDot = $this->DumpAsXDot();
  774. if (strpos($sDot, 'digraph') === false)
  775. {
  776. throw new Exception($sDot);
  777. }
  778. $aChunks = explode(";", $sDot);
  779. foreach($aChunks as $sChunk)
  780. {
  781. if(preg_match('/"([^"]+)".+pos="([0-9\\.]+),([0-9\\.]+)"/ms', $sChunk, $aMatches))
  782. {
  783. $sId = $aMatches[1];
  784. $xPos = $aMatches[2];
  785. $yPos = $aMatches[3];
  786. $oNode = $this->GetNode($sId);
  787. $oNode->x = (float)$xPos;
  788. $oNode->y = (float)$yPos;
  789. }
  790. }
  791. }
  792. public function GetBoundingBox()
  793. {
  794. $xMin = null;
  795. $xMax = null;
  796. $yMin = null;
  797. $yMax = null;
  798. $oIterator = new RelationTypeIterator($this, 'Node');
  799. foreach($oIterator as $sId => $oNode)
  800. {
  801. if ($xMin === null) // First element in the loop
  802. {
  803. $xMin = $oNode->x - $oNode->GetWidth();
  804. $xMax = $oNode->x + $oNode->GetWidth();
  805. $yMin = $oNode->y - $oNode->GetHeight();
  806. $yMax = $oNode->y + $oNode->GetHeight();
  807. }
  808. else
  809. {
  810. $xMin = min($xMin, $oNode->x - $oNode->GetWidth() / 2);
  811. $xMax = max($xMax, $oNode->x + $oNode->GetWidth() / 2);
  812. $yMin = min($yMin, $oNode->y - $oNode->GetHeight() / 2);
  813. $yMax = max($yMax, $oNode->y + $oNode->GetHeight() / 2);
  814. }
  815. }
  816. return array('xmin' => $xMin, 'xmax' => $xMax, 'ymin' => $yMin, 'ymax' => $yMax);
  817. }
  818. function Translate($dx, $dy)
  819. {
  820. $oIterator = new RelationTypeIterator($this, 'Node');
  821. foreach($oIterator as $sId => $oNode)
  822. {
  823. $oNode->x += $dx;
  824. $oNode->y += $dy;
  825. }
  826. }
  827. public function UpdatePositions($aPositions)
  828. {
  829. foreach($aPositions as $sNodeId => $aPos)
  830. {
  831. $oNode = $this->GetNode($sNodeId);
  832. if ($oNode != null)
  833. {
  834. $oNode->x = $aPos['x'];
  835. $oNode->y = $aPos['y'];
  836. }
  837. }
  838. }
  839. /**
  840. * Renders as JSON string suitable for loading into the simple_graph widget
  841. */
  842. function GetAsJSON($sContextKey)
  843. {
  844. $aContextDefs = $this->GetContextDefinitions($sContextKey, false);
  845. $aData = array('nodes' => array(), 'edges' => array());
  846. $iGroupIdx = 0;
  847. $oIterator = new RelationTypeIterator($this, 'Node');
  848. foreach($oIterator as $sId => $oNode)
  849. {
  850. if ($oNode instanceof DisplayableGroupNode)
  851. {
  852. $aGroups[] = $oNode->GetObjects();
  853. $oNode->SetProperty('group_index', $iGroupIdx);
  854. $iGroupIdx++;
  855. }
  856. $aData['nodes'][] = $oNode->GetForRaphael($aContextDefs);
  857. }
  858. $oIterator = new RelationTypeIterator($this, 'Edge');
  859. foreach($oIterator as $sId => $oEdge)
  860. {
  861. $aEdge = array();
  862. $aEdge['id'] = $oEdge->GetId();
  863. $aEdge['source_node_id'] = $oEdge->GetSourceNode()->GetId();
  864. $aEdge['sink_node_id'] = $oEdge->GetSinkNode()->GetId();
  865. $fOpacity = ($oEdge->GetSinkNode()->GetProperty('is_reached') && $oEdge->GetSourceNode()->GetProperty('is_reached') ? 1 : 0.2);
  866. $aEdge['attr'] = array('opacity' => $fOpacity, 'stroke' => '#000');
  867. $aData['edges'][] = $aEdge;
  868. }
  869. return json_encode($aData);
  870. }
  871. /**
  872. * Renders the graph in a PDF document: centered in the current page
  873. * @param PDFPage $oPage The PDFPage representing the PDF document to draw into
  874. * @param string $sComments An optional comment to display next to the graph (HTML entities will be escaped, \n replaced by <br/>)
  875. * @param string $sContextKey The key to fetch the queries in the configuration. Example: itop-tickets/relation_context/UserRequest/impacts/down
  876. * @param float $xMin Left coordinate of the bounding box to display the graph
  877. * @param float $xMax Right coordinate of the bounding box to display the graph
  878. * @param float $yMin Top coordinate of the bounding box to display the graph
  879. * @param float $yMax Bottom coordinate of the bounding box to display the graph
  880. */
  881. function RenderAsPDF(PDFPage $oPage, $sComments = '', $sContextKey, $xMin = -1, $xMax = -1, $yMin = -1, $yMax = -1)
  882. {
  883. $aContextDefs = $this->GetContextDefinitions($sContextKey, false); // No need to develop the parameters
  884. $oPdf = $oPage->get_tcpdf();
  885. $aBB = $this->GetBoundingBox();
  886. $this->Translate(-$aBB['xmin'], -$aBB['ymin']);
  887. $aMargins = $oPdf->getMargins();
  888. if ($xMin == -1)
  889. {
  890. $xMin = $aMargins['left'];
  891. }
  892. if ($xMax == -1)
  893. {
  894. $xMax = $oPdf->getPageWidth() - $aMargins['right'];
  895. }
  896. if ($yMin == -1)
  897. {
  898. $yMin = $aMargins['top'];
  899. }
  900. if ($yMax == -1)
  901. {
  902. $yMax = $oPdf->getPageHeight() - $aMargins['bottom'];
  903. }
  904. $fBreakMargin = $oPdf->getBreakMargin();
  905. $oPdf->SetAutoPageBreak(false);
  906. $aRemainingArea = $this->RenderKey($oPdf, $sComments, $xMin, $yMin, $xMax, $yMax, $aContextDefs);
  907. $xMin = $aRemainingArea['xmin'];
  908. $xMax = $aRemainingArea['xmax'];
  909. $yMin = $aRemainingArea['ymin'];
  910. $yMax = $aRemainingArea['ymax'];
  911. //$oPdf->Rect($xMin, $yMin, $xMax - $xMin, $yMax - $yMin, 'D', array(), array(225, 50, 50));
  912. $fPageW = $xMax - $xMin;
  913. $fPageH = $yMax - $yMin;
  914. $w = $aBB['xmax'] - $aBB['xmin'];
  915. $h = $aBB['ymax'] - $aBB['ymin'] + 10; // Extra space for the labels which may appear "below" the icons
  916. $fScale = min($fPageW / $w, $fPageH / $h);
  917. $dx = ($fPageW - $fScale * $w) / 2;
  918. $dy = ($fPageH - $fScale * $h) / 2;
  919. $this->Translate(($xMin + $dx)/$fScale, ($yMin + $dy)/$fScale);
  920. $oIterator = new RelationTypeIterator($this, 'Edge');
  921. $iLoopTimeLimit = MetaModel::GetConfig()->Get('max_execution_time_per_loop');
  922. foreach($oIterator as $sId => $oEdge)
  923. {
  924. set_time_limit($iLoopTimeLimit);
  925. $oEdge->RenderAsPDF($oPdf, $this, $fScale, $aContextDefs);
  926. }
  927. $oIterator = new RelationTypeIterator($this, 'Node');
  928. foreach($oIterator as $sId => $oNode)
  929. {
  930. set_time_limit($iLoopTimeLimit);
  931. $oNode->RenderAsPDF($oPdf, $this, $fScale, $aContextDefs);
  932. }
  933. $oIterator = new RelationTypeIterator($this, 'Node');
  934. $oPdf->SetAutoPageBreak(true, $fBreakMargin);
  935. $oPdf->SetAlpha(1);
  936. }
  937. /**
  938. * Renders (in PDF) the key (legend) of the graphics vertically to the left of the specified zone (xmin,ymin, xmax,ymax),
  939. * and the comment (if any) at the bottom of the page. Returns the position of remaining area.
  940. * @param TCPDF $oPdf
  941. * @param string $sComments
  942. * @param float $xMin
  943. * @param float $yMin
  944. * @param float $xMax
  945. * @param float $yMax
  946. * @param hash $aContextDefs
  947. * @return hash An array ('xmin' => , 'xmax' => ,'ymin' => , 'ymax' => ) of the remaining available area to paint the graph
  948. */
  949. protected function RenderKey(TCPDF $oPdf, $sComments, $xMin, $yMin, $xMax, $yMax, $aContextDefs)
  950. {
  951. $fFontSize = 7; // in mm
  952. $fIconSize = 6; // in mm
  953. $fPadding = 1; // in mm
  954. $oIterator = new RelationTypeIterator($this, 'Node');
  955. $fMaxWidth = max($oPdf->GetStringWidth(Dict::S('UI:Relation:Key')) - $fIconSize, $oPdf->GetStringWidth(Dict::S('UI:Relation:Comments')) - $fIconSize);
  956. $aClasses = array();
  957. $aIcons = array();
  958. $aContexts = array();
  959. $aContextIcons = array();
  960. $oPdf->SetFont('dejavusans', '', $fFontSize, '', true);
  961. foreach($oIterator as $sId => $oNode)
  962. {
  963. if ($sClass = $oNode->GetProperty('class'))
  964. {
  965. if (!array_key_exists($sClass, $aClasses))
  966. {
  967. $sClassLabel = MetaModel::GetName($sClass);
  968. $width = $oPdf->GetStringWidth($sClassLabel);
  969. $fMaxWidth = max($width, $fMaxWidth);
  970. $aClasses[$sClass] = $sClassLabel;
  971. $sIconUrl = $oNode->GetProperty('icon_url');
  972. $sIconPath = str_replace(utils::GetAbsoluteUrlModulesRoot(), APPROOT.'env-'.utils::GetCurrentEnvironment().'/', $sIconUrl);
  973. $aIcons[$sClass] = $sIconPath;
  974. }
  975. }
  976. $aContextRootCauses = $oNode->GetProperty('context_root_causes');
  977. if (!is_null($aContextRootCauses))
  978. {
  979. foreach($aContextRootCauses as $key => $aObjects)
  980. {
  981. $aContexts[$key] = Dict::S($aContextDefs[$key]['dict']);
  982. $aContextIcons[$key] = APPROOT.'env-'.utils::GetCurrentEnvironment().'/'.$aContextDefs[$key]['icon'];
  983. }
  984. }
  985. }
  986. $oPdf->SetXY($xMin + $fPadding, $yMin + $fPadding);
  987. $yPos = $yMin + $fPadding;
  988. $oPdf->SetFillColor(225, 225, 225);
  989. $oPdf->Cell($fIconSize + $fPadding + $fMaxWidth, $fIconSize + $fPadding, Dict::S('UI:Relation:Key'), 0 /* border */, 1 /* ln */, 'C', true /* fill */);
  990. $yPos += $fIconSize + 2*$fPadding;
  991. foreach($aClasses as $sClass => $sLabel)
  992. {
  993. $oPdf->SetX($xMin + $fIconSize + $fPadding);
  994. $oPdf->Cell(0, $fIconSize + 2*$fPadding, $sLabel, 0 /* border */, 1 /* ln */);
  995. $oPdf->Image($aIcons[$sClass], $xMin+1, $yPos, $fIconSize, $fIconSize);
  996. $yPos += $fIconSize + 2*$fPadding;
  997. }
  998. foreach($aContexts as $key => $sLabel)
  999. {
  1000. $oPdf->SetX($xMin + $fIconSize + $fPadding);
  1001. $oPdf->Cell(0, $fIconSize + 2*$fPadding, $sLabel, 0 /* border */, 1 /* ln */);
  1002. $oPdf->Image($aContextIcons[$key], $xMin+1+$fIconSize*0.125, $yPos+$fIconSize*0.125, $fIconSize*0.75, $fIconSize*0.75);
  1003. $yPos += $fIconSize + 2*$fPadding;
  1004. }
  1005. $oPdf->Rect($xMin, $yMin, $fMaxWidth + $fIconSize + 3*$fPadding, $yMax - $yMin, 'D');
  1006. if ($sComments != '')
  1007. {
  1008. // Draw the comment text (surrounded by a rectangle)
  1009. $xPos = $xMin + $fMaxWidth + $fIconSize + 4*$fPadding;
  1010. $w = $xMax - $xPos - 2*$fPadding;
  1011. $iNbLines = 1;
  1012. $sText = '<p>'.str_replace("\n", '<br/>', htmlentities($sComments, ENT_QUOTES, 'UTF-8'), $iNbLines).'</p>';
  1013. $fLineHeight = $oPdf->getStringHeight($w, $sText);
  1014. $h = (1+$iNbLines) * $fLineHeight;
  1015. $yPos = $yMax - 2*$fPadding - $h;
  1016. $oPdf->writeHTMLCell($w, $h, $xPos + $fPadding, $yPos + $fPadding, $sText, 0 /* border */, 1 /* ln */);
  1017. $oPdf->Rect($xPos, $yPos, $w + 2*$fPadding, $h + 2*$fPadding, 'D');
  1018. $yMax = $yPos - $fPadding;
  1019. }
  1020. return array('xmin' => $xMin + $fMaxWidth + $fIconSize + 4*$fPadding, 'xmax' => $xMax, 'ymin' => $yMin, 'ymax' => $yMax);
  1021. }
  1022. /**
  1023. * Get the context definitions from the parameters / configuration. The format of the "key" string is:
  1024. * <module>/relation_context/<class>/<relation>/<direction>
  1025. * The values will be retrieved for the given class and all its parents and merged together as a single array.
  1026. * Entries with an invalid query are removed from the list.
  1027. * @param string $sContextKey The key to fetch the queries in the configuration. Example: itop-tickets/relation_context/UserRequest/impacts/down
  1028. * @param bool $bDevelopParams Whether or not to substitute the parameters inside the queries with the supplied "context params"
  1029. * @param array $aContextParams Arguments for the queries (via ToArgs()) if $bDevelopParams == true
  1030. * @return multitype:multitype:string
  1031. */
  1032. public function GetContextDefinitions($sContextKey, $bDevelopParams = true, $aContextParams = array())
  1033. {
  1034. $aLevels = explode('/', $sContextKey);
  1035. $sLeafClass = $aLevels[2];
  1036. $aRelationContext = MetaModel::GetConfig()->GetModuleSetting($aLevels[0], $aLevels[1], array());
  1037. $aContextDefs = array();
  1038. foreach(MetaModel::EnumParentClasses($sLeafClass, ENUM_PARENT_CLASSES_ALL) as $sClass)
  1039. {
  1040. if (isset($aRelationContext[$sClass][$aLevels[3]][$aLevels[4]]['items']))
  1041. {
  1042. $aContextDefs = array_merge($aContextDefs, $aRelationContext[$sClass][$aLevels[3]][$aLevels[4]]['items']);
  1043. }
  1044. }
  1045. // Check if the queries are valid
  1046. foreach($aContextDefs as $sKey => $sDefs)
  1047. {
  1048. $sOQL = $aContextDefs[$sKey]['oql'];
  1049. try
  1050. {
  1051. // Expand the parameters. If anything goes wrong, then the query is considered as invalid and removed from the list
  1052. $oSearch = DBObjectSearch::FromOQL($sOQL);
  1053. $aContextDefs[$sKey]['oql'] = $oSearch->ToOQL($bDevelopParams, $aContextParams);
  1054. }
  1055. catch(Exception $e)
  1056. {
  1057. unset($aContextDefs[$sKey]);
  1058. }
  1059. }
  1060. return $aContextDefs;
  1061. }
  1062. /**
  1063. * Display the graph inside the given page, with the "filter" drawer above it
  1064. * @param WebPage $oP
  1065. * @param hash $aResults
  1066. * @param string $sRelation
  1067. * @param ApplicationContext $oAppContext
  1068. * @param array $aExcludedObjects
  1069. */
  1070. function Display(WebPage $oP, $aResults, $sRelation, ApplicationContext $oAppContext, $aExcludedObjects = array(), $sObjClass = null, $iObjKey = null, $sContextKey, $aContextParams = array())
  1071. {
  1072. $aContextDefs = $this->GetContextDefinitions($sContextKey, true, $aContextParams);
  1073. $aExcludedByClass = array();
  1074. foreach($aExcludedObjects as $oObj)
  1075. {
  1076. if (!array_key_exists(get_class($oObj), $aExcludedByClass))
  1077. {
  1078. $aExcludedByClass[get_class($oObj)] = array();
  1079. }
  1080. $aExcludedByClass[get_class($oObj)][] = $oObj->GetKey();
  1081. }
  1082. $oP->add("<div id=\"ds_flash\" class=\"SearchDrawer\" style=\"display:none;\">\n");
  1083. $oP->add_ready_script(
  1084. <<<EOF
  1085. $( "#tabbedContent_0" ).tabs({ heightStyle: "fill" });
  1086. $("#dh_flash").click( function() {
  1087. $("#ds_flash").slideToggle('normal', function() { $("#ds_flash").parent().resize(); } );
  1088. $("#dh_flash").toggleClass('open');
  1089. });
  1090. $('#ReloadMovieBtn').button().button('disable');
  1091. EOF
  1092. );
  1093. $aSortedElements = array();
  1094. foreach($aResults as $sClassIdx => $aObjects)
  1095. {
  1096. foreach($aObjects as $oCurrObj)
  1097. {
  1098. $sSubClass = get_class($oCurrObj);
  1099. $aSortedElements[$sSubClass] = MetaModel::GetName($sSubClass);
  1100. }
  1101. }
  1102. asort($aSortedElements);
  1103. $idx = 0;
  1104. foreach($aSortedElements as $sSubClass => $sClassName)
  1105. {
  1106. $oP->add("<span style=\"padding-right:2em; white-space:nowrap;\"><input type=\"checkbox\" id=\"exclude_$idx\" name=\"excluded[]\" value=\"$sSubClass\" checked onChange=\"$('#ReloadMovieBtn').button('enable')\"><label for=\"exclude_$idx\">&nbsp;".MetaModel::GetClassIcon($sSubClass)."&nbsp;$sClassName</label></span> ");
  1107. $idx++;
  1108. }
  1109. $oP->add("<p style=\"text-align:right\"><button type=\"button\" id=\"ReloadMovieBtn\" onClick=\"DoReload()\">".Dict::S('UI:Button:Refresh')."</button></p>");
  1110. $oP->add("</div>\n");
  1111. $oP->add("<div class=\"HRDrawer\"></div>\n");
  1112. $oP->add("<div id=\"dh_flash\" class=\"DrawerHandle\">".Dict::S('UI:ElementsDisplayed')."</div>\n");
  1113. $aAdditionalContexts = array();
  1114. foreach($aContextDefs as $sKey => $aDefinition)
  1115. {
  1116. $aAdditionalContexts[] = array('key' => $sKey, 'label' => Dict::S($aDefinition['dict']), 'oql' => $aDefinition['oql']);
  1117. }
  1118. $sDirection = utils::ReadParam('d', 'horizontal');
  1119. $iGroupingThreshold = utils::ReadParam('g', 5);
  1120. $oP->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/fraphael.js');
  1121. $oP->add_linked_stylesheet(utils::GetAbsoluteUrlAppRoot().'css/jquery.contextMenu.css');
  1122. $oP->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/jquery.contextMenu.js');
  1123. $oP->add_linked_script(utils::GetAbsoluteUrlAppRoot().'js/simple_graph.js');
  1124. try
  1125. {
  1126. $this->InitFromGraphviz();
  1127. $sExportAsPdfURL = '';
  1128. $sExportAsPdfURL = utils::GetAbsoluteUrlAppRoot().'pages/ajax.render.php?operation=relation_pdf&relation='.$sRelation.'&direction='.($this->bDirectionDown ? 'down' : 'up');
  1129. $oAppcontext = new ApplicationContext();
  1130. $sContext = $oAppContext->GetForLink();
  1131. $sDrillDownURL = utils::GetAbsoluteUrlAppRoot().'pages/UI.php?operation=details&class=%1$s&id=%2$s&'.$sContext;
  1132. $sExportAsDocumentURL = utils::GetAbsoluteUrlAppRoot().'pages/ajax.render.php?operation=relation_attachment&relation='.$sRelation.'&direction='.($this->bDirectionDown ? 'down' : 'up');
  1133. $sLoadFromURL = utils::GetAbsoluteUrlAppRoot().'pages/ajax.render.php?operation=relation_json&relation='.$sRelation.'&direction='.($this->bDirectionDown ? 'down' : 'up');
  1134. $sAttachmentExportTitle = '';
  1135. if (($sObjClass != null) && ($iObjKey != null))
  1136. {
  1137. $oTargetObj = MetaModel::GetObject($sObjClass, $iObjKey, false);
  1138. if ($oTargetObj)
  1139. {
  1140. $sAttachmentExportTitle = Dict::Format('UI:Relation:AttachmentExportOptions_Name', $oTargetObj->GetName());
  1141. }
  1142. }
  1143. $sId = 'graph';
  1144. $oP->add('<div id="'.$sId.'" class="simple-graph"></div>');
  1145. $aParams = array(
  1146. 'source_url' => $sLoadFromURL,
  1147. 'sources' => ($this->bDirectionDown ? $this->aSourceObjects : $this->aSinkObjects),
  1148. 'excluded' => $aExcludedByClass,
  1149. 'grouping_threshold' => $iGroupingThreshold,
  1150. 'export_as_pdf' => array('url' => $sExportAsPdfURL, 'label' => Dict::S('UI:Relation:ExportAsPDF')),
  1151. 'export_as_attachment' => array('url' => $sExportAsDocumentURL, 'label' => Dict::S('UI:Relation:ExportAsAttachment'), 'obj_class' => $sObjClass, 'obj_key' => $iObjKey),
  1152. 'drill_down' => array('url' => $sDrillDownURL, 'label' => Dict::S('UI:Relation:DrillDown')),
  1153. 'labels' => array(
  1154. 'export_pdf_title' => Dict::S('UI:Relation:PDFExportOptions'),
  1155. 'export_as_attachment_title' => $sAttachmentExportTitle,
  1156. 'export' => Dict::S('UI:Button:Export'),
  1157. 'cancel' => Dict::S('UI:Button:Cancel'),
  1158. 'title' => Dict::S('UI:RelationOption:Title'),
  1159. 'untitled' => Dict::S('UI:RelationOption:Untitled'),
  1160. 'include_list' => Dict::S('UI:RelationOption:IncludeList'),
  1161. 'comments' => Dict::S('UI:RelationOption:Comments'),
  1162. 'grouping_threshold' => Dict::S('UI:RelationOption:GroupingThreshold'),
  1163. 'refresh' => Dict::S('UI:Button:Refresh'),
  1164. 'check_all' => Dict::S('UI:SearchValue:CheckAll'),
  1165. 'uncheck_all' => Dict::S('UI:SearchValue:UncheckAll'),
  1166. 'none_selected' => Dict::S('UI:Relation:NoneSelected'),
  1167. 'nb_selected' => Dict::S('UI:SearchValue:NbSelected'),
  1168. 'additional_context_info' => Dict::S('UI:Relation:AdditionalContextInfo'),
  1169. 'zoom' => Dict::S('UI:Relation:Zoom'),
  1170. 'loading' => Dict::S('UI:Loading'),
  1171. ),
  1172. 'page_format' => array(
  1173. 'label' => Dict::S('UI:Relation:PDFExportPageFormat'),
  1174. 'values' => array(
  1175. 'A3' => Dict::S('UI:PageFormat_A3'),
  1176. 'A4' => Dict::S('UI:PageFormat_A4'),
  1177. 'Letter' => Dict::S('UI:PageFormat_Letter'),
  1178. ),
  1179. ),
  1180. 'page_orientation' => array(
  1181. 'label' => Dict::S('UI:Relation:PDFExportPageOrientation'),
  1182. 'values' => array(
  1183. 'P' => Dict::S('UI:PageOrientation_Portrait'),
  1184. 'L' => Dict::S('UI:PageOrientation_Landscape'),
  1185. ),
  1186. ),
  1187. 'additional_contexts' => $aAdditionalContexts,
  1188. 'context_key' => $sContextKey,
  1189. );
  1190. if (!extension_loaded('gd'))
  1191. {
  1192. // PDF export requires GD
  1193. unset($aParams['export_as_pdf']);
  1194. }
  1195. if (!extension_loaded('gd') || is_null($sObjClass) || is_null($iObjKey))
  1196. {
  1197. // Export as Attachment requires GD (for building the PDF) AND a valid objclass/objkey couple
  1198. unset($aParams['export_as_attachment']);
  1199. }
  1200. $oP->add_ready_script("$('#$sId').simple_graph(".json_encode($aParams).");");
  1201. }
  1202. catch(Exception $e)
  1203. {
  1204. $oP->add('<div>'.$e->getMessage().'</div>');
  1205. }
  1206. $oP->add_script(
  1207. <<<EOF
  1208. function DoReload()
  1209. {
  1210. $('#ReloadMovieBtn').button('disable');
  1211. try
  1212. {
  1213. var aExcluded = [];
  1214. $('input[name^=excluded]').each( function() {
  1215. if (!$(this).prop('checked'))
  1216. {
  1217. aExcluded.push($(this).val());
  1218. }
  1219. } );
  1220. $('#graph').simple_graph('option', {excluded_classes: aExcluded});
  1221. $('#graph').simple_graph('reload');
  1222. }
  1223. catch(err)
  1224. {
  1225. alert(err);
  1226. }
  1227. }
  1228. EOF
  1229. );
  1230. }
  1231. }