drawio-offline
Version:
diagrams.net desktop
1,661 lines (1,476 loc) • 208 kB
JavaScript
/**
* Copyright (c) 2006-2020, JGraph Ltd
* Copyright (c) 2006-2020, draw.io AG
*/
/**
* Constructs a new point for the optional x and y coordinates. If no
* coordinates are given, then the default values for <x> and <y> are used.
* @constructor
* @class Implements a basic 2D point. Known subclassers = {@link mxRectangle}.
* @param {number} x X-coordinate of the point.
* @param {number} y Y-coordinate of the point.
*/
App = function(editor, container, lightbox)
{
EditorUi.call(this, editor, container, (lightbox != null) ? lightbox :
(urlParams['lightbox'] == '1' || (uiTheme == 'min' &&
urlParams['chrome'] != '0')));
// Logs unloading of window with modifications for Google Drive file
if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp)
{
window.onunload = mxUtils.bind(this, function()
{
var file = this.getCurrentFile();
if (file != null && file.isModified())
{
var evt = {category: 'DISCARD-FILE-' + file.getHash(),
action: ((file.savingFile) ? 'saving' : '') +
((file.savingFile && file.savingFileTime != null) ? '_' +
Math.round((Date.now() - file.savingFileTime.getTime()) / 1000) : '') +
((file.saveLevel != null) ? ('-sl_' + file.saveLevel) : '') +
'-age_' + ((file.ageStart != null) ? Math.round((Date.now() - file.ageStart.getTime()) / 1000) : 'x') +
((this.editor.autosave) ? '' : '-nosave') +
((file.isAutosave()) ? '' : '-noauto') +
'-open_' + ((file.opened != null) ? Math.round((Date.now() - file.opened.getTime()) / 1000) : 'x') +
'-save_' + ((file.lastSaved != null) ? Math.round((Date.now() - file.lastSaved.getTime()) / 1000) : 'x') +
'-change_' + ((file.lastChanged != null) ? Math.round((Date.now() - file.lastChanged.getTime()) / 1000) : 'x') +
'-alive_' + Math.round((Date.now() - App.startTime.getTime()) / 1000),
label: (file.sync != null) ? ('client_' + file.sync.clientId) : 'nosync'};
if (file.constructor == DriveFile && file.desc != null && this.drive != null)
{
evt.label += ((this.drive.user != null) ? ('-user_' + this.drive.user.id) : '-nouser') + '-rev_' +
file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + '-size_' + file.getSize() +
'-mime_' + file.desc.mimeType;
}
EditorUi.logEvent(evt);
}
});
}
// Logs changes to autosave
this.editor.addListener('autosaveChanged', mxUtils.bind(this, function()
{
var file = this.getCurrentFile();
if (file != null)
{
EditorUi.logEvent({category: ((this.editor.autosave) ? 'ON' : 'OFF') +
'-AUTOSAVE-FILE-' + file.getHash(), action: 'changed',
label: 'autosave_' + ((this.editor.autosave) ? 'on' : 'off')});
}
}));
// Pre-fetches images
if (mxClient.IS_SVG)
{
mxGraph.prototype.warningImage.src = '';
}
else
{
var img = new Image();
img.src = mxGraph.prototype.warningImage.src;
}
// Global helper method to deal with popup blockers
window.openWindow = mxUtils.bind(this, function(url, pre, fallback)
{
if (urlParams['openInSameWin'] == '1')
{
fallback();
return;
}
var wnd = null;
try
{
wnd = window.open(url);
}
catch (e)
{
// ignore
}
if (wnd == null || wnd === undefined)
{
this.showDialog(new PopupDialog(this, url, pre, fallback).container, 320, 140, true, true);
}
else if (pre != null)
{
pre();
}
});
// Initial state for toolbar items is disabled
this.updateDocumentTitle();
this.updateUi();
// Global helper method to display error messages
window.showOpenAlert = mxUtils.bind(this, function(message)
{
// Cancel must be called before showing error message
if (window.openFile != null)
{
window.openFile.cancel(true);
}
this.handleError(message);
});
// Handles opening files via drag and drop
if (!this.editor.chromeless || this.editor.editable)
{
this.addFileDropHandler([document]);
}
// Process the queue for waiting plugins
if (App.DrawPlugins != null)
{
for (var i = 0; i < App.DrawPlugins.length; i++)
{
try
{
App.DrawPlugins[i](this);
}
catch (e)
{
if (window.console != null)
{
console.log('Plugin Error:', e, App.DrawPlugins[i]);
}
}
finally
{
App.embedModePluginsCount--;
this.initializeEmbedMode();
}
}
// Installs global callback for plugins
window.Draw.loadPlugin = mxUtils.bind(this, function(callback)
{
try
{
callback(this);
}
finally
{
App.embedModePluginsCount--;
this.initializeEmbedMode();
}
});
//Set a timeout in case a plugin doesn't load quickly or doesn't load at all
setTimeout(mxUtils.bind(this, function()
{
//Force finish loading if its not yet called
if (App.embedModePluginsCount > 0)
{
App.embedModePluginsCount = 0;
this.initializeEmbedMode();
}
}), 5000); //5 sec timeout
}
this.load();
};
/**
* Timeout error
*/
App.ERROR_TIMEOUT = 'timeout';
/**
* Busy error
*/
App.ERROR_BUSY = 'busy';
/**
* Unknown error
*/
App.ERROR_UNKNOWN = 'unknown';
/**
* Google drive mode
*/
App.MODE_GOOGLE = 'google';
/**
* Dropbox mode
*/
App.MODE_DROPBOX = 'dropbox';
/**
* OneDrive Mode
*/
App.MODE_ONEDRIVE = 'onedrive';
/**
* Github Mode
*/
App.MODE_GITHUB = 'github';
/**
* Gitlab mode
*/
App.MODE_GITLAB = 'gitlab';
/**
* Device Mode
*/
App.MODE_DEVICE = 'device';
/**
* Browser Mode
*/
App.MODE_BROWSER = 'browser';
/**
* Trello App Mode
*/
App.MODE_TRELLO = 'trello';
/**
* Embed App Mode
*/
App.MODE_EMBED = 'embed';
/**
* Sets the delay for autosave in milliseconds. Default is 2000.
*/
App.DROPBOX_APPKEY = window.DRAWIO_DROPBOX_ID;
/**
* Sets URL to load the Dropbox SDK from
*/
App.DROPBOX_URL = window.DRAWIO_BASE_URL + '/js/dropbox/Dropbox-sdk.min.js';
/**
* Sets URL to load the Dropbox dropins JS from.
*/
App.DROPINS_URL = 'https://www.dropbox.com/static/api/2/dropins.js';
/**
* OneDrive Client JS (file/folder picker). This is a slightly modified version to allow using accessTokens
* But it doesn't work for IE11, so we fallback to the original one
*/
App.ONEDRIVE_URL = mxClient.IS_IE11? 'https://js.live.net/v7.2/OneDrive.js' : window.DRAWIO_BASE_URL + '/js/onedrive/OneDrive.js';
App.ONEDRIVE_INLINE_PICKER_URL = window.DRAWIO_BASE_URL + '/js/onedrive/mxODPicker.js';
/**
* Trello URL
*/
App.TRELLO_URL = 'https://api.trello.com/1/client.js';
/**
* Trello JQuery dependency
*/
App.TRELLO_JQUERY_URL = 'https://code.jquery.com/jquery-1.7.1.min.js';
/**
* Specifies the key for the pusher project.
*/
App.PUSHER_KEY = '1e756b07a690c5bdb054';
/**
* Specifies the key for the pusher project.
*/
App.PUSHER_CLUSTER = 'eu';
/**
* Specifies the URL for the pusher API.
*/
App.PUSHER_URL = 'https://js.pusher.com/4.3/pusher.min.js';
/**
* Socket.io library
*/
App.SOCKET_IO_URL = window.DRAWIO_BASE_URL + '/js/socket.io/socket.io.min.js';
App.SIMPLE_PEER_URL = window.DRAWIO_BASE_URL + '/js/socket.io/simplepeer9.10.0.min.js';
App.SOCKET_IO_SRV = 'http://localhost:3030';
/**
* Google APIs to load. The realtime API is needed to notify collaborators of conversion
* of the realtime files, but after Dec 11 it's read-only and hence no longer needed.
*/
App.GOOGLE_APIS = 'drive-share';
/**
* Function: authorize
*
* Authorizes the client, gets the userId and calls <open>.
*/
App.startTime = new Date();
/**
* Defines plugin IDs for loading via p URL parameter. Update the table at
* https://www.diagrams.net/doc/faq/supported-url-parameters
*/
App.pluginRegistry = {'4xAKTrabTpTzahoLthkwPNUn': 'plugins/explore.js',
'ex': 'plugins/explore.js', 'p1': 'plugins/p1.js',
'ac': 'plugins/connect.js', 'acj': 'plugins/connectJira.js',
'ac148': 'plugins/cConf-1-4-8.js', 'ac148cmnt': 'plugins/cConf-comments.js',
'voice': 'plugins/voice.js',
'tips': 'plugins/tooltips.js', 'svgdata': 'plugins/svgdata.js',
'electron': 'plugins/electron.js',
'number': 'plugins/number.js', 'sql': 'plugins/sql.js',
'props': 'plugins/props.js', 'text': 'plugins/text.js',
'anim': 'plugins/animation.js', 'update': 'plugins/update.js',
'trees': 'plugins/trees/trees.js', 'import': 'plugins/import.js',
'replay': 'plugins/replay.js', 'anon': 'plugins/anonymize.js',
'tr': 'plugins/trello.js', 'f5': 'plugins/rackF5.js',
'tickets': 'plugins/tickets.js', 'flow': 'plugins/flow.js',
'webcola': 'plugins/webcola/webcola.js', 'rnd': 'plugins/random.js',
'page': 'plugins/page.js', 'gd': 'plugins/googledrive.js',
'tags': 'plugins/tags.js'};
App.publicPlugin = [
'ex',
'voice',
'tips',
'svgdata',
'number',
'sql',
'props',
'text',
'anim',
'update',
'trees',
// 'import',
'replay',
'anon',
'tickets',
'flow',
'webcola',
// 'rnd', 'page', 'gd',
'tags'
];
/**
* Loads all given scripts and invokes onload after
* all scripts have finished loading.
*/
App.loadScripts = function(scripts, onload)
{
var n = scripts.length;
for (var i = 0; i < n; i++)
{
mxscript(scripts[i], function()
{
if (--n == 0 && onload != null)
{
onload();
}
});
}
};
/**
* Function: getStoredMode
*
* Returns the current mode.
*/
App.getStoredMode = function()
{
var mode = null;
if (mode == null && isLocalStorage)
{
mode = localStorage.getItem('.mode');
}
if (mode == null && typeof(Storage) != 'undefined')
{
var cookies = document.cookie.split(";");
for (var i = 0; i < cookies.length; i++)
{
// Removes spaces around cookie
var cookie = mxUtils.trim(cookies[i]);
if (cookie.substring(0, 5) == 'MODE=')
{
mode = cookie.substring(5);
break;
}
}
if (mode != null && isLocalStorage)
{
// Moves to local storage
var expiry = new Date();
expiry.setYear(expiry.getFullYear() - 1);
document.cookie = 'MODE=; expires=' + expiry.toUTCString();
localStorage.setItem('.mode', mode);
}
}
return mode;
};
/**
* Static Application initializer executed at load-time.
*/
(function()
{
if (!mxClient.IS_CHROMEAPP)
{
if (urlParams['offline'] != '1')
{
// Switches to dropbox mode for db.draw.io
if (window.location.hostname == 'db.draw.io' && urlParams['mode'] == null)
{
urlParams['mode'] = 'dropbox';
}
App.mode = urlParams['mode'];
}
if (App.mode == null)
{
// Stored mode overrides preferred mode
App.mode = App.getStoredMode();
}
/**
* Lazy loading backends.
*/
if (window.mxscript != null)
{
// Loads gapi for all browsers but IE8 and below if not disabled or if enabled and in embed mode
if (urlParams['embed'] != '1')
{
if (typeof window.DriveClient === 'function')
{
if (urlParams['gapi'] != '0' && isSvgBrowser &&
(document.documentMode == null || document.documentMode >= 10))
{
// Immediately loads client
if (App.mode == App.MODE_GOOGLE || (urlParams['state'] != null &&
window.location.hash == '') || (window.location.hash != null &&
window.location.hash.substring(0, 2) == '#G'))
{
mxscript('https://apis.google.com/js/api.js');
}
// Keeps lazy loading for fallback to authenticated Google file if not public in loadFile
else if (urlParams['chrome'] == '0' && (window.location.hash == null ||
window.location.hash.substring(0, 45) !== '#Uhttps%3A%2F%2Fdrive.google.com%2Fuc%3Fid%3D'))
{
// Disables loading of client
window.DriveClient = null;
}
}
else
{
// Disables loading of client
window.DriveClient = null;
}
}
// Loads dropbox for all browsers but IE8 and below (no CORS) if not disabled or if enabled and in embed mode
// KNOWN: Picker does not work in IE11 (https://dropbox.zendesk.com/requests/1650781)
if (typeof window.DropboxClient === 'function')
{
if (urlParams['db'] != '0' && isSvgBrowser &&
(document.documentMode == null || document.documentMode > 9))
{
// Immediately loads client
if (App.mode == App.MODE_DROPBOX || (window.location.hash != null &&
window.location.hash.substring(0, 2) == '#D'))
{
mxscript(App.DROPBOX_URL, function()
{
// Must load this after the dropbox SDK since they use the same namespace
mxscript(App.DROPINS_URL, null, 'dropboxjs', App.DROPBOX_APPKEY, true);
});
}
else if (urlParams['chrome'] == '0')
{
window.DropboxClient = null;
}
}
else
{
// Disables loading of client
window.DropboxClient = null;
}
}
// Loads OneDrive for all browsers but IE6/IOS if not disabled or if enabled and in embed mode
if (typeof window.OneDriveClient === 'function')
{
if (urlParams['od'] != '0' && (navigator.userAgent == null ||
navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10))
{
// Immediately loads client
if (App.mode == App.MODE_ONEDRIVE || (window.location.hash != null &&
window.location.hash.substring(0, 2) == '#W'))
{
if (urlParams['inlinePicker'] == '1' || mxClient.IS_ANDROID || mxClient.IS_IOS)
{
mxscript(App.ONEDRIVE_INLINE_PICKER_URL, function()
{
window.OneDrive = {}; //Needed to allow code that check its existance to work BUT it's not used
});
}
else
{
mxscript(App.ONEDRIVE_URL);
}
}
else if (urlParams['chrome'] == '0')
{
window.OneDriveClient = null;
}
}
else
{
// Disables loading of client
window.OneDriveClient = null;
}
}
// Loads Trello for all browsers but < IE10 if not disabled or if enabled and in embed mode
if (typeof window.TrelloClient === 'function')
{
if (urlParams['tr'] == '1' && isSvgBrowser && !mxClient.IS_IE11 &&
(document.documentMode == null || document.documentMode >= 10))
{
// Immediately loads client
if (App.mode == App.MODE_TRELLO || (window.location.hash != null &&
window.location.hash.substring(0, 2) == '#T'))
{
mxscript(App.TRELLO_JQUERY_URL, function()
{
mxscript(App.TRELLO_URL);
});
}
else if (urlParams['chrome'] == '0')
{
window.TrelloClient = null;
}
}
else
{
// Disables loading of client
window.TrelloClient = null;
}
}
}
}
}
})();
/**
* Clears the PWA cache.
*/
App.clearServiceWorker = function(success)
{
navigator.serviceWorker.getRegistrations().then(function(registrations)
{
if (registrations != null && registrations.length > 0)
{
for (var i = 0; i < registrations.length; i++)
{
registrations[i].unregister();
}
if (success != null)
{
success();
}
}
});
};
/**
* Program flow starts here.
*
* Optional callback is called with the app instance.
*/
App.main = function(callback, createUi)
{
// Logs uncaught errors
window.onerror = function(message, url, linenumber, colno, err)
{
EditorUi.logError('Global: ' + ((message != null) ? message : ''),
url, linenumber, colno, err, null, true);
};
// Blocks stand-alone mode for certain subdomains
if (window.top == window.self &&
(/ac\.draw\.io$/.test(window.location.hostname) ||
/ac-ent\.draw\.io$/.test(window.location.hostname) ||
/aj\.draw\.io$/.test(window.location.hostname)))
{
document.body.innerHTML = '<div style="margin-top:10%;text-align:center;">Stand-alone mode not allowed for this domain.</div>';
return;
}
// Removes info text in embed mode
if (urlParams['embed'] == '1' || urlParams['lightbox'] == '1')
{
var geInfo = document.getElementById('geInfo');
if (geInfo != null)
{
geInfo.parentNode.removeChild(geInfo);
}
}
// Redirects to the latest AWS icons
if (document.referrer != null && urlParams['libs'] == 'aws3' &&
document.referrer.substring(0, 42) == 'https://aws.amazon.com/architecture/icons/')
{
urlParams['libs'] = 'aws4';
}
if (window.mxscript != null)
{
// Checks for script content changes to avoid CSP errors in production
if (urlParams['dev'] == '1' && CryptoJS != null && App.mode != App.MODE_DROPBOX && App.mode != App.MODE_TRELLO)
{
var scripts = document.getElementsByTagName('script');
// Checks bootstrap script
if (scripts != null && scripts.length > 0)
{
var content = mxUtils.getTextContent(scripts[0]);
if (CryptoJS.MD5(content).toString() != 'b02227617087e21bd49f2faa15164112')
{
console.log('Change bootstrap script MD5 in the previous line:', CryptoJS.MD5(content).toString());
alert('[Dev] Bootstrap script change requires update of CSP');
}
}
// Checks main script
if (scripts != null && scripts.length > 1)
{
var content = mxUtils.getTextContent(scripts[scripts.length - 1]);
if (CryptoJS.MD5(content).toString() != 'd53805dd6f0bbba2da4966491ca0a505')
{
console.log('Change main script MD5 in the previous line:', CryptoJS.MD5(content).toString());
alert('[Dev] Main script change requires update of CSP');
}
}
}
try
{
// Removes PWA cache on www.draw.io to force use of new domain via redirect
if (Editor.enableServiceWorker && (urlParams['offline'] == '0' ||
/www\.draw\.io$/.test(window.location.hostname) ||
(urlParams['offline'] != '1' && urlParams['dev'] == '1')))
{
App.clearServiceWorker(function()
{
if (urlParams['offline'] == '0')
{
alert('Cache cleared');
}
});
}
else if (Editor.enableServiceWorker)
{
// Runs as progressive web app if service workers are supported
navigator.serviceWorker.register('/service-worker.js');
}
}
catch (e)
{
if (window.console != null)
{
console.error(e);
}
}
// Loads Pusher API
if (('ArrayBuffer' in window) && !mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp &&
DrawioFile.SYNC == 'auto' && (urlParams['embed'] != '1' || urlParams['embedRT'] == '1') && urlParams['local'] != '1' &&
(urlParams['chrome'] != '0' || urlParams['rt'] == '1') &&
urlParams['stealth'] != '1' && urlParams['offline'] != '1')
{
// TODO: Check if async loading is fast enough
mxscript(App.PUSHER_URL);
if (urlParams['rtCursors'] == '1')
{
mxscript(App.SOCKET_IO_URL);
mxscript(App.SIMPLE_PEER_URL);
}
}
// Loads plugins
if (urlParams['plugins'] != '0' && urlParams['offline'] != '1')
{
// mxSettings is not yet initialized in configure mode, redirect parameter
// to p URL parameter in caller for plugins in embed mode
var plugins = (mxSettings.settings != null) ? mxSettings.getPlugins() : null;
// Configured plugins in embed mode with configure=1 URL should be loaded so we
// look ahead here and parse the config to fetch the list of custom plugins
if (mxSettings.settings == null && isLocalStorage && typeof(JSON) !== 'undefined')
{
try
{
var temp = JSON.parse(localStorage.getItem(mxSettings.key));
if (temp != null)
{
plugins = temp.plugins;
}
}
catch (e)
{
// ignore
}
}
var temp = urlParams['p'];
App.initPluginCallback();
if (temp != null)
{
// Mapping from key to URL in App.plugins
App.loadPlugins(temp.split(';'));
}
if (plugins != null && plugins.length > 0 && urlParams['plugins'] != '0')
{
// Loading plugins inside the asynchronous block below stops the page from loading so a
// hardcoded message for the warning dialog is used since the resources are loadd below
var warning = 'The page has requested to load the following plugin(s):\n \n {1}\n \n Would you like to load these plugin(s) now?\n \n NOTE : Only allow plugins to run if you fully understand the security implications of doing so.\n';
var tmp = window.location.protocol + '//' + window.location.host;
var local = true;
for (var i = 0; i < plugins.length && local; i++)
{
if (plugins[i].charAt(0) != '/' && plugins[i].substring(0, tmp.length) != tmp)
{
local = false;
}
}
if (local || mxUtils.confirm(mxResources.replacePlaceholders(warning, [plugins.join('\n')]).replace(/\\n/g, '\n')))
{
for (var i = 0; i < plugins.length; i++)
{
try
{
if (App.pluginsLoaded[plugins[i]] == null)
{
App.pluginsLoaded[plugins[i]] = true;
App.embedModePluginsCount++;
if (plugins[i].charAt(0) == '/')
{
plugins[i] = PLUGINS_BASE_PATH + plugins[i];
}
mxscript(plugins[i]);
}
}
catch (e)
{
// ignore
}
}
}
}
}
// Loads gapi for all browsers but IE8 and below if not disabled or if enabled and in embed mode
// Special case: Cannot load in asynchronous code below
if (typeof window.DriveClient === 'function' &&
(typeof gapi === 'undefined' && (((urlParams['embed'] != '1' && urlParams['gapi'] != '0') ||
(urlParams['embed'] == '1' && urlParams['gapi'] == '1')) && isSvgBrowser &&
isLocalStorage && (document.documentMode == null || document.documentMode >= 10))))
{
mxscript('https://apis.google.com/js/api.js?onload=DrawGapiClientCallback', null, null, null, mxClient.IS_SVG);
}
// Disables client
else if (typeof window.gapi === 'undefined')
{
window.DriveClient = null;
}
}
/**
* Asynchronous MathJax extension.
*/
if (urlParams['math'] != '0')
{
Editor.initMath();
}
function doLoad(bundle)
{
// Prefetches asynchronous requests so that below code runs synchronous
// Loading the correct bundle (one file) via the fallback system in mxResources. The stylesheet
// is compiled into JS in the build process and is only needed for local development.
mxUtils.getAll((urlParams['dev'] != '1') ? [bundle] : [bundle,
STYLE_PATH + '/default.xml', STYLE_PATH + '/dark-default.xml'], function(xhr)
{
// Adds bundle text to resources
mxResources.parse(xhr[0].getText());
// Configuration mode
if (isLocalStorage && localStorage != null && window.location.hash != null &&
window.location.hash.substring(0, 9) == '#_CONFIG_')
{
try
{
var trustedPlugins = {};
for (var key in App.pluginRegistry)
{
trustedPlugins[App.pluginRegistry[key]] = true;
}
// Only allows trusted plugins
function checkPlugins(plugins)
{
if (plugins != null)
{
for (var i = 0; i < plugins.length; i++)
{
if (!trustedPlugins[plugins[i]])
{
throw new Error(mxResources.get('invalidInput') + ' "' + plugins[i]) + '"';
}
}
}
return true;
};
var value = JSON.parse(Graph.decompress(window.location.hash.substring(9)));
if (value != null && checkPlugins(value.plugins))
{
EditorUi.debug('Setting configuration', JSON.stringify(value));
if (value.merge != null)
{
var temp = localStorage.getItem(Editor.configurationKey);
if (temp != null)
{
try
{
var config = JSON.parse(temp);
for (var key in value.merge)
{
config[key] = value.merge[key];
}
value = config;
}
catch (e)
{
window.location.hash = '';
alert(e);
}
}
else
{
value = value.merge;
}
}
if (confirm(mxResources.get('configLinkWarn')) &&
confirm(mxResources.get('configLinkConfirm')))
{
localStorage.setItem(Editor.configurationKey, JSON.stringify(value));
window.location.hash = '';
window.location.reload();
}
}
window.location.hash = '';
}
catch (e)
{
window.location.hash = '';
alert(e);
}
}
// Prepares themes with mapping from old default-style to old XML file
if (xhr.length > 2)
{
Graph.prototype.defaultThemes['default-style2'] = xhr[1].getDocumentElement();
Graph.prototype.defaultThemes['darkTheme'] = xhr[2].getDocumentElement();
}
// Main
function realMain()
{
var ui = (createUi != null) ? createUi() : new App(new Editor(
urlParams['chrome'] == '0' || uiTheme == 'min',
null, null, null, urlParams['chrome'] != '0'));
if (window.mxscript != null)
{
// Loads dropbox for all browsers but IE8 and below (no CORS) if not disabled or if enabled and in embed mode
// KNOWN: Picker does not work in IE11 (https://dropbox.zendesk.com/requests/1650781)
if (typeof window.DropboxClient === 'function' &&
(window.Dropbox == null && window.DrawDropboxClientCallback != null &&
(((urlParams['embed'] != '1' && urlParams['db'] != '0') ||
(urlParams['embed'] == '1' && urlParams['db'] == '1')) &&
isSvgBrowser && (document.documentMode == null || document.documentMode > 9))))
{
mxscript(App.DROPBOX_URL, function()
{
// Must load this after the dropbox SDK since they use the same namespace
mxscript(App.DROPINS_URL, function()
{
DrawDropboxClientCallback();
}, 'dropboxjs', App.DROPBOX_APPKEY);
});
}
// Disables client
else if (typeof window.Dropbox === 'undefined' || typeof window.Dropbox.choose === 'undefined')
{
window.DropboxClient = null;
}
// Loads OneDrive for all browsers but IE6/IOS if not disabled or if enabled and in embed mode
if (typeof window.OneDriveClient === 'function' &&
(typeof OneDrive === 'undefined' && window.DrawOneDriveClientCallback != null &&
(((urlParams['embed'] != '1' && urlParams['od'] != '0') || (urlParams['embed'] == '1' &&
urlParams['od'] == '1')) && (navigator.userAgent == null ||
navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10))))
{
if (urlParams['inlinePicker'] == '1' || mxClient.IS_ANDROID || mxClient.IS_IOS)
{
mxscript(App.ONEDRIVE_INLINE_PICKER_URL, function()
{
window.OneDrive = {}; //Needed to allow code that check its existance to work BUT it's not used
window.DrawOneDriveClientCallback();
});
}
else
{
mxscript(App.ONEDRIVE_URL, window.DrawOneDriveClientCallback);
}
}
// Disables client
else if (typeof window.OneDrive === 'undefined')
{
window.OneDriveClient = null;
}
// Loads Trello for all browsers but < IE10 if not disabled or if enabled and in embed mode
if (typeof window.TrelloClient === 'function' && !mxClient.IS_IE11 &&
typeof window.Trello === 'undefined' && window.DrawTrelloClientCallback != null &&
urlParams['tr'] == '1' && (navigator.userAgent == null ||
navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10))
{
mxscript(App.TRELLO_JQUERY_URL, function()
{
// Must load this after the dropbox SDK since they use the same namespace
mxscript(App.TRELLO_URL, function()
{
DrawTrelloClientCallback();
});
});
}
// Disables client
else if (typeof window.Trello === 'undefined')
{
window.TrelloClient = null;
}
}
if (callback != null)
{
callback(ui);
}
/**
* For developers only
*/
if (urlParams['chrome'] != '0' && urlParams['test'] == '1')
{
EditorUi.debug('App.start', [ui, (new Date().getTime() - t0.getTime()) + 'ms']);
if (urlParams['export'] != null)
{
EditorUi.debug('Export:', EXPORT_URL);
}
}
};
if (urlParams['dev'] == '1' || EditorUi.isElectronApp) //TODO check if we can remove these scripts loading from index.html
{
realMain();
}
else
{
mxStencilRegistry.allowEval = false;
App.loadScripts(['js/shapes-14-6-5.min.js', 'js/stencils.min.js',
'js/extensions.min.js'], realMain);
}
}, function(xhr)
{
var st = document.getElementById('geStatus');
if (st != null)
{
st.innerHTML = 'Error loading page. <a>Please try refreshing.</a>';
// Tries reload with default resources in case any language resources were not available
st.getElementsByTagName('a')[0].onclick = function()
{
mxLanguage = 'en';
doLoad(mxResources.getDefaultBundle(RESOURCE_BASE, mxLanguage) ||
mxResources.getSpecialBundle(RESOURCE_BASE, mxLanguage));
};
}
});
};
function doMain()
{
// Optional override for autosaveDelay and defaultEdgeLength
try
{
if (mxSettings.settings != null)
{
document.body.style.backgroundColor = (uiTheme == 'dark' ||
mxSettings.settings.darkMode) ? '#2a2a2a' : '#ffffff';
if (mxSettings.settings.autosaveDelay != null)
{
var val = parseInt(mxSettings.settings.autosaveDelay);
if (!isNaN(val) && val > 0)
{
DrawioFile.prototype.autosaveDelay = val;
EditorUi.debug('Setting autosaveDelay', val);
}
else
{
EditorUi.debug('Invalid autosaveDelay', val);
}
}
if (mxSettings.settings.defaultEdgeLength != null)
{
var val = parseInt(mxSettings.settings.defaultEdgeLength);
if (!isNaN(val) && val > 0)
{
Graph.prototype.defaultEdgeLength = val;
EditorUi.debug('Using defaultEdgeLength', val);
}
else
{
EditorUi.debug('Invalid defaultEdgeLength', val);
}
}
}
}
catch (e)
{
if (window.console != null)
{
console.error(e);
}
}
// Prefetches default fonts with URLs
if (Menus.prototype.defaultFonts != null)
{
for (var i = 0; i < Menus.prototype.defaultFonts.length; i++)
{
var value = Menus.prototype.defaultFonts[i];
if (typeof value !== 'string' &&
value.fontFamily != null &&
value.fontUrl != null)
{
Graph.addFont(value.fontFamily, value.fontUrl);
}
}
}
// Adds required resources (disables loading of fallback properties, this can only
// be used if we know that all keys are defined in the language specific file)
mxResources.loadDefaultBundle = false;
doLoad(mxResources.getDefaultBundle(RESOURCE_BASE, mxLanguage) ||
mxResources.getSpecialBundle(RESOURCE_BASE, mxLanguage));
};
// Sends load event if configuration is requested and waits for configure message
if (urlParams['configure'] == '1')
{
var op = window.opener || window.parent;
var configHandler = function(evt)
{
if (evt.source == op)
{
try
{
var data = JSON.parse(evt.data);
if (data != null && data.action == 'configure')
{
mxEvent.removeListener(window, 'message', configHandler);
Editor.configure(data.config, true);
mxSettings.load();
doMain();
}
}
catch (e)
{
if (window.console != null)
{
console.log('Error in configure message: ' + e, evt.data);
}
}
}
};
// Receives XML message from opener and puts it into the graph
mxEvent.addListener(window, 'message', configHandler);
op.postMessage(JSON.stringify({event: 'configure'}), '*');
}
else
{
if (Editor.config == null)
{
// Loads configuration from global scope or local storage
if (window.DRAWIO_CONFIG != null)
{
try
{
EditorUi.debug('Using global configuration', window.DRAWIO_CONFIG);
Editor.configure(window.DRAWIO_CONFIG);
mxSettings.load();
}
catch (e)
{
if (window.console != null)
{
console.error(e);
}
}
}
// Loads configuration from local storage
if (isLocalStorage && localStorage != null && urlParams['embed'] != '1')
{
var configData = localStorage.getItem(Editor.configurationKey);
if (configData != null)
{
try
{
configData = JSON.parse(configData);
if (configData != null)
{
EditorUi.debug('Using local configuration', configData);
Editor.configure(configData);
mxSettings.load();
}
}
catch (e)
{
if (window.console != null)
{
console.error(e);
}
}
}
}
}
doMain();
}
};
//Extends EditorUi
mxUtils.extend(App, EditorUi);
/**
* Executes the first step for connecting to Google Drive.
*/
App.prototype.defaultUserPicture = IMAGE_PATH + '/default-user.jpg';
/**
*
*/
App.prototype.shareImage = '';
/**
*
*/
App.prototype.chevronUpImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/chevron-up.png' : '';
/**
*
*/
App.prototype.chevronDownImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/chevron-down.png' : '';
/**
*
*/
App.prototype.formatShowImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/format-show.png' : '';
/**
*
*/
App.prototype.formatHideImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/format-hide.png' : '';
/**
*
*/
App.prototype.fullscreenImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/fullscreen.png' : '';
/**
* Interval to show dialog for unsaved data if autosave is on.
* Default is 300000 (5 minutes).
*/
App.prototype.warnInterval = 300000;
/**
*
*/
App.prototype.compactMode = false;
/**
*
*/
App.prototype.fullscreenMode = false;
/**
* Overriden UI settings depending on mode.
*/
if (urlParams['embed'] != '1')
{
App.prototype.menubarHeight = 64;
}
else
{
App.prototype.footerHeight = 0;
}
/**
* Queue for loading plugins and wait for UI instance
*/
App.initPluginCallback = function()
{
if (App.DrawPlugins == null)
{
// Workaround for need to load plugins now but wait for UI instance
App.DrawPlugins = [];
// Global entry point for plugins is Draw.loadPlugin. This is the only
// long-term supported solution for access to the EditorUi instance.
window.Draw = new Object();
window.Draw.loadPlugin = function(callback)
{
App.DrawPlugins.push(callback);
};
}
};
/**
*
*/
App.pluginsLoaded = {};
App.embedModePluginsCount = 0;
/**
* Queue for loading plugins and wait for UI instance
*/
App.loadPlugins = function(plugins, useInclude)
{
EditorUi.debug('Loading plugins', plugins);
for (var i = 0; i < plugins.length; i++)
{
if (plugins[i] != null && plugins[i].length > 0)
{
try
{
var url = PLUGINS_BASE_PATH + App.pluginRegistry[plugins[i]];
if (url != null)
{
if (App.pluginsLoaded[url] == null)
{
App.pluginsLoaded[url] = true;
App.embedModePluginsCount++;
if (typeof window.drawDevUrl === 'undefined')
{
if (useInclude)
{
mxinclude(url);
}
else
{
mxscript(url);
}
}
else
{
if (useInclude)
{
mxinclude(url);
}
else
{
mxscript(drawDevUrl + url);
}
}
}
}
else if (window.console != null)
{
console.log('Unknown plugin:', plugins[i]);
}
}
catch (e)
{
if (window.console != null)
{
console.log('Error loading plugin:', plugins[i], e);
}
}
}
}
};
/**
* Delay embed mode initialization until all plugins are loaded
*/
App.prototype.initializeEmbedMode = function()
{
if (urlParams['embed'] == '1')
{
if (window.location.hostname == 'app.diagrams.net')
{
this.showBanner('EmbedDeprecationFooter', 'app.diagrams.net will stop working for embed mode. Please use embed.diagrams.net.');
}
if (App.embedModePluginsCount > 0 || this.initEmbedDone)
{
return; //Wait for plugins to load, or this is a duplicate call due to timeout
}
else
{
this.initEmbedDone = true;
}
EditorUi.prototype.initializeEmbedMode.apply(this, arguments);
}
};
/**
* TODO: Define viewer protocol and implement new viewer style toolbar
*/
App.prototype.initializeViewerMode = function()
{
var parent = window.opener || window.parent;
if (parent != null)
{
this.editor.graph.addListener(mxEvent.SIZE, mxUtils.bind(this, function()
{
parent.postMessage(JSON.stringify(this.createLoadMessage('size')), '*');
}));
}
};
/**
* Translates this point by the given vector.
*
* @param {number} dx X-coordinate of the translation.
* @param {number} dy Y-coordinate of the translation.
*/
App.prototype.init = function()
{
EditorUi.prototype.init.apply(this, arguments);
/**
* Specifies the default filename.
*/
this.defaultLibraryName = mxResources.get('untitledLibrary');
/**
* Holds the listener for description changes.
*/
this.descriptorChangedListener = mxUtils.bind(this, this.descriptorChanged);
/**
* Creates github client.
*/
this.gitHub = (!mxClient.IS_IE || document.documentMode == 10 ||
mxClient.IS_IE11 || mxClient.IS_EDGE) &&
(urlParams['gh'] != '0' && (urlParams['embed'] != '1' ||
urlParams['gh'] == '1')) ? new GitHubClient(this) : null;
if (this.gitHub != null)
{
this.gitHub.addListener('userChanged', mxUtils.bind(this, function()
{
this.updateUserElement();
this.restoreLibraries();
}))
}
/**
* Creates gitlab client.
*/
this.gitLab = (!mxClient.IS_IE || document.documentMode == 10 ||
mxClient.IS_IE11 || mxClient.IS_EDGE) &&
(urlParams['gl'] != '0' && (urlParams['embed'] != '1' ||
urlParams['gl'] == '1')) ? new GitLabClient(this) : null;
if (this.gitLab != null)
{
this.gitLab.addListener('userChanged', mxUtils.bind(this, function()
{
this.updateUserElement();
this.restoreLibraries();
}));
}
/**
* Lazy-loading for individual backends
*/
if (urlParams['embed'] != '1' || urlParams['od'] == '1')
{
/**
* Creates onedrive client if all required libraries are available.
*/
var initOneDriveClient = mxUtils.bind(this, function()
{
if (typeof OneDrive !== 'undefined')
{
/**
* Holds the x-coordinate of the point.
*/
this.oneDrive = new OneDriveClient(this);
this.oneDrive.addListener('userChanged', mxUtils.bind(this, function()
{
this.updateUserElement();
this.restoreLibraries();
}));
// Notifies listeners of new client
this.fireEvent(new mxEventObject('clientLoaded', 'client', this.oneDrive));
}
else if (window.DrawOneDriveClientCallback == null)
{
window.DrawOneDriveClientCallback = initOneDriveClient;
}
});
initOneDriveClient();
}
/**
* Lazy-loading for Trello
*/
if (urlParams['embed'] != '1' || urlParams['tr'] == '1')
{
/**
* Creates Trello client if all required libraries are available.
*/
var initTrelloClient = mxUtils.bind(this, function()
{
if (typeof window.Trello !== 'undefined')
{
try
{
this.trello = new TrelloClient(this);
//TODO we have no user info from Trello so we don't set a user
this.trello.addListener('userChanged', mxUtils.bind(this, function()
{
this.updateUserElement();
this.restoreLibraries();
}));
// Notifies listeners of new client
this.fireEvent(new mxEventObject('clientLoaded', 'client', this.trello));
}
catch (e)
{
if (window.console != null)
{
console.error(e);
}
}
}
else if (window.DrawTrelloClientCallback == null)
{
window.DrawTrelloClientCallback = initTrelloClient;
}
});
initTrelloClient();
}
/**
* Creates drive client with all required libraries are available.
*/
if (urlParams['embed'] != '1' || urlParams['gapi'] == '1')
{
var initDriveClient = mxUtils.bind(this, function()
{
/**
* Creates google drive client if all required libraries are available.
*/
if (typeof gapi !== 'undefined')
{
var doInit = mxUtils.bind(this, function()
{
this.drive = new DriveClient(this);
this.drive.addListener('userChanged', mxUtils.bind(this, function()
{
this.updateUserElement();
this.restoreLibraries();
this.checkLicense();
}))
// Notifies listeners of new client
this.fireEvent(new mxEventObject('clientLoaded', 'client', this.drive));
});
if (window.DrawGapiClientCallback != null)
{
gapi.load(((urlParams['picker'] != '0') ? 'picker,': '') + App.GOOGLE_APIS, doInit);
/**
* Clears any callbacks.
*/
window.DrawGapiClientCallback = null;
}
else
{
doInit();
}
}
else if (window.DrawGapiClientCallback == null)
{
window.DrawGapiClientCallback = initDriveClient;
}
});
initDriveClient();
}
if (urlParams['embed'] != '1' || urlParams['db'] == '1')
{
/**
* Creates dropbox client if all required libraries are available.
*/
var initDropboxClient = mxUtils.bind(this, function()
{
if (typeof Dropbox === 'function' && typeof Dropbox.choose !== 'undefined')
{
/**
* Clears dropbox client callback.
*/
window.DrawDropboxClientCallback = null;
/**
* Holds the x-coordinate of the point.
*/
try
{
this.dropbox = new DropboxClient(this);
this.dropbox.addListener('userChanged', mxUtils.bind(this, function()
{
this.updateUserElement();
this.restoreLibraries();
}));
// Notifies listeners of new client
this.fireEvent(new mxEventObject('clientLoaded', 'client', this.dropbox));
}
catch (e)
{
if (window.console != null)
{
console.error(e);
}
}
}
else if (window.DrawDropboxClientCallback == null)
{
window.DrawDropboxClientCallback = initDropboxClient;
}
});
initDropboxClient();
}
if (urlParams['embed'] != '1')
{
/**
* Holds the background element.
*/
this.bg = this.createBackground();
document.body.appendChild(this.bg);
this.diagramContainer.style.visibility = 'hidden';
this.formatContainer.style.visibility = 'hidden';
this.hsplit.style.display = 'none';
this.sidebarContainer.style.display = 'none';
this.sidebarFooterContainer.style.display = 'none';
// Sets the initial mode
if (urlParams['local'] == '1')
{
this.setMode(App.MODE_DEVICE);
}
else
{
this.mode = App.mode;
}
// Add to Home Screen dialog for mobile devices
if ('serviceWorker' in navigator && !this.editor.isChromelessView() &&
(mxClient.IS_ANDROID || mxClient.IS_IOS))
{
window.addEventListener('beforeinstallprompt', mxUtils.bind(this, function(e)
{
this.showBanner('AddToHomeScreenFooter', mxResourc