UNPKG

drawio-offline

Version:
1,773 lines (1,533 loc) 297 kB
/** * Copyright (c) 2006-2020, JGraph Ltd * Copyright (c) 2006-2020, draw.io AG */ var StorageDialog = function(editorUi, fn, rowLimit) { rowLimit = (rowLimit != null) ? rowLimit : 2; var div = document.createElement('div'); div.style.textAlign = 'center'; div.style.whiteSpace = 'nowrap'; div.style.paddingTop = '0px'; div.style.paddingBottom = '20px'; var buttons = document.createElement('div'); buttons.style.border = '1px solid #d3d3d3'; buttons.style.borderWidth = '1px 0px 1px 0px'; buttons.style.padding = '10px 0px 20px 0px'; var count = 0; var container = document.createElement('div'); container.style.paddingTop = '2px'; buttons.appendChild(container); var p3 = document.createElement('p'); function addLogo(img, title, mode, clientName, labels, clientFn) { if (++count > rowLimit) { mxUtils.br(container); count = 0; } var button = document.createElement('a'); button.style.overflow = 'hidden'; button.style.display = 'inline-block'; button.className = 'geBaseButton'; button.style.boxSizing = 'border-box'; button.style.fontSize = '11px'; button.style.position = 'relative'; button.style.margin = '4px'; button.style.marginTop = '8px'; button.style.marginBottom = '0px'; button.style.padding = '8px 10px 8px 10px'; button.style.width = '88px'; button.style.height = '100px'; button.style.whiteSpace = 'nowrap'; button.setAttribute('title', title); var label = document.createElement('div'); label.style.textOverflow = 'ellipsis'; label.style.overflow = 'hidden'; label.style.position = 'absolute'; label.style.bottom = '8px'; label.style.left = '0px'; label.style.right = '0px'; mxUtils.write(label, title); button.appendChild(label); if (img != null) { var logo = document.createElement('img'); logo.setAttribute('src', img); logo.setAttribute('border', '0'); logo.setAttribute('align', 'absmiddle'); logo.style.width = '60px'; logo.style.height = '60px'; logo.style.paddingBottom = '6px'; button.appendChild(logo); } else { label.style.paddingTop = '5px'; label.style.whiteSpace = 'normal'; // Handles special case if (mxClient.IS_IOS) { button.style.padding = '0px 10px 20px 10px'; button.style.top = '6px'; } else if (mxClient.IS_FF) { label.style.paddingTop = '0px'; label.style.marginTop = '-2px'; } } if (labels != null) { for (var i = 0; i < labels.length; i++) { mxUtils.br(label); mxUtils.write(label, labels[i]); } } function initButton() { mxEvent.addListener(button, 'click', (clientFn != null) ? clientFn : function() { // Special case: Redirect all drive users to draw.io pro if (mode == App.MODE_GOOGLE && !editorUi.isDriveDomain()) { window.location.hostname = DriveClient.prototype.newAppHostname; } else if (mode == App.MODE_GOOGLE && editorUi.spinner.spin(document.body, mxResources.get('authorizing'))) { // Tries immediate authentication editorUi.drive.checkToken(mxUtils.bind(this, function() { editorUi.spinner.stop(); editorUi.setMode(mode, true); fn(); })); } else if (mode == App.MODE_ONEDRIVE && editorUi.spinner.spin(document.body, mxResources.get('authorizing'))) { // Tries immediate authentication editorUi.oneDrive.checkToken(mxUtils.bind(this, function() { editorUi.spinner.stop(); editorUi.setMode(mode, true); fn(); })); } else { editorUi.setMode(mode, true); fn(); } }); }; // Supports lazy loading if (clientName != null && editorUi[clientName] == null) { logo.style.visibility = 'hidden'; mxUtils.setOpacity(label, 10); var size = 12; var spinner = new Spinner({ lines: 12, // The number of lines to draw length: size, // The length of each line width: 5, // The line thickness radius: 10, // The radius of the inner circle rotate: 0, // The rotation offset color: Editor.isDarkMode() ? '#c0c0c0' : '#000', // #rgb or #rrggbb speed: 1.5, // Rounds per second trail: 60, // Afterglow percentage shadow: false, // Whether to render a shadow hwaccel: false, // Whether to use hardware acceleration top: '40%', zIndex: 2e9 // The z-index (defaults to 2000000000) }); spinner.spin(button); // Timeout after 30 secs var timeout = window.setTimeout(function() { if (editorUi[clientName] == null) { spinner.stop(); button.style.display = 'none'; } }, 30000); editorUi.addListener('clientLoaded', mxUtils.bind(this, function(sender, evt) { if (editorUi[clientName] != null && evt.getProperty('client') == editorUi[clientName]) { window.clearTimeout(timeout); mxUtils.setOpacity(label, 100); logo.style.visibility = ''; spinner.stop(); initButton(); if (clientName == 'drive' && p3.parentNode != null) { p3.parentNode.removeChild(p3); } } })); } else { initButton(); } container.appendChild(button); }; var hd = document.createElement('p'); hd.style.cssText = 'font-size:22px;padding:4px 0 16px 0;margin:0;color:gray;'; mxUtils.write(hd, mxResources.get('saveDiagramsTo') + ':'); div.appendChild(hd); var addButtons = function() { count = 0; if (typeof window.DriveClient === 'function') { addLogo(IMAGE_PATH + '/google-drive-logo.svg', mxResources.get('googleDrive'), App.MODE_GOOGLE, 'drive'); } if (typeof window.OneDriveClient === 'function') { addLogo(IMAGE_PATH + '/onedrive-logo.svg', mxResources.get('oneDrive'), App.MODE_ONEDRIVE, 'oneDrive'); } addLogo(IMAGE_PATH + '/osa_drive-harddisk.png', mxResources.get('device'), App.MODE_DEVICE); if (isLocalStorage && (urlParams['browser'] == '1' || urlParams['offline'] == '1')) { addLogo(IMAGE_PATH + '/osa_database.png', mxResources.get('browser'), App.MODE_BROWSER); } if (typeof window.DropboxClient === 'function') { addLogo(IMAGE_PATH + '/dropbox-logo.svg', mxResources.get('dropbox'), App.MODE_DROPBOX, 'dropbox'); } if (editorUi.gitHub != null) { addLogo(IMAGE_PATH + '/github-logo.svg', mxResources.get('github'), App.MODE_GITHUB, 'gitHub'); } if (editorUi.gitLab != null) { addLogo(IMAGE_PATH + '/gitlab-logo.svg', mxResources.get('gitlab'), App.MODE_GITLAB, 'gitLab'); } }; div.appendChild(buttons); addButtons(); var later = document.createElement('span'); later.style.cssText = 'position:absolute;cursor:pointer;bottom:27px;color:gray;userSelect:none;text-align:center;left:50%;'; mxUtils.setPrefixedStyle(later.style, 'transform', 'translate(-50%,0)'); mxUtils.write(later, mxResources.get('decideLater')); mxEvent.addListener(later, 'click', function() { editorUi.hideDialog(); var prev = Editor.useLocalStorage; editorUi.createFile(editorUi.defaultFilename, null, null, null, null, null, null, true); Editor.useLocalStorage = prev; }); div.appendChild(later); // Checks if Google Drive is missing after a 5 sec delay if (mxClient.IS_SVG && isLocalStorage && urlParams['gapi'] != '0' && (document.documentMode == null || document.documentMode >= 10)) { window.setTimeout(function() { if (editorUi.drive == null) { // To check for Disconnect plugin in chrome use mxClient.IS_GC and check for URL: // chrome-extension://jeoacafpbcihiomhlakheieifhpjdfeo/scripts/vendor/jquery/jquery-2.0.3.min.map p3.style.padding = '7px'; p3.style.fontSize = '9pt'; p3.style.marginTop = '-14px'; p3.innerHTML = '<a style="background-color:#dcdcdc;padding:6px;color:black;text-decoration:none;" ' + 'href="https://desk.draw.io/a/solutions/articles/16000074659" target="_blank">' + '<img border="0" src="' + mxGraph.prototype.warningImage.src + '" align="absmiddle" ' + 'style="margin-top:-4px"> ' + mxResources.get('googleDriveMissingClickHere') + '</a>'; div.appendChild(p3); } }, 5000); } this.container = div; }; /** * Constructs a dialog for creating new files from templates. */ var SplashDialog = function(editorUi) { var div = document.createElement('div'); div.style.textAlign = 'center'; if (mxClient.IS_CHROMEAPP || EditorUi.isElectronApp) { var elt = editorUi.addLanguageMenu(div, true); if (elt != null) { elt.style.bottom = '19px'; } } var serviceCount = editorUi.getServiceCount(); var logo = document.createElement('img'); logo.setAttribute('border', '0'); logo.setAttribute('align', 'absmiddle'); logo.style.width = '32px'; logo.style.height = '32px'; logo.style.marginRight = '8px'; logo.style.marginTop = '-4px'; var buttons = document.createElement('div'); buttons.style.margin = '8px 0px 0px 0px'; buttons.style.padding = '18px 0px 24px 0px'; var service = ''; if (editorUi.mode == App.MODE_GOOGLE) { logo.src = IMAGE_PATH + '/google-drive-logo.svg'; service = mxResources.get('googleDrive'); } else if (editorUi.mode == App.MODE_DROPBOX) { logo.src = IMAGE_PATH + '/dropbox-logo.svg'; service = mxResources.get('dropbox'); } else if (editorUi.mode == App.MODE_ONEDRIVE) { logo.src = IMAGE_PATH + '/onedrive-logo.svg'; service = mxResources.get('oneDrive'); } else if (editorUi.mode == App.MODE_GITHUB) { logo.src = IMAGE_PATH + '/github-logo.svg'; service = mxResources.get('github'); } else if (editorUi.mode == App.MODE_GITLAB) { logo.src = IMAGE_PATH + '/gitlab-logo.svg'; service = mxResources.get('gitlab'); } else if (editorUi.mode == App.MODE_BROWSER) { logo.src = IMAGE_PATH + '/osa_database.png'; service = mxResources.get('browser'); } else if (editorUi.mode == App.MODE_TRELLO) { logo.src = IMAGE_PATH + '/trello-logo.svg'; service = mxResources.get('trello'); } else { logo.src = IMAGE_PATH + '/osa_drive-harddisk.png'; buttons.style.paddingBottom = '10px'; buttons.style.paddingTop = '30px'; service = mxResources.get('device'); } var btn = document.createElement('button'); btn.className = 'geBigButton'; btn.style.marginBottom = '8px'; btn.style.fontSize = '18px'; btn.style.padding = '10px'; btn.style.width = '340px'; if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp) { buttons.style.border = '1px solid #d3d3d3'; buttons.style.borderWidth = '1px 0px 1px 0px'; var table = document.createElement('table'); var tbody = document.createElement('tbody'); var row = document.createElement('tr'); var left = document.createElement('td'); var right = document.createElement('td'); table.setAttribute('align', 'center'); left.appendChild(logo); var title = document.createElement('div'); title.style.fontSize = '22px'; title.style.paddingBottom = '6px'; title.style.color = 'gray'; mxUtils.write(title, service); right.style.textAlign = 'left'; right.appendChild(title); row.appendChild(left); row.appendChild(right); tbody.appendChild(row); table.appendChild(tbody); div.appendChild(table); var change = document.createElement('span'); change.style.cssText = 'position:absolute;cursor:pointer;bottom:27px;color:gray;userSelect:none;text-align:center;left:50%;'; mxUtils.setPrefixedStyle(change.style, 'transform', 'translate(-50%,0)'); mxUtils.write(change, mxResources.get('changeStorage')); mxEvent.addListener(change, 'click', function() { editorUi.hideDialog(false); editorUi.setMode(null); editorUi.clearMode(); editorUi.showSplash(true); }); div.appendChild(change); } else { buttons.style.padding = '42px 0px 56px 0px'; btn.style.marginBottom = '12px'; } mxUtils.write(btn, mxResources.get('createNewDiagram')); mxEvent.addListener(btn, 'click', function() { editorUi.hideDialog(); editorUi.actions.get('new').funct(); }); buttons.appendChild(btn); mxUtils.br(buttons); var btn = document.createElement('button'); btn.className = 'geBigButton'; btn.style.marginBottom = '22px'; btn.style.fontSize = '18px'; btn.style.padding = '10px'; btn.style.width = '340px'; mxUtils.write(btn, mxResources.get('openExistingDiagram')); mxEvent.addListener(btn, 'click', function() { editorUi.actions.get('open').funct(); }); buttons.appendChild(btn); var storage = 'undefined'; if (editorUi.mode == App.MODE_GOOGLE) { storage = mxResources.get('googleDrive'); } else if (editorUi.mode == App.MODE_DROPBOX) { storage = mxResources.get('dropbox'); } else if (editorUi.mode == App.MODE_ONEDRIVE) { storage = mxResources.get('oneDrive'); } else if (editorUi.mode == App.MODE_GITHUB) { storage = mxResources.get('github'); } else if (editorUi.mode == App.MODE_GITLAB) { storage = mxResources.get('gitlab'); } else if (editorUi.mode == App.MODE_TRELLO) { storage = mxResources.get('trello'); } else if (editorUi.mode == App.MODE_DEVICE) { storage = mxResources.get('device'); } else if (editorUi.mode == App.MODE_BROWSER) { storage = mxResources.get('browser'); } if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp) { function addLogout(logout) { btn.style.marginBottom = '24px'; var link = document.createElement('a'); link.style.display = 'inline-block'; link.style.color = 'gray'; link.style.cursor = 'pointer'; link.style.marginTop = '6px'; mxUtils.write(link, mxResources.get('signOut')); // Makes room after last big buttons btn.style.marginBottom = '16px'; buttons.style.paddingBottom = '18px'; mxEvent.addListener(link, 'click', function() { editorUi.confirm(mxResources.get('areYouSure'), function() { logout(); }); }); buttons.appendChild(link); }; if (editorUi.mode == App.MODE_GOOGLE && editorUi.drive != null) { var driveUsers =editorUi.drive.getUsersList(); if (driveUsers.length > 0) { var title = document.createElement('span'); title.style.marginTop = '6px'; mxUtils.write(title, mxResources.get('changeUser') + ':'); // Makes room after last big buttons btn.style.marginBottom = '16px'; buttons.style.paddingBottom = '18px'; buttons.appendChild(title); var usersSelect = document.createElement('select'); usersSelect.style.marginLeft = '4px'; usersSelect.style.width = '140px'; for (var i = 0; i < driveUsers.length; i++) { var option = document.createElement('option'); mxUtils.write(option, driveUsers[i].displayName); option.value = i; usersSelect.appendChild(option); //More info (email) about the user in a disabled option option = document.createElement('option'); option.innerHTML = '&nbsp;&nbsp;&nbsp;'; mxUtils.write(option, '<' + driveUsers[i].email + '>'); option.setAttribute('disabled', 'disabled'); usersSelect.appendChild(option); } //Add account option var option = document.createElement('option'); mxUtils.write(option, mxResources.get('addAccount')); option.value = driveUsers.length; usersSelect.appendChild(option); mxEvent.addListener(usersSelect, 'change', function() { var userIndex = usersSelect.value; var existingAccount = driveUsers.length != userIndex; if (existingAccount) { editorUi.drive.setUser(driveUsers[userIndex]); } editorUi.drive.authorize(existingAccount, function() { editorUi.setMode(App.MODE_GOOGLE); editorUi.hideDialog(); editorUi.showSplash(); }, function(resp) { editorUi.handleError(resp, null, function() { editorUi.hideDialog(); editorUi.showSplash(); }); }, true); }); buttons.appendChild(usersSelect); } else { addLogout(function() { editorUi.drive.logout(); }); } } else if (editorUi.mode == App.MODE_ONEDRIVE && editorUi.oneDrive != null && !editorUi.oneDrive.noLogout) { addLogout(function() { editorUi.oneDrive.logout(); }); } else if (editorUi.mode == App.MODE_GITHUB && editorUi.gitHub != null) { addLogout(function() { editorUi.gitHub.logout(); editorUi.openLink('https://www.github.com/logout'); }); } else if (editorUi.mode == App.MODE_GITLAB && editorUi.gitLab != null) { addLogout(function() { editorUi.gitLab.logout(); editorUi.openLink(DRAWIO_GITLAB_URL + '/users/sign_out'); }); } else if (editorUi.mode == App.MODE_TRELLO && editorUi.trello != null) { if (editorUi.trello.isAuthorized()) { addLogout(function() { editorUi.trello.logout(); }); } } else if (editorUi.mode == App.MODE_DROPBOX && editorUi.dropbox != null) { // NOTE: Dropbox has a logout option in the picker addLogout(function() { editorUi.dropbox.logout(); editorUi.openLink('https://www.dropbox.com/logout'); }); } } div.appendChild(buttons); this.container = div; }; /** * Constructs a new embed dialog */ var EmbedDialog = function(editorUi, result, timeout, ignoreSize, previewFn, title, tweet) { tweet = (tweet != null) ? tweet : 'Check out the diagram I made using @drawio'; var div = document.createElement('div'); var maxSize = 500000; var maxFbSize = 51200; var maxTwitterSize = 7168; // Checks if result is a link var validUrl = /^https?:\/\//.test(result) || /^mailto:\/\//.test(result); if (title != null) { mxUtils.write(div, title); } else { mxUtils.write(div, mxResources.get((result.length < maxSize) ? ((validUrl) ? 'link' : 'mainEmbedNotice') : 'preview') + ':'); } mxUtils.br(div); var size = document.createElement('div'); size.style.position = 'absolute'; size.style.top = '30px'; size.style.right = '30px'; size.style.color = 'gray'; mxUtils.write(size, editorUi.formatFileSize(result.length)); div.appendChild(size); // Using DIV for faster rendering var text = document.createElement('textarea'); text.setAttribute('autocomplete', 'off'); text.setAttribute('autocorrect', 'off'); text.setAttribute('autocapitalize', 'off'); text.setAttribute('spellcheck', 'false'); text.style.fontFamily = 'monospace'; text.style.wordBreak = 'break-all'; text.style.marginTop = '10px'; text.style.resize = 'none'; text.style.height = '150px'; text.style.width = '440px'; text.style.border = '1px solid gray'; text.value = mxResources.get('updatingDocument'); div.appendChild(text); mxUtils.br(div); this.init = function() { window.setTimeout(function() { if (result.length < maxSize) { text.value = result; text.focus(); if (mxClient.IS_GC || mxClient.IS_FF || document.documentMode >= 5) { text.select(); } else { document.execCommand('selectAll', false, null); } } else { text.setAttribute('readonly', 'true'); text.value = mxResources.get('tooLargeUseDownload'); } }, 0); }; var buttons = document.createElement('div'); buttons.style.position = 'absolute'; buttons.style.bottom = '36px'; buttons.style.right = '32px'; var previewBtn = null; // Loads forever in IE9 if (EmbedDialog.showPreviewOption && (!mxClient.IS_CHROMEAPP || validUrl) && !navigator.standalone && (validUrl || (mxClient.IS_SVG && (document.documentMode == null || document.documentMode > 9)))) { previewBtn = mxUtils.button(mxResources.get((result.length < maxSize) ? 'preview' : 'openInNewWindow'), function() { var value = (result.length < maxSize) ? text.value : result; if (previewFn != null) { previewFn(value); } else { if (validUrl) { try { var win = editorUi.openLink(value); if (win != null && (timeout == null || timeout > 0)) { window.setTimeout(mxUtils.bind(this, function() { try { if (win != null && win.location.href != null && win.location.href.substring(0, 8) != value.substring(0, 8)) { win.close(); editorUi.handleError({message: mxResources.get('drawingTooLarge')}); } } catch (e) { // ignore } }), timeout || 500); } } catch (e) { editorUi.handleError({message: e.message || mxResources.get('drawingTooLarge')}); } } else { var wnd = window.open(); var doc = (wnd != null) ? wnd.document : null; if (doc != null) { doc.writeln('<html><head><title>' + encodeURIComponent(mxResources.get('preview')) + '</title><meta charset="utf-8"></head>' + '<body>' + result + '</body></html>'); doc.close(); } else { editorUi.handleError({message: mxResources.get('errorUpdatingPreview')}); } } } }); previewBtn.className = 'geBtn'; buttons.appendChild(previewBtn); } if (!validUrl || result.length > 7500) { var downloadBtn = mxUtils.button(mxResources.get('download'), function() { editorUi.hideDialog(); editorUi.saveData('embed.txt', 'txt', result, 'text/plain'); }); downloadBtn.className = 'geBtn'; buttons.appendChild(downloadBtn); } // Twitter-intent does not allow more characters, must be pasted manually if (validUrl && (!editorUi.isOffline() || mxClient.IS_CHROMEAPP)) { if (result.length < maxFbSize) { var fbBtn = mxUtils.button('', function() { try { var url = 'https://www.facebook.com/sharer.php?p[url]=' + encodeURIComponent(text.value); editorUi.openLink(url); } catch (e) { editorUi.handleError({message: e.message || mxResources.get('drawingTooLarge')}); } }); var img = document.createElement('img'); img.setAttribute('src', Editor.facebookImage); img.setAttribute('width', '18'); img.setAttribute('height', '18'); img.setAttribute('border', '0'); fbBtn.appendChild(img); fbBtn.setAttribute('title', mxResources.get('facebook') + ' (' + editorUi.formatFileSize(maxFbSize) + ' max)'); fbBtn.style.verticalAlign = 'bottom'; fbBtn.style.paddingTop = '4px'; fbBtn.style.minWidth = '46px' fbBtn.className = 'geBtn'; buttons.appendChild(fbBtn); } if (result.length < maxTwitterSize) { var tweetBtn = mxUtils.button('', function() { try { var url = 'https://twitter.com/intent/tweet?text=' + encodeURIComponent(tweet) + '&url=' + encodeURIComponent(text.value); editorUi.openLink(url); } catch (e) { editorUi.handleError({message: e.message || mxResources.get('drawingTooLarge')}); } }); var img = document.createElement('img'); img.setAttribute('src', Editor.tweetImage); img.setAttribute('width', '18'); img.setAttribute('height', '18'); img.setAttribute('border', '0'); img.style.marginBottom = '5px' tweetBtn.appendChild(img); tweetBtn.setAttribute('title', mxResources.get('twitter') + ' (' + editorUi.formatFileSize(maxTwitterSize) + ' max)'); tweetBtn.style.verticalAlign = 'bottom'; tweetBtn.style.paddingTop = '4px'; tweetBtn.style.minWidth = '46px' tweetBtn.className = 'geBtn'; buttons.appendChild(tweetBtn); } } var closeBtn = mxUtils.button(mxResources.get('close'), function() { editorUi.hideDialog(); }); buttons.appendChild(closeBtn); var copyBtn = mxUtils.button(mxResources.get('copy'), function() { text.focus(); if (mxClient.IS_GC || mxClient.IS_FF || document.documentMode >= 5) { text.select(); } else { document.execCommand('selectAll', false, null); } document.execCommand('copy'); editorUi.alert(mxResources.get('copiedToClipboard')); }); if (result.length < maxSize) { // Does not work in Safari and shows annoying dialog for IE11- if (!mxClient.IS_SF && document.documentMode == null) { buttons.appendChild(copyBtn); copyBtn.className = 'geBtn gePrimaryBtn'; closeBtn.className = 'geBtn'; } else { closeBtn.className = 'geBtn gePrimaryBtn'; } } else { buttons.appendChild(previewBtn); closeBtn.className = 'geBtn'; previewBtn.className = 'geBtn gePrimaryBtn'; } div.appendChild(buttons); this.container = div; }; /** * Add embed dialog option. */ EmbedDialog.showPreviewOption = true; /** * Constructs a dialog for embedding the diagram in Google Sites. */ var GoogleSitesDialog = function(editorUi, publicUrl) { var div = document.createElement('div'); var graph = editorUi.editor.graph; var bounds = graph.getGraphBounds(); var scale = graph.view.scale; var x0 = Math.floor(bounds.x / scale - graph.view.translate.x); var y0 = Math.floor(bounds.y / scale - graph.view.translate.y); mxUtils.write(div, mxResources.get('googleGadget') + ':'); mxUtils.br(div); var gadgetInput = document.createElement('input'); gadgetInput.setAttribute('type', 'text'); gadgetInput.style.marginBottom = '8px'; gadgetInput.style.marginTop = '2px'; gadgetInput.style.width = '410px'; div.appendChild(gadgetInput); mxUtils.br(div); this.init = function() { gadgetInput.focus(); if (mxClient.IS_GC || mxClient.IS_FF || document.documentMode >= 5) { gadgetInput.select(); } else { document.execCommand('selectAll', false, null); } }; mxUtils.write(div, mxResources.get('top') + ':'); var topInput = document.createElement('input'); topInput.setAttribute('type', 'text'); topInput.setAttribute('size', '4'); topInput.style.marginRight = '16px'; topInput.style.marginLeft = '4px'; topInput.value = x0; div.appendChild(topInput); mxUtils.write(div, mxResources.get('height') + ':'); var heightInput = document.createElement('input'); heightInput.setAttribute('type', 'text'); heightInput.setAttribute('size', '4'); heightInput.style.marginLeft = '4px'; heightInput.value = Math.ceil(bounds.height / scale); div.appendChild(heightInput); mxUtils.br(div); var hr = document.createElement('hr'); hr.setAttribute('size', '1'); hr.style.marginBottom = '16px'; hr.style.marginTop = '16px'; div.appendChild(hr); mxUtils.write(div, mxResources.get('publicDiagramUrl') + ':'); mxUtils.br(div); var urlInput = document.createElement('input'); urlInput.setAttribute('type', 'text'); urlInput.setAttribute('size', '28'); urlInput.style.marginBottom = '8px'; urlInput.style.marginTop = '2px'; urlInput.style.width = '410px'; urlInput.value = publicUrl || ''; div.appendChild(urlInput); mxUtils.br(div); mxUtils.write(div, mxResources.get('borderWidth') + ':'); var borderInput = document.createElement('input'); borderInput.setAttribute('type', 'text'); borderInput.setAttribute('size', '3'); borderInput.style.marginBottom = '8px'; borderInput.style.marginLeft = '4px'; borderInput.value = '0'; div.appendChild(borderInput); mxUtils.br(div); var panCheckBox = document.createElement('input'); panCheckBox.setAttribute('type', 'checkbox'); panCheckBox.setAttribute('checked', 'checked'); panCheckBox.defaultChecked = true; panCheckBox.style.marginLeft = '16px'; div.appendChild(panCheckBox); mxUtils.write(div, mxResources.get('pan') + ' '); var zoomCheckBox = document.createElement('input'); zoomCheckBox.setAttribute('type', 'checkbox'); zoomCheckBox.setAttribute('checked', 'checked'); zoomCheckBox.defaultChecked = true; zoomCheckBox.style.marginLeft = '8px'; div.appendChild(zoomCheckBox); mxUtils.write(div, mxResources.get('zoom') + ' '); var editCheckBox = document.createElement('input'); editCheckBox.setAttribute('type', 'checkbox'); editCheckBox.style.marginLeft = '8px'; editCheckBox.setAttribute('title', window.location.href); div.appendChild(editCheckBox); mxUtils.write(div, mxResources.get('edit') + ' '); var editBlankCheckBox = document.createElement('input'); editBlankCheckBox.setAttribute('type', 'checkbox'); editBlankCheckBox.style.marginLeft = '8px'; div.appendChild(editBlankCheckBox); mxUtils.write(div, mxResources.get('asNew') + ' '); mxUtils.br(div); var resizeCheckBox = document.createElement('input'); resizeCheckBox.setAttribute('type', 'checkbox'); resizeCheckBox.setAttribute('checked', 'checked'); resizeCheckBox.defaultChecked = true; resizeCheckBox.style.marginLeft = '16px'; div.appendChild(resizeCheckBox); mxUtils.write(div, mxResources.get('resize') + ' '); var fitCheckBox = document.createElement('input'); fitCheckBox.setAttribute('type', 'checkbox'); fitCheckBox.style.marginLeft = '8px'; div.appendChild(fitCheckBox); mxUtils.write(div, mxResources.get('fit') + ' '); var embedCheckBox = document.createElement('input'); embedCheckBox.setAttribute('type', 'checkbox'); embedCheckBox.style.marginLeft = '8px'; div.appendChild(embedCheckBox); mxUtils.write(div, mxResources.get('embed') + ' '); var node = null; var s = editorUi.getBasenames().join(';'); var file = editorUi.getCurrentFile(); function update() { var title = (file != null && file.getTitle() != null) ? file.getTitle() : this.defaultFilename; if (embedCheckBox.checked && urlInput.value != '') { var encUrl = encodeURIComponent(mxUtils.htmlEntities(urlInput.value)); var gurl = 'https://www.draw.io/gadget.xml?type=4&diagram=' + encUrl; if (title != null) { gurl += '&title=' + encodeURIComponent(title); } if (s.length > 0) { gurl += '&s=' + s; } if (borderInput.value != '' && borderInput.value != '0') { gurl += '&border=' + borderInput.value; } if (heightInput.value != '') { gurl += '&height=' + heightInput.value; } gurl += '&pan=' + ((panCheckBox.checked) ? '1': '0'); gurl += '&zoom=' + ((zoomCheckBox.checked) ? '1': '0'); gurl += '&fit=' + ((fitCheckBox.checked) ? '1': '0'); gurl += '&resize=' + ((resizeCheckBox.checked) ? '1': '0'); gurl += '&x0=' + Number(topInput.value); gurl += '&y0=' + y0; if (graph.mathEnabled) { gurl += '&math=1'; } if (editBlankCheckBox.checked) { gurl += '&edit=_blank'; } else if (editCheckBox.checked) { gurl += '&edit=' + encodeURIComponent(mxUtils.htmlEntities(window.location.href)); } gadgetInput.value = gurl; } else if (file.constructor == DriveFile || file.constructor == DropboxFile) { var gurl = 'https://www.draw.io/gadget.xml?embed=0&diagram='; if (urlInput.value != '') { gurl += encodeURIComponent(mxUtils.htmlEntities(urlInput.value)) + '&type=3'; } else { gurl += file.getHash().substring(1); if (file.constructor == DropboxFile) { gurl += '&type=2'; } else { gurl += '&type=1'; } } if (title != null) { gurl += '&title=' + encodeURIComponent(title); } if (heightInput.value != '') { var h = parseInt(heightInput.value) + parseInt(topInput.value); gurl += '&height=' + h; } gadgetInput.value = gurl; } else { gadgetInput.value = ''; } }; mxEvent.addListener(panCheckBox, 'change', update); mxEvent.addListener(zoomCheckBox, 'change', update); mxEvent.addListener(resizeCheckBox, 'change', update); mxEvent.addListener(fitCheckBox, 'change', update); mxEvent.addListener(editCheckBox, 'change', update); mxEvent.addListener(editBlankCheckBox, 'change', update); mxEvent.addListener(embedCheckBox, 'change', update); mxEvent.addListener(heightInput, 'change', update); mxEvent.addListener(topInput, 'change', update); mxEvent.addListener(borderInput, 'change', update); mxEvent.addListener(urlInput, 'change', update); update(); mxEvent.addListener(gadgetInput, 'click', function() { gadgetInput.focus(); if (mxClient.IS_GC || mxClient.IS_FF || document.documentMode >= 5) { gadgetInput.select(); } else { document.execCommand('selectAll', false, null); } }); var buttons = document.createElement('div'); buttons.style.paddingTop = '12px'; buttons.style.textAlign = 'right'; var closeBtn = mxUtils.button(mxResources.get('close'), function() { editorUi.hideDialog(); }); closeBtn.className = 'geBtn gePrimaryBtn'; buttons.appendChild(closeBtn); div.appendChild(buttons); this.container = div; }; /** * Constructs a new parse dialog. */ var CreateGraphDialog = function(editorUi, title, type) { var div = document.createElement('div'); div.style.textAlign = 'right'; this.init = function() { var container = document.createElement('div'); container.style.position = 'relative'; container.style.border = '1px solid gray'; container.style.width = '100%'; container.style.height = '360px'; container.style.overflow = 'hidden'; container.style.marginBottom = '16px'; mxEvent.disableContextMenu(container); div.appendChild(container); var graph = new Graph(container); graph.setCellsCloneable(true); graph.setPanning(true); graph.setAllowDanglingEdges(false); graph.connectionHandler.select = false; graph.view.setTranslate(20, 20); graph.border = 20; graph.panningHandler.useLeftButtonForPanning = true; var vertexStyle = 'rounded=1;'; var edgeStyle = 'curved=1;'; var startStyle = 'ellipse'; // FIXME: Does not work in iPad var mxCellRendererInstallCellOverlayListeners = mxCellRenderer.prototype.installCellOverlayListeners; graph.cellRenderer.installCellOverlayListeners = function(state, overlay, shape) { mxCellRenderer.prototype.installCellOverlayListeners.apply(this, arguments); mxEvent.addListener(shape.node, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', function (evt) { overlay.fireEvent(new mxEventObject('pointerdown', 'event', evt, 'state', state)); }); if (!mxClient.IS_POINTER && mxClient.IS_TOUCH) { mxEvent.addListener(shape.node, 'touchstart', function (evt) { overlay.fireEvent(new mxEventObject('pointerdown', 'event', evt, 'state', state)); }); } }; graph.getAllConnectionConstraints = function() { return null; }; // Keeps highlight behind overlays graph.connectionHandler.marker.highlight.keepOnTop = false; graph.connectionHandler.createEdgeState = function(me) { var edge = graph.createEdge(null, null, null, null, null, edgeStyle); return new mxCellState(this.graph.view, edge, this.graph.getCellStyle(edge)); }; // Gets the default parent for inserting new cells. This // is normally the first child of the root (ie. layer 0). var parent = graph.getDefaultParent(); var addOverlay = mxUtils.bind(this, function(cell) { // Creates a new overlay with an image and a tooltip var overlay = new mxCellOverlay(this.connectImage, 'Add outgoing'); overlay.cursor = 'hand'; // Installs a handler for clicks on the overlay overlay.addListener(mxEvent.CLICK, function(sender, evt2) { // TODO: Add menu for picking next shape graph.connectionHandler.reset(); graph.clearSelection(); var geo = graph.getCellGeometry(cell); var v2; executeLayout(function() { v2 = graph.insertVertex(parent, null, 'Entry', geo.x, geo.y, 80, 30, vertexStyle); addOverlay(v2); graph.view.refresh(v2); var e1 = graph.insertEdge(parent, null, '', cell, v2, edgeStyle); }, function() { graph.scrollCellToVisible(v2); }); }); // FIXME: Does not work in iPad (inserts loop) overlay.addListener('pointerdown', function(sender, eo) { var evt2 = eo.getProperty('event'); var state = eo.getProperty('state'); graph.popupMenuHandler.hideMenu(); graph.stopEditing(false); var pt = mxUtils.convertPoint(graph.container, mxEvent.getClientX(evt2), mxEvent.getClientY(evt2)); graph.connectionHandler.start(state, pt.x, pt.y); graph.isMouseDown = true; graph.isMouseTrigger = mxEvent.isMouseEvent(evt2); mxEvent.consume(evt2); }); // Sets the overlay for the cell in the graph graph.addCellOverlay(cell, overlay); }); // Adds cells to the model in a single step graph.getModel().beginUpdate(); var v1; try { v1 = graph.insertVertex(parent, null, 'Start', 0, 0, 80, 30, startStyle); addOverlay(v1); } finally { // Updates the display graph.getModel().endUpdate(); } var layout; if (type == 'horizontalTree') { layout = new mxCompactTreeLayout(graph); layout.edgeRouting = false; layout.levelDistance = 30; edgeStyle = 'edgeStyle=elbowEdgeStyle;elbow=horizontal;'; } else if (type == 'verticalTree') { layout = new mxCompactTreeLayout(graph, false); layout.edgeRouting = false; layout.levelDistance = 30; edgeStyle = 'edgeStyle=elbowEdgeStyle;elbow=vertical;'; } else if (type == 'radialTree') { layout = new mxRadialTreeLayout(graph, false); layout.edgeRouting = false; layout.levelDistance = 80; } else if (type == 'verticalFlow') { layout = new mxHierarchicalLayout(graph, mxConstants.DIRECTION_NORTH); } else if (type == 'horizontalFlow') { layout = new mxHierarchicalLayout(graph, mxConstants.DIRECTION_WEST); } else if (type == 'organic') { layout = new mxFastOrganicLayout(graph, false); layout.forceConstant = 80; } else if (type == 'circle') { layout = new mxCircleLayout(graph); } if (layout != null) { var executeLayout = function(change, post) { graph.getModel().beginUpdate(); try { if (change != null) { change(); } layout.execute(graph.getDefaultParent(), v1); } catch (e) { throw e; } finally { // New API for animating graph layout results asynchronously var morph = new mxMorphing(graph); morph.addListener(mxEvent.DONE, mxUtils.bind(this, function() { graph.getModel().endUpdate(); if (post != null) { post(); } })); morph.startAnimation(); } }; var edgeHandleConnect = mxEdgeHandler.prototype.connect; mxEdgeHandler.prototype.connect = function(edge, terminal, isSource, isClone, me) { edgeHandleConnect.apply(this, arguments); executeLayout(); }; graph.resizeCell = function() { mxGraph.prototype.resizeCell.apply(this, arguments); executeLayout(); }; graph.connectionHandler.addListener(mxEvent.CONNECT, function() { executeLayout(); }); } var cancelBtn = mxUtils.button(mxResources.get('close'), function() { editorUi.confirm(mxResources.get('areYouSure'), function() { if (container.parentNode != null) { graph.destroy(); container.parentNode.removeChild(container); } editorUi.hideDialog(); }); }) cancelBtn.className = 'geBtn'; if (editorUi.editor.cancelFirst) { div.appendChild(cancelBtn); } var okBtn = mxUtils.button(mxResources.get('insert'), function(evt) { graph.clearCellOverlays(); var cells = graph.getModel().getChildren(graph.getDefaultParent()); var pt = (mxEvent.isAltDown(evt)) ? editorUi.editor.graph.getFreeInsertPoint() : editorUi.editor.graph.getCenterInsertPoint( graph.getBoundingBoxFromGeometry(cells, true)); cells = editorUi.editor.graph.importCells(cells, pt.x, pt.y); var view = editorUi.editor.graph.view; var temp = view.getBounds(cells); temp.x -= view.translate.x; temp.y -= view.translate.y; editorUi.editor.graph.scrollRectToVisible(temp); editorUi.editor.graph.setSelectionCells(cells); if (container.parentNode != null) { graph.destroy(); container.parentNode.removeChild(container); } editorUi.hideDialog(); }); div.appendChild(okBtn); okBtn.className = 'geBtn gePrimaryBtn'; if (!editorUi.editor.cancelFirst) { div.appendChild(cancelBtn); } }; this.container = div; }; /** * */ CreateGraphDialog.prototype.connectImage = new mxImage((mxClient.IS_SVG) ? '' : IMAGE_PATH + '/handle-connect.png', 26, 26); /** * Constructs a new parse dialog. */ var BackgroundImageDialog = function(editorUi, applyFn, img) { var div = document.createElement('div'); div.style.whiteSpace = 'nowrap'; var h3 = document.createElement('h2'); mxUtils.write(h3, mxResources.get('backgroundImage')); h3.style.marginTop = '0px'; div.appendChild(h3); mxUtils.write(div, mxResources.get('image') + ' ' + mxResources.get('url') + ':'); mxUtils.br(div); var urlInput = document.createElement('input'); urlInput.setAttribute('type', 'text'); urlInput.style.marginTop = '4px'; urlInput.style.marginBottom = '4px'; urlInput.style.width = '350px'; urlInput.value = (img != null) ? img.src : ''; var resetting = false; var ignoreEvt = false; var urlChanged = function(evt, done) { // Skips blur event if called from apply button if (evt == null || !ignoreEvt) { urlInput.value = mxUtils.trim(urlInput.value); if (!resetting && urlInput.value != '' && !editorUi.isOffline()) { editorUi.loadImage(urlInput.value, function(img) { widthInput.value = img.width; heightInput.value = img.height; if (done != null) { done(urlInput.value); } }, function() { editorUi.showError(mxResources.get('error'), mxResources.get('fileNotFound'), mxResources.get('ok')); widthInput.value = ''; heightInput.value = ''; if (done != null) { done(null); } }); } else { widthInput.value = ''; heightInput.value = ''; if (done != null) { done(''); } } } }; this.init = function() { urlInput.focus(); // Installs drag and drop handler for local images and links if (Graph.fileSupport) { urlInput.setAttribute('placeholder', mxResources.get('dragImagesHere')); // Setup the dnd listeners var dlg = div.parentNode; var graph = editorUi.editor.graph; var dropElt = null; mxEvent.addListener(dlg, 'dragleave', function(evt) { if (dropElt != null) { dropElt.parentNode.removeChild(dropElt); dropElt = null; } evt.stopPropagation(); evt.preventDefault(); }); mxEvent.addListener(dlg, 'dragover', mxUtils.bind(this, function(evt) { // IE 10 does not implement pointer-events so it can't have a drop highlight if (dropElt == null && (!mxClient.IS_IE || document.documentMode > 10)) { dropElt = editorUi.highlightElement(dlg); } evt.stopPropagation(); evt.preventDefault(); })); mxEvent.addListener(dlg, 'drop', mxUtils.bind(this, function(evt) { if (dropElt != null) { dropElt.parentNode.removeChild(dropElt); dropElt = null; } if (evt.dataTransfer.files.length > 0) { editorUi.importFiles(evt.dataTransfer.files, 0, 0, editorUi.maxBackgroundSize, function(data, mimeType, x, y, w, h) { urlInput.value = data; urlChanged(); }, function() { // No post processing }, function(file) { // Handles only images return file.type.substring(0, 6) == 'image/'; }, function(queue) { // Invokes elements of queue in order for (var i = 0; i < queue.length; i++) { queue[i](); } }, true, editorUi.maxBackgroundBytes, editorUi.maxBackgroundBytes, true); } else if (mxUtils.indexOf(evt.dataTransfer.types, 'text/uri-list') >= 0) { var uri = evt.dataTransfer.getData('text/uri-list'); if ((/\.(gif|jpg|jpeg|tiff|png|svg)$/i).test(uri)) { urlInput.value = decodeURIComponent(uri); urlChanged(); } } evt.stopPropagation(); evt.preventDefault(); }), false); } }; div.appendChild(urlInput); mxUtils.br(div); mxUtils.br(div); mxUtils.write(div, mxResources.get('width') + ':'); var widthInput = document.createElement('input'); widthInput.setAttribute('type', 'text'); widthInput.style.width = '60px'; widthInput.style.marginLeft = '4px'; widthInput.style.marginRight = '16px'; widthInput.value = (img != null) ? img.width : ''; div.appendChild(widthInput); mxUtils.write(div, mxResources.get('height') + ':'); var heightInput = document.createElement('input'); heightInput.setAttribute('type', 'text'); heightInput.style.width = '60px'; heightInput.style.marginLeft = '4px'; heightInput.style.marginRight = '16px'; heightInput.value = (img != null) ? img.height : ''; div.appendChild(heightInput); var resetBtn = mxUtils.button(mxResources.get('reset'), function() { urlInput.value = ''; widthInput.value = ''; heightInput.value = ''; resetting = false; }); mxEvent.addGestureListeners(resetBtn, function() { // Blocks processing a image URL while clicking reset resetting = true; }); resetBtn.className = 'geBtn'; resetBtn.width = '100'; div.appendChild(resetBtn); mxUtils.br(div); mxEvent.addListener(urlInput, 'change', urlChanged); ImageDialog.filePicked = function(data) { if (data.action == google.picker.Action.PICKED) { if (data.docs[0].thumbnails != null) { var thumb = data.docs[0].thumbnails[data.docs[0].thumbnails.length - 1]; if (thumb != null) { urlInput.value = thumb.url; urlChanged(); } } } urlInput.focus(); }; var btns = document.createElement('div'); btns.style.marginTop = '40px'; btns.style.textAlign = 'right'; var cancelBtn = mxUtils.button(mxResources.get('cancel'), function() { resetting = true; editorUi.hideDialog(); }); cancelBtn.className = 'geBtn'; if (editorUi.editor.cancelFirst) { btns.appendChild(cancelBtn); } applyBtn = mxUtils.button(mxResources.get('apply'), function() { editorUi.hideDialog(); urlChanged(null, function(url) { applyFn((url != '' && url != null) ? new mxImage(urlInput.value, widthInput.value, heightInput.value) : null, url == null); }); }); mxEvent.addGestureListeners(applyBtn, function() { ignoreEvt = true; }); applyBtn.className = 'geBtn gePrimaryBtn'; btns.appendChild(applyBtn); if (!editorUi.editor.cancelFirst) { btns.appendChild(cancelBtn); } div.appendChild(btns); this.container = div; }; /** * Constructs a new parse dialog. */ var ParseDialog = function(editorUi, title, defaultType) { var plantUmlExample = '@startuml\nskinparam shadowing false\nAlice -> Bob: Authentication Request\nBob --> Alice: Authentication Response\n\nAlice -> Bob: Another authentication Request\nAlice <-- Bob: Another authentication Response\n@enduml'; var insertPoint = editorUi.editor.graph.getFreeInsertPoint(); function parse(text, type, evt) { var lines = text.split('\n'); if (type == 'plantUmlPng' || type == 'plantUmlSvg' || type == 'plantUmlTxt') { if (editorUi.spinner.spin(document.body, mxResources.get('inserting'))) { var graph = editorUi.editor.graph; var format = (type == 'plantUmlTxt') ? 'txt' : ((type == 'plantUmlPng') ? 'png' : 'svg'); function insertPlantUmlImage(text, format, data, w, h) { insertPoint