diff --git a/rsrc/remarkup-image.css b/rsrc/remarkup-image.css --- a/rsrc/remarkup-image.css +++ b/rsrc/remarkup-image.css @@ -2,7 +2,62 @@ * @provides diagram-remarkup-image-css */ -.diagram-container > .diagram-content { +.diagram-content { + display: flex; +} + +.diagram-content.diagram-center { + justify-content: center; +} + +.diagram-content.diagram-right { + justify-content: right; +} + +.diagram-content.diagram-float-left { + float: left; + margin-right: 1em; max-width: 100%; - cursor: pointer; +} + +.diagram-content.diagram-float-right { + float: right; + margin-left: 1em; + max-width: 100%; +} + +.diagram-content > .diagram-container { + max-width: 100%; + overflow: hidden !important; +} + +.diagram-content > .diagram-container.full { + flex-grow: 1; +} + +/* GraphViewer is only rendering when container has width */ +.diagram-content > .diagram-container:empty { + min-width: 1px; +} + +/* Fixing images in GraphViewer lightbox toolbar */ +.geDiagramContainer + .geAdaptiveAsset + div > span > img { + display: initial; +} + +/* Fixing centering of button in GraphViewer toolbar */ +body > div[style*="align-items:"]:not([class]) { + line-height: initial; + display: flex; +} + +/* reset some styles for correct svg rendering */ + +.diagram-content > .diagram-container, .geDiagramContainer { + line-height: initial; +} + +.diagram-content > .diagram-container td, +.geDiagramContainer td { + padding: revert-layer; } diff --git a/rsrc/remarkup-image.js b/rsrc/remarkup-image.js --- a/rsrc/remarkup-image.js +++ b/rsrc/remarkup-image.js @@ -4,24 +4,58 @@ JX.onload(function() { - var singleClickTimeout = null; - - JX.Stratcom.listen( - 'click', - ['diagram-remarkup-image'], - function(evt) { - var detail = evt.getRawEvent().detail; - - if (detail === 1) { - singleClickTimeout = window.setTimeout(function() { - window.open('/diagram/data/' + evt.getTarget().dataset.diagramVersion); - singleClickTimeout = null; - }, 300); - } else if (detail === 2) { - window.clearTimeout(singleClickTimeout); - window.open('/diagram/DIAG' + evt.getTarget().dataset.diagramId); + var viewerScriptURI = 'https://viewer.diagrams.net/js/viewer-static.min.js'; + var viewerScriptPrepended = false; + var viewerScriptBlocked = false; + + document.addEventListener('securitypolicyviolation', function(e) { + if (viewerScriptURI === e.blockedURI) { + viewerScriptBlocked = true; + processElements(); + } + }); + + function processViewer(viewer) { + viewer.toolbar.appendChild( + viewer.createToolbarButton( + function() { window.open(viewer.graphConfig.edit); }, + '', + null, + true + ) + ); + } + + function processElements() { + console.log('processElements: ', document.querySelectorAll('.diagram-container:empty')); + document.querySelectorAll('.diagram-container:empty').forEach(function(container) { + if (window.GraphViewer) { + GraphViewer.createViewerForElement(container, processViewer); + } else if (viewerScriptBlocked) { + container.innerHTML = 'Diagram Viewer could not be loaded.'; } + }); + } + + function init() { + if (!document.querySelector('.diagram-container')) { + return; } - ); + if (viewerScriptPrepended || viewerScriptBlocked) { + processElements(); + return; + } + + var viewerScript = document.createElement('script'); + viewerScript.setAttribute('src', viewerScriptURI); + viewerScript.addEventListener('load', processElements); + + document.body.prepend(viewerScript); + + viewerScriptPrepended = true; + } + + init(); + JX.Request.listen('done', function() { setTimeout(init); }); }); diff --git a/src/__phutil_library_map__.php b/src/__phutil_library_map__.php --- a/src/__phutil_library_map__.php +++ b/src/__phutil_library_map__.php @@ -26,6 +26,7 @@ 'DiagramTransactionType' => 'xaction/DiagramTransactionType.php', 'DiagramUploadConduitAPIMethod' => 'conduit/DiagramUploadConduitAPIMethod.php', 'DiagramVersion' => 'storage/DiagramVersion.php', + 'DrawioPngParser' => 'parser/DrawioPngParser.php', 'PhabricatorDiagramQuery' => 'query/PhabricatorDiagramQuery.php', 'PhabricatorDiagramTransactionQuery' => 'query/PhabricatorDiagramTransactionQuery.php', 'PhabricatorDiagramVersionQuery' => 'query/PhabricatorDiagramVersionQuery.php', @@ -61,6 +62,7 @@ 'PhabricatorDestructibleInterface', 'PhabricatorPolicyInterface', ), + 'DrawioPngParser' => 'Phobject', 'PhabricatorDiagramQuery' => 'PhabricatorCursorPagedPolicyAwareQuery', 'PhabricatorDiagramTransactionQuery' => 'PhabricatorApplicationTransactionQuery', 'PhabricatorDiagramVersionQuery' => 'PhabricatorCursorPagedPolicyAwareQuery', diff --git a/src/application/DiagramApplication.php b/src/application/DiagramApplication.php --- a/src/application/DiagramApplication.php +++ b/src/application/DiagramApplication.php @@ -1,6 +1,11 @@ addContentSecurityPolicyURI('script-src', 'https://viewer.diagrams.net'); + } + public function getName() { return pht('Diagrams'); } diff --git a/src/celerity/map.php b/src/celerity/map.php --- a/src/celerity/map.php +++ b/src/celerity/map.php @@ -11,15 +11,15 @@ 'diagram-extension.css' => '7ad39d5d', 'iframe-toolbtn.css' => '35ad6f49', 'iframe-toolbtn.js' => '26d75a35', - 'remarkup-image.css' => '42b46bf1', - 'remarkup-image.js' => '64e7e9e1', + 'remarkup-image.css' => '51ddb637', + 'remarkup-image.js' => '842accb1', ), 'symbols' => array( 'diagram-css-extension' => '7ad39d5d', 'diagram-css-iframe-toolbtn' => '35ad6f49', 'diagram-js-iframe-toolbtn' => '26d75a35', - 'diagram-remarkup-image-css' => '42b46bf1', - 'diagram-remarkup-image-js' => '64e7e9e1', + 'diagram-remarkup-image-css' => '51ddb637', + 'diagram-remarkup-image-js' => '842accb1', 'javelin-behavior-diagram-extension' => 'a0b36dca', ), 'requires' => array(), diff --git a/src/controller/DiagramController.php b/src/controller/DiagramController.php --- a/src/controller/DiagramController.php +++ b/src/controller/DiagramController.php @@ -243,53 +243,19 @@ * They are cut out before the 2 strings are compared. */ public static function equalPngMetaData($base64_1, $base64_2) { - $base64 = array($base64_1, $base64_2); - $textData = array(); - for ($i = 0; $i < 2; $i++) { - $data = base64_decode($base64[$i]); - $fp = fopen('data://text/plain;base64,' . base64_encode($data), 'rb'); - $sig = fread($fp, 8); - if ($sig != "\x89PNG\x0d\x0a\x1a\x0a") { - fclose($fp); - return false; - } - $textData[$i] = array(); - while (!feof($fp)) { - try { - $chunk = unpack('Nlength/a4type', fread($fp, 8)); - } catch (Exception $e) { - // invalid base64 data - return false; - } - if ($chunk['type'] == 'IEND') break; - if ($chunk['type'] == 'tEXt') { - list($key, $val) = explode("\0", fread($fp, $chunk['length'])); - if ($key == 'mxfile') { - // Decode the URL-encoded XML data - $decodedVal = urldecode($val); - // Load the XML and remove the modified and etag attributes - $xml = simplexml_load_string($decodedVal); - unset($xml->attributes()->modified); - unset($xml->attributes()->etag); - // Save the modified XML as the value - $val = $xml->asXML(); - } - $textData[$i][$key] = $val; - fseek($fp, 4, SEEK_CUR); - } else { - fseek($fp, $chunk['length'] + 4, SEEK_CUR); - } - } - fclose($fp); - } + $mxfile1 = DrawioPngParser::getMxfile($base64_1); + $mxfile2 = DrawioPngParser::getMxfile($base64_2); - if (isset($textData[0]['mxfile']) && isset($textData[1]['mxfile'])) { - // Both arrays contain the mxfile key, compare their values - return $textData[0]['mxfile'] == $textData[1]['mxfile']; - } else { - // At least one of the arrays doesn't contain mxfile key, return false + if ($mxfile1 === null || $mxfile2 === null) { return false; } + + unset($mxfile1->attributes()->modified); + unset($mxfile1->attributes()->etag); + unset($mxfile2->attributes()->modified); + unset($mxfile2->attributes()->etag); + + return $mxfile1->asXML() == $mxfile2->asXML(); } /** diff --git a/src/parser/DrawioPngParser.php b/src/parser/DrawioPngParser.php new file mode 100644 --- /dev/null +++ b/src/parser/DrawioPngParser.php @@ -0,0 +1,38 @@ +getEngine()->getConfig('viewer'); @@ -31,70 +27,66 @@ PhabricatorObjectHandle $handle, $options) { - if ($options) { - $params = explode(',', $options); - $params = array_map('trim', $params); - } else { - $params = array(); + $options = $this->getOptions($options); + $fullSize = !!$options['full']; + $contentClass = 'diagram-content'; + $contentStyle = ''; + + switch ($options['layout'] && !$fullSize) { + case 'right': + case 'center': + $contentClass .= ' diagram-' . $options['layout']; + break; + case 'left': + default: + $contentClass .= ' diagram-left'; + break; } - // Generate the appropriate HTML using the data from the Diagram and - // file objects. - $style = ''; - $class = 'diagram-content'; - $alt = ''; - - $has_layout = false; - foreach ($params as $param) { - if (strpos($param, '=') !== false) { - list($key, $value) = explode('=', $param, 2); - } else { - $key = $param; - $value = null; - } - switch ($key) { - case 'layout': - $has_layout = true; - if ($value === 'left') { - $class .= ' phabricator-remarkup-embed-layout-left'; - } else if ($value === 'right') { - $class .= ' phabricator-remarkup-embed-layout-right'; - } + if ($options['float'] && !$fullSize) { + switch ($options['layout']) { + case 'right': + $contentClass .= ' diagram-float-right'; break; - case 'float': - $class .= ' phabricator-remarkup-embed-float-left'; - break; - case 'size': - if ($value === 'full') { - $style .= 'width: 100%;'; - } - break; - case 'alt': - $alt = phutil_escape_html($value); + case 'left': + default: + $contentClass .= ' diagram-float-left'; break; } } - if ($has_layout == false) { - $class .= ' phabricator-remarkup-embed-layout-left'; + $width = $options['width']; + if ($width && !$fullSize && preg_match('/^(?:\d*\\.)?\d+(%|px)?$/', $width)) { + if (is_numeric($width)) { + $contentStyle = 'width: ' . $width . 'px;'; + } else { + $contentStyle = 'width: ' . $width . ';'; + } } + $mxGraphConfig = array( + 'responsive' => $fullSize, + 'resize' => true, + 'editable' => true, + 'edit' => PhabricatorEnv::getURI('/diagram/DIAG' . $diagram->getDiagramID()), + 'lightbox' => true, + 'toolbar' => 'pages' . ($fullSize ? '' : ' zoom'), + 'toolbar-position' => 'inline', + 'page' => $options['page'], + 'xml' => DrawioPngParser::getMxfile($diagram->getBase64Data())->asXML() + ); + $output = phutil_tag( 'div', array( - 'class' => 'diagram-container', + 'class' => $contentClass, + 'style' => $contentStyle, ), phutil_tag( - 'img', + 'div', array( - 'style' => $style, - 'class' => $class, - 'src' => 'data:image/png;base64,' . $diagram->getBase64Data(), - 'alt' => $alt, - 'data-sigil' => 'diagram-remarkup-image', - 'data-diagram-version' => $diagram->getPHID(), - 'data-diagram-id' => $diagram->getDiagramID(), - 'title' => 'Double click to edit...' + 'class' => 'diagram-container' . ($fullSize ? ' full' : ''), + 'data-mxgraph' => json_encode($mxGraphConfig) ) ) ); @@ -102,6 +94,24 @@ return $output; } + private function getOptions($option_string) { + $options = array( + 'full' => false, + 'layout' => 'left', + 'float' => false, + 'width' => '', + 'page' => 0 + ); + + if ($option_string) { + $option_string = trim($option_string, ', '); + $parser = new PhutilSimpleOptions(); + $options = $parser->parse($option_string) + $options; + } + + return $options; + } + public function getDocumentation() { return <<