1/** 2 * Copyright (c) 2006-2020, JGraph Ltd 3 * Copyright (c) 2006-2020, draw.io AG 4 */ 5 6/** 7 * Constructs a new point for the optional x and y coordinates. If no 8 * coordinates are given, then the default values for <x> and <y> are used. 9 * @constructor 10 * @class Implements a basic 2D point. Known subclassers = {@link mxRectangle}. 11 * @param {number} x X-coordinate of the point. 12 * @param {number} y Y-coordinate of the point. 13 */ 14App = function(editor, container, lightbox) 15{ 16 EditorUi.call(this, editor, container, (lightbox != null) ? lightbox : 17 (urlParams['lightbox'] == '1' || (uiTheme == 'min' && 18 urlParams['chrome'] != '0'))); 19 20 // Logs unloading of window with modifications for Google Drive file 21 if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp) 22 { 23 window.onunload = mxUtils.bind(this, function() 24 { 25 var file = this.getCurrentFile(); 26 27 if (file != null && file.isModified()) 28 { 29 var evt = {category: 'DISCARD-FILE-' + file.getHash(), 30 action: ((file.savingFile) ? 'saving' : '') + 31 ((file.savingFile && file.savingFileTime != null) ? '_' + 32 Math.round((Date.now() - file.savingFileTime.getTime()) / 1000) : '') + 33 ((file.saveLevel != null) ? ('-sl_' + file.saveLevel) : '') + 34 '-age_' + ((file.ageStart != null) ? Math.round((Date.now() - file.ageStart.getTime()) / 1000) : 'x') + 35 ((this.editor.autosave) ? '' : '-nosave') + 36 ((file.isAutosave()) ? '' : '-noauto') + 37 '-open_' + ((file.opened != null) ? Math.round((Date.now() - file.opened.getTime()) / 1000) : 'x') + 38 '-save_' + ((file.lastSaved != null) ? Math.round((Date.now() - file.lastSaved.getTime()) / 1000) : 'x') + 39 '-change_' + ((file.lastChanged != null) ? Math.round((Date.now() - file.lastChanged.getTime()) / 1000) : 'x') + 40 '-alive_' + Math.round((Date.now() - App.startTime.getTime()) / 1000), 41 label: (file.sync != null) ? ('client_' + file.sync.clientId) : 'nosync'}; 42 43 if (file.constructor == DriveFile && file.desc != null && this.drive != null) 44 { 45 evt.label += ((this.drive.user != null) ? ('-user_' + this.drive.user.id) : '-nouser') + '-rev_' + 46 file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + '-size_' + file.getSize() + 47 '-mime_' + file.desc.mimeType; 48 } 49 50 EditorUi.logEvent(evt); 51 } 52 }); 53 } 54 55 // Logs changes to autosave 56 this.editor.addListener('autosaveChanged', mxUtils.bind(this, function() 57 { 58 var file = this.getCurrentFile(); 59 60 if (file != null) 61 { 62 EditorUi.logEvent({category: ((this.editor.autosave) ? 'ON' : 'OFF') + 63 '-AUTOSAVE-FILE-' + file.getHash(), action: 'changed', 64 label: 'autosave_' + ((this.editor.autosave) ? 'on' : 'off')}); 65 } 66 })); 67 68 // Pre-fetches images 69 if (mxClient.IS_SVG) 70 { 71 mxGraph.prototype.warningImage.src = ''; 72 } 73 else 74 { 75 var img = new Image(); 76 img.src = mxGraph.prototype.warningImage.src; 77 } 78 79 // Global helper method to deal with popup blockers 80 window.openWindow = mxUtils.bind(this, function(url, pre, fallback) 81 { 82 if (urlParams['openInSameWin'] == '1' || navigator.standalone) 83 { 84 fallback(); 85 return; 86 } 87 88 var wnd = null; 89 90 try 91 { 92 wnd = window.open(url); 93 } 94 catch (e) 95 { 96 // ignore 97 } 98 99 if (wnd == null || wnd === undefined) 100 { 101 this.showDialog(new PopupDialog(this, url, pre, fallback).container, 320, 140, true, true); 102 } 103 else if (pre != null) 104 { 105 pre(); 106 } 107 }); 108 109 // Initial state for toolbar items is disabled 110 this.updateDocumentTitle(); 111 this.updateUi(); 112 113 // Global helper method to display error messages 114 window.showOpenAlert = mxUtils.bind(this, function(message) 115 { 116 // Cancel must be called before showing error message 117 if (window.openFile != null) 118 { 119 window.openFile.cancel(true); 120 } 121 122 this.handleError(message); 123 }); 124 125 // Handles opening files via drag and drop 126 if (!this.editor.chromeless || this.editor.editable) 127 { 128 this.addFileDropHandler([document]); 129 } 130 131 // Process the queue for waiting plugins 132 if (App.DrawPlugins != null) 133 { 134 for (var i = 0; i < App.DrawPlugins.length; i++) 135 { 136 try 137 { 138 App.DrawPlugins[i](this); 139 } 140 catch (e) 141 { 142 if (window.console != null) 143 { 144 console.log('Plugin Error:', e, App.DrawPlugins[i]); 145 } 146 } 147 finally 148 { 149 App.embedModePluginsCount--; 150 this.initializeEmbedMode(); 151 } 152 } 153 154 // Installs global callback for plugins 155 window.Draw.loadPlugin = mxUtils.bind(this, function(callback) 156 { 157 try 158 { 159 callback(this); 160 } 161 finally 162 { 163 App.embedModePluginsCount--; 164 this.initializeEmbedMode(); 165 } 166 }); 167 168 //Set a timeout in case a plugin doesn't load quickly or doesn't load at all 169 setTimeout(mxUtils.bind(this, function() 170 { 171 //Force finish loading if its not yet called 172 if (App.embedModePluginsCount > 0) 173 { 174 App.embedModePluginsCount = 0; 175 this.initializeEmbedMode(); 176 } 177 }), 5000); //5 sec timeout 178 } 179 180 this.load(); 181}; 182 183/** 184 * Timeout error 185 */ 186App.ERROR_TIMEOUT = 'timeout'; 187 188/** 189 * Busy error 190 */ 191App.ERROR_BUSY = 'busy'; 192 193/** 194 * Unknown error 195 */ 196App.ERROR_UNKNOWN = 'unknown'; 197 198/** 199 * Google drive mode 200 */ 201App.MODE_GOOGLE = 'google'; 202 203/** 204 * Dropbox mode 205 */ 206App.MODE_DROPBOX = 'dropbox'; 207 208/** 209 * OneDrive Mode 210 */ 211App.MODE_ONEDRIVE = 'onedrive'; 212 213/** 214 * Github Mode 215 */ 216App.MODE_GITHUB = 'github'; 217 218/** 219 * Gitlab mode 220 */ 221App.MODE_GITLAB = 'gitlab'; 222 223/** 224 * Device Mode 225 */ 226App.MODE_DEVICE = 'device'; 227 228/** 229 * Browser Mode 230 */ 231App.MODE_BROWSER = 'browser'; 232 233/** 234 * Trello App Mode 235 */ 236App.MODE_TRELLO = 'trello'; 237 238/** 239 * Notion App Mode 240 */ 241App.MODE_NOTION = 'notion'; 242 243/** 244 * Embed App Mode 245 */ 246App.MODE_EMBED = 'embed'; 247 248/** 249 * Atlas App Mode 250 */ 251App.MODE_ATLAS = 'atlas'; 252 253/** 254 * Sets the delay for autosave in milliseconds. Default is 2000. 255 */ 256App.DROPBOX_APPKEY = window.DRAWIO_DROPBOX_ID; 257 258/** 259 * Sets URL to load the Dropbox SDK from 260 */ 261App.DROPBOX_URL = window.DRAWIO_BASE_URL + '/js/dropbox/Dropbox-sdk.min.js'; 262 263/** 264 * Sets URL to load the Dropbox dropins JS from. 265 */ 266App.DROPINS_URL = 'https://www.dropbox.com/static/api/2/dropins.js'; 267 268/** 269 * OneDrive Client JS (file/folder picker). This is a slightly modified version to allow using accessTokens 270 * But it doesn't work for IE11, so we fallback to the original one 271 */ 272App.ONEDRIVE_URL = mxClient.IS_IE11? 'https://js.live.net/v7.2/OneDrive.js' : window.DRAWIO_BASE_URL + '/js/onedrive/OneDrive.js'; 273 274/** 275 * Trello URL 276 */ 277App.TRELLO_URL = 'https://api.trello.com/1/client.js'; 278 279/** 280 * Trello JQuery dependency 281 */ 282App.TRELLO_JQUERY_URL = window.DRAWIO_BASE_URL + '/js/jquery/jquery-3.3.1.min.js'; 283 284/** 285 * Specifies the key for the pusher project. 286 */ 287App.PUSHER_KEY = '1e756b07a690c5bdb054'; 288 289/** 290 * Specifies the key for the pusher project. 291 */ 292App.PUSHER_CLUSTER = 'eu'; 293 294/** 295 * Specifies the URL for the pusher API. 296 */ 297App.PUSHER_URL = 'https://js.pusher.com/4.3/pusher.min.js'; 298 299/** 300 * Socket.io library 301 */ 302App.SOCKET_IO_URL = window.DRAWIO_BASE_URL + '/js/socket.io/socket.io.min.js'; 303App.SIMPLE_PEER_URL = window.DRAWIO_BASE_URL + '/js/socket.io/simplepeer9.10.0.min.js'; 304App.SOCKET_IO_SRV = 'http://localhost:3030'; 305 306/** 307 * Google APIs to load. The realtime API is needed to notify collaborators of conversion 308 * of the realtime files, but after Dec 11 it's read-only and hence no longer needed. 309 */ 310App.GOOGLE_APIS = 'drive-share'; 311 312/** 313 * Function: authorize 314 * 315 * Authorizes the client, gets the userId and calls <open>. 316 */ 317App.startTime = new Date(); 318 319/** 320 * Defines plugin IDs for loading via p URL parameter. Update the table at 321 * https://www.diagrams.net/doc/faq/supported-url-parameters 322 */ 323App.pluginRegistry = {'4xAKTrabTpTzahoLthkwPNUn': 'plugins/explore.js', 324 'ex': 'plugins/explore.js', 'p1': 'plugins/p1.js', 325 'ac': 'plugins/connect.js', 'acj': 'plugins/connectJira.js', 326 'ac148': 'plugins/cConf-1-4-8.js', 'ac148cmnt': 'plugins/cConf-comments.js', 327 'voice': 'plugins/voice.js', 328 'tips': 'plugins/tooltips.js', 'svgdata': 'plugins/svgdata.js', 329 'electron': 'plugins/electron.js', 330 'number': 'plugins/number.js', 'sql': 'plugins/sql.js', 331 'props': 'plugins/props.js', 'text': 'plugins/text.js', 332 'anim': 'plugins/animation.js', 'update': 'plugins/update.js', 333 'trees': 'plugins/trees/trees.js', 'import': 'plugins/import.js', 334 'replay': 'plugins/replay.js', 'anon': 'plugins/anonymize.js', 335 'tr': 'plugins/trello.js', 'f5': 'plugins/rackF5.js', 336 'tickets': 'plugins/tickets.js', 'flow': 'plugins/flow.js', 337 'webcola': 'plugins/webcola/webcola.js', 'rnd': 'plugins/random.js', 338 'page': 'plugins/page.js', 'gd': 'plugins/googledrive.js', 339 'tags': 'plugins/tags.js'}; 340 341App.publicPlugin = [ 342 'ex', 343 'voice', 344 'tips', 345 'svgdata', 346 'number', 347 'sql', 348 'props', 349 'text', 350 'anim', 351 'update', 352 'trees', 353// 'import', 354 'replay', 355 'anon', 356 'tickets', 357 'flow', 358 'webcola', 359// 'rnd', 'page', 'gd', 360 'tags' 361]; 362 363/** 364 * Loads all given scripts and invokes onload after 365 * all scripts have finished loading. 366 */ 367App.loadScripts = function(scripts, onload) 368{ 369 var n = scripts.length; 370 371 for (var i = 0; i < scripts.length; i++) 372 { 373 mxscript(scripts[i], function() 374 { 375 if (--n == 0 && onload != null) 376 { 377 onload(); 378 } 379 }); 380 } 381}; 382 383/** 384 * Function: getStoredMode 385 * 386 * Returns the current mode. 387 */ 388App.getStoredMode = function() 389{ 390 var mode = null; 391 392 if (mode == null && isLocalStorage) 393 { 394 mode = localStorage.getItem('.mode'); 395 } 396 397 if (mode == null && typeof(Storage) != 'undefined') 398 { 399 var cookies = document.cookie.split(";"); 400 401 for (var i = 0; i < cookies.length; i++) 402 { 403 // Removes spaces around cookie 404 var cookie = mxUtils.trim(cookies[i]); 405 406 if (cookie.substring(0, 5) == 'MODE=') 407 { 408 mode = cookie.substring(5); 409 break; 410 } 411 } 412 413 if (mode != null && isLocalStorage) 414 { 415 // Moves to local storage 416 var expiry = new Date(); 417 expiry.setYear(expiry.getFullYear() - 1); 418 document.cookie = 'MODE=; expires=' + expiry.toUTCString(); 419 localStorage.setItem('.mode', mode); 420 } 421 } 422 423 return mode; 424}; 425 426/** 427 * Static Application initializer executed at load-time. 428 */ 429(function() 430{ 431 if (!mxClient.IS_CHROMEAPP) 432 { 433 if (urlParams['offline'] != '1') 434 { 435 // Switches to dropbox mode for db.draw.io 436 if (window.location.hostname == 'db.draw.io' && urlParams['mode'] == null) 437 { 438 urlParams['mode'] = 'dropbox'; 439 } 440 441 App.mode = urlParams['mode']; 442 } 443 444 if (App.mode == null) 445 { 446 // Stored mode overrides preferred mode 447 App.mode = App.getStoredMode(); 448 } 449 450 /** 451 * Lazy loading backends. 452 */ 453 if (window.mxscript != null) 454 { 455 // Loads gapi for all browsers but IE8 and below if not disabled or if enabled and in embed mode 456 if (urlParams['embed'] != '1') 457 { 458 if (typeof window.DriveClient === 'function') 459 { 460 if (urlParams['gapi'] != '0' && isSvgBrowser && 461 (document.documentMode == null || document.documentMode >= 10)) 462 { 463 // Immediately loads client 464 if (App.mode == App.MODE_GOOGLE || (urlParams['state'] != null && 465 window.location.hash == '') || (window.location.hash != null && 466 window.location.hash.substring(0, 2) == '#G')) 467 { 468 mxscript('https://apis.google.com/js/api.js'); 469 } 470 // Keeps lazy loading for fallback to authenticated Google file if not public in loadFile 471 else if (urlParams['chrome'] == '0' && (window.location.hash == null || 472 window.location.hash.substring(0, 45) !== '#Uhttps%3A%2F%2Fdrive.google.com%2Fuc%3Fid%3D')) 473 { 474 // Disables loading of client 475 window.DriveClient = null; 476 } 477 } 478 else 479 { 480 // Disables loading of client 481 window.DriveClient = null; 482 } 483 } 484 485 // Loads dropbox for all browsers but IE8 and below (no CORS) if not disabled or if enabled and in embed mode 486 // KNOWN: Picker does not work in IE11 (https://dropbox.zendesk.com/requests/1650781) 487 if (typeof window.DropboxClient === 'function') 488 { 489 if (urlParams['db'] != '0' && isSvgBrowser && 490 (document.documentMode == null || document.documentMode > 9)) 491 { 492 // Immediately loads client 493 if (App.mode == App.MODE_DROPBOX || (window.location.hash != null && 494 window.location.hash.substring(0, 2) == '#D')) 495 { 496 mxscript(App.DROPBOX_URL, function() 497 { 498 // Must load this after the dropbox SDK since they use the same namespace 499 mxscript(App.DROPINS_URL, null, 'dropboxjs', App.DROPBOX_APPKEY, true); 500 }); 501 } 502 else if (urlParams['chrome'] == '0') 503 { 504 window.DropboxClient = null; 505 } 506 } 507 else 508 { 509 // Disables loading of client 510 window.DropboxClient = null; 511 } 512 } 513 514 // Loads OneDrive for all browsers but IE6/IOS if not disabled or if enabled and in embed mode 515 if (typeof window.OneDriveClient === 'function') 516 { 517 if (urlParams['od'] != '0' && (navigator.userAgent == null || 518 navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10)) 519 { 520 // Immediately loads client 521 if (App.mode == App.MODE_ONEDRIVE || (window.location.hash != null && 522 window.location.hash.substring(0, 2) == '#W')) 523 { 524 //Editor.oneDriveInlinePicker can be set with configuration which is done later, so load it all time 525 mxscript(App.ONEDRIVE_URL); 526 } 527 else if (urlParams['chrome'] == '0') 528 { 529 window.OneDriveClient = null; 530 } 531 } 532 else 533 { 534 // Disables loading of client 535 window.OneDriveClient = null; 536 } 537 } 538 539 // Loads Trello for all browsers but < IE10 if not disabled or if enabled and in embed mode 540 if (typeof window.TrelloClient === 'function') 541 { 542 if (urlParams['tr'] == '1' && isSvgBrowser && !mxClient.IS_IE11 && 543 (document.documentMode == null || document.documentMode >= 10)) 544 { 545 // Immediately loads client 546 if (App.mode == App.MODE_TRELLO || (window.location.hash != null && 547 window.location.hash.substring(0, 2) == '#T')) 548 { 549 mxscript(App.TRELLO_JQUERY_URL, function() 550 { 551 mxscript(App.TRELLO_URL); 552 }); 553 } 554 else if (urlParams['chrome'] == '0') 555 { 556 window.TrelloClient = null; 557 } 558 } 559 else 560 { 561 // Disables loading of client 562 window.TrelloClient = null; 563 } 564 } 565 } 566 } 567 } 568})(); 569 570/** 571 * Clears the PWA cache. 572 */ 573App.clearServiceWorker = function(success) 574{ 575 navigator.serviceWorker.getRegistrations().then(function(registrations) 576 { 577 if (registrations != null && registrations.length > 0) 578 { 579 for (var i = 0; i < registrations.length; i++) 580 { 581 registrations[i].unregister(); 582 } 583 584 if (success != null) 585 { 586 success(); 587 } 588 } 589 }); 590}; 591 592/** 593 * Program flow starts here. 594 * 595 * Optional callback is called with the app instance. 596 */ 597App.main = function(callback, createUi) 598{ 599 // Logs uncaught errors 600 window.onerror = function(message, url, linenumber, colno, err) 601 { 602 EditorUi.logError('Global: ' + ((message != null) ? message : ''), 603 url, linenumber, colno, err, null, true); 604 }; 605 606 // Blocks stand-alone mode for certain subdomains 607 if (window.top == window.self && 608 (/ac\.draw\.io$/.test(window.location.hostname) || 609 /ac-ent\.draw\.io$/.test(window.location.hostname) || 610 /aj\.draw\.io$/.test(window.location.hostname))) 611 { 612 document.body.innerHTML = '<div style="margin-top:10%;text-align:center;">Stand-alone mode not allowed for this domain.</div>'; 613 614 return; 615 } 616 617 // Removes info text in embed mode 618 if (urlParams['embed'] == '1' || urlParams['lightbox'] == '1') 619 { 620 var geInfo = document.getElementById('geInfo'); 621 622 if (geInfo != null) 623 { 624 geInfo.parentNode.removeChild(geInfo); 625 } 626 } 627 628 // Redirects to the latest AWS icons 629 if (document.referrer != null && urlParams['libs'] == 'aws3' && 630 document.referrer.substring(0, 42) == 'https://aws.amazon.com/architecture/icons/') 631 { 632 urlParams['libs'] = 'aws4'; 633 } 634 635 if (window.mxscript != null) 636 { 637 // Checks for script content changes to avoid CSP errors in production 638 if (urlParams['dev'] == '1' && CryptoJS != null && App.mode != App.MODE_DROPBOX && App.mode != App.MODE_TRELLO) 639 { 640 var scripts = document.getElementsByTagName('script'); 641 642 // Checks bootstrap script 643 if (scripts != null && scripts.length > 0) 644 { 645 var content = mxUtils.getTextContent(scripts[0]); 646 647 if (CryptoJS.MD5(content).toString() != 'b02227617087e21bd49f2faa15164112') 648 { 649 console.log('Change bootstrap script MD5 in the previous line:', CryptoJS.MD5(content).toString()); 650 alert('[Dev] Bootstrap script change requires update of CSP'); 651 } 652 } 653 654 // Checks main script 655 if (scripts != null && scripts.length > 1) 656 { 657 var content = mxUtils.getTextContent(scripts[scripts.length - 1]); 658 659 if (CryptoJS.MD5(content).toString() != 'd53805dd6f0bbba2da4966491ca0a505') 660 { 661 console.log('Change main script MD5 in the previous line:', CryptoJS.MD5(content).toString()); 662 alert('[Dev] Main script change requires update of CSP'); 663 } 664 } 665 } 666 667 try 668 { 669 // Removes PWA cache on www.draw.io to force use of new domain via redirect 670 if (Editor.enableServiceWorker && (urlParams['offline'] == '0' || 671 /www\.draw\.io$/.test(window.location.hostname) || 672 (urlParams['offline'] != '1' && urlParams['dev'] == '1'))) 673 { 674 App.clearServiceWorker(function() 675 { 676 if (urlParams['offline'] == '0') 677 { 678 alert('Cache cleared'); 679 } 680 }); 681 } 682 else if (Editor.enableServiceWorker) 683 { 684 // Runs as progressive web app if service workers are supported 685 navigator.serviceWorker.register('/service-worker.js'); 686 } 687 } 688 catch (e) 689 { 690 if (window.console != null) 691 { 692 console.error(e); 693 } 694 } 695 696 // Loads Pusher API 697 if (('ArrayBuffer' in window) && !mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && 698 DrawioFile.SYNC == 'auto' && (urlParams['embed'] != '1' || urlParams['embedRT'] == '1') && urlParams['local'] != '1' && 699 (urlParams['chrome'] != '0' || urlParams['rt'] == '1') && 700 urlParams['stealth'] != '1' && urlParams['offline'] != '1') 701 { 702 // TODO: Check if async loading is fast enough 703 mxscript(App.PUSHER_URL); 704 705 if (urlParams['rtCursors'] == '1') 706 { 707 mxscript(App.SOCKET_IO_URL); 708 mxscript(App.SIMPLE_PEER_URL); 709 } 710 } 711 712 // Loads plugins 713 if (urlParams['plugins'] != '0' && urlParams['offline'] != '1') 714 { 715 // mxSettings is not yet initialized in configure mode, redirect parameter 716 // to p URL parameter in caller for plugins in embed mode 717 var plugins = (mxSettings.settings != null) ? mxSettings.getPlugins() : null; 718 719 // Configured plugins in embed mode with configure=1 URL should be loaded so we 720 // look ahead here and parse the config to fetch the list of custom plugins 721 if (mxSettings.settings == null && isLocalStorage && typeof(JSON) !== 'undefined') 722 { 723 try 724 { 725 var temp = JSON.parse(localStorage.getItem(mxSettings.key)); 726 727 if (temp != null) 728 { 729 plugins = temp.plugins; 730 } 731 } 732 catch (e) 733 { 734 // ignore 735 } 736 } 737 738 var temp = urlParams['p']; 739 App.initPluginCallback(); 740 741 if (temp != null) 742 { 743 // Mapping from key to URL in App.plugins 744 App.loadPlugins(temp.split(';')); 745 } 746 747 if (plugins != null && plugins.length > 0 && urlParams['plugins'] != '0') 748 { 749 // Loading plugins inside the asynchronous block below stops the page from loading so a 750 // hardcoded message for the warning dialog is used since the resources are loadd below 751 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'; 752 var tmp = window.location.protocol + '//' + window.location.host; 753 var local = true; 754 755 for (var i = 0; i < plugins.length && local; i++) 756 { 757 if (plugins[i].charAt(0) != '/' && plugins[i].substring(0, tmp.length) != tmp) 758 { 759 local = false; 760 } 761 } 762 763 if (local || mxUtils.confirm(mxResources.replacePlaceholders(warning, [plugins.join('\n')]).replace(/\\n/g, '\n'))) 764 { 765 for (var i = 0; i < plugins.length; i++) 766 { 767 try 768 { 769 if (App.pluginsLoaded[plugins[i]] == null) 770 { 771 App.pluginsLoaded[plugins[i]] = true; 772 App.embedModePluginsCount++; 773 774 if (plugins[i].charAt(0) == '/') 775 { 776 plugins[i] = PLUGINS_BASE_PATH + plugins[i]; 777 } 778 779 mxscript(plugins[i]); 780 } 781 } 782 catch (e) 783 { 784 // ignore 785 } 786 } 787 } 788 } 789 } 790 791 // Loads gapi for all browsers but IE8 and below if not disabled or if enabled and in embed mode 792 // Special case: Cannot load in asynchronous code below 793 if (typeof window.DriveClient === 'function' && 794 (typeof gapi === 'undefined' && (((urlParams['embed'] != '1' && urlParams['gapi'] != '0') || 795 (urlParams['embed'] == '1' && urlParams['gapi'] == '1')) && isSvgBrowser && 796 isLocalStorage && (document.documentMode == null || document.documentMode >= 10)))) 797 { 798 mxscript('https://apis.google.com/js/api.js?onload=DrawGapiClientCallback', null, null, null, mxClient.IS_SVG); 799 } 800 // Disables client 801 else if (typeof window.gapi === 'undefined') 802 { 803 window.DriveClient = null; 804 } 805 } 806 807 /** 808 * Asynchronous MathJax extension. 809 */ 810 if (urlParams['math'] != '0') 811 { 812 Editor.initMath(); 813 } 814 815 function doLoad(bundle) 816 { 817 // Prefetches asynchronous requests so that below code runs synchronous 818 // Loading the correct bundle (one file) via the fallback system in mxResources. The stylesheet 819 // is compiled into JS in the build process and is only needed for local development. 820 mxUtils.getAll((urlParams['dev'] != '1') ? [bundle] : [bundle, 821 STYLE_PATH + '/default.xml'], function(xhr) 822 { 823 // Adds bundle text to resources 824 mxResources.parse(xhr[0].getText()); 825 826 // Configuration mode 827 if (isLocalStorage && localStorage != null && window.location.hash != null && 828 window.location.hash.substring(0, 9) == '#_CONFIG_') 829 { 830 try 831 { 832 var trustedPlugins = {}; 833 834 for (var key in App.pluginRegistry) 835 { 836 trustedPlugins[App.pluginRegistry[key]] = true; 837 } 838 839 // Only allows trusted plugins 840 function checkPlugins(plugins) 841 { 842 if (plugins != null) 843 { 844 for (var i = 0; i < plugins.length; i++) 845 { 846 if (!trustedPlugins[plugins[i]]) 847 { 848 throw new Error(mxResources.get('invalidInput') + ' "' + plugins[i]) + '"'; 849 } 850 } 851 } 852 853 return true; 854 }; 855 856 var value = JSON.parse(Graph.decompress(window.location.hash.substring(9))); 857 858 if (value != null && checkPlugins(value.plugins)) 859 { 860 EditorUi.debug('Setting configuration', JSON.stringify(value)); 861 862 if (value.merge != null) 863 { 864 var temp = localStorage.getItem(Editor.configurationKey); 865 866 if (temp != null) 867 { 868 869 try 870 { 871 var config = JSON.parse(temp); 872 873 for (var key in value.merge) 874 { 875 config[key] = value.merge[key]; 876 } 877 878 value = config; 879 } 880 catch (e) 881 { 882 window.location.hash = ''; 883 alert(e); 884 } 885 } 886 else 887 { 888 value = value.merge; 889 } 890 } 891 892 if (confirm(mxResources.get('configLinkWarn')) && 893 confirm(mxResources.get('configLinkConfirm'))) 894 { 895 localStorage.setItem(Editor.configurationKey, JSON.stringify(value)); 896 window.location.hash = ''; 897 window.location.reload(); 898 } 899 } 900 901 window.location.hash = ''; 902 } 903 catch (e) 904 { 905 window.location.hash = ''; 906 alert(e); 907 } 908 } 909 910 // Prepares themes with mapping from old default-style to old XML file 911 if (xhr.length > 1) 912 { 913 Graph.prototype.defaultThemes['default-style2'] = xhr[1].getDocumentElement(); 914 Graph.prototype.defaultThemes['darkTheme'] = xhr[1].getDocumentElement(); 915 } 916 917 // Main 918 function realMain() 919 { 920 var ui = (createUi != null) ? createUi() : new App(new Editor( 921 urlParams['chrome'] == '0' || uiTheme == 'min', 922 null, null, null, urlParams['chrome'] != '0')); 923 924 if (window.mxscript != null) 925 { 926 // Loads dropbox for all browsers but IE8 and below (no CORS) if not disabled or if enabled and in embed mode 927 // KNOWN: Picker does not work in IE11 (https://dropbox.zendesk.com/requests/1650781) 928 if (typeof window.DropboxClient === 'function' && 929 (window.Dropbox == null && window.DrawDropboxClientCallback != null && 930 (((urlParams['embed'] != '1' && urlParams['db'] != '0') || 931 (urlParams['embed'] == '1' && urlParams['db'] == '1')) && 932 isSvgBrowser && (document.documentMode == null || document.documentMode > 9)))) 933 { 934 mxscript(App.DROPBOX_URL, function() 935 { 936 // Must load this after the dropbox SDK since they use the same namespace 937 mxscript(App.DROPINS_URL, function() 938 { 939 DrawDropboxClientCallback(); 940 }, 'dropboxjs', App.DROPBOX_APPKEY); 941 }); 942 } 943 // Disables client 944 else if (typeof window.Dropbox === 'undefined' || typeof window.Dropbox.choose === 'undefined') 945 { 946 window.DropboxClient = null; 947 } 948 949 // Loads OneDrive for all browsers but IE6/IOS if not disabled or if enabled and in embed mode 950 if (typeof window.OneDriveClient === 'function' && 951 (typeof OneDrive === 'undefined' && window.DrawOneDriveClientCallback != null && 952 (((urlParams['embed'] != '1' && urlParams['od'] != '0') || (urlParams['embed'] == '1' && 953 urlParams['od'] == '1')) && (navigator.userAgent == null || 954 navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10)))) 955 { 956 //Editor.oneDriveInlinePicker can be set with configuration which is done later, so load it all time 957 mxscript(App.ONEDRIVE_URL, window.DrawOneDriveClientCallback); 958 } 959 // Disables client 960 else if (typeof window.OneDrive === 'undefined') 961 { 962 window.OneDriveClient = null; 963 } 964 965 // Loads Trello for all browsers but < IE10 if not disabled or if enabled and in embed mode 966 if (typeof window.TrelloClient === 'function' && !mxClient.IS_IE11 && 967 typeof window.Trello === 'undefined' && window.DrawTrelloClientCallback != null && 968 urlParams['tr'] == '1' && (navigator.userAgent == null || 969 navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10)) 970 { 971 mxscript(App.TRELLO_JQUERY_URL, function() 972 { 973 // Must load this after the dropbox SDK since they use the same namespace 974 mxscript(App.TRELLO_URL, function() 975 { 976 DrawTrelloClientCallback(); 977 }); 978 }); 979 } 980 // Disables client 981 else if (typeof window.Trello === 'undefined') 982 { 983 window.TrelloClient = null; 984 } 985 986 } 987 988 if (callback != null) 989 { 990 callback(ui); 991 } 992 993 /** 994 * For developers only 995 */ 996 if (urlParams['chrome'] != '0' && urlParams['test'] == '1') 997 { 998 EditorUi.debug('App.start', [ui, (new Date().getTime() - t0.getTime()) + 'ms']); 999 1000 if (urlParams['export'] != null) 1001 { 1002 EditorUi.debug('Export:', EXPORT_URL); 1003 } 1004 } 1005 }; 1006 1007 if (urlParams['dev'] == '1' || EditorUi.isElectronApp) //TODO check if we can remove these scripts loading from index.html 1008 { 1009 realMain(); 1010 } 1011 else 1012 { 1013 mxStencilRegistry.allowEval = false; 1014 App.loadScripts(['js/shapes-14-6-5.min.js', 'js/stencils.min.js', 1015 'js/extensions.min.js'], realMain); 1016 } 1017 }, function(xhr) 1018 { 1019 var st = document.getElementById('geStatus'); 1020 1021 if (st != null) 1022 { 1023 st.innerHTML = 'Error loading page. <a>Please try refreshing.</a>'; 1024 1025 // Tries reload with default resources in case any language resources were not available 1026 st.getElementsByTagName('a')[0].onclick = function() 1027 { 1028 mxLanguage = 'en'; 1029 doLoad(mxResources.getDefaultBundle(RESOURCE_BASE, mxLanguage) || 1030 mxResources.getSpecialBundle(RESOURCE_BASE, mxLanguage)); 1031 }; 1032 } 1033 }); 1034 }; 1035 1036 function doMain() 1037 { 1038 // Optional override for autosaveDelay and defaultEdgeLength 1039 try 1040 { 1041 if (mxSettings.settings != null) 1042 { 1043 document.body.style.backgroundColor = (uiTheme == 'dark' || 1044 mxSettings.settings.darkMode) ? Editor.darkColor : '#ffffff'; 1045 1046 if (mxSettings.settings.autosaveDelay != null) 1047 { 1048 var val = parseInt(mxSettings.settings.autosaveDelay); 1049 1050 if (!isNaN(val) && val > 0) 1051 { 1052 DrawioFile.prototype.autosaveDelay = val; 1053 EditorUi.debug('Setting autosaveDelay', val); 1054 } 1055 else 1056 { 1057 EditorUi.debug('Invalid autosaveDelay', val); 1058 } 1059 } 1060 1061 if (mxSettings.settings.defaultEdgeLength != null) 1062 { 1063 var val = parseInt(mxSettings.settings.defaultEdgeLength); 1064 1065 if (!isNaN(val) && val > 0) 1066 { 1067 Graph.prototype.defaultEdgeLength = val; 1068 EditorUi.debug('Using defaultEdgeLength', val); 1069 } 1070 else 1071 { 1072 EditorUi.debug('Invalid defaultEdgeLength', val); 1073 } 1074 } 1075 } 1076 } 1077 catch (e) 1078 { 1079 if (window.console != null) 1080 { 1081 console.error(e); 1082 } 1083 } 1084 1085 // Prefetches default fonts with URLs 1086 if (Menus.prototype.defaultFonts != null) 1087 { 1088 for (var i = 0; i < Menus.prototype.defaultFonts.length; i++) 1089 { 1090 var value = Menus.prototype.defaultFonts[i]; 1091 1092 if (typeof value !== 'string' && 1093 value.fontFamily != null && 1094 value.fontUrl != null) 1095 { 1096 Graph.addFont(value.fontFamily, value.fontUrl); 1097 } 1098 } 1099 } 1100 1101 // Adds required resources (disables loading of fallback properties, this can only 1102 // be used if we know that all keys are defined in the language specific file) 1103 mxResources.loadDefaultBundle = false; 1104 doLoad(mxResources.getDefaultBundle(RESOURCE_BASE, mxLanguage) || 1105 mxResources.getSpecialBundle(RESOURCE_BASE, mxLanguage)); 1106 }; 1107 1108 // Sends load event if configuration is requested and waits for configure message 1109 if (urlParams['configure'] == '1') 1110 { 1111 var op = window.opener || window.parent; 1112 1113 var configHandler = function(evt) 1114 { 1115 if (evt.source == op) 1116 { 1117 try 1118 { 1119 var data = JSON.parse(evt.data); 1120 1121 if (data != null && data.action == 'configure') 1122 { 1123 mxEvent.removeListener(window, 'message', configHandler); 1124 Editor.configure(data.config, true); 1125 mxSettings.load(); 1126 doMain(); 1127 } 1128 } 1129 catch (e) 1130 { 1131 if (window.console != null) 1132 { 1133 console.log('Error in configure message: ' + e, evt.data); 1134 } 1135 } 1136 } 1137 }; 1138 1139 // Receives XML message from opener and puts it into the graph 1140 mxEvent.addListener(window, 'message', configHandler); 1141 op.postMessage(JSON.stringify({event: 'configure'}), '*'); 1142 } 1143 else 1144 { 1145 if (Editor.config == null) 1146 { 1147 // Loads configuration from global scope or local storage 1148 if (window.DRAWIO_CONFIG != null) 1149 { 1150 try 1151 { 1152 EditorUi.debug('Using global configuration', window.DRAWIO_CONFIG); 1153 Editor.configure(window.DRAWIO_CONFIG); 1154 mxSettings.load(); 1155 } 1156 catch (e) 1157 { 1158 if (window.console != null) 1159 { 1160 console.error(e); 1161 } 1162 } 1163 } 1164 1165 // Loads configuration from local storage 1166 if (isLocalStorage && localStorage != null && urlParams['embed'] != '1') 1167 { 1168 var configData = localStorage.getItem(Editor.configurationKey); 1169 1170 if (configData != null) 1171 { 1172 try 1173 { 1174 configData = JSON.parse(configData); 1175 1176 if (configData != null) 1177 { 1178 EditorUi.debug('Using local configuration', configData); 1179 Editor.configure(configData); 1180 mxSettings.load(); 1181 } 1182 } 1183 catch (e) 1184 { 1185 if (window.console != null) 1186 { 1187 console.error(e); 1188 } 1189 } 1190 } 1191 } 1192 } 1193 1194 doMain(); 1195 } 1196}; 1197 1198//Extends EditorUi 1199mxUtils.extend(App, EditorUi); 1200 1201/** 1202 * Executes the first step for connecting to Google Drive. 1203 */ 1204App.prototype.defaultUserPicture = IMAGE_PATH + '/default-user.jpg'; 1205 1206/** 1207 * 1208 */ 1209App.prototype.shareImage = ''; 1210 1211/** 1212 * 1213 */ 1214App.prototype.chevronUpImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/chevron-up.png' : ''; 1215 1216/** 1217 * 1218 */ 1219App.prototype.chevronDownImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/chevron-down.png' : ''; 1220 1221/** 1222 * 1223 */ 1224App.prototype.formatShowImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/format-show.png' : ''; 1225 1226/** 1227 * 1228 */ 1229App.prototype.formatHideImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/format-hide.png' : ''; 1230 1231/** 1232 * 1233 */ 1234App.prototype.fullscreenImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/fullscreen.png' : ''; 1235 1236/** 1237 * Interval to show dialog for unsaved data if autosave is on. 1238 * Default is 300000 (5 minutes). 1239 */ 1240App.prototype.warnInterval = 300000; 1241 1242/** 1243 * 1244 */ 1245App.prototype.compactMode = false; 1246 1247/** 1248 * 1249 */ 1250App.prototype.fullscreenMode = false; 1251 1252/** 1253 * Overriden UI settings depending on mode. 1254 */ 1255if (urlParams['embed'] != '1') 1256{ 1257 App.prototype.menubarHeight = 64; 1258} 1259else 1260{ 1261 App.prototype.footerHeight = 0; 1262} 1263 1264/** 1265 * Queue for loading plugins and wait for UI instance 1266 */ 1267App.initPluginCallback = function() 1268{ 1269 if (App.DrawPlugins == null) 1270 { 1271 // Workaround for need to load plugins now but wait for UI instance 1272 App.DrawPlugins = []; 1273 1274 // Global entry point for plugins is Draw.loadPlugin. This is the only 1275 // long-term supported solution for access to the EditorUi instance. 1276 window.Draw = new Object(); 1277 window.Draw.loadPlugin = function(callback) 1278 { 1279 App.DrawPlugins.push(callback); 1280 }; 1281 } 1282}; 1283 1284/** 1285 * 1286 */ 1287App.pluginsLoaded = {}; 1288App.embedModePluginsCount = 0; 1289 1290/** 1291 * Queue for loading plugins and wait for UI instance 1292 */ 1293App.loadPlugins = function(plugins, useInclude) 1294{ 1295 EditorUi.debug('Loading plugins', plugins); 1296 1297 for (var i = 0; i < plugins.length; i++) 1298 { 1299 if (plugins[i] != null && plugins[i].length > 0) 1300 { 1301 try 1302 { 1303 var url = PLUGINS_BASE_PATH + App.pluginRegistry[plugins[i]]; 1304 1305 if (url != null) 1306 { 1307 if (App.pluginsLoaded[url] == null) 1308 { 1309 App.pluginsLoaded[url] = true; 1310 App.embedModePluginsCount++; 1311 1312 if (typeof window.drawDevUrl === 'undefined') 1313 { 1314 if (useInclude) 1315 { 1316 mxinclude(url); 1317 } 1318 else 1319 { 1320 mxscript(url); 1321 } 1322 } 1323 else 1324 { 1325 if (useInclude) 1326 { 1327 mxinclude(url); 1328 } 1329 else 1330 { 1331 mxscript(drawDevUrl + url); 1332 } 1333 } 1334 } 1335 } 1336 else if (window.console != null) 1337 { 1338 console.log('Unknown plugin:', plugins[i]); 1339 } 1340 } 1341 catch (e) 1342 { 1343 if (window.console != null) 1344 { 1345 console.log('Error loading plugin:', plugins[i], e); 1346 } 1347 } 1348 } 1349 } 1350}; 1351 1352/** 1353 * Delay embed mode initialization until all plugins are loaded 1354 */ 1355App.prototype.initializeEmbedMode = function() 1356{ 1357 if (urlParams['embed'] == '1') 1358 { 1359 if (window.location.hostname == 'app.diagrams.net') 1360 { 1361 this.showBanner('EmbedDeprecationFooter', 'app.diagrams.net will stop working for embed mode. Please use embed.diagrams.net.'); 1362 } 1363 1364 if (App.embedModePluginsCount > 0 || this.initEmbedDone) 1365 { 1366 return; //Wait for plugins to load, or this is a duplicate call due to timeout 1367 } 1368 else 1369 { 1370 this.initEmbedDone = true; 1371 } 1372 1373 EditorUi.prototype.initializeEmbedMode.apply(this, arguments); 1374 } 1375}; 1376 1377/** 1378 * TODO: Define viewer protocol and implement new viewer style toolbar 1379 */ 1380App.prototype.initializeViewerMode = function() 1381{ 1382 var parent = window.opener || window.parent; 1383 1384 if (parent != null) 1385 { 1386 this.editor.graph.addListener(mxEvent.SIZE, mxUtils.bind(this, function() 1387 { 1388 parent.postMessage(JSON.stringify(this.createLoadMessage('size')), '*'); 1389 })); 1390 } 1391}; 1392 1393/** 1394 * Translates this point by the given vector. 1395 * 1396 * @param {number} dx X-coordinate of the translation. 1397 * @param {number} dy Y-coordinate of the translation. 1398 */ 1399App.prototype.init = function() 1400{ 1401 EditorUi.prototype.init.apply(this, arguments); 1402 1403 /** 1404 * Specifies the default filename. 1405 */ 1406 this.defaultLibraryName = mxResources.get('untitledLibrary'); 1407 1408 /** 1409 * Holds the listener for description changes. 1410 */ 1411 this.descriptorChangedListener = mxUtils.bind(this, this.descriptorChanged); 1412 1413 /** 1414 * Creates github client. 1415 */ 1416 this.gitHub = (!mxClient.IS_IE || document.documentMode == 10 || 1417 mxClient.IS_IE11 || mxClient.IS_EDGE) && 1418 (urlParams['gh'] != '0' && (urlParams['embed'] != '1' || 1419 urlParams['gh'] == '1')) ? new GitHubClient(this) : null; 1420 1421 if (this.gitHub != null) 1422 { 1423 this.gitHub.addListener('userChanged', mxUtils.bind(this, function() 1424 { 1425 this.updateUserElement(); 1426 this.restoreLibraries(); 1427 1428 this.showBanner('GithubFooter', 'Click to install GitHub app', mxUtils.bind(this, function() 1429 { 1430 this.openLink('https://github.com/apps/draw-io-app'); 1431 })); 1432 })); 1433 } 1434 1435 /** 1436 * Creates gitlab client. 1437 */ 1438 this.gitLab = (!mxClient.IS_IE || document.documentMode == 10 || 1439 mxClient.IS_IE11 || mxClient.IS_EDGE) && 1440 (urlParams['gl'] != '0' && (urlParams['embed'] != '1' || 1441 urlParams['gl'] == '1')) ? new GitLabClient(this) : null; 1442 1443 if (this.gitLab != null) 1444 { 1445 this.gitLab.addListener('userChanged', mxUtils.bind(this, function() 1446 { 1447 this.updateUserElement(); 1448 this.restoreLibraries(); 1449 })); 1450 } 1451 1452 /** 1453 * Creates notion client. 1454 */ 1455 this.notion = 1456 //TODO disabled by default 1457 /*(!mxClient.IS_IE || document.documentMode == 10 || 1458 mxClient.IS_IE11 || mxClient.IS_EDGE) && 1459 (urlParams['ntn'] != '0' && (urlParams['embed'] != '1' || 1460 urlParams['ntn'] == '1')) */ 1461 urlParams['ntn'] == '1' ? new NotionClient(this) : null; 1462 1463 if (this.notion != null) 1464 { 1465 this.notion.addListener('userChanged', mxUtils.bind(this, function() 1466 { 1467 this.updateUserElement(); 1468 this.restoreLibraries(); 1469 })); 1470 } 1471 1472 /** 1473 * Lazy-loading for individual backends 1474 */ 1475 if (urlParams['embed'] != '1' || urlParams['od'] == '1') 1476 { 1477 /** 1478 * Creates onedrive client if all required libraries are available. 1479 */ 1480 var initOneDriveClient = mxUtils.bind(this, function() 1481 { 1482 if (typeof OneDrive !== 'undefined') 1483 { 1484 /** 1485 * Holds the x-coordinate of the point. 1486 */ 1487 this.oneDrive = new OneDriveClient(this); 1488 1489 this.oneDrive.addListener('userChanged', mxUtils.bind(this, function() 1490 { 1491 this.updateUserElement(); 1492 this.restoreLibraries(); 1493 })); 1494 1495 // Notifies listeners of new client 1496 this.fireEvent(new mxEventObject('clientLoaded', 'client', this.oneDrive)); 1497 } 1498 else if (window.DrawOneDriveClientCallback == null) 1499 { 1500 window.DrawOneDriveClientCallback = initOneDriveClient; 1501 } 1502 }); 1503 1504 initOneDriveClient(); 1505 } 1506 1507 /** 1508 * Lazy-loading for Trello 1509 */ 1510 if (urlParams['embed'] != '1' || urlParams['tr'] == '1') 1511 { 1512 /** 1513 * Creates Trello client if all required libraries are available. 1514 */ 1515 var initTrelloClient = mxUtils.bind(this, function() 1516 { 1517 if (typeof window.Trello !== 'undefined') 1518 { 1519 try 1520 { 1521 this.trello = new TrelloClient(this); 1522 1523 //TODO we have no user info from Trello so we don't set a user 1524 this.trello.addListener('userChanged', mxUtils.bind(this, function() 1525 { 1526 this.updateUserElement(); 1527 this.restoreLibraries(); 1528 })); 1529 1530 // Notifies listeners of new client 1531 this.fireEvent(new mxEventObject('clientLoaded', 'client', this.trello)); 1532 } 1533 catch (e) 1534 { 1535 if (window.console != null) 1536 { 1537 console.error(e); 1538 } 1539 } 1540 } 1541 else if (window.DrawTrelloClientCallback == null) 1542 { 1543 window.DrawTrelloClientCallback = initTrelloClient; 1544 } 1545 }); 1546 1547 initTrelloClient(); 1548 } 1549 1550 /** 1551 * Creates drive client with all required libraries are available. 1552 */ 1553 if (urlParams['embed'] != '1' || urlParams['gapi'] == '1') 1554 { 1555 var initDriveClient = mxUtils.bind(this, function() 1556 { 1557 /** 1558 * Creates google drive client if all required libraries are available. 1559 */ 1560 if (typeof gapi !== 'undefined') 1561 { 1562 var doInit = mxUtils.bind(this, function() 1563 { 1564 this.drive = new DriveClient(this); 1565 1566 this.drive.addListener('userChanged', mxUtils.bind(this, function() 1567 { 1568 this.updateUserElement(); 1569 this.restoreLibraries(); 1570 this.checkLicense(); 1571 })) 1572 1573 // Notifies listeners of new client 1574 this.fireEvent(new mxEventObject('clientLoaded', 'client', this.drive)); 1575 }); 1576 1577 if (window.DrawGapiClientCallback != null) 1578 { 1579 gapi.load(((urlParams['picker'] != '0') ? 'picker,': '') + App.GOOGLE_APIS, doInit); 1580 1581 /** 1582 * Clears any callbacks. 1583 */ 1584 window.DrawGapiClientCallback = null; 1585 } 1586 else 1587 { 1588 doInit(); 1589 } 1590 } 1591 else if (window.DrawGapiClientCallback == null) 1592 { 1593 window.DrawGapiClientCallback = initDriveClient; 1594 } 1595 }); 1596 1597 initDriveClient(); 1598 } 1599 1600 if (urlParams['embed'] != '1' || urlParams['db'] == '1') 1601 { 1602 /** 1603 * Creates dropbox client if all required libraries are available. 1604 */ 1605 var initDropboxClient = mxUtils.bind(this, function() 1606 { 1607 if (typeof Dropbox === 'function' && typeof Dropbox.choose !== 'undefined') 1608 { 1609 /** 1610 * Clears dropbox client callback. 1611 */ 1612 window.DrawDropboxClientCallback = null; 1613 1614 /** 1615 * Holds the x-coordinate of the point. 1616 */ 1617 try 1618 { 1619 this.dropbox = new DropboxClient(this); 1620 1621 this.dropbox.addListener('userChanged', mxUtils.bind(this, function() 1622 { 1623 this.updateUserElement(); 1624 this.restoreLibraries(); 1625 })); 1626 1627 // Notifies listeners of new client 1628 this.fireEvent(new mxEventObject('clientLoaded', 'client', this.dropbox)); 1629 } 1630 catch (e) 1631 { 1632 if (window.console != null) 1633 { 1634 console.error(e); 1635 } 1636 } 1637 } 1638 else if (window.DrawDropboxClientCallback == null) 1639 { 1640 window.DrawDropboxClientCallback = initDropboxClient; 1641 } 1642 }); 1643 1644 initDropboxClient(); 1645 } 1646 1647 if (urlParams['embed'] != '1') 1648 { 1649 /** 1650 * Holds the background element. 1651 */ 1652 this.bg = this.createBackground(); 1653 document.body.appendChild(this.bg); 1654 this.diagramContainer.style.visibility = 'hidden'; 1655 this.formatContainer.style.visibility = 'hidden'; 1656 this.hsplit.style.display = 'none'; 1657 this.sidebarContainer.style.display = 'none'; 1658 this.sidebarFooterContainer.style.display = 'none'; 1659 1660 // Sets the initial mode 1661 if (urlParams['local'] == '1') 1662 { 1663 this.setMode(App.MODE_DEVICE); 1664 } 1665 else 1666 { 1667 this.mode = App.mode; 1668 } 1669 1670 // Add to Home Screen dialog for mobile devices 1671 if ('serviceWorker' in navigator && !this.editor.isChromelessView() && 1672 (mxClient.IS_ANDROID || mxClient.IS_IOS)) 1673 { 1674 window.addEventListener('beforeinstallprompt', mxUtils.bind(this, function(e) 1675 { 1676 this.showBanner('AddToHomeScreenFooter', mxResources.get('installApp'), function() 1677 { 1678 e.prompt(); 1679 }); 1680 })); 1681 } 1682 1683 if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && !this.isOffline() && 1684 !mxClient.IS_ANDROID && !mxClient.IS_IOS && urlParams['open'] == null && 1685 (!this.editor.chromeless || this.editor.editable)) 1686 { 1687 this.editor.addListener('fileLoaded', mxUtils.bind(this, function() 1688 { 1689 var file = this.getCurrentFile(); 1690 var mode = (file != null) ? file.getMode() : null; 1691 1692 if (urlParams['extAuth'] != '1' && (mode == App.MODE_DEVICE || mode == App.MODE_BROWSER)) 1693 { 1694 this.showDownloadDesktopBanner(); 1695 } 1696 else if (urlParams['embed'] != '1' && this.getServiceName() == 'draw.io') 1697 1698 { 1699 // just app.diagrams.net users 1700 // this.showNameConfBanner(); 1701 } 1702 })); 1703 } 1704 1705 if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && urlParams['embed'] != '1' && DrawioFile.SYNC == 'auto' && 1706 urlParams['local'] != '1' && urlParams['stealth'] != '1' && !this.isOffline() && 1707 (!this.editor.chromeless || this.editor.editable)) 1708 { 1709 // Checks if the cache is alive 1710 var acceptResponse = true; 1711 1712 var timeoutThread = window.setTimeout(mxUtils.bind(this, function() 1713 { 1714 acceptResponse = false; 1715 1716 // Switches to manual sync if cache cannot be reached 1717 DrawioFile.SYNC = 'manual'; 1718 1719 var file = this.getCurrentFile(); 1720 1721 if (file != null && file.sync != null) 1722 { 1723 file.sync.destroy(); 1724 file.sync = null; 1725 1726 var status = mxUtils.htmlEntities(mxResources.get('timeout')); 1727 this.editor.setStatus('<div title="'+ status + 1728 '" class="geStatusAlert">' + status + '</div>'); 1729 } 1730 1731 EditorUi.logEvent({category: 'TIMEOUT-CACHE-CHECK', action: 'timeout', label: 408}); 1732 }), Editor.cacheTimeout); 1733 1734 var t0 = new Date().getTime(); 1735 1736 mxUtils.get(EditorUi.cacheUrl + '?alive', mxUtils.bind(this, function(req) 1737 { 1738 window.clearTimeout(timeoutThread); 1739 })); 1740 } 1741 } 1742 else if (this.menubar != null) 1743 { 1744 this.menubar.container.style.paddingTop = '0px'; 1745 } 1746 1747 this.updateHeader(); 1748 1749 if (this.menubar != null) 1750 { 1751 this.buttonContainer = document.createElement('div'); 1752 this.buttonContainer.style.display = 'inline-block'; 1753 this.buttonContainer.style.paddingRight = '48px'; 1754 this.buttonContainer.style.position = 'absolute'; 1755 this.buttonContainer.style.right = '0px'; 1756 1757 this.menubar.container.appendChild(this.buttonContainer); 1758 } 1759 1760 if ((uiTheme == 'atlas' || urlParams['atlas'] == '1') && this.menubar != null) 1761 { 1762 if (this.toggleElement != null) 1763 { 1764 this.toggleElement.click(); 1765 this.toggleElement.style.display = 'none'; 1766 } 1767 1768 this.icon = document.createElement('img'); 1769 this.icon.setAttribute('src', IMAGE_PATH + '/logo-flat-small.png'); 1770 this.icon.setAttribute('title', mxResources.get('draw.io')); 1771 this.icon.style.padding = urlParams['atlas'] == '1'? '7px' : '6px'; 1772 this.icon.style.cursor = 'pointer'; 1773 1774 mxEvent.addListener(this.icon, 'click', mxUtils.bind(this, function(evt) 1775 { 1776 this.appIconClicked(evt); 1777 })); 1778 1779 this.menubar.container.insertBefore(this.icon, this.menubar.container.firstChild); 1780 } 1781 1782 if (this.editor.graph.isViewer()) 1783 { 1784 this.initializeViewerMode(); 1785 } 1786}; 1787 1788/** 1789 * Schedules a sanity check. 1790 */ 1791App.prototype.scheduleSanityCheck = function() 1792{ 1793 if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && 1794 this.sanityCheckThread == null) 1795 { 1796 this.sanityCheckThread = window.setTimeout(mxUtils.bind(this, function() 1797 { 1798 this.sanityCheckThread = null; 1799 this.sanityCheck(); 1800 }), this.warnInterval); 1801 } 1802}; 1803 1804/** 1805 * Stops sanity checks. 1806 */ 1807App.prototype.stopSanityCheck = function() 1808{ 1809 if (this.sanityCheckThread != null) 1810 { 1811 window.clearTimeout(this.sanityCheckThread); 1812 this.sanityCheckThread = null; 1813 } 1814}; 1815 1816/** 1817 * Shows a warning after some time with unsaved changes and autosave. 1818 */ 1819App.prototype.sanityCheck = function() 1820{ 1821 var file = this.getCurrentFile(); 1822 1823 if (file != null && file.isModified() && file.isAutosave() && file.isOverdue()) 1824 { 1825 var evt = {category: 'WARN-FILE-' + file.getHash(), 1826 action: ((file.savingFile) ? 'saving' : '') + 1827 ((file.savingFile && file.savingFileTime != null) ? '_' + 1828 Math.round((Date.now() - file.savingFileTime.getTime()) / 1000) : '') + 1829 ((file.saveLevel != null) ? ('-sl_' + file.saveLevel) : '') + 1830 '-age_' + ((file.ageStart != null) ? Math.round((Date.now() - file.ageStart.getTime()) / 1000) : 'x') + 1831 ((this.editor.autosave) ? '' : '-nosave') + 1832 ((file.isAutosave()) ? '' : '-noauto') + 1833 '-open_' + ((file.opened != null) ? Math.round((Date.now() - file.opened.getTime()) / 1000) : 'x') + 1834 '-save_' + ((file.lastSaved != null) ? Math.round((Date.now() - file.lastSaved.getTime()) / 1000) : 'x') + 1835 '-change_' + ((file.lastChanged != null) ? Math.round((Date.now() - file.lastChanged.getTime()) / 1000) : 'x')+ 1836 '-alive_' + Math.round((Date.now() - App.startTime.getTime()) / 1000), 1837 label: (file.sync != null) ? ('client_' + file.sync.clientId) : 'nosync'}; 1838 1839 if (file.constructor == DriveFile && file.desc != null && this.drive != null) 1840 { 1841 evt.label += ((this.drive.user != null) ? ('-user_' + this.drive.user.id) : '-nouser') + '-rev_' + 1842 file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + '-size_' + file.getSize() + 1843 '-mime_' + file.desc.mimeType; 1844 } 1845 1846 EditorUi.logEvent(evt); 1847 1848 var msg = mxResources.get('ensureDataSaved'); 1849 1850 if (file.lastSaved != null) 1851 { 1852 var str = this.timeSince(file.lastSaved); 1853 1854 if (str == null) 1855 { 1856 str = mxResources.get('lessThanAMinute'); 1857 } 1858 1859 msg = mxResources.get('lastSaved', [str]); 1860 } 1861 1862 // Resets possible stale state 1863 this.spinner.stop(); 1864 1865 this.showError(mxResources.get('unsavedChanges'), msg, mxResources.get('ignore'), 1866 mxUtils.bind(this, function() 1867 { 1868 this.hideDialog(); 1869 }), null, mxResources.get('save'), mxUtils.bind(this, function() 1870 { 1871 this.stopSanityCheck(); 1872 this.actions.get((this.mode == null || !file.isEditable()) ? 1873 'saveAs' : 'save').funct(); 1874 }), null, null, 360, 120, null, mxUtils.bind(this, function() 1875 { 1876 this.scheduleSanityCheck(); 1877 })); 1878 } 1879}; 1880 1881/** 1882 * Returns true if the current domain is for the new drive app. 1883 */ 1884App.prototype.isDriveDomain = function() 1885{ 1886 return urlParams['drive'] != '0' && 1887 (window.location.hostname == 'test.draw.io' || 1888 window.location.hostname == 'www.draw.io' || 1889 window.location.hostname == 'drive.draw.io' || 1890 window.location.hostname == 'app.diagrams.net' || 1891 window.location.hostname == 'jgraph.github.io'); 1892}; 1893 1894/** 1895 * Returns the pusher instance for notifications. Creates the instance of none exists. 1896 */ 1897App.prototype.getPusher = function() 1898{ 1899 if (this.pusher == null && typeof window.Pusher === 'function') 1900 { 1901 this.pusher = new Pusher(App.PUSHER_KEY, 1902 { 1903 cluster: App.PUSHER_CLUSTER, 1904 encrypted: true 1905 }); 1906 } 1907 1908 return this.pusher; 1909}; 1910 1911/** 1912 * Shows a footer to download the desktop version once per session. 1913 */ 1914App.prototype.showNameChangeBanner = function() 1915{ 1916 this.showBanner('DiagramsFooter', 'draw.io is now diagrams.net', mxUtils.bind(this, function() 1917 { 1918 this.openLink('https://www.diagrams.net/blog/move-diagrams-net'); 1919 })); 1920}; 1921 1922/** 1923 * Shows a footer to download the desktop version once per session. 1924 */ 1925App.prototype.showNameConfBanner = function() 1926{ 1927 this.showBanner('ConfFooter', 'Try draw.io for Confluence', mxUtils.bind(this, function() 1928 { 1929 this.openLink('https://marketplace.atlassian.com/apps/1210933/draw-io-diagrams-for-confluence'); 1930 }), true); 1931}; 1932 1933/** 1934 * Shows a footer to download the desktop version once per session. 1935 */ 1936App.prototype.showDownloadDesktopBanner = function() 1937{ 1938 this.showBanner('DesktopFooter', mxResources.get('downloadDesktop'), mxUtils.bind(this, function() 1939 { 1940 this.openLink('https://get.diagrams.net/'); 1941 })); 1942}; 1943 1944/** 1945 * Shows a footer to download the desktop version once per session. 1946 */ 1947App.prototype.showRatingBanner = function() 1948{ 1949 if (!this.bannerShowing && !this['hideBanner' + 'ratingFooter'] && 1950 (!isLocalStorage || mxSettings.settings == null || 1951 mxSettings.settings['close' + 'ratingFooter'] == null)) 1952 { 1953 var banner = document.createElement('div'); 1954 banner.style.cssText = 'position:absolute;bottom:10px;left:50%;max-width:90%;padding:18px 34px 12px 20px;' + 1955 'font-size:16px;font-weight:bold;white-space:nowrap;cursor:pointer;z-index:' + mxPopupMenu.prototype.zIndex + ';'; 1956 mxUtils.setPrefixedStyle(banner.style, 'box-shadow', '1px 1px 2px 0px #ddd'); 1957 mxUtils.setPrefixedStyle(banner.style, 'transform', 'translate(-50%,120%)'); 1958 mxUtils.setPrefixedStyle(banner.style, 'transition', 'all 1s ease'); 1959 banner.className = 'geBtn gePrimaryBtn'; 1960 1961 var img = document.createElement('img'); 1962 img.setAttribute('src', Dialog.prototype.closeImage); 1963 img.setAttribute('title', mxResources.get('close')); 1964 img.setAttribute('border', '0'); 1965 img.style.cssText = 'position:absolute;right:10px;top:12px;filter:invert(1);padding:6px;margin:-6px;cursor:default;'; 1966 banner.appendChild(img); 1967 1968 var star = '' + 1969 'XdvcmtzIENTM5jWRgMAAAQRdEVYdFhNTDpjb20uYWRvYmUueG1wADw/eHBhY2tldCBiZWdpbj0iICAgIiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+Cjx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8i' + 1970 'IHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDQuMS1jMDM0IDQ2LjI3Mjk3NiwgU2F0IEphbiAyNyAyMDA3IDIyOjExOjQxICAgICAgICAiPgogICA8cmRmOlJERiB4bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDI' + 1971 'vMjItcmRmLXN5bnRheC1ucyMiPgogICAgICA8cmRmOkRlc2NyaXB0aW9uIHJkZjphYm91dD0iIgogICAgICAgICAgICB4bWxuczp4YXA9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC8iPgogICAgICAgICA8eGFwOkNyZW' + 1972 'F0b3JUb29sPkFkb2JlIEZpcmV3b3JrcyBDUzM8L3hhcDpDcmVhdG9yVG9vbD4KICAgICAgICAgPHhhcDpDcmVhdGVEYXRlPjIwMDgtMDItMTdUMDI6MzY6NDVaPC94YXA6Q3JlYXRlRGF0ZT4KICAgICAgICAgPHhhcDpNb2RpZ' + 1973 'nlEYXRlPjIwMDktMDMtMTdUMTQ6MTI6MDJaPC94YXA6TW9kaWZ5RGF0ZT4KICAgICAgPC9yZGY6RGVzY3JpcHRpb24+CiAgICAgIDxyZGY6RGVzY3JpcHRpb24gcmRmOmFib3V0PSIiCiAgICAgICAgICAgIHhtbG5zOmRjPSJo' + 1974 'dHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyI+CiAgICAgICAgIDxkYzpmb3JtYXQ+aW1hZ2UvcG5nPC9kYzpmb3JtYXQ+CiAgICAgIDwvcmRmOkRlc2NyaXB0aW9uPgogICA8L3JkZjpSREY+CjwveDp4bXBtZXRhPgo' + 1975 'gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgIC' + 1976 'AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgI' + 1977 'CAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIIImu8AAAAAVdEVYdENyZWF0aW9uIFRpbWUAMi8xNy8wOCCcqlgAAAHuSURBVDiNlZJBi1JRGIbfk+fc0ZuMXorJe4XujWoMdREaA23HICj6AQeLINr0C4I27ab2' + 1978 '7VqOI9+q/sH8gMDceG1RkIwgClEXFMbRc5zTZgZURmG+5fu9PN/7Hg6wZohoh4h21nn4uqXW+q0xZgzg+SrPlTXX73uet+26bp6ICpcGaK1fua57M5vN3tZav7gUgIiSqVTqcRAEm0EQbCaTyQoRXb3Iy4hoG8CT6XSaY4xtMMa' + 1979 'SQohMPp8v+r7vAEC3243CMGwqpfoApsaYE8uyfgM45ABOjDEvXdfNlMvlzFINAIDneY7neZVzvdlsDgaDQYtzfsjOIjtKqU+e5+0Wi0V3VV8ACMOw3+/3v3HOX0sp/7K53te11h/S6fRuoVAIhBAL76OUOm2320dRFH0VQuxJKf' + 1980 '8BAFu+UKvVvpRKpWe2bYt5fTweq0ajQUKIN1LK43N94SMR0Y1YLLYlhBBKqQUw51wkEol7WmuzoC8FuJtIJLaUUoii6Ljb7f4yxpz6vp9zHMe2bfvacDi8BeDHKkBuNps5rVbr52QyaVuW9ZExttHpdN73ej0/Ho+nADxYCdBaV' + 1981 '0aj0RGAz5ZlHUgpx2erR/V6/d1wOHwK4CGA/QsBnPN9AN+llH+WkqFare4R0QGAO/M6M8Ysey81/wGqa8MlVvHPNAAAAABJRU5ErkJggg=='; 1982 1983 mxUtils.write(banner, 'Please rate us'); 1984 document.body.appendChild(banner); 1985 1986 var star1 = document.createElement('img'); 1987 star1.setAttribute('border', '0'); 1988 star1.setAttribute('align', 'absmiddle'); 1989 star1.setAttribute('title', '1 star'); 1990 star1.setAttribute('style', 'margin-top:-6px;cursor:pointer;margin-left:8px;'); 1991 star1.setAttribute('src', star); 1992 banner.appendChild(star1); 1993 1994 var star2 = document.createElement('img'); 1995 star2.setAttribute('border', '0'); 1996 star2.setAttribute('align', 'absmiddle'); 1997 star2.setAttribute('title', '2 star'); 1998 star2.setAttribute('style', 'margin-top:-6px;margin-left:3px;cursor:pointer;'); 1999 star2.setAttribute('src', star); 2000 banner.appendChild(star2); 2001 2002 var star3 = document.createElement('img'); 2003 star3.setAttribute('border', '0'); 2004 star3.setAttribute('align', 'absmiddle'); 2005 star3.setAttribute('title', '3 star'); 2006 star3.setAttribute('style', 'margin-top:-6px;margin-left:3px;cursor:pointer;'); 2007 star3.setAttribute('src', star); 2008 banner.appendChild(star3); 2009 2010 var star4 = document.createElement('img'); 2011 star4.setAttribute('border', '0'); 2012 star4.setAttribute('align', 'absmiddle'); 2013 star4.setAttribute('title', '4 star'); 2014 star4.setAttribute('style', 'margin-top:-6px;margin-left:3px;cursor:pointer;'); 2015 star4.setAttribute('src', star); 2016 banner.appendChild(star4); 2017 2018 this.bannerShowing = true; 2019 2020 var onclose = mxUtils.bind(this, function() 2021 { 2022 if (banner.parentNode != null) 2023 { 2024 banner.parentNode.removeChild(banner); 2025 this.bannerShowing = false; 2026 2027 this['hideBanner' + 'ratingFooter'] = true; 2028 2029 if (isLocalStorage && mxSettings.settings != null) 2030 { 2031 mxSettings.settings['close' + 'ratingFooter'] = Date.now(); 2032 mxSettings.save(); 2033 } 2034 } 2035 }); 2036 2037 mxEvent.addListener(img, 'click', mxUtils.bind(this, function(e) 2038 { 2039 mxEvent.consume(e); 2040 onclose(); 2041 })); 2042 mxEvent.addListener(star1, 'click', mxUtils.bind(this, function(e) 2043 { 2044 mxEvent.consume(e); 2045 onclose(); 2046 })); 2047 mxEvent.addListener(star2, 'click', mxUtils.bind(this, function(e) 2048 { 2049 mxEvent.consume(e); 2050 onclose(); 2051 })); 2052 mxEvent.addListener(star3, 'click', mxUtils.bind(this, function(e) 2053 { 2054 mxEvent.consume(e); 2055 onclose(); 2056 })); 2057 mxEvent.addListener(star4, 'click', mxUtils.bind(this, function(e) 2058 { 2059 mxEvent.consume(e); 2060 window.open('https://marketplace.atlassian.com/apps/1210933/draw-io-diagrams-for-confluence?hosting=datacenter&tab=reviews'); 2061 onclose(); 2062 })); 2063 2064 var hide = mxUtils.bind(this, function() 2065 { 2066 mxUtils.setPrefixedStyle(banner.style, 'transform', 'translate(-50%,120%)'); 2067 2068 window.setTimeout(mxUtils.bind(this, function() 2069 { 2070 onclose(); 2071 }), 1000); 2072 }); 2073 2074 window.setTimeout(mxUtils.bind(this, function() 2075 { 2076 mxUtils.setPrefixedStyle(banner.style, 'transform', 'translate(-50%,0%)'); 2077 }), 500); 2078 2079 window.setTimeout(hide, 60000); 2080 } 2081}; 2082 2083/** 2084 * Checks license in the case of Google Drive storage. 2085 * IMPORTANT: Do not change this function without consulting 2086 * the privacy lead. No personal information must be sent. 2087 */ 2088App.prototype.checkLicense = function() 2089{ 2090 var driveUser = this.drive.getUser(); 2091 var email = (driveUser != null) ? driveUser.email : null; 2092 2093 if (!this.isOffline() && !this.editor.chromeless && email != null && driveUser.id != null) 2094 { 2095 // Only the domain and hashed user ID are transmitted. This code was reviewed and deemed 2096 // compliant by dbenson 2021-09-01. 2097 var at = email.lastIndexOf('@'); 2098 var domain = (at >= 0) ? email.substring(at + 1) : ''; 2099 var userId = Editor.crc32(driveUser.id); 2100 2101 // Timestamp is workaround for cached response in certain environments 2102 mxUtils.post('/license', 'domain=' + encodeURIComponent(domain) + '&id=' + encodeURIComponent(userId) + 2103 '&ts=' + new Date().getTime(), 2104 mxUtils.bind(this, function(req) 2105 { 2106 try 2107 { 2108 if (req.getStatus() >= 200 && req.getStatus() <= 299) 2109 { 2110 var value = req.getText(); 2111 2112 if (value.length > 0) 2113 { 2114 var lic = JSON.parse(value); 2115 2116 if (lic != null) 2117 { 2118 this.handleLicense(lic, domain); 2119 } 2120 } 2121 } 2122 } 2123 catch (e) 2124 { 2125 // ignore 2126 } 2127 })); 2128 } 2129}; 2130 2131/** 2132 * Returns true if the current domain is for the new drive app. 2133 */ 2134App.prototype.handleLicense = function(lic, domain) 2135{ 2136 if (lic != null && lic.plugins != null) 2137 { 2138 App.loadPlugins(lic.plugins.split(';'), true); 2139 } 2140}; 2141 2142/** 2143 * 2144 */ 2145App.prototype.getEditBlankXml = function() 2146{ 2147 var file = this.getCurrentFile(); 2148 2149 if (file != null && this.editor.isChromelessView() && this.editor.graph.isLightboxView()) 2150 { 2151 return file.getData(); 2152 } 2153 else 2154 { 2155 return this.getFileData(true); 2156 } 2157}; 2158 2159/** 2160 * Updates action states depending on the selection. 2161 */ 2162App.prototype.updateActionStates = function() 2163{ 2164 EditorUi.prototype.updateActionStates.apply(this, arguments); 2165 2166 this.actions.get('revisionHistory').setEnabled(this.isRevisionHistoryEnabled()); 2167}; 2168 2169/** 2170 * Adds the specified entry to the recent file list in local storage 2171 */ 2172App.prototype.addRecent = function(entry) 2173{ 2174 if (isLocalStorage && localStorage != null) 2175 { 2176 var recent = this.getRecent(); 2177 2178 if (recent == null) 2179 { 2180 recent = []; 2181 } 2182 else 2183 { 2184 for (var i = 0; i < recent.length; i++) 2185 { 2186 if (recent[i].id == entry.id) 2187 { 2188 recent.splice(i, 1); 2189 } 2190 } 2191 } 2192 2193 if (recent != null) 2194 { 2195 recent.unshift(entry); 2196 recent = recent.slice(0, 10); 2197 localStorage.setItem('.recent', JSON.stringify(recent)); 2198 } 2199 } 2200}; 2201 2202/** 2203 * Returns the recent file list from local storage 2204 */ 2205App.prototype.getRecent = function() 2206{ 2207 if (isLocalStorage && localStorage != null) 2208 { 2209 try 2210 { 2211 var recent = localStorage.getItem('.recent'); 2212 2213 if (recent != null) 2214 { 2215 return JSON.parse(recent); 2216 } 2217 } 2218 catch (e) 2219 { 2220 // ignore 2221 } 2222 2223 return null; 2224 } 2225}; 2226 2227/** 2228 * Clears the recent file list in local storage 2229 */ 2230App.prototype.resetRecent = function(entry) 2231{ 2232 if (isLocalStorage && localStorage != null) 2233 { 2234 try 2235 { 2236 localStorage.removeItem('.recent'); 2237 } 2238 catch (e) 2239 { 2240 // ignore 2241 } 2242 } 2243}; 2244 2245/** 2246 * Sets the onbeforeunload for the application 2247 */ 2248App.prototype.onBeforeUnload = function() 2249{ 2250 if (urlParams['embed'] == '1' && this.editor.modified) 2251 { 2252 return mxResources.get('allChangesLost'); 2253 } 2254 else 2255 { 2256 var file = this.getCurrentFile(); 2257 2258 if (file != null) 2259 { 2260 // KNOWN: Message is ignored by most browsers 2261 if (file.constructor == LocalFile && file.getHash() == '' && !file.isModified() && 2262 urlParams['nowarn'] != '1' && !this.isDiagramEmpty() && urlParams['url'] == null && 2263 !this.editor.isChromelessView() && file.fileHandle == null) 2264 { 2265 return mxResources.get('ensureDataSaved'); 2266 } 2267 else if (file.isModified()) 2268 { 2269 return mxResources.get('allChangesLost'); 2270 } 2271 else 2272 { 2273 file.close(true); 2274 } 2275 } 2276 } 2277}; 2278 2279/** 2280 * Translates this point by the given vector. 2281 * 2282 * @param {number} dx X-coordinate of the translation. 2283 * @param {number} dy Y-coordinate of the translation. 2284 */ 2285App.prototype.updateDocumentTitle = function() 2286{ 2287 if (!this.editor.graph.isLightboxView()) 2288 { 2289 var title = this.editor.appName; 2290 var file = this.getCurrentFile(); 2291 2292 if (this.isOfflineApp()) 2293 { 2294 title += ' app'; 2295 } 2296 2297 if (file != null) 2298 { 2299 var filename = (file.getTitle() != null) ? file.getTitle() : this.defaultFilename; 2300 title = filename + ' - ' + title; 2301 } 2302 2303 if (document.title != title) 2304 { 2305 document.title = title; 2306 var graph = this.editor.graph; 2307 graph.invalidateDescendantsWithPlaceholders(graph.model.getRoot()); 2308 graph.view.validate(); 2309 } 2310 } 2311}; 2312 2313/** 2314 * Returns a thumbnail of the current file. 2315 */ 2316App.prototype.getThumbnail = function(width, fn) 2317{ 2318 var result = false; 2319 2320 try 2321 { 2322 var acceptResponse = true; 2323 2324 var timeoutThread = window.setTimeout(mxUtils.bind(this, function() 2325 { 2326 acceptResponse = false; 2327 fn(null); 2328 }), this.timeout); 2329 2330 var success = mxUtils.bind(this, function(canvas) 2331 { 2332 window.clearTimeout(timeoutThread); 2333 2334 if (acceptResponse) 2335 { 2336 fn(canvas); 2337 } 2338 }); 2339 2340 if (this.thumbImageCache == null) 2341 { 2342 this.thumbImageCache = new Object(); 2343 } 2344 2345 var graph = this.editor.graph; 2346 var bgImg = graph.backgroundImage; 2347 2348 // Exports PNG for first page while other page is visible by creating a graph 2349 // LATER: Add caching for the graph or SVG while not on first page 2350 // To avoid refresh during save dark theme uses separate graph instance 2351 var darkTheme = graph.themes != null && graph.defaultThemeName == 'darkTheme'; 2352 2353 if (this.pages != null && (darkTheme || this.currentPage != this.pages[0])) 2354 { 2355 var graphGetGlobalVariable = graph.getGlobalVariable; 2356 graph = this.createTemporaryGraph(graph.getStylesheet()); 2357 graph.setBackgroundImage = this.editor.graph.setBackgroundImage; 2358 var page = this.pages[0]; 2359 2360 if (this.currentPage == page) 2361 { 2362 graph.setBackgroundImage(bgImg); 2363 } 2364 else if (page.viewState != null && page.viewState != null) 2365 { 2366 bgImg = page.viewState.backgroundImage; 2367 graph.setBackgroundImage(bgImg); 2368 } 2369 2370 graph.getGlobalVariable = function(name) 2371 { 2372 if (name == 'page') 2373 { 2374 return page.getName(); 2375 } 2376 else if (name == 'pagenumber') 2377 { 2378 return 1; 2379 } 2380 2381 return graphGetGlobalVariable.apply(this, arguments); 2382 }; 2383 2384 graph.getGlobalVariable = graphGetGlobalVariable; 2385 document.body.appendChild(graph.container); 2386 graph.model.setRoot(page.root); 2387 } 2388 2389 // Uses client-side canvas export 2390 if (mxClient.IS_CHROMEAPP || this.useCanvasForExport) 2391 { 2392 this.editor.exportToCanvas(mxUtils.bind(this, function(canvas) 2393 { 2394 try 2395 { 2396 // Removes temporary graph from DOM 2397 if (graph != this.editor.graph && graph.container.parentNode != null) 2398 { 2399 graph.container.parentNode.removeChild(graph.container); 2400 } 2401 } 2402 catch (e) 2403 { 2404 canvas = null; 2405 } 2406 2407 success(canvas); 2408 }), width, this.thumbImageCache, '#ffffff', function() 2409 { 2410 // Continues with null in error case 2411 success(); 2412 }, null, null, null, null, null, null, graph, null, null, null, 2413 null, 'diagram', null); 2414 2415 result = true; 2416 } 2417 else if (this.canvasSupported && this.getCurrentFile() != null) 2418 { 2419 var canvas = document.createElement('canvas'); 2420 var bounds = graph.getGraphBounds(); 2421 var t = graph.view.translate; 2422 var s = graph.view.scale; 2423 2424 if (bgImg != null) 2425 { 2426 bounds = mxRectangle.fromRectangle(bounds); 2427 bounds.add(new mxRectangle( 2428 (t.x + bgImg.x) * s, (t.y + bgImg.y) * s, 2429 bgImg.width * s, bgImg.height * s)); 2430 } 2431 2432 var scale = width / bounds.width; 2433 2434 // Limits scale to 1 or 2 * width / height 2435 scale = Math.min(1, Math.min((width * 3) / (bounds.height * 4), scale)); 2436 2437 var x0 = Math.floor(bounds.x); 2438 var y0 = Math.floor(bounds.y); 2439 2440 canvas.setAttribute('width', Math.ceil(scale * (bounds.width + 4))); 2441 canvas.setAttribute('height', Math.ceil(scale * (bounds.height + 4))); 2442 2443 var ctx = canvas.getContext('2d'); 2444 2445 // Configures the canvas 2446 ctx.scale(scale, scale); 2447 ctx.translate(-x0, -y0); 2448 2449 // Paint white background instead of transparent 2450 var bg = graph.background; 2451 2452 if (bg == null || bg == '' || bg == mxConstants.NONE) 2453 { 2454 bg = '#ffffff'; 2455 } 2456 2457 // Paints background 2458 ctx.save(); 2459 ctx.fillStyle = bg; 2460 ctx.fillRect(x0, y0, Math.ceil(bounds.width + 4), Math.ceil(bounds.height + 4)); 2461 ctx.restore(); 2462 2463 // Paints background image 2464 if (bgImg != null) 2465 { 2466 var img = new Image(); 2467 img.src = bgImg.src; 2468 2469 ctx.drawImage(img, bgImg.x * scale, bgImg.y * scale, 2470 bgImg.width * scale, bgImg.height * scale); 2471 } 2472 2473 var htmlCanvas = new mxJsCanvas(canvas); 2474 2475 // NOTE: htmlCanvas passed into async canvas is only used for image 2476 // and canvas caching (canvas caching not used in this case as we do 2477 // not render text). To reuse that cache via the thumbImageCache we 2478 // pass that into the async canvas and override the image cache in 2479 // the newly created html canvas with that of the thumbImageCache. 2480 // LATER: Is clear thumbImageCache needed if file changes? 2481 var asynCanvas = new mxAsyncCanvas(this.thumbImageCache); 2482 htmlCanvas.images = this.thumbImageCache.images; 2483 2484 // Render graph 2485 var imgExport = new mxImageExport(); 2486 2487 imgExport.drawShape = function(state, canvas) 2488 { 2489 if (state.shape instanceof mxShape && state.shape.checkBounds()) 2490 { 2491 canvas.save(); 2492 canvas.translate(0.5, 0.5); 2493 state.shape.paint(canvas); 2494 canvas.translate(-0.5, -0.5); 2495 canvas.restore(); 2496 } 2497 }; 2498 2499 imgExport.drawText = function(state, canvas) 2500 { 2501 // No text output for thumbnails 2502 }; 2503 2504 imgExport.drawState(graph.getView().getState(graph.model.root), asynCanvas); 2505 2506 asynCanvas.finish(mxUtils.bind(this, function() 2507 { 2508 try 2509 { 2510 imgExport.drawState(graph.getView().getState(graph.model.root), htmlCanvas); 2511 2512 // Removes temporary graph from DOM 2513 if (graph != this.editor.graph && graph.container.parentNode != null) 2514 { 2515 graph.container.parentNode.removeChild(graph.container); 2516 } 2517 } 2518 catch (e) 2519 { 2520 canvas = null; 2521 } 2522 2523 success(canvas); 2524 })); 2525 2526 result = true; 2527 } 2528 } 2529 catch (e) 2530 { 2531 result = false; 2532 2533 // Removes temporary graph from DOM 2534 if (graph != null && graph != this.editor.graph && graph.container.parentNode != null) 2535 { 2536 graph.container.parentNode.removeChild(graph.container); 2537 } 2538 } 2539 2540 if (!result) 2541 { 2542 window.clearTimeout(timeoutThread); 2543 } 2544 2545 return result; 2546}; 2547 2548/** 2549 * Translates this point by the given vector. 2550 * 2551 * @param {number} dx X-coordinate of the translation. 2552 * @param {number} dy Y-coordinate of the translation. 2553 */ 2554App.prototype.createBackground = function() 2555{ 2556 var bg = this.createDiv('background'); 2557 bg.style.position = 'absolute'; 2558 bg.style.background = 'white'; 2559 bg.style.left = '0px'; 2560 bg.style.top = '0px'; 2561 bg.style.bottom = '0px'; 2562 bg.style.right = '0px'; 2563 2564 mxUtils.setOpacity(bg, 100); 2565 2566 return bg; 2567}; 2568 2569/** 2570 * Translates this point by the given vector. 2571 * 2572 * @param {number} dx X-coordinate of the translation. 2573 * @param {number} dy Y-coordinate of the translation. 2574 */ 2575(function() 2576{ 2577 var editorUiSetMode = EditorUi.prototype.setMode; 2578 2579 App.prototype.setMode = function(mode, remember) 2580 { 2581 editorUiSetMode.apply(this, arguments); 2582 2583 // Note: UseLocalStorage affects the file dialogs 2584 // and should not be modified if mode is undefined 2585 if (this.mode != null) 2586 { 2587 Editor.useLocalStorage = this.mode == App.MODE_BROWSER; 2588 } 2589 2590 if (this.appIcon != null) 2591 { 2592 var file = this.getCurrentFile(); 2593 mode = (file != null) ? file.getMode() : mode; 2594 2595 if (mode == App.MODE_GOOGLE) 2596 { 2597 this.appIcon.setAttribute('title', mxResources.get('openIt', [mxResources.get('googleDrive')])); 2598 this.appIcon.style.cursor = 'pointer'; 2599 } 2600 else if (mode == App.MODE_DROPBOX) 2601 { 2602 this.appIcon.setAttribute('title', mxResources.get('openIt', [mxResources.get('dropbox')])); 2603 this.appIcon.style.cursor = 'pointer'; 2604 } 2605 else if (mode == App.MODE_ONEDRIVE) 2606 { 2607 this.appIcon.setAttribute('title', mxResources.get('openIt', [mxResources.get('oneDrive')])); 2608 this.appIcon.style.cursor = 'pointer'; 2609 } 2610 else 2611 { 2612 this.appIcon.removeAttribute('title'); 2613 this.appIcon.style.cursor = (mode == App.MODE_DEVICE) ? 'pointer' : 'default'; 2614 } 2615 } 2616 2617 if (remember) 2618 { 2619 try 2620 { 2621 if (isLocalStorage) 2622 { 2623 localStorage.setItem('.mode', mode); 2624 } 2625 else if (typeof(Storage) != 'undefined') 2626 { 2627 var expiry = new Date(); 2628 expiry.setYear(expiry.getFullYear() + 1); 2629 document.cookie = 'MODE=' + mode + '; expires=' + expiry.toUTCString(); 2630 } 2631 } 2632 catch (e) 2633 { 2634 // ignore possible access denied 2635 } 2636 } 2637 }; 2638})(); 2639 2640/** 2641 * Function: authorize 2642 * 2643 * Authorizes the client, gets the userId and calls <open>. 2644 */ 2645App.prototype.appIconClicked = function(evt) 2646{ 2647 if (mxEvent.isAltDown(evt)) 2648 { 2649 this.showSplash(true); 2650 } 2651 else 2652 { 2653 var file = this.getCurrentFile(); 2654 var mode = (file != null) ? file.getMode() : null; 2655 2656 if (mode == App.MODE_GOOGLE) 2657 { 2658 if (file != null && file.desc != null && file.desc.parents != null && 2659 file.desc.parents.length > 0 && !mxEvent.isShiftDown(evt)) 2660 { 2661 // Opens containing folder 2662 this.openLink('https://drive.google.com/drive/folders/' + file.desc.parents[0].id); 2663 } 2664 else if (file != null && file.getId() != null) 2665 { 2666 this.openLink('https://drive.google.com/open?id=' + file.getId()); 2667 } 2668 else 2669 { 2670 this.openLink('https://drive.google.com/?authuser=0'); 2671 } 2672 } 2673 else if (mode == App.MODE_ONEDRIVE) 2674 { 2675 if (file != null && file.meta != null && file.meta.webUrl != null) 2676 { 2677 var url = file.meta.webUrl; 2678 var name = encodeURIComponent(file.meta.name); 2679 2680 if (url.substring(url.length - name.length, url.length) == name) 2681 { 2682 url = url.substring(0, url.length - name.length); 2683 } 2684 2685 this.openLink(url); 2686 } 2687 else 2688 { 2689 this.openLink('https://onedrive.live.com/'); 2690 } 2691 } 2692 else if (mode == App.MODE_DROPBOX) 2693 { 2694 if (file != null && file.stat != null && file.stat.path_display != null) 2695 { 2696 var url = 'https://www.dropbox.com/home/Apps/drawio' + file.stat.path_display; 2697 2698 if (!mxEvent.isShiftDown(evt)) 2699 { 2700 url = url.substring(0, url.length - file.stat.name.length); 2701 } 2702 2703 this.openLink(url); 2704 } 2705 else 2706 { 2707 this.openLink('https://www.dropbox.com/'); 2708 } 2709 } 2710 else if (mode == App.MODE_TRELLO) 2711 { 2712 this.openLink('https://trello.com/'); 2713 } 2714 else if (mode == App.MODE_NOTION) 2715 { 2716 this.openLink('https://www.notion.so/'); 2717 } 2718 else if (mode == App.MODE_GITHUB) 2719 { 2720 if (file != null && file.constructor == GitHubFile) 2721 { 2722 this.openLink(file.meta.html_url); 2723 } 2724 else 2725 { 2726 this.openLink('https://github.com/'); 2727 } 2728 } 2729 else if (mode == App.MODE_GITLAB) 2730 { 2731 if (file != null && file.constructor == GitLabFile) 2732 { 2733 this.openLink(file.meta.html_url); 2734 } 2735 else 2736 { 2737 this.openLink(DRAWIO_GITLAB_URL); 2738 } 2739 } 2740 else if (mode == App.MODE_DEVICE) 2741 { 2742 this.openLink('https://get.draw.io/'); 2743 } 2744 } 2745 2746 mxEvent.consume(evt); 2747}; 2748 2749/** 2750 * Function: authorize 2751 * 2752 * Authorizes the client, gets the userId and calls <open>. 2753 */ 2754App.prototype.clearMode = function() 2755{ 2756 if (isLocalStorage) 2757 { 2758 localStorage.removeItem('.mode'); 2759 } 2760 else if (typeof(Storage) != 'undefined') 2761 { 2762 var expiry = new Date(); 2763 expiry.setYear(expiry.getFullYear() - 1); 2764 document.cookie = 'MODE=; expires=' + expiry.toUTCString(); 2765 } 2766}; 2767 2768/** 2769 * Translates this point by the given vector. 2770 * 2771 * @param {number} dx X-coordinate of the translation. 2772 * @param {number} dy Y-coordinate of the translation. 2773 */ 2774App.prototype.getDiagramId = function() 2775{ 2776 var id = window.location.hash; 2777 2778 // Strips the hash sign 2779 if (id != null && id.length > 0) 2780 { 2781 id = id.substring(1); 2782 } 2783 2784 // Workaround for Trello client appending data after hash 2785 if (id != null && id.length > 1 && id.charAt(0) == 'T') 2786 { 2787 var idx = id.indexOf('#'); 2788 2789 if (idx > 0) 2790 { 2791 id = id.substring(0, idx); 2792 } 2793 } 2794 2795 return id; 2796}; 2797 2798/** 2799 * Opens any file specified in the URL parameters. 2800 */ 2801App.prototype.open = function() 2802{ 2803 // Cross-domain window access is not allowed in FF, so if we 2804 // were opened from another domain then this will fail. 2805 try 2806 { 2807 // If the create URL param is used in embed mode then 2808 // we try to open the XML from window.opener[value]. 2809 // Use this for embedding via tab to bypass the timing 2810 // issues when passing messages without onload event. 2811 if (window.opener != null) 2812 { 2813 var value = urlParams['create']; 2814 2815 if (value != null) 2816 { 2817 value = decodeURIComponent(value); 2818 } 2819 2820 if (value != null && value.length > 0 && value.substring(0, 7) != 'http://' && 2821 value.substring(0, 8) != 'https://') 2822 { 2823 var doc = mxUtils.parseXml(window.opener[value]); 2824 this.editor.setGraphXml(doc.documentElement); 2825 } 2826 else if (window.opener.openFile != null) 2827 { 2828 window.opener.openFile.setConsumer(mxUtils.bind(this, function(xml, filename, temp) 2829 { 2830 this.spinner.stop(); 2831 2832 if (filename == null) 2833 { 2834 var title = urlParams['title']; 2835 temp = true; 2836 2837 if (title != null) 2838 { 2839 filename = decodeURIComponent(title); 2840 } 2841 else 2842 { 2843 filename = this.defaultFilename; 2844 } 2845 } 2846 2847 // Replaces PNG with XML extension 2848 var dot = (!this.useCanvasForExport) ? filename.substring(filename.length - 4) == '.png' : -1; 2849 2850 if (dot > 0) 2851 { 2852 filename = filename.substring(0, filename.length - 4) + '.drawio'; 2853 } 2854 2855 this.fileLoaded((mxClient.IS_IOS) ? 2856 new StorageFile(this, xml, filename) : 2857 new LocalFile(this, xml, filename, temp)); 2858 })); 2859 } 2860 } 2861 } 2862 catch(e) 2863 { 2864 // ignore 2865 } 2866}; 2867 2868App.prototype.loadGapi = function(then) 2869{ 2870 if (typeof gapi !== 'undefined') 2871 { 2872 gapi.load(((urlParams['picker'] != '0') ? 'picker,': '') + App.GOOGLE_APIS, then); 2873 } 2874}; 2875 2876/** 2877 * Main function. Program starts here. 2878 * 2879 * @param {number} dx X-coordinate of the translation. 2880 * @param {number} dy Y-coordinate of the translation. 2881 */ 2882App.prototype.load = function() 2883{ 2884 // Checks if we're running in embedded mode 2885 if (urlParams['embed'] != '1') 2886 { 2887 if (this.spinner.spin(document.body, mxResources.get('starting'))) 2888 { 2889 try 2890 { 2891 this.stateArg = (urlParams['state'] != null && this.drive != null) ? JSON.parse(decodeURIComponent(urlParams['state'])) : null; 2892 } 2893 catch (e) 2894 { 2895 // ignores invalid state args 2896 } 2897 2898 this.editor.graph.setEnabled(this.getCurrentFile() != null); 2899 2900 // Passes the userId from the state parameter to the client 2901 if ((window.location.hash == null || window.location.hash.length == 0) && 2902 this.drive != null && this.stateArg != null && this.stateArg.userId != null) 2903 { 2904 this.drive.setUserId(this.stateArg.userId); 2905 } 2906 2907 // Legacy support for fileId parameter which is moved to the hash tag 2908 if (urlParams['fileId'] != null) 2909 { 2910 window.location.hash = 'G' + urlParams['fileId']; 2911 window.location.search = this.getSearch(['fileId']); 2912 } 2913 else 2914 { 2915 // Asynchronous or disabled loading of client 2916 if (this.drive == null) 2917 { 2918 if (this.mode == App.MODE_GOOGLE) 2919 { 2920 this.mode = null; 2921 } 2922 2923 this.start(); 2924 } 2925 else 2926 { 2927 this.loadGapi(mxUtils.bind(this, function() 2928 { 2929 this.start(); 2930 })); 2931 } 2932 } 2933 } 2934 } 2935 else 2936 { 2937 this.restoreLibraries(); 2938 2939 if (urlParams['gapi'] == '1') 2940 { 2941 this.loadGapi(function() {}); 2942 } 2943 } 2944}; 2945 2946/** 2947 * Adds the listener for automatically saving the diagram for local changes. 2948 */ 2949App.prototype.showRefreshDialog = function(title, message) 2950{ 2951 if (!this.showingRefreshDialog) 2952 { 2953 this.showingRefreshDialog = true; 2954 2955 this.showError(title || mxResources.get('externalChanges'), 2956 message || mxResources.get('redirectToNewApp'), 2957 mxResources.get('refresh'), mxUtils.bind(this, function() 2958 { 2959 var file = this.getCurrentFile(); 2960 2961 if (file != null) 2962 { 2963 file.setModified(false); 2964 } 2965 2966 this.spinner.spin(document.body, mxResources.get('connecting')); 2967 this.editor.graph.setEnabled(false); 2968 window.location.reload(); 2969 }), null, null, null, null, null, 340, 180); 2970 2971 // Adds important notice to dialog 2972 if (this.dialog != null && this.dialog.container != null) 2973 { 2974 var alert = this.createRealtimeNotice(); 2975 alert.style.left = '0'; 2976 alert.style.right = '0'; 2977 alert.style.borderRadius = '0'; 2978 alert.style.borderLeftStyle = 'none'; 2979 alert.style.borderRightStyle = 'none'; 2980 alert.style.marginBottom = '26px'; 2981 alert.style.padding = '8px 0 8px 0'; 2982 2983 this.dialog.container.appendChild(alert); 2984 } 2985 } 2986}; 2987 2988/** 2989 * Called in start after the spinner stops. 2990 */ 2991App.prototype.showAlert = function(message) 2992{ 2993 if (message != null && message.length > 0) 2994 { 2995 var div = document.createElement('div'); 2996 div.className = 'geAlert'; 2997 div.style.zIndex = 2e9; 2998 div.style.left = '50%'; 2999 div.style.top = '-100%'; 3000 //Limit width to 80% max with word wrapping 3001 div.style.maxWidth = '80%'; 3002 div.style.width = 'max-content'; 3003 div.style.whiteSpace = 'pre-wrap'; 3004 mxUtils.setPrefixedStyle(div.style, 'transform', 'translate(-50%,0%)'); 3005 mxUtils.setPrefixedStyle(div.style, 'transition', 'all 1s ease'); 3006 3007 div.innerHTML = message; 3008 3009 var close = document.createElement('a'); 3010 close.className = 'geAlertLink'; 3011 close.style.textAlign = 'right'; 3012 close.style.marginTop = '20px'; 3013 close.style.display = 'block'; 3014 close.setAttribute('title', mxResources.get('close')); 3015 close.innerHTML = mxResources.get('close'); 3016 div.appendChild(close); 3017 3018 mxEvent.addListener(close, 'click', function(evt) 3019 { 3020 if (div.parentNode != null) 3021 { 3022 div.parentNode.removeChild(div); 3023 mxEvent.consume(evt); 3024 } 3025 }); 3026 3027 document.body.appendChild(div); 3028 3029 // Delayed to get smoother animation after DOM rendering 3030 window.setTimeout(function() 3031 { 3032 div.style.top = '30px'; 3033 }, 10); 3034 3035 // Fades out the alert after 15 secs 3036 window.setTimeout(function() 3037 { 3038 mxUtils.setPrefixedStyle(div.style, 'transition', 'all 2s ease'); 3039 div.style.opacity = '0'; 3040 3041 window.setTimeout(function() 3042 { 3043 if (div.parentNode != null) 3044 { 3045 div.parentNode.removeChild(div); 3046 } 3047 }, 2000); 3048 }, 15000); 3049 } 3050}; 3051 3052/** 3053 * Translates this point by the given vector. 3054 * 3055 * @param {number} dx X-coordinate of the translation. 3056 * @param {number} dy Y-coordinate of the translation. 3057 */ 3058App.prototype.start = function() 3059{ 3060 if (this.bg != null && this.bg.parentNode != null) 3061 { 3062 this.bg.parentNode.removeChild(this.bg); 3063 } 3064 3065 this.restoreLibraries(); 3066 this.spinner.stop(); 3067 3068 try 3069 { 3070 // Handles all errors 3071 var ui = this; 3072 3073 window.onerror = function(message, url, linenumber, colno, err) 3074 { 3075 // Ignores Grammarly error [1344] 3076 if (message != 'ResizeObserver loop limit exceeded') 3077 { 3078 EditorUi.logError('Uncaught: ' + ((message != null) ? message : ''), 3079 url, linenumber, colno, err, null, true); 3080 ui.handleError({message: message}, mxResources.get('unknownError'), 3081 null, null, null, null, true); 3082 } 3083 }; 3084 3085 // Listens to changes of the hash if not in embed or client mode 3086 if (urlParams['client'] != '1' && urlParams['embed'] != '1') 3087 { 3088 // Installs listener to claim current draft if there is one 3089 try 3090 { 3091 if (isLocalStorage) 3092 { 3093 window.addEventListener('storage', mxUtils.bind(this, function(evt) 3094 { 3095 var file = this.getCurrentFile(); 3096 EditorUi.debug('storage event', evt, file); 3097 3098 if (file != null && evt.key == '.draft-alive-check' && evt.newValue != null && file.draftId != null) 3099 { 3100 this.draftAliveCheck = evt.newValue; 3101 file.saveDraft(); 3102 } 3103 })); 3104 } 3105 3106 if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && !this.isOfflineApp() && 3107 urlParams['open'] == null && /www\.draw\.io$/.test(window.location.hostname) && 3108 (!this.editor.chromeless || this.editor.editable)) 3109 { 3110 this.showNameChangeBanner(); 3111 } 3112 } 3113 catch (e) 3114 { 3115 // ignore 3116 } 3117 3118 mxEvent.addListener(window, 'hashchange', mxUtils.bind(this, function(evt) 3119 { 3120 try 3121 { 3122 this.hideDialog(); 3123 var id = this.getDiagramId(); 3124 var file = this.getCurrentFile(); 3125 3126 if (file == null || file.getHash() != id) 3127 { 3128 this.loadFile(id, true); 3129 } 3130 } 3131 catch (e) 3132 { 3133 // Workaround for possible scrollWidth of null in Dialog ctor 3134 if (document.body != null) 3135 { 3136 this.handleError(e, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function() 3137 { 3138 var file = this.getCurrentFile(); 3139 window.location.hash = (file != null) ? file.getHash() : ''; 3140 })); 3141 } 3142 } 3143 })); 3144 } 3145 3146 // Descriptor for CSV import 3147 if ((window.location.hash == null || window.location.hash.length <= 1) && urlParams['desc'] != null) 3148 { 3149 try 3150 { 3151 this.loadDescriptor(JSON.parse(Graph.decompress(urlParams['desc'])), 3152 null, mxUtils.bind(this, function(e) 3153 { 3154 this.handleError(e, mxResources.get('errorLoadingFile')); 3155 })); 3156 } 3157 catch (e) 3158 { 3159 this.handleError(e, mxResources.get('errorLoadingFile')); 3160 } 3161 } 3162 // Redirects old url URL parameter to new #U format 3163 else if ((window.location.hash == null || window.location.hash.length <= 1) && urlParams['url'] != null) 3164 { 3165 this.loadFile('U' + urlParams['url'], true); 3166 } 3167 else if (this.getCurrentFile() == null) 3168 { 3169 var done = mxUtils.bind(this, function() 3170 { 3171 // Starts in client mode and waits for data 3172 if (urlParams['client'] == '1' && (window.location.hash == null || 3173 window.location.hash.length == 0 || window.location.hash.substring(0, 2) == '#P')) 3174 { 3175 var doLoadFile = mxUtils.bind(this, function(xml) 3176 { 3177 // Extracts graph model from PNG 3178 if (xml.substring(0, 22) == 'data:image/png;base64,') 3179 { 3180 xml = this.extractGraphModelFromPng(xml); 3181 } 3182 3183 var title = urlParams['title']; 3184 3185 if (title != null) 3186 { 3187 title = decodeURIComponent(title); 3188 } 3189 else 3190 { 3191 title = this.defaultFilename; 3192 } 3193 3194 var file = new LocalFile(this, xml, title, true); 3195 3196 if (window.location.hash != null && window.location.hash.substring(0, 2) == '#P') 3197 { 3198 file.getHash = function() 3199 { 3200 return window.location.hash.substring(1); 3201 }; 3202 } 3203 3204 this.fileLoaded(file); 3205 this.getCurrentFile().setModified(!this.editor.chromeless); 3206 }); 3207 3208 var parent = window.opener || window.parent; 3209 3210 if (parent != window) 3211 { 3212 var value = urlParams['create']; 3213 3214 if (value != null) 3215 { 3216 doLoadFile(parent[decodeURIComponent(value)]); 3217 } 3218 else 3219 { 3220 value = urlParams['data']; 3221 3222 if (value != null) 3223 { 3224 doLoadFile(decodeURIComponent(value)); 3225 } 3226 else 3227 { 3228 this.installMessageHandler(mxUtils.bind(this, function(xml, evt) 3229 { 3230 // Ignores messages from other windows 3231 if (evt.source == parent) 3232 { 3233 doLoadFile(xml); 3234 } 3235 })); 3236 } 3237 } 3238 } 3239 } 3240 // Checks if no earlier loading errors are showing 3241 else if (this.dialog == null) 3242 { 3243 if (urlParams['demo'] == '1') 3244 { 3245 var prev = Editor.useLocalStorage; 3246 this.createFile(this.defaultFilename, null, null, null, null, null, null, true); 3247 Editor.useLocalStorage = prev; 3248 } 3249 else 3250 { 3251 var waiting = false; 3252 3253 // Checks if we're waiting for some asynchronous file to be loaded 3254 // Cross-domain window access is not allowed in FF, so if we 3255 // were opened from another domain then this will fail. 3256 try 3257 { 3258 waiting = window.opener != null && window.opener.openFile != null; 3259 } 3260 catch(e) 3261 { 3262 // ignore 3263 } 3264 3265 if (waiting) 3266 { 3267 // Spinner is stopped in App.open 3268 this.spinner.spin(document.body, mxResources.get('loading')) 3269 } 3270 else 3271 { 3272 var id = this.getDiagramId(); 3273 3274 3275 if (EditorUi.enableDrafts && (urlParams['mode'] == null || EditorUi.isElectronApp) && 3276 this.getServiceName() == 'draw.io' && (id == null || id.length == 0) && 3277 !this.editor.isChromelessView()) 3278 { 3279 this.checkDrafts(); 3280 } 3281 else if (id != null && id.length > 0) 3282 { 3283 this.loadFile(id, null, null, mxUtils.bind(this, function() 3284 { 3285 var temp = decodeURIComponent(urlParams['viewbox'] || ''); 3286 3287 if (temp != '') 3288 { 3289 try 3290 { 3291 var bounds = JSON.parse(temp); 3292 this.editor.graph.fitWindow(bounds, bounds.border); 3293 } 3294 catch (e) 3295 { 3296 // Ignore invalid viewport 3297 console.error(e); 3298 } 3299 } 3300 })); 3301 } 3302 else if (urlParams['splash'] != '0') 3303 { 3304 this.loadFile(); 3305 } 3306 else 3307 { 3308 this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true); 3309 } 3310 } 3311 } 3312 } 3313 }); 3314 3315 var value = decodeURIComponent(urlParams['create'] || ''); 3316 3317 if ((window.location.hash == null || window.location.hash.length <= 1) && 3318 value != null && value.length > 0 && this.spinner.spin(document.body, mxResources.get('loading'))) 3319 { 3320 var reconnect = mxUtils.bind(this, function() 3321 { 3322 // Removes URL parameter and reloads the page 3323 if (this.spinner.spin(document.body, mxResources.get('reconnecting'))) 3324 { 3325 window.location.search = this.getSearch(['create', 'title']); 3326 }; 3327 }); 3328 3329 var showCreateDialog = mxUtils.bind(this, function(xml) 3330 { 3331 this.spinner.stop(); 3332 3333 // Resets mode for dialog - local file is only for preview 3334 if (urlParams['splash'] != '0') 3335 { 3336 this.fileLoaded(new LocalFile(this, xml, null)); 3337 3338 this.editor.graph.setEnabled(false); 3339 this.mode = urlParams['mode']; 3340 var title = urlParams['title']; 3341 3342 if (title != null) 3343 { 3344 title = decodeURIComponent(title); 3345 } 3346 else 3347 { 3348 title = this.defaultFilename; 3349 } 3350 3351 var serviceCount = this.getServiceCount(true); 3352 3353 if (isLocalStorage) 3354 { 3355 serviceCount++; 3356 } 3357 3358 var rowLimit = (serviceCount <= 4) ? 2 : (serviceCount > 6 ? 4 : 3); 3359 3360 var dlg = new CreateDialog(this, title, mxUtils.bind(this, function(filename, mode) 3361 { 3362 if (mode == null) 3363 { 3364 this.hideDialog(); 3365 var prev = Editor.useLocalStorage; 3366 this.createFile((filename.length > 0) ? filename : this.defaultFilename, 3367 this.getFileData(), null, null, null, true, null, true); 3368 Editor.useLocalStorage = prev; 3369 } 3370 else 3371 { 3372 this.pickFolder(mode, mxUtils.bind(this, function(folderId) 3373 { 3374 this.createFile(filename, this.getFileData(true), 3375 null, mode, null, true, folderId); 3376 })); 3377 } 3378 }), null, null, null, null, urlParams['browser'] == '1', 3379 null, null, true, rowLimit, null, null, null, 3380 this.editor.fileExtensions); 3381 this.showDialog(dlg.container, 400, (serviceCount > rowLimit) ? 390 : 270, 3382 true, false, mxUtils.bind(this, function(cancel) 3383 { 3384 if (cancel && this.getCurrentFile() == null) 3385 { 3386 this.showSplash(); 3387 } 3388 })); 3389 dlg.init(); 3390 } 3391 }); 3392 3393 value = decodeURIComponent(value); 3394 3395 if (value.substring(0, 7) != 'http://' && value.substring(0, 8) != 'https://') 3396 { 3397 // Cross-domain window access is not allowed in FF, so if we 3398 // were opened from another domain then this will fail. 3399 try 3400 { 3401 if (window.opener != null && window.opener[value] != null) 3402 { 3403 showCreateDialog(window.opener[value]); 3404 } 3405 else 3406 { 3407 this.handleError(null, mxResources.get('errorLoadingFile')); 3408 } 3409 } 3410 catch (e) 3411 { 3412 this.handleError(e, mxResources.get('errorLoadingFile')); 3413 } 3414 } 3415 else 3416 { 3417 this.loadTemplate(value, function(text) 3418 { 3419 showCreateDialog(text); 3420 }, mxUtils.bind(this, function() 3421 { 3422 this.handleError(null, mxResources.get('errorLoadingFile'), reconnect); 3423 })); 3424 } 3425 } 3426 else 3427 { 3428 // Passes the fileId from the state parameter to the hash tag and reloads 3429 // the page without the state parameter 3430 if ((window.location.hash == null || window.location.hash.length <= 1) && 3431 urlParams['state'] != null && this.stateArg != null && this.stateArg.action == 'open') 3432 { 3433 if (this.stateArg.ids != null) 3434 { 3435 if (window.history && window.history.replaceState) 3436 { 3437 // Removes state URL parameter without reloading the page 3438 window.history.replaceState(null, null, window.location.pathname + 3439 this.getSearch(['state'])); 3440 } 3441 3442 window.location.hash = 'G' + this.stateArg.ids[0]; 3443 } 3444 } 3445 else if ((window.location.hash == null || window.location.hash.length <= 1) && 3446 this.drive != null && this.stateArg != null && this.stateArg.action == 'create') 3447 { 3448 if (window.history && window.history.replaceState) 3449 { 3450 // Removes state URL parameter without reloading the page 3451 window.history.replaceState(null, null, window.location.pathname + 3452 this.getSearch(['state'])); 3453 } 3454 3455 this.setMode(App.MODE_GOOGLE); 3456 3457 if (urlParams['splash'] == '0') 3458 { 3459 this.createFile((urlParams['title'] != null) ? 3460 decodeURIComponent(urlParams['title']) : 3461 this.defaultFilename); 3462 } 3463 else 3464 { 3465 this.actions.get('new').funct(); 3466 } 3467 } 3468 else 3469 { 3470 // Removes open URL parameter. Hash is also updated in Init to load client. 3471 if (urlParams['open'] != null && window.history && window.history.replaceState) 3472 { 3473 window.history.replaceState(null, null, window.location.pathname + 3474 this.getSearch(['open'])); 3475 window.location.hash = urlParams['open']; 3476 } 3477 3478 done(); 3479 } 3480 } 3481 } 3482 } 3483 catch (e) 3484 { 3485 this.handleError(e); 3486 } 3487}; 3488 3489/** 3490 * Checks for orphaned drafts. 3491 */ 3492App.prototype.loadDraft = function(xml, success) 3493{ 3494 this.createFile(this.defaultFilename, xml, null, null, mxUtils.bind(this, function() 3495 { 3496 window.setTimeout(mxUtils.bind(this, function() 3497 { 3498 var file = this.getCurrentFile(); 3499 3500 if (file != null) 3501 { 3502 file.fileChanged(); 3503 3504 if (success != null) 3505 { 3506 success(); 3507 } 3508 } 3509 }), 0); 3510 }), null, null, true); 3511}; 3512 3513/** 3514 * Checks for orphaned drafts. 3515 */ 3516App.prototype.checkDrafts = function() 3517{ 3518 try 3519 { 3520 // Triggers storage event for other windows to mark active drafts 3521 var guid = Editor.guid(); 3522 localStorage.setItem('.draft-alive-check', guid); 3523 3524 window.setTimeout(mxUtils.bind(this, function() 3525 { 3526 localStorage.removeItem('.draft-alive-check'); 3527 3528 this.getDatabaseItems(mxUtils.bind(this, function(items) 3529 { 3530 // Collects orphaned drafts 3531 var drafts = []; 3532 3533 for (var i = 0; i < items.length; i++) 3534 { 3535 try 3536 { 3537 var key = items[i].key; 3538 3539 if (key != null && key.substring(0, 7) == '.draft_') 3540 { 3541 var obj = JSON.parse(items[i].data); 3542 3543 if (obj != null && obj.type == 'draft' && obj.aliveCheck != guid) 3544 { 3545 obj.key = key; 3546 drafts.push(obj); 3547 } 3548 } 3549 } 3550 catch (e) 3551 { 3552 // ignore 3553 } 3554 } 3555 3556 if (drafts.length == 1) 3557 { 3558 this.loadDraft(drafts[0].data, mxUtils.bind(this, function() 3559 { 3560 this.removeDatabaseItem(drafts[0].key); 3561 })); 3562 } 3563 else if (drafts.length > 1) 3564 { 3565 var ts = new Date(drafts[0].modified); 3566 3567 var dlg = new DraftDialog(this, (drafts.length > 1) ? mxResources.get('selectDraft') : 3568 mxResources.get('draftFound', [ts.toLocaleDateString() + ' ' + ts.toLocaleTimeString()]), 3569 (drafts.length > 1) ? null : drafts[0].data, mxUtils.bind(this, function(index) 3570 { 3571 this.hideDialog(); 3572 index = (index != '') ? index : 0; 3573 3574 this.loadDraft(drafts[index].data, mxUtils.bind(this, function() 3575 { 3576 this.removeDatabaseItem(drafts[index].key); 3577 })); 3578 }), mxUtils.bind(this, function(index, success) 3579 { 3580 index = (index != '') ? index : 0; 3581 3582 // Discard draft 3583 this.confirm(mxResources.get('areYouSure'), null, mxUtils.bind(this, function() 3584 { 3585 this.removeDatabaseItem(drafts[index].key); 3586 3587 if (success != null) 3588 { 3589 success(); 3590 } 3591 }), mxResources.get('no'), mxResources.get('yes')); 3592 }), null, null, null, (drafts.length > 1) ? drafts : null); 3593 this.showDialog(dlg.container, 640, 480, true, false, mxUtils.bind(this, function(cancel) 3594 { 3595 if (urlParams['splash'] != '0') 3596 { 3597 this.loadFile(); 3598 } 3599 else 3600 { 3601 this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true); 3602 } 3603 })); 3604 dlg.init(); 3605 } 3606 else if (urlParams['splash'] != '0') 3607 { 3608 this.loadFile(); 3609 } 3610 else 3611 { 3612 this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true); 3613 } 3614 }), mxUtils.bind(this, function() 3615 { 3616 if (urlParams['splash'] != '0') 3617 { 3618 this.loadFile(); 3619 } 3620 else 3621 { 3622 this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true); 3623 } 3624 })); 3625 }), 0); 3626 } 3627 catch (e) 3628 { 3629 // ignore 3630 } 3631}; 3632 3633/** 3634 * Translates this point by the given vector. 3635 * 3636 * @param {number} dx X-coordinate of the translation. 3637 * @param {number} dy Y-coordinate of the translation. 3638 */ 3639App.prototype.showSplash = function(force) 3640{ 3641 //Splash dialog shouldn't be shownn when running without a file menu 3642 if (urlParams['noFileMenu'] == '1') 3643 { 3644 return; 3645 } 3646 3647 var serviceCount = this.getServiceCount(true); 3648 3649 var showSecondDialog = mxUtils.bind(this, function() 3650 { 3651 var dlg = new SplashDialog(this); 3652 3653 this.showDialog(dlg.container, 340, (mxClient.IS_CHROMEAPP || EditorUi.isElectronApp) ? 200 : 230, true, true, 3654 mxUtils.bind(this, function(cancel) 3655 { 3656 // Creates a blank diagram if the dialog is closed 3657 if (cancel && !mxClient.IS_CHROMEAPP) 3658 { 3659 var prev = Editor.useLocalStorage; 3660 this.createFile(this.defaultFilename + (EditorUi.isElectronApp? '.drawio' : ''), null, null, null, null, null, null, 3661 urlParams['local'] != '1'); 3662 Editor.useLocalStorage = prev; 3663 } 3664 }), true); 3665 }); 3666 3667 if (this.editor.isChromelessView()) 3668 { 3669 this.handleError({message: mxResources.get('noFileSelected')}, 3670 mxResources.get('errorLoadingFile'), mxUtils.bind(this, function() 3671 { 3672 this.showSplash(); 3673 })); 3674 } 3675 else if (!mxClient.IS_CHROMEAPP && (this.mode == null || force)) 3676 { 3677 var rowLimit = (serviceCount == 4) ? 2 : 3; 3678 3679 var dlg = new StorageDialog(this, mxUtils.bind(this, function() 3680 { 3681 this.hideDialog(); 3682 showSecondDialog(); 3683 }), rowLimit); 3684 3685 this.showDialog(dlg.container, (rowLimit < 3) ? 200 : 300, 3686 ((serviceCount > 3) ? 320 : 210), true, false); 3687 } 3688 else if (urlParams['create'] == null) 3689 { 3690 showSecondDialog(); 3691 } 3692}; 3693 3694/** 3695 * Translates this point by the given vector. 3696 * 3697 * @param {number} dx X-coordinate of the translation. 3698 * @param {number} dy Y-coordinate of the translation. 3699 */ 3700App.prototype.addLanguageMenu = function(elt, addLabel) 3701{ 3702 var img = null; 3703 var langMenu = this.menus.get('language'); 3704 3705 if (langMenu != null) 3706 { 3707 img = document.createElement('div'); 3708 img.setAttribute('title', mxResources.get('language')); 3709 img.className = 'geIcon geSprite geSprite-globe'; 3710 img.style.position = 'absolute'; 3711 img.style.cursor = 'pointer'; 3712 img.style.bottom = '20px'; 3713 img.style.right = '20px'; 3714 3715 if (addLabel) 3716 { 3717 img.style.direction = 'rtl'; 3718 img.style.textAlign = 'right'; 3719 img.style.right = '24px'; 3720 3721 var label = document.createElement('span'); 3722 label.style.display = 'inline-block'; 3723 label.style.fontSize = '12px'; 3724 label.style.margin = '5px 24px 0 0'; 3725 label.style.color = 'gray'; 3726 label.style.userSelect = 'none'; 3727 3728 mxUtils.write(label, mxResources.get('language')); 3729 img.appendChild(label); 3730 } 3731 3732 mxEvent.addListener(img, 'click', mxUtils.bind(this, function(evt) 3733 { 3734 this.editor.graph.popupMenuHandler.hideMenu(); 3735 var menu = new mxPopupMenu(this.menus.get('language').funct); 3736 menu.div.className += ' geMenubarMenu'; 3737 menu.smartSeparators = true; 3738 menu.showDisabled = true; 3739 menu.autoExpand = true; 3740 3741 // Disables autoexpand and destroys menu when hidden 3742 menu.hideMenu = mxUtils.bind(this, function() 3743 { 3744 mxPopupMenu.prototype.hideMenu.apply(menu, arguments); 3745 menu.destroy(); 3746 }); 3747 3748 var offset = mxUtils.getOffset(img); 3749 menu.popup(offset.x, offset.y + img.offsetHeight, null, evt); 3750 3751 // Allows hiding by clicking on document 3752 this.setCurrentMenu(menu); 3753 })); 3754 3755 elt.appendChild(img); 3756 } 3757 3758 return img; 3759}; 3760 3761/** 3762 * Loads the given file handle as a local file. 3763 */ 3764App.prototype.loadFileSystemEntry = function(fileHandle, success, error) 3765{ 3766 error = (error != null) ? error : mxUtils.bind(this, function(e) 3767 { 3768 this.handleError(e); 3769 }); 3770 3771 try 3772 { 3773 fileHandle.getFile().then(mxUtils.bind(this, function(file) 3774 { 3775 var reader = new FileReader(); 3776 3777 reader.onload = mxUtils.bind(this, function(e) 3778 { 3779 try 3780 { 3781 if (success != null) 3782 { 3783 var data = e.target.result; 3784 3785 if (file.type == 'image/png') 3786 { 3787 data = this.extractGraphModelFromPng(data); 3788 } 3789 3790 success(new LocalFile(this, data, file.name, null, fileHandle, file)); 3791 } 3792 else 3793 { 3794 this.openFileHandle(e.target.result, file.name, file, false, fileHandle); 3795 } 3796 } 3797 catch(e) 3798 { 3799 error(e); 3800 } 3801 }); 3802 3803 reader.onerror = error; 3804 3805 if ((file.type.substring(0, 5) === 'image' || 3806 file.type === 'application/pdf') && 3807 file.type.substring(0, 9) !== 'image/svg') 3808 { 3809 reader.readAsDataURL(file); 3810 } 3811 else 3812 { 3813 reader.readAsText(file); 3814 } 3815 }), error); 3816 } 3817 catch (e) 3818 { 3819 error(e); 3820 } 3821}; 3822 3823/** 3824 * Loads the given file handle as a local file. 3825 */ 3826App.prototype.createFileSystemOptions = function(name) 3827{ 3828 var ext = []; 3829 var temp = null; 3830 3831 if (name != null) 3832 { 3833 var idx = name.lastIndexOf('.'); 3834 3835 if (idx > 0) 3836 { 3837 temp = name.substring(idx + 1); 3838 } 3839 } 3840 3841 for (var i = 0; i < this.editor.diagramFileTypes.length; i++) 3842 { 3843 var obj = {description: mxResources.get(this.editor.diagramFileTypes[i].description) + 3844 ((mxClient.IS_MAC) ? ' (.' + this.editor.diagramFileTypes[i].extension + ')' : ''), 3845 accept: {}}; 3846 obj.accept[this.editor.diagramFileTypes[i].mimeType] = ['.' + this.editor.diagramFileTypes[i].extension]; 3847 3848 if (this.editor.diagramFileTypes[i].extension == temp) 3849 { 3850 ext.splice(0, 0, obj); 3851 } 3852 else 3853 { 3854 if (this.editor.diagramFileTypes[i].extension == temp) 3855 { 3856 ext.splice(0, 0, obj); 3857 } 3858 else 3859 { 3860 ext.push(obj); 3861 } 3862 } 3863 } 3864 3865 // TODO: Specify default filename 3866 return {types: ext, fileName: name}; 3867}; 3868 3869/** 3870 * Loads the given file handle as a local file. 3871 */ 3872App.prototype.showSaveFilePicker = function(success, error, opts) 3873{ 3874 error = (error != null) ? error : mxUtils.bind(this, function(e) 3875 { 3876 if (e.name != 'AbortError') 3877 { 3878 this.handleError(e); 3879 } 3880 }); 3881 3882 opts = (opts != null) ? opts : this.createFileSystemOptions(); 3883 3884 window.showSaveFilePicker(opts).then(mxUtils.bind(this, function(fileHandle) 3885 { 3886 if (fileHandle != null) 3887 { 3888 fileHandle.getFile().then(mxUtils.bind(this, function(desc) 3889 { 3890 success(fileHandle, desc); 3891 }), error); 3892 } 3893 }), error); 3894}; 3895 3896/** 3897 * Translates this point by the given vector. 3898 * 3899 * @param {number} dx X-coordinate of the translation. 3900 * @param {number} dy Y-coordinate of the translation. 3901 */ 3902App.prototype.pickFile = function(mode) 3903{ 3904 try 3905 { 3906 mode = (mode != null) ? mode : this.mode; 3907 3908 if (mode == App.MODE_GOOGLE) 3909 { 3910 if (this.drive != null && typeof(google) != 'undefined' && typeof(google.picker) != 'undefined') 3911 { 3912 this.drive.pickFile(); 3913 } 3914 else 3915 { 3916 this.openLink('https://drive.google.com'); 3917 } 3918 } 3919 else 3920 { 3921 var peer = this.getPeerForMode(mode); 3922 3923 if (peer != null) 3924 { 3925 peer.pickFile(); 3926 } 3927 else if (mode == App.MODE_DEVICE && EditorUi.nativeFileSupport) 3928 { 3929 window.showOpenFilePicker().then(mxUtils.bind(this, function(fileHandles) 3930 { 3931 if (fileHandles != null && fileHandles.length > 0 && 3932 this.spinner.spin(document.body, mxResources.get('loading'))) 3933 { 3934 this.loadFileSystemEntry(fileHandles[0]); 3935 } 3936 }), mxUtils.bind(this, function(e) 3937 { 3938 if (e.name != 'AbortError') 3939 { 3940 this.handleError(e); 3941 } 3942 })); 3943 } 3944 else if (mode == App.MODE_DEVICE && Graph.fileSupport) 3945 { 3946 if (this.openFileInputElt == null) 3947 { 3948 var input = document.createElement('input'); 3949 input.setAttribute('type', 'file'); 3950 3951 mxEvent.addListener(input, 'change', mxUtils.bind(this, function() 3952 { 3953 if (input.files != null) 3954 { 3955 this.openFiles(input.files); 3956 3957 // Resets input to force change event for 3958 // same file (type reset required for IE) 3959 input.type = ''; 3960 input.type = 'file'; 3961 input.value = ''; 3962 } 3963 })); 3964 3965 input.style.display = 'none'; 3966 document.body.appendChild(input); 3967 this.openFileInputElt = input; 3968 } 3969 3970 this.openFileInputElt.click(); 3971 } 3972 else 3973 { 3974 this.hideDialog(); 3975 window.openNew = this.getCurrentFile() != null && !this.isDiagramEmpty(); 3976 window.baseUrl = this.getUrl(); 3977 window.openKey = 'open'; 3978 3979 window.listBrowserFiles = mxUtils.bind(this, function(success, error) 3980 { 3981 StorageFile.listFiles(this, 'F', success, error); 3982 }); 3983 3984 window.openBrowserFile = mxUtils.bind(this, function(title, success, error) 3985 { 3986 StorageFile.getFileContent(this, title, success, error); 3987 }); 3988 3989 window.deleteBrowserFile = mxUtils.bind(this, function(title, success, error) 3990 { 3991 StorageFile.deleteFile(this, title, success, error); 3992 }); 3993 3994 var prevValue = Editor.useLocalStorage; 3995 Editor.useLocalStorage = (mode == App.MODE_BROWSER); 3996 this.openFile(); 3997 3998 // Installs local handler for opened files in same window 3999 window.openFile.setConsumer(mxUtils.bind(this, function(xml, filename) 4000 { 4001 var doOpenFile = mxUtils.bind(this, function() 4002 { 4003 // Replaces PNG with XML extension 4004 var dot = !this.useCanvasForExport && filename.substring(filename.length - 4) == '.png'; 4005 4006 if (dot) 4007 { 4008 filename = filename.substring(0, filename.length - 4) + '.drawio'; 4009 } 4010 4011 this.fileLoaded((mode == App.MODE_BROWSER) ? 4012 new StorageFile(this, xml, filename) : 4013 new LocalFile(this, xml, filename)); 4014 }); 4015 4016 var currentFile = this.getCurrentFile(); 4017 4018 if (currentFile == null || !currentFile.isModified()) 4019 { 4020 doOpenFile(); 4021 } 4022 else 4023 { 4024 this.confirm(mxResources.get('allChangesLost'), null, doOpenFile, 4025 mxResources.get('cancel'), mxResources.get('discardChanges')); 4026 } 4027 })); 4028 4029 // Extends dialog close to show splash screen 4030 var dlg = this.dialog; 4031 var dlgClose = dlg.close; 4032 4033 this.dialog.close = mxUtils.bind(this, function(cancel) 4034 { 4035 Editor.useLocalStorage = prevValue; 4036 dlgClose.apply(dlg, arguments); 4037 4038 if (this.getCurrentFile() == null) 4039 { 4040 this.showSplash(); 4041 } 4042 }); 4043 } 4044 } 4045 } 4046 catch (e) 4047 { 4048 this.handleError(e); 4049 } 4050}; 4051 4052/** 4053 * Translates this point by the given vector. 4054 * 4055 * @param {number} dx X-coordinate of the translation. 4056 * @param {number} dy Y-coordinate of the translation. 4057 */ 4058App.prototype.pickLibrary = function(mode) 4059{ 4060 mode = (mode != null) ? mode : this.mode; 4061 4062 if (mode == App.MODE_GOOGLE || mode == App.MODE_DROPBOX || mode == App.MODE_ONEDRIVE || 4063 mode == App.MODE_GITHUB || mode == App.MODE_GITLAB || mode == App.MODE_TRELLO || 4064 mode == App.MODE_NOTION) 4065 { 4066 var peer = (mode == App.MODE_GOOGLE) ? this.drive : 4067 ((mode == App.MODE_ONEDRIVE) ? this.oneDrive : 4068 ((mode == App.MODE_GITHUB) ? this.gitHub : 4069 ((mode == App.MODE_GITLAB) ? this.gitLab : 4070 ((mode == App.MODE_TRELLO) ? this.trello : 4071 ((mode == App.MODE_NOTION) ? this.notion : 4072 this.dropbox))))); 4073 4074 if (peer != null) 4075 { 4076 peer.pickLibrary(mxUtils.bind(this, function(id, optionalFile) 4077 { 4078 if (optionalFile != null) 4079 { 4080 try 4081 { 4082 this.loadLibrary(optionalFile); 4083 } 4084 catch (e) 4085 { 4086 this.handleError(e, mxResources.get('errorLoadingFile')); 4087 } 4088 } 4089 else 4090 { 4091 if (this.spinner.spin(document.body, mxResources.get('loading'))) 4092 { 4093 peer.getLibrary(id, mxUtils.bind(this, function(file) 4094 { 4095 this.spinner.stop(); 4096 4097 try 4098 { 4099 this.loadLibrary(file); 4100 } 4101 catch (e) 4102 { 4103 this.handleError(e, mxResources.get('errorLoadingFile')); 4104 } 4105 }), mxUtils.bind(this, function(resp) 4106 { 4107 this.handleError(resp, (resp != null) ? mxResources.get('errorLoadingFile') : null); 4108 })); 4109 } 4110 } 4111 })); 4112 } 4113 } 4114 else if (mode == App.MODE_DEVICE && Graph.fileSupport) 4115 { 4116 if (this.libFileInputElt == null) 4117 { 4118 var input = document.createElement('input'); 4119 input.setAttribute('type', 'file'); 4120 4121 mxEvent.addListener(input, 'change', mxUtils.bind(this, function() 4122 { 4123 if (input.files != null) 4124 { 4125 for (var i = 0; i < input.files.length; i++) 4126 { 4127 (mxUtils.bind(this, function(file) 4128 { 4129 var reader = new FileReader(); 4130 4131 reader.onload = mxUtils.bind(this, function(e) 4132 { 4133 try 4134 { 4135 this.loadLibrary(new LocalLibrary(this, e.target.result, file.name)); 4136 } 4137 catch (e) 4138 { 4139 this.handleError(e, mxResources.get('errorLoadingFile')); 4140 } 4141 }); 4142 4143 reader.readAsText(file); 4144 }))(input.files[i]); 4145 } 4146 4147 // Resets input to force change event for same file (type reset required for IE) 4148 input.type = ''; 4149 input.type = 'file'; 4150 input.value = ''; 4151 } 4152 })); 4153 4154 input.style.display = 'none'; 4155 document.body.appendChild(input); 4156 this.libFileInputElt = input; 4157 } 4158 4159 this.libFileInputElt.click(); 4160 } 4161 else 4162 { 4163 window.openNew = false; 4164 window.openKey = 'open'; 4165 4166 window.listBrowserFiles = mxUtils.bind(this, function(success, error) 4167 { 4168 StorageFile.listFiles(this, 'L', success, error); 4169 }); 4170 4171 window.openBrowserFile = mxUtils.bind(this, function(title, success, error) 4172 { 4173 StorageFile.getFileContent(this, title, success, error); 4174 }); 4175 4176 window.deleteBrowserFile = mxUtils.bind(this, function(title, success, error) 4177 { 4178 StorageFile.deleteFile(this, title, success, error); 4179 }); 4180 4181 var prevValue = Editor.useLocalStorage; 4182 Editor.useLocalStorage = mode == App.MODE_BROWSER; 4183 4184 // Closes dialog after open 4185 window.openFile = new OpenFile(mxUtils.bind(this, function(cancel) 4186 { 4187 this.hideDialog(cancel); 4188 })); 4189 4190 window.openFile.setConsumer(mxUtils.bind(this, function(xml, filename) 4191 { 4192 try 4193 { 4194 this.loadLibrary((mode == App.MODE_BROWSER) ? new StorageLibrary(this, xml, filename) : 4195 new LocalLibrary(this, xml, filename)); 4196 } 4197 catch (e) 4198 { 4199 this.handleError(e, mxResources.get('errorLoadingFile')); 4200 } 4201 })); 4202 4203 // Removes openFile if dialog is closed 4204 this.showDialog(new OpenDialog(this).container, (Editor.useLocalStorage) ? 640 : 360, 4205 (Editor.useLocalStorage) ? 480 : 220, true, true, function() 4206 { 4207 Editor.useLocalStorage = prevValue; 4208 window.openFile = null; 4209 }); 4210 } 4211}; 4212 4213/** 4214 * Translates this point by the given vector. 4215 * 4216 * @param {number} dx X-coordinate of the translation. 4217 * @param {number} dy Y-coordinate of the translation. 4218 */ 4219App.prototype.saveLibrary = function(name, images, file, mode, noSpin, noReload, fn) 4220{ 4221 try 4222 { 4223 mode = (mode != null) ? mode : this.mode; 4224 noSpin = (noSpin != null) ? noSpin : false; 4225 noReload = (noReload != null) ? noReload : false; 4226 var xml = this.createLibraryDataFromImages(images); 4227 4228 var error = mxUtils.bind(this, function(resp) 4229 { 4230 this.spinner.stop(); 4231 4232 if (fn != null) 4233 { 4234 fn(); 4235 } 4236 4237 this.handleError(resp, (resp != null) ? mxResources.get('errorSavingFile') : null); 4238 }); 4239 4240 // Handles special case for local libraries 4241 if (file == null && mode == App.MODE_DEVICE) 4242 { 4243 file = new LocalLibrary(this, xml, name); 4244 } 4245 4246 if (file == null) 4247 { 4248 this.pickFolder(mode, mxUtils.bind(this, function(folderId) 4249 { 4250 if (mode == App.MODE_GOOGLE && this.drive != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4251 { 4252 this.drive.insertFile(name, xml, folderId, mxUtils.bind(this, function(newFile) 4253 { 4254 this.spinner.stop(); 4255 this.hideDialog(true); 4256 this.libraryLoaded(newFile, images); 4257 }), error, this.drive.libraryMimeType); 4258 } 4259 else if (mode == App.MODE_GITHUB && this.gitHub != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4260 { 4261 this.gitHub.insertLibrary(name, xml, mxUtils.bind(this, function(newFile) 4262 { 4263 this.spinner.stop(); 4264 this.hideDialog(true); 4265 this.libraryLoaded(newFile, images); 4266 }), error, folderId); 4267 } 4268 else if (mode == App.MODE_GITLAB && this.gitLab != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4269 { 4270 this.gitLab.insertLibrary(name, xml, mxUtils.bind(this, function(newFile) 4271 { 4272 this.spinner.stop(); 4273 this.hideDialog(true); 4274 this.libraryLoaded(newFile, images); 4275 }), error, folderId); 4276 } 4277 else if (mode == App.MODE_NOTION && this.notion != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4278 { 4279 this.notion.insertLibrary(name, xml, mxUtils.bind(this, function(newFile) 4280 { 4281 this.spinner.stop(); 4282 this.hideDialog(true); 4283 this.libraryLoaded(newFile, images); 4284 }), error, folderId); 4285 } 4286 else if (mode == App.MODE_TRELLO && this.trello != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4287 { 4288 this.trello.insertLibrary(name, xml, mxUtils.bind(this, function(newFile) 4289 { 4290 this.spinner.stop(); 4291 this.hideDialog(true); 4292 this.libraryLoaded(newFile, images); 4293 }), error, folderId); 4294 } 4295 else if (mode == App.MODE_DROPBOX && this.dropbox != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4296 { 4297 this.dropbox.insertLibrary(name, xml, mxUtils.bind(this, function(newFile) 4298 { 4299 this.spinner.stop(); 4300 this.hideDialog(true); 4301 this.libraryLoaded(newFile, images); 4302 }), error, folderId); 4303 } 4304 else if (mode == App.MODE_ONEDRIVE && this.oneDrive != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4305 { 4306 this.oneDrive.insertLibrary(name, xml, mxUtils.bind(this, function(newFile) 4307 { 4308 this.spinner.stop(); 4309 this.hideDialog(true); 4310 this.libraryLoaded(newFile, images); 4311 }), error, folderId); 4312 } 4313 else if (mode == App.MODE_BROWSER) 4314 { 4315 var fn = mxUtils.bind(this, function() 4316 { 4317 var file = new StorageLibrary(this, xml, name); 4318 4319 // Inserts data into local storage 4320 file.saveFile(name, false, mxUtils.bind(this, function() 4321 { 4322 this.hideDialog(true); 4323 this.libraryLoaded(file, images); 4324 }), error); 4325 }); 4326 4327 if (localStorage.getItem(name) == null) 4328 { 4329 fn(); 4330 } 4331 else 4332 { 4333 this.confirm(mxResources.get('replaceIt', [name]), fn); 4334 } 4335 } 4336 else 4337 { 4338 this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}); 4339 } 4340 })); 4341 } 4342 else if (noSpin || this.spinner.spin(document.body, mxResources.get('saving'))) 4343 { 4344 file.setData(xml); 4345 4346 var doSave = mxUtils.bind(this, function() 4347 { 4348 file.save(true, mxUtils.bind(this, function(resp) 4349 { 4350 this.spinner.stop(); 4351 this.hideDialog(true); 4352 4353 if (!noReload) 4354 { 4355 this.libraryLoaded(file, images); 4356 } 4357 4358 if (fn != null) 4359 { 4360 fn(); 4361 } 4362 }), error); 4363 }); 4364 4365 if (name != file.getTitle()) 4366 { 4367 var oldHash = file.getHash(); 4368 4369 file.rename(name, mxUtils.bind(this, function(resp) 4370 { 4371 // Change hash in stored settings 4372 if (file.constructor != LocalLibrary && oldHash != file.getHash()) 4373 { 4374 mxSettings.removeCustomLibrary(oldHash); 4375 mxSettings.addCustomLibrary(file.getHash()); 4376 } 4377 4378 // Workaround for library files changing hash so 4379 // the old library cannot be removed from the 4380 // sidebar using the updated file in libraryLoaded 4381 this.removeLibrarySidebar(oldHash); 4382 4383 doSave(); 4384 }), error) 4385 } 4386 else 4387 { 4388 doSave(); 4389 } 4390 } 4391 } 4392 catch (e) 4393 { 4394 this.handleError(e); 4395 } 4396}; 4397 4398/** 4399 * Adds the label menu items to the given menu and parent. 4400 */ 4401App.prototype.saveFile = function(forceDialog, success) 4402{ 4403 var file = this.getCurrentFile(); 4404 4405 if (file != null) 4406 { 4407 // FIXME: Invoke for local files 4408 var done = mxUtils.bind(this, function() 4409 { 4410 if (EditorUi.enableDrafts) 4411 { 4412 file.removeDraft(); 4413 } 4414 4415 if (this.getCurrentFile() != file && !file.isModified()) 4416 { 4417 // Workaround for possible status update while save as dialog is showing 4418 // is to show no saved status for device files 4419 if (file.getMode() != App.MODE_DEVICE) 4420 { 4421 this.editor.setStatus(mxUtils.htmlEntities(mxResources.get('allChangesSaved'))); 4422 } 4423 else 4424 { 4425 this.editor.setStatus(''); 4426 } 4427 } 4428 4429 if (success != null) 4430 { 4431 success(); 4432 } 4433 }); 4434 4435 if (!forceDialog && file.getTitle() != null && file.invalidFileHandle == null && this.mode != null) 4436 { 4437 this.save(file.getTitle(), done); 4438 } 4439 else if (file != null && file.constructor == LocalFile && file.fileHandle != null) 4440 { 4441 this.showSaveFilePicker(mxUtils.bind(this, function(fileHandle, desc) 4442 { 4443 file.invalidFileHandle = null; 4444 file.fileHandle = fileHandle; 4445 file.title = desc.name; 4446 file.desc = desc; 4447 this.save(desc.name, done); 4448 }), null, this.createFileSystemOptions(file.getTitle())); 4449 } 4450 else 4451 { 4452 var filename = (file.getTitle() != null) ? file.getTitle() : this.defaultFilename; 4453 var allowTab = !mxClient.IS_IOS || !navigator.standalone; 4454 var prev = this.mode; 4455 var serviceCount = this.getServiceCount(true); 4456 4457 if (isLocalStorage) 4458 { 4459 serviceCount++; 4460 } 4461 4462 var rowLimit = (serviceCount <= 4) ? 2 : (serviceCount > 6 ? 4 : 3); 4463 4464 var dlg = new CreateDialog(this, filename, mxUtils.bind(this, function(name, mode, input) 4465 { 4466 if (name != null && name.length > 0) 4467 { 4468 // Handles special case where PDF export is detected 4469 if (/(\.pdf)$/i.test(name)) 4470 { 4471 this.confirm(mxResources.get('didYouMeanToExportToPdf'), mxUtils.bind(this, function() 4472 { 4473 this.hideDialog(); 4474 this.actions.get('exportPdf').funct(); 4475 }), mxUtils.bind(this, function() 4476 { 4477 input.value = name.split('.').slice(0, -1).join('.'); 4478 input.focus(); 4479 4480 if (mxClient.IS_GC || mxClient.IS_FF || document.documentMode >= 5) 4481 { 4482 input.select(); 4483 } 4484 else 4485 { 4486 document.execCommand('selectAll', false, null); 4487 } 4488 }), mxResources.get('yes'), mxResources.get('no')); 4489 } 4490 else 4491 { 4492 this.hideDialog(); 4493 4494 if (prev == null && mode == App.MODE_DEVICE) 4495 { 4496 if (file != null && EditorUi.nativeFileSupport) 4497 { 4498 this.showSaveFilePicker(mxUtils.bind(this, function(fileHandle, desc) 4499 { 4500 file.fileHandle = fileHandle; 4501 file.mode = App.MODE_DEVICE; 4502 file.title = desc.name; 4503 file.desc = desc; 4504 4505 this.setMode(App.MODE_DEVICE); 4506 this.save(desc.name, done); 4507 }), mxUtils.bind(this, function(e) 4508 { 4509 if (e.name != 'AbortError') 4510 { 4511 this.handleError(e); 4512 } 4513 }), this.createFileSystemOptions(name)); 4514 } 4515 else 4516 { 4517 this.setMode(App.MODE_DEVICE); 4518 this.save(name, done); 4519 } 4520 } 4521 else if (mode == 'download') 4522 { 4523 var tmp = new LocalFile(this, null, name); 4524 tmp.save(); 4525 } 4526 else if (mode == '_blank') 4527 { 4528 window.openFile = new OpenFile(function() 4529 { 4530 window.openFile = null; 4531 }); 4532 4533 // Do not use a filename to use undefined mode 4534 window.openFile.setData(this.getFileData(true)); 4535 this.openLink(this.getUrl(window.location.pathname), null, true); 4536 } 4537 else if (prev != mode) 4538 { 4539 this.pickFolder(mode, mxUtils.bind(this, function(folderId) 4540 { 4541 this.createFile(name, this.getFileData(/(\.xml)$/i.test(name) || 4542 name.indexOf('.') < 0 || /(\.drawio)$/i.test(name), 4543 /(\.svg)$/i.test(name), /(\.html)$/i.test(name)), 4544 null, mode, done, this.mode == null, folderId); 4545 })); 4546 } 4547 else if (mode != null) 4548 { 4549 this.save(name, done); 4550 } 4551 } 4552 } 4553 }), mxUtils.bind(this, function() 4554 { 4555 this.hideDialog(); 4556 }), mxResources.get('saveAs'), mxResources.get('download'), null, null, allowTab, 4557 null, true, rowLimit, null, null, null, this.editor.fileExtensions, false); 4558 this.showDialog(dlg.container, 400, (serviceCount > rowLimit) ? 390 : 270, true, true); 4559 dlg.init(); 4560 } 4561 } 4562}; 4563 4564/** 4565 * Translates this point by the given vector. 4566 * 4567 * @param {number} dx X-coordinate of the translation. 4568 * @param {number} dy Y-coordinate of the translation. 4569 */ 4570App.prototype.loadTemplate = function(url, onload, onerror, templateFilename, asLibrary) 4571{ 4572 var base64 = false; 4573 var realUrl = url; 4574 4575 if (!this.editor.isCorsEnabledForUrl(realUrl)) 4576 { 4577 // Always uses base64 response to check magic numbers for file type 4578 var nocache = 't=' + new Date().getTime(); 4579 realUrl = PROXY_URL + '?url=' + encodeURIComponent(url) + '&base64=1&' + nocache; 4580 base64 = true; 4581 } 4582 4583 var filterFn = (templateFilename != null) ? templateFilename : url; 4584 4585 this.editor.loadUrl(realUrl, mxUtils.bind(this, function(responseData) 4586 { 4587 try 4588 { 4589 var data = (!base64) ? responseData : ((window.atob && !mxClient.IS_IE && !mxClient.IS_IE11) ? 4590 atob(responseData) : Base64.decode(responseData)); 4591 var isVisioFilename = /(\.v(dx|sdx?))($|\?)/i.test(filterFn) || 4592 /(\.vs(x|sx?))($|\?)/i.test(filterFn); 4593 4594 if (isVisioFilename || this.isVisioData(data)) 4595 { 4596 // Adds filename to control converter code 4597 if (!isVisioFilename) 4598 { 4599 if (asLibrary) 4600 { 4601 filterFn = this.isRemoteVisioData(data) ? 'raw.vss' : 'raw.vssx'; 4602 } 4603 else 4604 { 4605 filterFn = this.isRemoteVisioData(data) ? 'raw.vsd' : 'raw.vsdx'; 4606 } 4607 } 4608 4609 this.importVisio(this.base64ToBlob(responseData.substring(responseData.indexOf(',') + 1)), function(xml) 4610 { 4611 onload(xml); 4612 }, onerror, filterFn); 4613 } 4614 else if (!this.isOffline() && new XMLHttpRequest().upload && this.isRemoteFileFormat(data, filterFn)) 4615 { 4616 // Asynchronous parsing via server 4617 this.parseFile(new Blob([data], {type: 'application/octet-stream'}), mxUtils.bind(this, function(xhr) 4618 { 4619 if (xhr.readyState == 4 && xhr.status >= 200 && xhr.status <= 299 && 4620 xhr.responseText.substring(0, 13) == '<mxGraphModel') 4621 { 4622 onload(xhr.responseText); 4623 } 4624 }), url); 4625 } 4626 else if (this.isLucidChartData(data)) 4627 { 4628 this.convertLucidChart(data, mxUtils.bind(this, function(xml) 4629 { 4630 onload(xml); 4631 }), mxUtils.bind(this, function(e) 4632 { 4633 onerror(e); 4634 })); 4635 } 4636 else 4637 { 4638 if (/(\.png)($|\?)/i.test(filterFn) || this.isPngData(data)) 4639 { 4640 data = this.extractGraphModelFromPng(responseData); 4641 } 4642 4643 onload(data); 4644 } 4645 } 4646 catch (e) 4647 { 4648 onerror(e); 4649 } 4650 }), onerror, /(\.png)($|\?)/i.test(filterFn) || /(\.v(dx|sdx?))($|\?)/i.test(filterFn) || 4651 /(\.vs(x|sx?))($|\?)/i.test(filterFn), null, null, base64); 4652}; 4653 4654/** 4655 * Translates this point by the given vector. 4656 * 4657 * @param {number} dx X-coordinate of the translation. 4658 * @param {number} dy Y-coordinate of the translation. 4659 */ 4660App.prototype.getPeerForMode = function(mode) 4661{ 4662 if (mode == App.MODE_GOOGLE) 4663 { 4664 return this.drive; 4665 } 4666 else if (mode == App.MODE_GITHUB) 4667 { 4668 return this.gitHub; 4669 } 4670 else if (mode == App.MODE_GITLAB) 4671 { 4672 return this.gitLab; 4673 } 4674 else if (mode == App.MODE_DROPBOX) 4675 { 4676 return this.dropbox; 4677 } 4678 else if (mode == App.MODE_ONEDRIVE) 4679 { 4680 return this.oneDrive; 4681 } 4682 else if (mode == App.MODE_TRELLO) 4683 { 4684 return this.trello; 4685 } 4686 else if (mode == App.MODE_NOTION) 4687 { 4688 return this.notion; 4689 } 4690 else 4691 { 4692 return null; 4693 } 4694}; 4695 4696/** 4697 * Translates this point by the given vector. 4698 * 4699 * @param {number} dx X-coordinate of the translation. 4700 * @param {number} dy Y-coordinate of the translation. 4701 */ 4702App.prototype.createFile = function(title, data, libs, mode, done, replace, folderId, tempFile, clibs) 4703{ 4704 mode = (tempFile) ? null : ((mode != null) ? mode : this.mode); 4705 4706 if (title != null && this.spinner.spin(document.body, mxResources.get('inserting'))) 4707 { 4708 data = (data != null) ? data : this.emptyDiagramXml; 4709 4710 var complete = mxUtils.bind(this, function() 4711 { 4712 this.spinner.stop(); 4713 }); 4714 4715 var error = mxUtils.bind(this, function(resp) 4716 { 4717 complete(); 4718 4719 if (resp == null && this.getCurrentFile() == null && this.dialog == null) 4720 { 4721 this.showSplash(); 4722 } 4723 else if (resp != null) 4724 { 4725 this.handleError(resp); 4726 } 4727 }); 4728 4729 try 4730 { 4731 if (mode == App.MODE_GOOGLE && this.drive != null) 4732 { 4733 if (folderId == null && this.stateArg != null && this.stateArg.folderId != null) 4734 { 4735 folderId = this.stateArg.folderId; 4736 } 4737 4738 this.drive.insertFile(title, data, folderId, mxUtils.bind(this, function(file) 4739 { 4740 complete(); 4741 this.fileCreated(file, libs, replace, done, clibs); 4742 }), error); 4743 } 4744 else if (mode == App.MODE_GITHUB && this.gitHub != null) 4745 { 4746 this.gitHub.insertFile(title, data, mxUtils.bind(this, function(file) 4747 { 4748 complete(); 4749 this.fileCreated(file, libs, replace, done, clibs); 4750 }), error, false, folderId); 4751 } 4752 else if (mode == App.MODE_GITLAB && this.gitLab != null) 4753 { 4754 this.gitLab.insertFile(title, data, mxUtils.bind(this, function(file) 4755 { 4756 complete(); 4757 this.fileCreated(file, libs, replace, done, clibs); 4758 }), error, false, folderId); 4759 } 4760 else if (mode == App.MODE_NOTION && this.notion != null) 4761 { 4762 this.notion.insertFile(title, data, mxUtils.bind(this, function(file) 4763 { 4764 complete(); 4765 this.fileCreated(file, libs, replace, done, clibs); 4766 }), error, false, folderId); 4767 } 4768 else if (mode == App.MODE_TRELLO && this.trello != null) 4769 { 4770 this.trello.insertFile(title, data, mxUtils.bind(this, function(file) 4771 { 4772 complete(); 4773 this.fileCreated(file, libs, replace, done, clibs); 4774 }), error, false, folderId); 4775 } 4776 else if (mode == App.MODE_DROPBOX && this.dropbox != null) 4777 { 4778 this.dropbox.insertFile(title, data, mxUtils.bind(this, function(file) 4779 { 4780 complete(); 4781 this.fileCreated(file, libs, replace, done, clibs); 4782 }), error); 4783 } 4784 else if (mode == App.MODE_ONEDRIVE && this.oneDrive != null) 4785 { 4786 this.oneDrive.insertFile(title, data, mxUtils.bind(this, function(file) 4787 { 4788 complete(); 4789 this.fileCreated(file, libs, replace, done, clibs); 4790 }), error, false, folderId); 4791 } 4792 else if (mode == App.MODE_BROWSER) 4793 { 4794 StorageFile.insertFile(this, title, data, mxUtils.bind(this, function(file) 4795 { 4796 complete(); 4797 this.fileCreated(file, libs, replace, done, clibs); 4798 }), error); 4799 } 4800 else if (!tempFile && mode == App.MODE_DEVICE && EditorUi.nativeFileSupport) 4801 { 4802 complete(); 4803 4804 this.showSaveFilePicker(mxUtils.bind(this, function(fileHandle, desc) 4805 { 4806 var file = new LocalFile(this, data, desc.name, null, fileHandle, desc); 4807 4808 file.saveFile(desc.name, false, mxUtils.bind(this, function() 4809 { 4810 this.fileCreated(file, libs, replace, done, clibs); 4811 }), error, true); 4812 }), mxUtils.bind(this, function(e) 4813 { 4814 if (e.name != 'AbortError') 4815 { 4816 error(e); 4817 } 4818 }), this.createFileSystemOptions(title)); 4819 } 4820 else 4821 { 4822 complete(); 4823 this.fileCreated(new LocalFile(this, data, title, mode == null), libs, replace, done, clibs); 4824 } 4825 } 4826 catch (e) 4827 { 4828 complete(); 4829 this.handleError(e); 4830 } 4831 } 4832}; 4833 4834/** 4835 * Translates this point by the given vector. 4836 * 4837 * @param {number} dx X-coordinate of the translation. 4838 * @param {number} dy Y-coordinate of the translation. 4839 */ 4840App.prototype.fileCreated = function(file, libs, replace, done, clibs) 4841{ 4842 var url = window.location.pathname; 4843 4844 if (libs != null && libs.length > 0) 4845 { 4846 url += '?libs=' + libs; 4847 } 4848 4849 if (clibs != null && clibs.length > 0) 4850 { 4851 url += '?clibs=' + clibs; 4852 } 4853 4854 url = this.getUrl(url); 4855 4856 // Always opens a new tab for local files to avoid losing changes 4857 if (file.getMode() != App.MODE_DEVICE) 4858 { 4859 url += '#' + file.getHash(); 4860 } 4861 4862 // Makes sure to produce consistent output with finalized files via createFileData this needs 4863 // to save the file again since it needs the newly created file ID for redirecting in HTML 4864 if (this.spinner.spin(document.body, mxResources.get('inserting'))) 4865 { 4866 var data = file.getData(); 4867 var dataNode = (data.length > 0) ? this.editor.extractGraphModel( 4868 mxUtils.parseXml(data).documentElement, true) : null; 4869 var redirect = window.location.protocol + '//' + window.location.hostname + url; 4870 var node = dataNode; 4871 var graph = null; 4872 4873 // Handles special case where SVG files need a rendered graph to be saved 4874 if (dataNode != null && /\.svg$/i.test(file.getTitle())) 4875 { 4876 graph = this.createTemporaryGraph(this.editor.graph.getStylesheet()); 4877 document.body.appendChild(graph.container); 4878 node = this.decodeNodeIntoGraph(node, graph); 4879 } 4880 4881 file.setData(this.createFileData(dataNode, graph, file, redirect)); 4882 4883 if (graph != null) 4884 { 4885 graph.container.parentNode.removeChild(graph.container); 4886 } 4887 4888 var complete = mxUtils.bind(this, function() 4889 { 4890 this.spinner.stop(); 4891 }); 4892 4893 var fn = mxUtils.bind(this, function() 4894 { 4895 complete(); 4896 4897 var currentFile = this.getCurrentFile(); 4898 4899 if (replace == null && currentFile != null) 4900 { 4901 replace = !currentFile.isModified() && currentFile.getMode() == null; 4902 } 4903 4904 var fn3 = mxUtils.bind(this, function() 4905 { 4906 window.openFile = null; 4907 this.fileLoaded(file); 4908 4909 if (replace) 4910 { 4911 file.addAllSavedStatus(); 4912 } 4913 4914 if (libs != null) 4915 { 4916 this.sidebar.showEntries(libs); 4917 } 4918 4919 if (clibs != null) 4920 { 4921 var temp = []; 4922 var tokens = clibs.split(';'); 4923 4924 for (var i = 0; i < tokens.length; i++) 4925 { 4926 temp.push(decodeURIComponent(tokens[i])); 4927 } 4928 4929 this.loadLibraries(temp); 4930 } 4931 }); 4932 4933 var fn2 = mxUtils.bind(this, function() 4934 { 4935 if (replace || currentFile == null || !currentFile.isModified()) 4936 { 4937 fn3(); 4938 } 4939 else 4940 { 4941 this.confirm(mxResources.get('allChangesLost'), null, fn3, 4942 mxResources.get('cancel'), mxResources.get('discardChanges')); 4943 } 4944 }); 4945 4946 if (done != null) 4947 { 4948 done(); 4949 } 4950 4951 // Opens the file in a new window 4952 if (replace != null && !replace) 4953 { 4954 // Opens local file in a new window 4955 if (file.constructor == LocalFile) 4956 { 4957 window.openFile = new OpenFile(function() 4958 { 4959 window.openFile = null; 4960 }); 4961 4962 window.openFile.setData(file.getData(), file.getTitle(), file.getMode() == null); 4963 } 4964 4965 if (done != null) 4966 { 4967 done(); 4968 } 4969 4970 window.openWindow(url, null, fn2); 4971 } 4972 else 4973 { 4974 fn2(); 4975 } 4976 }); 4977 4978 // Updates data in memory for local files 4979 if (file.constructor == LocalFile) 4980 { 4981 fn(); 4982 } 4983 else 4984 { 4985 file.saveFile(file.getTitle(), false, mxUtils.bind(this, function() 4986 { 4987 fn(); 4988 }), mxUtils.bind(this, function(resp) 4989 { 4990 complete(); 4991 this.handleError(resp); 4992 })); 4993 } 4994 } 4995}; 4996 4997/** 4998 * Translates this point by the given vector. 4999 * 5000 * @param {number} dx X-coordinate of the translation. 5001 * @param {number} dy Y-coordinate of the translation. 5002 */ 5003App.prototype.loadFile = function(id, sameWindow, file, success, force) 5004{ 5005 if (urlParams['openInSameWin'] == '1' || navigator.standalone) 5006 { 5007 sameWindow = true; 5008 } 5009 5010 this.hideDialog(); 5011 5012 var fn2 = mxUtils.bind(this, function() 5013 { 5014 if (id == null || id.length == 0) 5015 { 5016 this.editor.setStatus(''); 5017 this.fileLoaded(null); 5018 } 5019 else if (this.spinner.spin(document.body, mxResources.get('loading'))) 5020 { 5021 // Handles files from localStorage 5022 if (id.charAt(0) == 'L') 5023 { 5024 this.spinner.stop(); 5025 5026 if (!isLocalStorage) 5027 { 5028 this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function() 5029 { 5030 var tempFile = this.getCurrentFile(); 5031 window.location.hash = (tempFile != null) ? tempFile.getHash() : ''; 5032 })); 5033 } 5034 else 5035 { 5036 var error = mxUtils.bind(this, function (e) 5037 { 5038 this.handleError(e, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function() 5039 { 5040 var tempFile = this.getCurrentFile(); 5041 window.location.hash = (tempFile != null) ? tempFile.getHash() : ''; 5042 })); 5043 }); 5044 5045 id = decodeURIComponent(id.substring(1)); 5046 5047 StorageFile.getFileContent(this, id, mxUtils.bind(this, function(data) 5048 { 5049 if (data != null) 5050 { 5051 this.fileLoaded(new StorageFile(this, data, id)); 5052 5053 if (success != null) 5054 { 5055 success(); 5056 } 5057 } 5058 else 5059 { 5060 error({message: mxResources.get('fileNotFound')}); 5061 } 5062 }), error); 5063 } 5064 } 5065 else if (file != null) 5066 { 5067 // File already loaded 5068 this.spinner.stop(); 5069 this.fileLoaded(file); 5070 5071 if (success != null) 5072 { 5073 success(); 5074 } 5075 } 5076 else if (id.charAt(0) == 'S') 5077 { 5078 this.spinner.stop(); 5079 5080 this.alert('[Deprecation] #S is no longer supported, go to https://app.diagrams.net/?desc=' + id.substring(1).substring(0, 10), mxUtils.bind(this, function() 5081 { 5082 window.location.href = 'https://app.diagrams.net/?desc=' + id.substring(1); 5083 })); 5084 } 5085 else if (id.charAt(0) == 'R') 5086 { 5087 // Raw file encoded into URL 5088 this.spinner.stop(); 5089 var data = decodeURIComponent(id.substring(1)); 5090 5091 if (data.charAt(0) != '<') 5092 { 5093 data = Graph.decompress(data); 5094 } 5095 5096 var tempFile = new LocalFile(this, data, (urlParams['title'] != null) ? 5097 decodeURIComponent(urlParams['title']) : this.defaultFilename, true); 5098 tempFile.getHash = function() 5099 { 5100 return id; 5101 }; 5102 this.fileLoaded(tempFile); 5103 5104 if (success != null) 5105 { 5106 success(); 5107 } 5108 } 5109 else if (id.charAt(0) == 'E') // Embed file 5110 { 5111 //Currently we only reload current file. Id is not used! 5112 var currentFile = this.getCurrentFile(); 5113 5114 if (currentFile == null) 5115 { 5116 this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile')); 5117 } 5118 else 5119 { 5120 this.remoteInvoke('getDraftFileContent', null, null, mxUtils.bind(this, function(data, desc) 5121 { 5122 this.spinner.stop(); 5123 this.fileLoaded(new EmbedFile(this, data, desc)); 5124 5125 if (success != null) 5126 { 5127 success(); 5128 } 5129 }), mxUtils.bind(this, function() 5130 { 5131 this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile')); 5132 })); 5133 } 5134 } 5135 else if (id.charAt(0) == 'U') 5136 { 5137 var url = decodeURIComponent(id.substring(1)); 5138 5139 var doFallback = mxUtils.bind(this, function() 5140 { 5141 // Fallback for non-public Google Drive files 5142 if (url.substring(0, 31) == 'https://drive.google.com/uc?id=' && 5143 (this.drive != null || typeof window.DriveClient === 'function')) 5144 { 5145 this.hideDialog(); 5146 5147 var fallback = mxUtils.bind(this, function() 5148 { 5149 this.spinner.stop(); 5150 5151 if (this.drive != null) 5152 { 5153 var tempId = url.substring(31, url.lastIndexOf('&ex')); 5154 5155 this.loadFile('G' + tempId, sameWindow, null, mxUtils.bind(this, function() 5156 { 5157 var currentFile = this.getCurrentFile(); 5158 5159 if (currentFile != null && this.editor.chromeless && !this.editor.editable) 5160 { 5161 currentFile.getHash = function() 5162 { 5163 return 'G' + tempId; 5164 }; 5165 5166 window.location.hash = '#' + currentFile.getHash(); 5167 } 5168 5169 if (success != null) 5170 { 5171 success(); 5172 } 5173 })); 5174 5175 return true; 5176 } 5177 else 5178 { 5179 return false; 5180 } 5181 }); 5182 5183 if (!fallback() && this.spinner.spin(document.body, mxResources.get('loading'))) 5184 { 5185 this.addListener('clientLoaded', fallback); 5186 } 5187 5188 return true; 5189 } 5190 else 5191 { 5192 return false; 5193 } 5194 }); 5195 5196 this.loadTemplate(url, mxUtils.bind(this, function(text) 5197 { 5198 this.spinner.stop(); 5199 5200 if (text != null && text.length > 0) 5201 { 5202 var filename = this.defaultFilename; 5203 5204 // Tries to find name from URL with valid extensions 5205 if (urlParams['title'] == null && urlParams['notitle'] != '1') 5206 { 5207 var tmp = url; 5208 var dot = url.lastIndexOf('.'); 5209 var slash = tmp.lastIndexOf('/'); 5210 5211 if (dot > slash && slash > 0) 5212 { 5213 tmp = tmp.substring(slash + 1, dot); 5214 var ext = url.substring(dot); 5215 5216 if (!this.useCanvasForExport && ext == '.png') 5217 { 5218 ext = '.drawio'; 5219 } 5220 5221 if (ext === '.svg' || ext === '.xml' || 5222 ext === '.html' || ext === '.png' || 5223 ext === '.drawio') 5224 { 5225 filename = tmp + ext; 5226 } 5227 } 5228 } 5229 5230 var tempFile = new LocalFile(this, text, (urlParams['title'] != null) ? 5231 decodeURIComponent(urlParams['title']) : filename, true); 5232 tempFile.getHash = function() 5233 { 5234 return id; 5235 }; 5236 5237 if (this.fileLoaded(tempFile, true)) 5238 { 5239 if (success != null) 5240 { 5241 success(); 5242 } 5243 } 5244 else if (!doFallback()) 5245 { 5246 this.handleError({message: mxResources.get('fileNotFound')}, 5247 mxResources.get('errorLoadingFile')); 5248 } 5249 } 5250 else if (!doFallback()) 5251 { 5252 this.handleError({message: mxResources.get('fileNotFound')}, 5253 mxResources.get('errorLoadingFile')); 5254 } 5255 }), mxUtils.bind(this, function() 5256 { 5257 if (!doFallback()) 5258 { 5259 this.spinner.stop(); 5260 this.handleError({message: mxResources.get('fileNotFound')}, 5261 mxResources.get('errorLoadingFile')); 5262 } 5263 }), (urlParams['template-filename'] != null) ? 5264 decodeURIComponent(urlParams['template-filename']) : null); 5265 } 5266 else 5267 { 5268 // Google Drive files are handled as default file types 5269 var peer = null; 5270 5271 if (id.charAt(0) == 'G') 5272 { 5273 peer = this.drive; 5274 } 5275 else if (id.charAt(0) == 'D') 5276 { 5277 peer = this.dropbox; 5278 } 5279 else if (id.charAt(0) == 'W') 5280 { 5281 peer = this.oneDrive; 5282 } 5283 else if (id.charAt(0) == 'H') 5284 { 5285 peer = this.gitHub; 5286 } 5287 else if (id.charAt(0) == 'A') 5288 { 5289 peer = this.gitLab; 5290 } 5291 else if (id.charAt(0) == 'T') 5292 { 5293 peer = this.trello; 5294 } 5295 else if (id.charAt(0) == 'N') 5296 { 5297 peer = this.notion; 5298 } 5299 5300 if (peer == null) 5301 { 5302 this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function() 5303 { 5304 var currentFile = this.getCurrentFile(); 5305 window.location.hash = (currentFile != null) ? currentFile.getHash() : ''; 5306 })); 5307 } 5308 else 5309 { 5310 var peerChar = id.charAt(0); 5311 id = decodeURIComponent(id.substring(1)); 5312 5313 peer.getFile(id, mxUtils.bind(this, function(file) 5314 { 5315 this.spinner.stop(); 5316 this.fileLoaded(file); 5317 var currentFile = this.getCurrentFile(); 5318 5319 if (currentFile == null) 5320 { 5321 window.location.hash = ''; 5322 this.showSplash(); 5323 } 5324 else if (this.editor.chromeless && !this.editor.editable) 5325 { 5326 // Keeps ID even for converted files in chromeless mode for refresh to work 5327 currentFile.getHash = function() 5328 { 5329 return peerChar + id; 5330 }; 5331 5332 window.location.hash = '#' + currentFile.getHash(); 5333 } 5334 else if (file == currentFile && file.getMode() == null) 5335 { 5336 // Shows a warning if a copy was opened which happens 5337 // eg. for .png files in IE as they cannot be written 5338 var status = mxResources.get('copyCreated'); 5339 this.editor.setStatus('<div title="'+ status + 5340 '" class="geStatusAlert">' + status + '</div>'); 5341 } 5342 5343 if (success != null) 5344 { 5345 success(); 5346 } 5347 }), mxUtils.bind(this, function(resp) 5348 { 5349 // Makes sure the file does not save the invalid UI model and overwrites anything important 5350 if (window.console != null && resp != null) 5351 { 5352 console.log('error in loadFile:', id, resp); 5353 } 5354 5355 this.handleError(resp, (resp != null) ? mxResources.get('errorLoadingFile') : null, mxUtils.bind(this, function() 5356 { 5357 var currentFile = this.getCurrentFile(); 5358 5359 if (currentFile == null) 5360 { 5361 window.location.hash = ''; 5362 this.showSplash(); 5363 } 5364 else 5365 { 5366 window.location.hash = '#' + currentFile.getHash(); 5367 } 5368 }), null, null, '#' + peerChar + id); 5369 })); 5370 } 5371 } 5372 } 5373 }); 5374 5375 var currentFile = this.getCurrentFile(); 5376 5377 var fn = mxUtils.bind(this, function() 5378 { 5379 if (force || currentFile == null || !currentFile.isModified()) 5380 { 5381 fn2(); 5382 } 5383 else 5384 { 5385 this.confirm(mxResources.get('allChangesLost'), mxUtils.bind(this, function() 5386 { 5387 if (currentFile != null) 5388 { 5389 window.location.hash = currentFile.getHash(); 5390 } 5391 }), fn2, mxResources.get('cancel'), mxResources.get('discardChanges')); 5392 } 5393 }); 5394 5395 if (id == null || id.length == 0) 5396 { 5397 fn(); 5398 } 5399 else if (currentFile != null && !sameWindow) 5400 { 5401 this.showDialog(new PopupDialog(this, this.getUrl() + '#' + id, 5402 null, fn).container, 320, 140, true, true); 5403 } 5404 else 5405 { 5406 fn(); 5407 } 5408}; 5409 5410/** 5411 * Translates this point by the given vector. 5412 * 5413 * @param {number} dx X-coordinate of the translation. 5414 * @param {number} dy Y-coordinate of the translation. 5415 */ 5416App.prototype.getLibraryStorageHint = function(file) 5417{ 5418 var tip = file.getTitle(); 5419 5420 if (file.constructor != LocalLibrary) 5421 { 5422 tip += '\n' + file.getHash(); 5423 } 5424 5425 if (file.constructor == DriveLibrary) 5426 { 5427 tip += ' (' + mxResources.get('googleDrive') + ')'; 5428 } 5429 else if (file.constructor == GitHubLibrary) 5430 { 5431 tip += ' (' + mxResources.get('github') + ')'; 5432 } 5433 else if (file.constructor == TrelloLibrary) 5434 { 5435 tip += ' (' + mxResources.get('trello') + ')'; 5436 } 5437 else if (file.constructor == DropboxLibrary) 5438 { 5439 tip += ' (' + mxResources.get('dropbox') + ')'; 5440 } 5441 else if (file.constructor == OneDriveLibrary) 5442 { 5443 tip += ' (' + mxResources.get('oneDrive') + ')'; 5444 } 5445 else if (file.constructor == StorageLibrary) 5446 { 5447 tip += ' (' + mxResources.get('browser') + ')'; 5448 } 5449 else if (file.constructor == LocalLibrary) 5450 { 5451 tip += ' (' + mxResources.get('device') + ')'; 5452 } 5453 5454 return tip; 5455}; 5456 5457/** 5458 * Updates action states depending on the selection. 5459 */ 5460App.prototype.restoreLibraries = function() 5461{ 5462 this.loadLibraries(mxSettings.getCustomLibraries(), mxUtils.bind(this, function() 5463 { 5464 this.loadLibraries((urlParams['clibs'] || '').split(';')); 5465 })); 5466}; 5467 5468/** 5469 * Updates action states depending on the selection. 5470 */ 5471App.prototype.loadLibraries = function(libs, done) 5472{ 5473 if (this.sidebar != null) 5474 { 5475 if (this.pendingLibraries == null) 5476 { 5477 this.pendingLibraries = new Object(); 5478 } 5479 5480 // Ignores this library next time 5481 var ignore = mxUtils.bind(this, function(id, keep) 5482 { 5483 if (!keep) 5484 { 5485 mxSettings.removeCustomLibrary(id); 5486 } 5487 5488 delete this.pendingLibraries[id]; 5489 }); 5490 5491 var waiting = 0; 5492 var files = []; 5493 5494 // Loads in order of libs array 5495 var checkDone = mxUtils.bind(this, function() 5496 { 5497 if (waiting == 0) 5498 { 5499 if (libs != null) 5500 { 5501 for (var i = libs.length - 1; i >= 0; i--) 5502 { 5503 if (files[i] != null) 5504 { 5505 this.loadLibrary(files[i]); 5506 } 5507 } 5508 } 5509 5510 if (done != null) 5511 { 5512 done(); 5513 } 5514 } 5515 }); 5516 5517 if (libs != null) 5518 { 5519 for (var i = 0; i < libs.length; i++) 5520 { 5521 var name = encodeURIComponent(decodeURIComponent(libs[i])); 5522 5523 (mxUtils.bind(this, function(id, index) 5524 { 5525 if (id != null && id.length > 0 && this.pendingLibraries[id] == null && 5526 this.sidebar.palettes[id] == null) 5527 { 5528 // Waits for all libraries to load 5529 waiting++; 5530 5531 var onload = mxUtils.bind(this, function(file) 5532 { 5533 delete this.pendingLibraries[id]; 5534 files[index] = file; 5535 waiting--; 5536 checkDone(); 5537 }); 5538 5539 var onerror = mxUtils.bind(this, function(keep) 5540 { 5541 ignore(id, keep); 5542 waiting--; 5543 checkDone(); 5544 }); 5545 5546 this.pendingLibraries[id] = true; 5547 var service = id.substring(0, 1); 5548 5549 if (service == 'L') 5550 { 5551 if (isLocalStorage || mxClient.IS_CHROMEAPP) 5552 { 5553 // Make asynchronous for barrier to work 5554 window.setTimeout(mxUtils.bind(this, function() 5555 { 5556 try 5557 { 5558 var name = decodeURIComponent(id.substring(1)); 5559 5560 StorageFile.getFileContent(this, name, mxUtils.bind(this, function(xml) 5561 { 5562 if (name == '.scratchpad' && xml == null) 5563 { 5564 xml = this.emptyLibraryXml; 5565 } 5566 5567 if (xml != null) 5568 { 5569 onload(new StorageLibrary(this, xml, name)); 5570 } 5571 else 5572 { 5573 onerror(); 5574 } 5575 }), onerror); 5576 } 5577 catch (e) 5578 { 5579 onerror(); 5580 } 5581 }), 0); 5582 } 5583 } 5584 else if (service == 'U') 5585 { 5586 var url = decodeURIComponent(id.substring(1)); 5587 5588 if (!this.isOffline()) 5589 { 5590 this.loadTemplate(url, mxUtils.bind(this, function(text) 5591 { 5592 if (text != null && text.length > 0) 5593 { 5594 // LATER: Convert mxfile to mxlibrary using code from libraryLoaded 5595 onload(new UrlLibrary(this, text, url)); 5596 } 5597 else 5598 { 5599 onerror(); 5600 } 5601 }), function() 5602 { 5603 onerror(); 5604 }, null, true); 5605 } 5606 } 5607 else if (service == 'R') 5608 { 5609 var libDesc = decodeURIComponent(id.substring(1)); 5610 5611 try 5612 { 5613 libDesc = JSON.parse(libDesc); 5614 var libObj = { 5615 id: libDesc[0], 5616 title: libDesc[1], 5617 downloadUrl: libDesc[2] 5618 } 5619 5620 this.remoteInvoke('getFileContent', [libObj.downloadUrl], null, mxUtils.bind(this, function(libContent) 5621 { 5622 try 5623 { 5624 onload(new RemoteLibrary(this, libContent, libObj)); 5625 } 5626 catch (e) 5627 { 5628 onerror(); 5629 } 5630 }), function() 5631 { 5632 onerror(); 5633 }); 5634 } 5635 catch (e) 5636 { 5637 onerror(); 5638 } 5639 } 5640 else if (service == 'S' && this.loadDesktopLib != null) 5641 { 5642 try 5643 { 5644 this.loadDesktopLib(decodeURIComponent(id.substring(1)), function(desktopLib) 5645 { 5646 onload(desktopLib); 5647 }, onerror); 5648 } 5649 catch (e) 5650 { 5651 onerror(); 5652 } 5653 } 5654 else 5655 { 5656 var peer = null; 5657 5658 if (service == 'G') 5659 { 5660 if (this.drive != null && this.drive.user != null) 5661 { 5662 peer = this.drive; 5663 } 5664 } 5665 else if (service == 'H') 5666 { 5667 if (this.gitHub != null && this.gitHub.getUser() != null) 5668 { 5669 peer = this.gitHub; 5670 } 5671 } 5672 else if (service == 'T') 5673 { 5674 if (this.trello != null && this.trello.isAuthorized()) 5675 { 5676 peer = this.trello; 5677 } 5678 } 5679 else if (service == 'D') 5680 { 5681 if (this.dropbox != null && this.dropbox.getUser() != null) 5682 { 5683 peer = this.dropbox; 5684 } 5685 } 5686 else if (service == 'W') 5687 { 5688 if (this.oneDrive != null && this.oneDrive.getUser() != null) 5689 { 5690 peer = this.oneDrive; 5691 } 5692 } 5693 5694 if (peer != null) 5695 { 5696 peer.getLibrary(decodeURIComponent(id.substring(1)), mxUtils.bind(this, function(file) 5697 { 5698 try 5699 { 5700 onload(file); 5701 } 5702 catch (e) 5703 { 5704 onerror(); 5705 } 5706 }), function(resp) 5707 { 5708 onerror(); 5709 }); 5710 } 5711 else 5712 { 5713 onerror(true); 5714 } 5715 } 5716 } 5717 }))(name, i); 5718 } 5719 5720 checkDone(); 5721 } 5722 else 5723 { 5724 checkDone(); 5725 } 5726 } 5727}; 5728 5729/** 5730 * Translates this point by the given vector. 5731 * 5732 * @param {number} dx X-coordinate of the translation. 5733 * @param {number} dy Y-coordinate of the translation. 5734 */ 5735App.prototype.updateButtonContainer = function() 5736{ 5737 if (this.buttonContainer != null) 5738 { 5739 var file = this.getCurrentFile(); 5740 5741 if (urlParams['embed'] == '1') 5742 { 5743 if (uiTheme == 'atlas' || urlParams['atlas'] == '1') 5744 { 5745 this.buttonContainer.style.paddingRight = '12px'; 5746 this.buttonContainer.style.paddingTop = '6px'; 5747 this.buttonContainer.style.right = urlParams['noLangIcon'] == '1'? '0' : '25px'; 5748 } 5749 else if (uiTheme != 'min') 5750 { 5751 this.buttonContainer.style.paddingRight = '38px'; 5752 this.buttonContainer.style.paddingTop = '6px'; 5753 } 5754 } 5755 5756 // Comments 5757 if (this.commentsSupported() && urlParams['sketch'] != '1') 5758 { 5759 if (this.commentButton == null) 5760 { 5761 this.commentButton = document.createElement('a'); 5762 this.commentButton.setAttribute('title', mxResources.get('comments')); 5763 this.commentButton.className = 'geToolbarButton'; 5764 this.commentButton.style.cssText = 'display:inline-block;position:relative;box-sizing:border-box;' + 5765 'margin-right:4px;float:left;cursor:pointer;width:24px;height:24px;background-size:24px 24px;' + 5766 'background-position:center center;background-repeat:no-repeat;background-image:' + 5767 'url(' + Editor.commentImage + ');'; 5768 5769 if (uiTheme == 'atlas') 5770 { 5771 this.commentButton.style.marginRight = '10px'; 5772 this.commentButton.style.marginTop = '-3px'; 5773 } 5774 else if (uiTheme == 'min') 5775 { 5776 this.commentButton.style.marginTop = '1px'; 5777 } 5778 else if (urlParams['atlas'] == '1') 5779 { 5780 this.commentButton.style.marginTop = '-2px'; 5781 } 5782 else 5783 { 5784 this.commentButton.style.marginTop = '-5px'; 5785 } 5786 5787 mxEvent.addListener(this.commentButton, 'click', mxUtils.bind(this, function() 5788 { 5789 this.actions.get('comments').funct(); 5790 })); 5791 5792 this.buttonContainer.appendChild(this.commentButton); 5793 5794 if (uiTheme == 'dark' || uiTheme == 'atlas') 5795 { 5796 this.commentButton.style.filter = 'invert(100%)'; 5797 } 5798 } 5799 } 5800 else if (this.commentButton != null) 5801 { 5802 this.commentButton.parentNode.removeChild(this.commentButton); 5803 this.commentButton = null; 5804 } 5805 5806 // Share 5807 if (urlParams['embed'] != '1' && this.getServiceName() == 'draw.io' && 5808 !mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && 5809 !this.isOfflineApp()) 5810 { 5811 if (file != null) 5812 { 5813 if (this.shareButton == null) 5814 { 5815 this.shareButton = document.createElement('div'); 5816 this.shareButton.className = 'geBtn gePrimaryBtn'; 5817 this.shareButton.style.display = 'inline-block'; 5818 this.shareButton.style.backgroundColor = '#F2931E'; 5819 this.shareButton.style.borderColor = '#F08705'; 5820 this.shareButton.style.backgroundImage = 'none'; 5821 this.shareButton.style.padding = '2px 10px 0 10px'; 5822 this.shareButton.style.marginTop = '-10px'; 5823 this.shareButton.style.height = '28px'; 5824 this.shareButton.style.lineHeight = '28px'; 5825 this.shareButton.style.minWidth = '0px'; 5826 this.shareButton.style.cssFloat = 'right'; 5827 this.shareButton.setAttribute('title', mxResources.get('share')); 5828 5829 var icon = document.createElement('img'); 5830 icon.setAttribute('src', this.shareImage); 5831 icon.setAttribute('align', 'absmiddle'); 5832 icon.style.marginRight = '4px'; 5833 icon.style.marginTop = '-3px'; 5834 this.shareButton.appendChild(icon); 5835 5836 if (!Editor.isDarkMode() && uiTheme != 'atlas') 5837 { 5838 this.shareButton.style.color = 'black'; 5839 icon.style.filter = 'invert(100%)'; 5840 } 5841 5842 mxUtils.write(this.shareButton, mxResources.get('share')); 5843 5844 mxEvent.addListener(this.shareButton, 'click', mxUtils.bind(this, function() 5845 { 5846 this.actions.get('share').funct(); 5847 })); 5848 5849 this.buttonContainer.appendChild(this.shareButton); 5850 } 5851 } 5852 else if (this.shareButton != null) 5853 { 5854 this.shareButton.parentNode.removeChild(this.shareButton); 5855 this.shareButton = null; 5856 } 5857 5858 //Fetch notifications 5859 if (urlParams['extAuth'] != '1') //Disable notification with external auth (e.g, Teams app) 5860 { 5861 this.fetchAndShowNotification('online', this.mode); 5862 } 5863 } 5864 else if (urlParams['notif'] != null) //Notif for embed mode 5865 { 5866 this.fetchAndShowNotification(urlParams['notif']); 5867 } 5868 } 5869}; 5870 5871 5872App.prototype.fetchAndShowNotification = function(target, subtarget) 5873{ 5874 if (this.fetchingNotif) 5875 { 5876 return; 5877 } 5878 5879 target = target || 'online'; 5880 var cachedNotifKey = '.notifCache'; 5881 var cachedNotif = null; 5882 5883 var processNotif = mxUtils.bind(this, function(notifs) 5884 { 5885 notifs = notifs.filter(function(notif) 5886 { 5887 return !notif.targets || notif.targets.indexOf(target) > -1 || 5888 (subtarget != null && notif.targets.indexOf(subtarget) > -1); 5889 }); 5890 5891 var lsReadFlag = target + 'NotifReadTS'; 5892 var lastRead = (localStorage != null) ? parseInt(localStorage.getItem(lsReadFlag)) : true; 5893 5894 for (var i = 0; i < notifs.length; i++) 5895 { 5896 notifs[i].isNew = (!lastRead || notifs[i].timestamp > lastRead); 5897 } 5898 5899 this.showNotification(notifs, lsReadFlag); 5900 }); 5901 5902 try 5903 { 5904 if (localStorage != null) 5905 { 5906 cachedNotif = JSON.parse(localStorage.getItem(cachedNotifKey)); 5907 } 5908 } 5909 catch(e) {} //Ignore 5910 5911 if (cachedNotif == null || cachedNotif.ts + 24 * 60 * 60 * 1000 < Date.now()) //Cache for one day 5912 { 5913 this.fetchingNotif = true; 5914 //Fetch all notifications and store them, then filter client-side 5915 mxUtils.get(NOTIFICATIONS_URL, mxUtils.bind(this, function(req) 5916 { 5917 if (req.getStatus() >= 200 && req.getStatus() <= 299) 5918 { 5919 var notifs = JSON.parse(req.getText()); 5920 5921 //Process and sort 5922 notifs.sort(function(a, b) 5923 { 5924 return b.timestamp - a.timestamp; 5925 }); 5926 5927 if (isLocalStorage) 5928 { 5929 localStorage.setItem(cachedNotifKey, JSON.stringify({ts: Date.now(), notifs: notifs})); 5930 } 5931 5932 this.fetchingNotif = false; 5933 processNotif(notifs); 5934 } 5935 })); 5936 } 5937 else 5938 { 5939 processNotif(cachedNotif.notifs); 5940 } 5941}; 5942 5943App.prototype.showNotification = function(notifs, lsReadFlag) 5944{ 5945 var newCount = notifs.length; 5946 5947 if (uiTheme == 'min') 5948 { 5949 newCount = 0; 5950 5951 for (var i = 0; i < notifs.length; i++) 5952 { 5953 if (notifs[i].isNew) 5954 { 5955 newCount++; 5956 } 5957 } 5958 } 5959 5960 if (newCount == 0) 5961 { 5962 if (this.notificationBtn != null) 5963 { 5964 this.notificationBtn.style.display = 'none'; 5965 this.editor.fireEvent(new mxEventObject('statusChanged')); 5966 } 5967 5968 return; 5969 } 5970 5971 function shouldAnimate(newNotif) 5972 { 5973 var countEl = document.querySelector('.geNotification-count'); 5974 5975 if (countEl == null) 5976 { 5977 return; 5978 } 5979 5980 countEl.innerHTML = newNotif; 5981 countEl.style.display = newNotif == 0? 'none' : ''; 5982 var notifBell = document.querySelector('.geNotification-bell'); 5983 notifBell.style.animation = newNotif == 0? 'none' : ''; 5984 notifBell.className = 'geNotification-bell' + (newNotif == 0? ' geNotification-bellOff' : ''); 5985 document.querySelector('.geBell-rad').style.animation = newNotif == 0? 'none' : ''; 5986 } 5987 5988 var markAllAsRead = mxUtils.bind(this, function() 5989 { 5990 this.notificationWin.style.display = 'none'; 5991 var unread = this.notificationWin.querySelectorAll('.circle.active'); 5992 5993 for (var i = 0; i < unread.length; i++) 5994 { 5995 unread[i].className = 'circle'; 5996 } 5997 5998 if (isLocalStorage && notifs[0]) 5999 { 6000 localStorage.setItem(lsReadFlag, notifs[0].timestamp); 6001 } 6002 }); 6003 6004 if (this.notificationBtn == null) 6005 { 6006 this.notificationBtn = document.createElement('div'); 6007 this.notificationBtn.className = 'geNotification-box'; 6008 6009 if (uiTheme == 'min') 6010 { 6011 this.notificationBtn.style.width = '30px'; 6012 this.notificationBtn.style.top = '4px'; 6013 } 6014 else if (urlParams['atlas'] == '1') 6015 { 6016 this.notificationBtn.style.top = '2px'; 6017 } 6018 6019 var notifCount = document.createElement('span'); 6020 notifCount.className = 'geNotification-count'; 6021 this.notificationBtn.appendChild(notifCount); 6022 6023 var notifBell = document.createElement('div'); 6024 notifBell.className = 'geNotification-bell'; 6025 notifBell.style.opacity = uiTheme == 'min'? '0.5' : ''; 6026 var bellPart = document.createElement('span'); 6027 bellPart.className = 'geBell-top'; 6028 notifBell.appendChild(bellPart); 6029 var bellPart = document.createElement('span'); 6030 bellPart.className = 'geBell-middle'; 6031 notifBell.appendChild(bellPart); 6032 var bellPart = document.createElement('span'); 6033 bellPart.className = 'geBell-bottom'; 6034 notifBell.appendChild(bellPart); 6035 var bellPart = document.createElement('span'); 6036 bellPart.className = 'geBell-rad'; 6037 notifBell.appendChild(bellPart); 6038 this.notificationBtn.appendChild(notifBell); 6039 6040 //Add as first child such that it is the left-most one 6041 this.buttonContainer.insertBefore(this.notificationBtn, this.buttonContainer.firstChild); 6042 6043 this.notificationWin = document.createElement('div'); 6044 this.notificationWin.className = 'geNotifPanel'; 6045 this.notificationWin.style.display = 'none'; 6046 document.body.appendChild(this.notificationWin); 6047 6048 var winHeader = document.createElement('div'); 6049 winHeader.className = 'header'; 6050 var winTitle = document.createElement('span'); 6051 winTitle.className = 'title'; 6052 winTitle.textContent = mxResources.get('notifications'); 6053 winHeader.appendChild(winTitle); 6054 var winClose = document.createElement('span'); 6055 winClose.className = 'closeBtn'; 6056 winClose.textContent = 'x'; 6057 winHeader.appendChild(winClose); 6058 this.notificationWin.appendChild(winHeader); 6059 6060 var winBody = document.createElement('div'); 6061 winBody.className = 'notifications clearfix'; 6062 var notifList = document.createElement('div'); 6063 notifList.setAttribute('id', 'geNotifList'); 6064 notifList.style.position = 'relative'; 6065 winBody.appendChild(notifList); 6066 this.notificationWin.appendChild(winBody); 6067 6068 mxEvent.addListener(this.notificationBtn, 'click', mxUtils.bind(this, function() 6069 { 6070 if (this.notificationWin.style.display == 'none') 6071 { 6072 this.notificationWin.style.display = ''; 6073 document.querySelector('.notifications').scrollTop = 0; 6074 var r = this.notificationBtn.getBoundingClientRect(); 6075 this.notificationWin.style.top = (r.top + this.notificationBtn.clientHeight) + 'px'; 6076 this.notificationWin.style.left = (r.right - this.notificationWin.clientWidth) + 'px'; 6077 shouldAnimate(0); //Stop animation once notifications are open 6078 } 6079 else 6080 { 6081 markAllAsRead(); 6082 } 6083 })); 6084 6085 mxEvent.addListener(winClose, 'click', markAllAsRead); 6086 } 6087 else 6088 { 6089 this.notificationBtn.style.display = ''; //In case it was hidden 6090 } 6091 6092 var newNotif = 0; 6093 var notifListEl = document.getElementById('geNotifList'); 6094 6095 if (notifListEl == null) 6096 { 6097 return; //This shouldn't happen and no meaning of continuing 6098 } 6099 else 6100 { 6101 notifListEl.innerHTML = '<div class="line"></div>'; 6102 6103 for (var i = 0; i < notifs.length; i++) 6104 { 6105 (function(editorUi, notif) 6106 { 6107 if (notif.isNew) 6108 { 6109 newNotif++; 6110 } 6111 6112 var notifEl = document.createElement('div'); 6113 notifEl.className = 'notification'; 6114 var ts = new Date(notif.timestamp); 6115 var str = editorUi.timeSince(ts); 6116 6117 if (str == null) 6118 { 6119 str = mxResources.get('lessThanAMinute'); 6120 } 6121 6122 notifEl.innerHTML = '<div class="circle' + (notif.isNew? ' active' : '') + '"></div><span class="time">' + 6123 mxUtils.htmlEntities(mxResources.get('timeAgo', [str], '{1} ago')) + '</span>' + 6124 '<p>' + mxUtils.htmlEntities(notif.content) + '</p>'; 6125 if (notif.link) 6126 { 6127 mxEvent.addListener(notifEl, 'click', function() 6128 { 6129 window.open(notif.link, 'notifWin'); 6130 }); 6131 } 6132 6133 notifListEl.appendChild(notifEl); 6134 })(this, notifs[i]); 6135 } 6136 } 6137 6138 shouldAnimate(newNotif); 6139}; 6140 6141/** 6142 * Translates this point by the given vector. 6143 * 6144 * @param {number} dx X-coordinate of the translation. 6145 * @param {number} dy Y-coordinate of the translation. 6146 */ 6147App.prototype.save = function(name, done) 6148{ 6149 var file = this.getCurrentFile(); 6150 6151 if (file != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6152 { 6153 this.editor.setStatus(''); 6154 6155 if (this.editor.graph.isEditing()) 6156 { 6157 this.editor.graph.stopEditing(); 6158 } 6159 6160 var success = mxUtils.bind(this, function() 6161 { 6162 file.handleFileSuccess(true); 6163 6164 if (done != null) 6165 { 6166 done(); 6167 } 6168 }); 6169 6170 var error = mxUtils.bind(this, function(err) 6171 { 6172 if (file.isModified()) 6173 { 6174 Editor.addRetryToError(err, mxUtils.bind(this, function() 6175 { 6176 this.save(name, done); 6177 })); 6178 } 6179 6180 file.handleFileError(err, true); 6181 }); 6182 6183 try 6184 { 6185 if (name == file.getTitle()) 6186 { 6187 file.save(true, success, error); 6188 } 6189 else 6190 { 6191 file.saveAs(name, success, error) 6192 } 6193 } 6194 catch (err) 6195 { 6196 error(err); 6197 } 6198 } 6199}; 6200 6201/** 6202 * Invokes callback with null if mode does not support folder or not null 6203 * if a valid folder was chosen for a mode that supports it. No callback 6204 * is made if no folder was chosen for a mode that supports it. 6205 */ 6206App.prototype.pickFolder = function(mode, fn, enabled, direct, force) 6207{ 6208 enabled = (enabled != null) ? enabled : true; 6209 var resume = this.spinner.pause(); 6210 6211 if (enabled && mode == App.MODE_GOOGLE && this.drive != null) 6212 { 6213 // Shows a save dialog 6214 this.drive.pickFolder(mxUtils.bind(this, function(evt) 6215 { 6216 resume(); 6217 6218 if (evt.action == google.picker.Action.PICKED) 6219 { 6220 var folderId = null; 6221 6222 if (evt.docs != null && evt.docs.length > 0 && evt.docs[0].type == 'folder') 6223 { 6224 folderId = evt.docs[0].id; 6225 } 6226 6227 fn(folderId); 6228 } 6229 }), force); 6230 } 6231 else if (enabled && mode == App.MODE_ONEDRIVE && this.oneDrive != null) 6232 { 6233 this.oneDrive.pickFolder(mxUtils.bind(this, function(files) 6234 { 6235 var folderId = null; 6236 resume(); 6237 6238 if (files != null && files.value != null && files.value.length > 0) 6239 { 6240 folderId = OneDriveFile.prototype.getIdOf(files.value[0]); 6241 fn(folderId); 6242 } 6243 }), direct); 6244 } 6245 else if (enabled && mode == App.MODE_GITHUB && this.gitHub != null) 6246 { 6247 this.gitHub.pickFolder(mxUtils.bind(this, function(folderPath) 6248 { 6249 resume(); 6250 fn(folderPath); 6251 })); 6252 } 6253 else if (enabled && mode == App.MODE_GITLAB && this.gitLab != null) 6254 { 6255 this.gitLab.pickFolder(mxUtils.bind(this, function(folderPath) 6256 { 6257 resume(); 6258 fn(folderPath); 6259 })); 6260 } 6261 else if (enabled && mode == App.MODE_NOTION && this.notion != null) 6262 { 6263 this.notion.pickFolder(mxUtils.bind(this, function(folderPath) 6264 { 6265 resume(); 6266 fn(folderPath); 6267 })); 6268 } 6269 else if (enabled && mode == App.MODE_TRELLO && this.trello != null) 6270 { 6271 this.trello.pickFolder(mxUtils.bind(this, function(cardId) 6272 { 6273 resume(); 6274 fn(cardId); 6275 })); 6276 } 6277 else 6278 { 6279 EditorUi.prototype.pickFolder.apply(this, arguments); 6280 } 6281}; 6282 6283/** 6284 * 6285 */ 6286App.prototype.exportFile = function(data, filename, mimeType, base64Encoded, mode, folderId) 6287{ 6288 if (mode == App.MODE_DROPBOX) 6289 { 6290 if (this.dropbox != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6291 { 6292 // LATER: Add folder picker 6293 this.dropbox.insertFile(filename, (base64Encoded) ? this.base64ToBlob(data, mimeType) : 6294 data, mxUtils.bind(this, function() 6295 { 6296 this.spinner.stop(); 6297 }), mxUtils.bind(this, function(resp) 6298 { 6299 this.spinner.stop(); 6300 this.handleError(resp); 6301 })); 6302 } 6303 } 6304 else if (mode == App.MODE_GOOGLE) 6305 { 6306 if (this.drive != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6307 { 6308 this.drive.insertFile(filename, data, folderId, mxUtils.bind(this, function(resp) 6309 { 6310 // TODO: Add callback with url param for clickable status message 6311 // "File exported. Click here to open folder." 6312// this.editor.setStatus('<div class="geStatusMessage">' + 6313// mxResources.get('saved') + '</div>'); 6314// 6315// // Installs click handler for opening 6316// if (this.statusContainer != null) 6317// { 6318// var links = this.statusContainer.getElementsByTagName('div'); 6319// 6320// if (links.length > 0) 6321// { 6322// links[0].style.cursor = 'pointer'; 6323// 6324// mxEvent.addListener(links[0], 'click', mxUtils.bind(this, function() 6325// { 6326// if (resp != null && resp.id != null) 6327// { 6328// window.open('https://drive.google.com/open?id=' + resp.id); 6329// } 6330// })); 6331// } 6332// } 6333 6334 this.spinner.stop(); 6335 }), mxUtils.bind(this, function(resp) 6336 { 6337 this.spinner.stop(); 6338 this.handleError(resp); 6339 }), mimeType, base64Encoded); 6340 } 6341 } 6342 else if (mode == App.MODE_ONEDRIVE) 6343 { 6344 if (this.oneDrive != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6345 { 6346 // KNOWN: OneDrive does not show .svg extension 6347 this.oneDrive.insertFile(filename, (base64Encoded) ? this.base64ToBlob(data, mimeType) : 6348 data, mxUtils.bind(this, function() 6349 { 6350 this.spinner.stop(); 6351 }), mxUtils.bind(this, function(resp) 6352 { 6353 this.spinner.stop(); 6354 this.handleError(resp); 6355 }), false, folderId); 6356 } 6357 } 6358 else if (mode == App.MODE_GITHUB) 6359 { 6360 if (this.gitHub != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6361 { 6362 // Must insert file as library to force the file to be written 6363 this.gitHub.insertFile(filename, data, mxUtils.bind(this, function() 6364 { 6365 this.spinner.stop(); 6366 }), mxUtils.bind(this, function(resp) 6367 { 6368 this.spinner.stop(); 6369 this.handleError(resp); 6370 }), true, folderId, base64Encoded); 6371 } 6372 } 6373 else if (mode == App.MODE_GITLAB) 6374 { 6375 if (this.gitHub != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6376 { 6377 // Must insert file as library to force the file to be written 6378 this.gitLab.insertFile(filename, data, mxUtils.bind(this, function() 6379 { 6380 this.spinner.stop(); 6381 }), mxUtils.bind(this, function(resp) 6382 { 6383 this.spinner.stop(); 6384 this.handleError(resp); 6385 }), true, folderId, base64Encoded); 6386 } 6387 } 6388 else if (mode == App.MODE_TRELLO) 6389 { 6390 if (this.trello != null && this.spinner.spin(document.body, mxResources.get('saving'))) 6391 { 6392 this.trello.insertFile(filename, (base64Encoded) ? this.base64ToBlob(data, mimeType) : 6393 data, mxUtils.bind(this, function() 6394 { 6395 this.spinner.stop(); 6396 }), mxUtils.bind(this, function(resp) 6397 { 6398 this.spinner.stop(); 6399 this.handleError(resp); 6400 }), false, folderId); 6401 } 6402 } 6403 else if (mode == App.MODE_BROWSER) 6404 { 6405 var fn = mxUtils.bind(this, function() 6406 { 6407 localStorage.setItem(filename, data); 6408 }); 6409 6410 if (localStorage.getItem(filename) == null) 6411 { 6412 fn(); 6413 } 6414 else 6415 { 6416 this.confirm(mxResources.get('replaceIt', [filename]), fn); 6417 } 6418 } 6419}; 6420 6421/** 6422 * Translates this point by the given vector. 6423 * 6424 * @param {number} dx X-coordinate of the translation. 6425 * @param {number} dy Y-coordinate of the translation. 6426 */ 6427App.prototype.descriptorChanged = function() 6428{ 6429 var file = this.getCurrentFile(); 6430 6431 if (file != null) 6432 { 6433 if (this.fname != null) 6434 { 6435 this.fnameWrapper.style.display = 'block'; 6436 this.fname.innerHTML = ''; 6437 var filename = (file.getTitle() != null) ? file.getTitle() : this.defaultFilename; 6438 mxUtils.write(this.fname, filename); 6439 this.fname.setAttribute('title', filename + ' - ' + mxResources.get('rename')); 6440 } 6441 6442 var graph = this.editor.graph; 6443 var editable = file.isEditable() && !file.invalidChecksum; 6444 6445 if (graph.isEnabled() && !editable) 6446 { 6447 graph.reset(); 6448 } 6449 6450 graph.setEnabled(editable); 6451 6452 // Ignores title and hash for revisions 6453 if (urlParams['rev'] == null) 6454 { 6455 this.updateDocumentTitle(); 6456 var newHash = file.getHash(); 6457 6458 if (newHash.length > 0) 6459 { 6460 window.location.hash = newHash; 6461 } 6462 else if (window.location.hash.length > 0) 6463 { 6464 window.location.hash = ''; 6465 } 6466 } 6467 } 6468 6469 this.updateUi(); 6470 6471 // Refresh if editable state has changed 6472 if (this.format != null && (file == null || 6473 this.fileEditable != file.isEditable()) && 6474 this.editor.graph.isSelectionEmpty()) 6475 { 6476 this.format.refresh(); 6477 this.fileEditable = (file != null) ? file.isEditable() : null; 6478 } 6479 6480 this.fireEvent(new mxEventObject('fileDescriptorChanged', 'file', file)); 6481}; 6482 6483/** 6484 * Adds the listener for automatically saving the diagram for local changes. 6485 */ 6486App.prototype.showAuthDialog = function(peer, showRememberOption, fn, closeFn) 6487{ 6488 var resume = this.spinner.pause(); 6489 6490 this.showDialog(new AuthDialog(this, peer, showRememberOption, mxUtils.bind(this, function(remember) 6491 { 6492 try 6493 { 6494 if (fn != null) 6495 { 6496 fn(remember, mxUtils.bind(this, function() 6497 { 6498 this.hideDialog(); 6499 resume(); 6500 })); 6501 } 6502 } 6503 catch (e) 6504 { 6505 this.editor.setStatus(mxUtils.htmlEntities(e.message)); 6506 } 6507 })).container, 300, (showRememberOption) ? 180 : 140, true, true, mxUtils.bind(this, function(cancel) 6508 { 6509 if (closeFn != null) 6510 { 6511 closeFn(cancel); 6512 } 6513 6514 if (cancel && this.getCurrentFile() == null && this.dialog == null) 6515 { 6516 this.showSplash(); 6517 } 6518 })); 6519}; 6520 6521/** 6522 * Checks if the client is authorized and calls the next step. The optional 6523 * readXml argument is used for import. Default is false. The optional 6524 * readLibrary argument is used for reading libraries. Default is false. 6525 */ 6526App.prototype.convertFile = function(url, filename, mimeType, extension, success, error, executeRequest, headers) 6527{ 6528 var name = filename; 6529 6530 // SVG file extensions are valid and needed for image import 6531 if (!/\.svg$/i.test(name)) 6532 { 6533 name = name.substring(0, filename.lastIndexOf('.')) + extension; 6534 } 6535 6536 var gitHubUrl = false; 6537 6538 if (this.gitHub != null && url.substring(0, this.gitHub.baseUrl.length) == this.gitHub.baseUrl) 6539 { 6540 gitHubUrl = true; 6541 } 6542 6543 // Workaround for wrong binary response with VSD(X) & VDX files 6544 if (/\.v(dx|sdx?)$/i.test(filename) && Graph.fileSupport && new XMLHttpRequest().upload && 6545 typeof new XMLHttpRequest().responseType === 'string') 6546 { 6547 var req = new XMLHttpRequest(); 6548 req.open('GET', url, true); 6549 6550 if (!gitHubUrl) 6551 { 6552 req.responseType = 'blob'; 6553 } 6554 6555 if (headers) 6556 { 6557 for (var key in headers) 6558 { 6559 req.setRequestHeader(key, headers[key]); 6560 } 6561 } 6562 6563 req.onload = mxUtils.bind(this, function() 6564 { 6565 if (req.status >= 200 && req.status <= 299) 6566 { 6567 var blob = null; 6568 6569 if (gitHubUrl) 6570 { 6571 var file = JSON.parse(req.responseText); 6572 blob = this.base64ToBlob(file.content, 'application/octet-stream'); 6573 } 6574 else 6575 { 6576 blob = new Blob([req.response], {type: 'application/octet-stream'}); 6577 } 6578 6579 this.importVisio(blob, mxUtils.bind(this, function(xml) 6580 { 6581 success(new LocalFile(this, xml, name, true)); 6582 }), error, filename) 6583 } 6584 else if (error != null) 6585 { 6586 error({message: mxResources.get('errorLoadingFile')}); 6587 } 6588 }); 6589 6590 req.onerror = error; 6591 req.send(); 6592 } 6593 else 6594 { 6595 var handleData = mxUtils.bind(this, function(data) 6596 { 6597 try 6598 { 6599 if (/\.pdf$/i.test(filename)) 6600 { 6601 var temp = Editor.extractGraphModelFromPdf(data); 6602 6603 if (temp != null && temp.length > 0) 6604 { 6605 success(new LocalFile(this, temp, name, true)); 6606 } 6607 } 6608 else if (/\.png$/i.test(filename)) 6609 { 6610 var temp = this.extractGraphModelFromPng(data); 6611 6612 if (temp != null) 6613 { 6614 success(new LocalFile(this, temp, name, true)); 6615 } 6616 else 6617 { 6618 success(new LocalFile(this, data, filename, true)); 6619 } 6620 } 6621 else if (Graph.fileSupport && new XMLHttpRequest().upload && this.isRemoteFileFormat(data, url)) 6622 { 6623 this.parseFile(new Blob([data], {type: 'application/octet-stream'}), mxUtils.bind(this, function(xhr) 6624 { 6625 if (xhr.readyState == 4) 6626 { 6627 if (xhr.status >= 200 && xhr.status <= 299) 6628 { 6629 success(new LocalFile(this, xhr.responseText, name, true)); 6630 } 6631 else if (error != null) 6632 { 6633 error({message: mxResources.get('errorLoadingFile')}); 6634 } 6635 } 6636 }), filename); 6637 } 6638 else 6639 { 6640 success(new LocalFile(this, data, name, true)); 6641 } 6642 } 6643 catch (e) 6644 { 6645 if (error != null) 6646 { 6647 error(e); 6648 } 6649 } 6650 }); 6651 6652 var binary = /\.png$/i.test(filename) || /\.jpe?g$/i.test(filename) || 6653 /\.pdf$/i.test(filename) || (mimeType != null && 6654 mimeType.substring(0, 6) == 'image/'); 6655 6656 // NOTE: Cannot force non-binary request via loadUrl so needs separate 6657 // code as decoding twice on content with binary data did not work 6658 if (gitHubUrl) 6659 { 6660 mxUtils.get(url, mxUtils.bind(this, function(req) 6661 { 6662 if (req.getStatus() >= 200 && req.getStatus() <= 299) 6663 { 6664 if (success != null) 6665 { 6666 var file = JSON.parse(req.getText()); 6667 var data = file.content; 6668 6669 if (file.encoding === 'base64') 6670 { 6671 if (/\.png$/i.test(filename)) 6672 { 6673 data = 'data:image/png;base64,' + data; 6674 } 6675 else if (/\.pdf$/i.test(filename)) 6676 { 6677 data = 'data:application/pdf;base64,' + data; 6678 } 6679 else 6680 { 6681 // Workaround for character encoding issues in IE10/11 6682 data = (window.atob && !mxClient.IS_IE && !mxClient.IS_IE11) ? atob(data) : Base64.decode(data); 6683 } 6684 } 6685 6686 handleData(data); 6687 } 6688 } 6689 else if (error != null) 6690 { 6691 error({code: App.ERROR_UNKNOWN}); 6692 } 6693 }), function() 6694 { 6695 if (error != null) 6696 { 6697 error({code: App.ERROR_UNKNOWN}); 6698 } 6699 }, false, this.timeout, function() 6700 { 6701 if (error != null) 6702 { 6703 error({code: App.ERROR_TIMEOUT, retry: fn}); 6704 } 6705 }, headers); 6706 } 6707 else if (executeRequest != null) 6708 { 6709 executeRequest(url, handleData, error, binary); 6710 } 6711 else 6712 { 6713 this.editor.loadUrl(url, handleData, error, binary, null, null, null, headers); 6714 } 6715 } 6716}; 6717 6718/** 6719 * Adds the listener for automatically saving the diagram for local changes. 6720 */ 6721App.prototype.updateHeader = function() 6722{ 6723 if (this.menubar != null) 6724 { 6725 this.appIcon = document.createElement('a'); 6726 this.appIcon.style.display = 'block'; 6727 this.appIcon.style.position = 'absolute'; 6728 this.appIcon.style.width = '32px'; 6729 this.appIcon.style.height = (this.menubarHeight - 28) + 'px'; 6730 this.appIcon.style.margin = '14px 0px 8px 16px'; 6731 this.appIcon.style.opacity = '0.85'; 6732 this.appIcon.style.borderRadius = '3px'; 6733 6734 if (uiTheme != 'dark') 6735 { 6736 this.appIcon.style.backgroundColor = '#f08705'; 6737 } 6738 6739 mxEvent.disableContextMenu(this.appIcon); 6740 6741 mxEvent.addListener(this.appIcon, 'click', mxUtils.bind(this, function(evt) 6742 { 6743 this.appIconClicked(evt); 6744 })); 6745 6746 // LATER: Use Alpha image loader in IE6 6747 // NOTE: This uses the diagram bit of the old logo as it looks better in this case 6748 //this.appIcon.style.filter = 'progid:DXImageTransform.Microsoft.AlphaImageLoader(src=' + IMAGE_PATH + '/logo-white.png,sizingMethod=\'scale\')'; 6749 var logo = (!mxClient.IS_SVG) ? 'url(\'' + IMAGE_PATH + '/logo-white.png\')' : 6750 ((uiTheme == 'dark') ? 'url()' : 6751 'url()'); 6752 this.appIcon.style.backgroundImage = logo; 6753 this.appIcon.style.backgroundPosition = 'center center'; 6754 this.appIcon.style.backgroundSize = '100% 100%'; 6755 this.appIcon.style.backgroundRepeat = 'no-repeat'; 6756 6757 mxUtils.setPrefixedStyle(this.appIcon.style, 'transition', 'all 125ms linear'); 6758 6759 mxEvent.addListener(this.appIcon, 'mouseover', mxUtils.bind(this, function() 6760 { 6761 var file = this.getCurrentFile(); 6762 6763 if (file != null) 6764 { 6765 var mode = file.getMode(); 6766 6767 if (mode == App.MODE_GOOGLE) 6768 { 6769 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/google-drive-logo-white.svg)'; 6770 this.appIcon.style.backgroundSize = '70% 70%'; 6771 } 6772 else if (mode == App.MODE_DROPBOX) 6773 { 6774 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/dropbox-logo-white.svg)'; 6775 this.appIcon.style.backgroundSize = '70% 70%'; 6776 } 6777 else if (mode == App.MODE_ONEDRIVE) 6778 { 6779 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/onedrive-logo-white.svg)'; 6780 this.appIcon.style.backgroundSize = '70% 70%'; 6781 } 6782 else if (mode == App.MODE_GITHUB) 6783 { 6784 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/github-logo-white.svg)'; 6785 this.appIcon.style.backgroundSize = '70% 70%'; 6786 } 6787 else if (mode == App.MODE_GITLAB) 6788 { 6789 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/gitlab-logo-white.svg)'; 6790 this.appIcon.style.backgroundSize = '100% 100%'; 6791 } 6792 else if (mode == App.MODE_NOTION) 6793 { 6794 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/notion-logo-white.svg)'; 6795 this.appIcon.style.backgroundSize = '70% 70%'; 6796 } 6797 else if (mode == App.MODE_TRELLO) 6798 { 6799 this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/trello-logo-white-orange.svg)'; 6800 this.appIcon.style.backgroundSize = '70% 70%'; 6801 } 6802 } 6803 })); 6804 6805 mxEvent.addListener(this.appIcon, 'mouseout', mxUtils.bind(this, function() 6806 { 6807 this.appIcon.style.backgroundImage = logo; 6808 this.appIcon.style.backgroundSize = '90% 90%'; 6809 })); 6810 6811 if (urlParams['embed'] != '1') 6812 { 6813 this.menubarContainer.appendChild(this.appIcon); 6814 } 6815 6816 this.fnameWrapper = document.createElement('div'); 6817 this.fnameWrapper.style.position = 'absolute'; 6818 this.fnameWrapper.style.right = '120px'; 6819 this.fnameWrapper.style.left = '60px'; 6820 this.fnameWrapper.style.top = '9px'; 6821 this.fnameWrapper.style.height = '26px'; 6822 this.fnameWrapper.style.display = 'none'; 6823 this.fnameWrapper.style.overflow = 'hidden'; 6824 this.fnameWrapper.style.textOverflow = 'ellipsis'; 6825 6826 this.fname = document.createElement('a'); 6827 this.fname.setAttribute('title', mxResources.get('rename')); 6828 this.fname.className = 'geItem'; 6829 this.fname.style.padding = '2px 8px 2px 8px'; 6830 this.fname.style.display = 'inline'; 6831 this.fname.style.fontSize = '18px'; 6832 this.fname.style.whiteSpace = 'nowrap'; 6833 6834 // Prevents focus 6835 mxEvent.addListener(this.fname, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', 6836 mxUtils.bind(this, function(evt) 6837 { 6838 evt.preventDefault(); 6839 })); 6840 6841 mxEvent.addListener(this.fname, 'click', mxUtils.bind(this, function(evt) 6842 { 6843 var file = this.getCurrentFile(); 6844 6845 if (file != null && file.isRenamable()) 6846 { 6847 if (this.editor.graph.isEditing()) 6848 { 6849 this.editor.graph.stopEditing(); 6850 } 6851 6852 this.actions.get('rename').funct(); 6853 } 6854 6855 mxEvent.consume(evt); 6856 })); 6857 6858 this.fnameWrapper.appendChild(this.fname); 6859 6860 if (urlParams['embed'] != '1') 6861 { 6862 this.menubarContainer.appendChild(this.fnameWrapper); 6863 6864 this.menubar.container.style.position = 'absolute'; 6865 this.menubar.container.style.paddingLeft = '59px'; 6866 this.toolbar.container.style.paddingLeft = '16px'; 6867 this.menubar.container.style.boxSizing = 'border-box'; 6868 this.menubar.container.style.top = '34px'; 6869 } 6870 6871 /** 6872 * Adds format panel toggle. 6873 */ 6874 this.toggleFormatElement = document.createElement('a'); 6875 this.toggleFormatElement.setAttribute('title', mxResources.get('formatPanel') + ' (' + Editor.ctrlKey + '+Shift+P)'); 6876 this.toggleFormatElement.style.position = 'absolute'; 6877 this.toggleFormatElement.style.display = 'inline-block'; 6878 this.toggleFormatElement.style.top = (uiTheme == 'atlas') ? '8px' : '6px'; 6879 this.toggleFormatElement.style.right = (uiTheme != 'atlas' && urlParams['embed'] != '1') ? '30px' : '10px'; 6880 this.toggleFormatElement.style.padding = '2px'; 6881 this.toggleFormatElement.style.fontSize = '14px'; 6882 this.toggleFormatElement.className = (uiTheme != 'atlas') ? 'geButton' : ''; 6883 this.toggleFormatElement.style.width = '16px'; 6884 this.toggleFormatElement.style.height = '16px'; 6885 this.toggleFormatElement.style.backgroundPosition = '50% 50%'; 6886 this.toggleFormatElement.style.backgroundRepeat = 'no-repeat'; 6887 this.toolbarContainer.appendChild(this.toggleFormatElement); 6888 6889 if (uiTheme == 'dark') 6890 { 6891 this.toggleFormatElement.style.filter = 'invert(100%)'; 6892 } 6893 6894 // Prevents focus 6895 mxEvent.addListener(this.toggleFormatElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', 6896 mxUtils.bind(this, function(evt) 6897 { 6898 evt.preventDefault(); 6899 })); 6900 6901 mxEvent.addListener(this.toggleFormatElement, 'click', mxUtils.bind(this, function(evt) 6902 { 6903 this.actions.get('formatPanel').funct(); 6904 mxEvent.consume(evt); 6905 })); 6906 6907 var toggleFormatPanel = mxUtils.bind(this, function() 6908 { 6909 if (this.formatWidth > 0) 6910 { 6911 this.toggleFormatElement.style.backgroundImage = 'url(\'' + this.formatShowImage + '\')'; 6912 } 6913 else 6914 { 6915 this.toggleFormatElement.style.backgroundImage = 'url(\'' + this.formatHideImage + '\')'; 6916 } 6917 }); 6918 6919 this.addListener('formatWidthChanged', toggleFormatPanel); 6920 toggleFormatPanel(); 6921 6922 this.fullscreenElement = document.createElement('a'); 6923 this.fullscreenElement.setAttribute('title', mxResources.get('fullscreen')); 6924 this.fullscreenElement.style.position = 'absolute'; 6925 this.fullscreenElement.style.display = 'inline-block'; 6926 this.fullscreenElement.style.top = (uiTheme == 'atlas') ? '8px' : '6px'; 6927 this.fullscreenElement.style.right = (uiTheme != 'atlas' && urlParams['embed'] != '1') ? '50px' : '30px'; 6928 this.fullscreenElement.style.padding = '2px'; 6929 this.fullscreenElement.style.fontSize = '14px'; 6930 this.fullscreenElement.className = (uiTheme != 'atlas') ? 'geButton' : ''; 6931 this.fullscreenElement.style.width = '16px'; 6932 this.fullscreenElement.style.height = '16px'; 6933 this.fullscreenElement.style.backgroundPosition = '50% 50%'; 6934 this.fullscreenElement.style.backgroundRepeat = 'no-repeat'; 6935 this.fullscreenElement.style.backgroundImage = 'url(\'' + this.fullscreenImage + '\')'; 6936 this.toolbarContainer.appendChild(this.fullscreenElement); 6937 6938 // Prevents focus 6939 mxEvent.addListener(this.fullscreenElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', 6940 mxUtils.bind(this, function(evt) 6941 { 6942 evt.preventDefault(); 6943 })); 6944 6945 // Some style changes in Atlas theme 6946 if (uiTheme == 'atlas') 6947 { 6948 mxUtils.setOpacity(this.toggleFormatElement, 70); 6949 mxUtils.setOpacity(this.fullscreenElement, 70); 6950 } 6951 6952 var initialPosition = this.hsplitPosition; 6953 6954 if (uiTheme == 'dark') 6955 { 6956 this.fullscreenElement.style.filter = 'invert(100%)'; 6957 } 6958 6959 mxEvent.addListener(this.fullscreenElement, 'click', mxUtils.bind(this, function(evt) 6960 { 6961 var visible = this.fullscreenMode; 6962 6963 if (uiTheme != 'atlas' && urlParams['embed'] != '1') 6964 { 6965 this.toggleCompactMode(visible); 6966 } 6967 6968 if (!visible) 6969 { 6970 initialPosition = this.hsplitPosition; 6971 } 6972 6973 this.hsplitPosition = (visible) ? initialPosition : 0; 6974 this.toggleFormatPanel(visible); 6975 this.fullscreenMode = !visible; 6976 mxEvent.consume(evt); 6977 })); 6978 6979 /** 6980 * Adds compact UI toggle. 6981 */ 6982 if (urlParams['embed'] != '1') 6983 { 6984 this.toggleElement = document.createElement('a'); 6985 this.toggleElement.setAttribute('title', mxResources.get('collapseExpand')); 6986 this.toggleElement.className = 'geButton'; 6987 this.toggleElement.style.position = 'absolute'; 6988 this.toggleElement.style.display = 'inline-block'; 6989 this.toggleElement.style.width = '16px'; 6990 this.toggleElement.style.height = '16px'; 6991 this.toggleElement.style.color = '#666'; 6992 this.toggleElement.style.top = (uiTheme == 'atlas') ? '8px' : '6px'; 6993 this.toggleElement.style.right = '10px'; 6994 this.toggleElement.style.padding = '2px'; 6995 this.toggleElement.style.fontSize = '14px'; 6996 this.toggleElement.style.textDecoration = 'none'; 6997 this.toggleElement.style.backgroundImage = 'url(\'' + this.chevronUpImage + '\')'; 6998 6999 this.toggleElement.style.backgroundPosition = '50% 50%'; 7000 this.toggleElement.style.backgroundRepeat = 'no-repeat'; 7001 7002 if (uiTheme == 'dark') 7003 { 7004 this.toggleElement.style.filter = 'invert(100%)'; 7005 } 7006 7007 // Prevents focus 7008 mxEvent.addListener(this.toggleElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', 7009 mxUtils.bind(this, function(evt) 7010 { 7011 evt.preventDefault(); 7012 })); 7013 7014 // Toggles compact mode 7015 mxEvent.addListener(this.toggleElement, 'click', mxUtils.bind(this, function(evt) 7016 { 7017 this.toggleCompactMode(); 7018 mxEvent.consume(evt); 7019 })); 7020 7021 if (uiTheme != 'atlas') 7022 { 7023 this.toolbarContainer.appendChild(this.toggleElement); 7024 } 7025 7026 // Enable compact mode for small screens except for Firefox where the height is wrong 7027 if (!mxClient.IS_FF && screen.height <= 740 && typeof this.toggleElement.click !== 'undefined') 7028 { 7029 window.setTimeout(mxUtils.bind(this, function() 7030 { 7031 this.toggleElement.click(); 7032 }), 0); 7033 } 7034 } 7035 } 7036}; 7037 7038/** 7039 * Adds the listener for automatically saving the diagram for local changes. 7040 */ 7041App.prototype.toggleCompactMode = function(visible) 7042{ 7043 visible = (visible != null) ? visible : this.compactMode; 7044 7045 if (visible) 7046 { 7047 this.menubar.container.style.position = 'absolute'; 7048 this.menubar.container.style.paddingLeft = '59px'; 7049 this.menubar.container.style.paddingTop = ''; 7050 this.menubar.container.style.paddingBottom = ''; 7051 this.menubar.container.style.top = '34px'; 7052 this.toolbar.container.style.paddingLeft = '16px'; 7053 this.buttonContainer.style.visibility = 'visible'; 7054 this.appIcon.style.display = 'block'; 7055 this.fnameWrapper.style.display = 'block'; 7056 this.fnameWrapper.style.visibility = 'visible'; 7057 this.menubarHeight = App.prototype.menubarHeight; 7058 this.refresh(); 7059 this.toggleElement.style.backgroundImage = 'url(\'' + this.chevronUpImage + '\')'; 7060 } 7061 else 7062 { 7063 this.menubar.container.style.position = 'relative'; 7064 this.menubar.container.style.paddingLeft = '4px'; 7065 this.menubar.container.style.paddingTop = '0px'; 7066 this.menubar.container.style.paddingBottom = '0px'; 7067 this.menubar.container.style.top = '0px'; 7068 this.toolbar.container.style.paddingLeft = '8px'; 7069 this.buttonContainer.style.visibility = 'hidden'; 7070 this.appIcon.style.display = 'none'; 7071 this.fnameWrapper.style.display = 'none'; 7072 this.fnameWrapper.style.visibility = 'hidden'; 7073 this.menubarHeight = EditorUi.prototype.menubarHeight; 7074 this.refresh(); 7075 this.toggleElement.style.backgroundImage = 'url(\'' + this.chevronDownImage + '\')'; 7076 } 7077 7078 this.compactMode = !visible; 7079}; 7080 7081/** 7082 * Adds the listener for automatically saving the diagram for local changes. 7083 */ 7084App.prototype.updateUserElement = function() 7085{ 7086 if ((this.drive == null || this.drive.getUser() == null) && 7087 (this.oneDrive == null || this.oneDrive.getUser() == null) && 7088 (this.dropbox == null || this.dropbox.getUser() == null) && 7089 (this.gitHub == null || this.gitHub.getUser() == null) && 7090 (this.gitLab == null || this.gitLab.getUser() == null) && 7091 (this.notion == null || this.notion.getUser() == null) && 7092 (this.trello == null || !this.trello.isAuthorized())) //TODO Trello no user issue 7093 { 7094 if (this.userElement != null) 7095 { 7096 this.userElement.parentNode.removeChild(this.userElement); 7097 this.userElement = null; 7098 } 7099 } 7100 else 7101 { 7102 if (this.userElement == null) 7103 { 7104 this.userElement = document.createElement('a'); 7105 this.userElement.className = 'geItem'; 7106 this.userElement.style.position = 'absolute'; 7107 this.userElement.style.fontSize = '8pt'; 7108 this.userElement.style.top = (uiTheme == 'atlas') ? '8px' : '2px'; 7109 this.userElement.style.right = '30px'; 7110 this.userElement.style.margin = '4px'; 7111 this.userElement.style.padding = '2px'; 7112 this.userElement.style.paddingRight = '16px'; 7113 this.userElement.style.verticalAlign = 'middle'; 7114 this.userElement.style.backgroundImage = 'url(' + IMAGE_PATH + '/expanded.gif)'; 7115 this.userElement.style.backgroundPosition = '100% 60%'; 7116 this.userElement.style.backgroundRepeat = 'no-repeat'; 7117 7118 this.menubarContainer.appendChild(this.userElement); 7119 7120 // Prevents focus 7121 mxEvent.addListener(this.userElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', 7122 mxUtils.bind(this, function(evt) 7123 { 7124 evt.preventDefault(); 7125 })); 7126 7127 mxEvent.addListener(this.userElement, 'click', mxUtils.bind(this, function(evt) 7128 { 7129 if (this.userPanel == null) 7130 { 7131 var div = document.createElement('div'); 7132 div.className = 'geDialog'; 7133 div.style.position = 'absolute'; 7134 div.style.top = (this.userElement.clientTop + 7135 this.userElement.clientHeight + 6) + 'px'; 7136 div.style.zIndex = 5; 7137 div.style.right = '36px'; 7138 div.style.padding = '0px'; 7139 div.style.cursor = 'default'; 7140 div.style.minWidth = '300px'; 7141 7142 this.userPanel = div; 7143 } 7144 7145 if (this.userPanel.parentNode != null) 7146 { 7147 this.userPanel.parentNode.removeChild(this.userPanel); 7148 } 7149 else 7150 { 7151 var connected = false; 7152 this.userPanel.innerHTML = ''; 7153 7154 var img = document.createElement('img'); 7155 7156 img.setAttribute('src', Dialog.prototype.closeImage); 7157 img.setAttribute('title', mxResources.get('close')); 7158 img.className = 'geDialogClose'; 7159 img.style.top = '8px'; 7160 img.style.right = '8px'; 7161 7162 mxEvent.addListener(img, 'click', mxUtils.bind(this, function() 7163 { 7164 if (this.userPanel.parentNode != null) 7165 { 7166 this.userPanel.parentNode.removeChild(this.userPanel); 7167 } 7168 })); 7169 7170 this.userPanel.appendChild(img); 7171 7172 if (this.drive != null) 7173 { 7174 var driveUsers = this.drive.getUsersList(); 7175 7176 if (driveUsers.length > 0) 7177 { 7178 // LATER: Cannot change user while file is open since close will not work with new 7179 // credentials and closing the file using fileLoaded(null) will show splash dialog. 7180 var closeFile = mxUtils.bind(this, function(callback, spinnerMsg) 7181 { 7182 var file = this.getCurrentFile(); 7183 7184 if (file != null && file.constructor == DriveFile) 7185 { 7186 this.spinner.spin(document.body, spinnerMsg); 7187 7188// file.close(); 7189 this.fileLoaded(null); 7190 7191 // LATER: Use callback to wait for thumbnail update 7192 window.setTimeout(mxUtils.bind(this, function() 7193 { 7194 this.spinner.stop(); 7195 callback(); 7196 }), 2000); 7197 } 7198 else 7199 { 7200 callback(); 7201 } 7202 }); 7203 7204 var createUserRow = mxUtils.bind(this, function (user) 7205 { 7206 var tr = document.createElement('tr'); 7207 tr.setAttribute('title', 'User ID: ' + user.id); 7208 7209 var td = document.createElement('td'); 7210 td.setAttribute('valig', 'middle'); 7211 td.style.height = '59px'; 7212 td.style.width = '66px'; 7213 7214 var img = document.createElement('img'); 7215 img.setAttribute('width', '50'); 7216 img.setAttribute('height', '50'); 7217 img.setAttribute('border', '0'); 7218 img.setAttribute('src', (user.pictureUrl != null) ? user.pictureUrl : this.defaultUserPicture); 7219 img.style.borderRadius = '50%'; 7220 img.style.margin = '4px 8px 0 8px'; 7221 td.appendChild(img); 7222 tr.appendChild(td); 7223 7224 var td = document.createElement('td'); 7225 td.setAttribute('valign', 'middle'); 7226 td.style.whiteSpace = 'nowrap'; 7227 td.style.paddingTop = '4px'; 7228 td.style.maxWidth = '0'; 7229 td.style.overflow = 'hidden'; 7230 td.style.textOverflow = 'ellipsis'; 7231 mxUtils.write(td, user.displayName + 7232 ((user.isCurrent && driveUsers.length > 1) ? 7233 ' (' + mxResources.get('default') + ')' : '')); 7234 7235 if (user.email != null) 7236 { 7237 mxUtils.br(td); 7238 7239 var small = document.createElement('small'); 7240 small.style.color = 'gray'; 7241 mxUtils.write(small, user.email); 7242 td.appendChild(small); 7243 } 7244 7245 var div = document.createElement('div'); 7246 div.style.marginTop = '4px'; 7247 7248 var i = document.createElement('i'); 7249 mxUtils.write(i, mxResources.get('googleDrive')); 7250 div.appendChild(i); 7251 td.appendChild(div); 7252 tr.appendChild(td); 7253 7254 if (!user.isCurrent) 7255 { 7256 tr.style.cursor = 'pointer'; 7257 tr.style.opacity = '0.3'; 7258 7259 mxEvent.addListener(tr, 'click', mxUtils.bind(this, function(evt) 7260 { 7261 closeFile(mxUtils.bind(this, function() 7262 { 7263 this.stateArg = null; 7264 this.drive.setUser(user); 7265 7266 this.drive.authorize(true, mxUtils.bind(this, function() 7267 { 7268 this.setMode(App.MODE_GOOGLE); 7269 this.hideDialog(); 7270 this.showSplash(); 7271 }), mxUtils.bind(this, function(resp) 7272 { 7273 this.handleError(resp); 7274 }), true); //Remember is true since add account imply keeping that account 7275 }), mxResources.get('closingFile') + '...'); 7276 7277 mxEvent.consume(evt); 7278 })); 7279 } 7280 7281 return tr; 7282 }); 7283 7284 connected = true; 7285 7286 var driveUserTable = document.createElement('table'); 7287 driveUserTable.style.borderSpacing = '0'; 7288 driveUserTable.style.fontSize = '10pt'; 7289 driveUserTable.style.width = '100%'; 7290 driveUserTable.style.padding = '10px'; 7291 7292 for (var i = 0; i < driveUsers.length; i++) 7293 { 7294 driveUserTable.appendChild(createUserRow(driveUsers[i])); 7295 } 7296 7297 this.userPanel.appendChild(driveUserTable); 7298 7299 var div = document.createElement('div'); 7300 div.style.textAlign = 'left'; 7301 div.style.padding = '10px'; 7302 div.style.whiteSpace = 'nowrap'; 7303 div.style.borderTop = '1px solid rgb(224, 224, 224)'; 7304 7305 var btn = mxUtils.button(mxResources.get('signOut'), mxUtils.bind(this, function() 7306 { 7307 this.confirm(mxResources.get('areYouSure'), mxUtils.bind(this, function() 7308 { 7309 closeFile(mxUtils.bind(this, function() 7310 { 7311 this.stateArg = null; 7312 this.drive.logout(); 7313 this.setMode(App.MODE_GOOGLE); 7314 this.hideDialog(); 7315 this.showSplash(); 7316 }), mxResources.get('signOut')); 7317 })); 7318 })); 7319 btn.className = 'geBtn'; 7320 btn.style.float = 'right'; 7321 div.appendChild(btn); 7322 7323 var btn = mxUtils.button(mxResources.get('addAccount'), mxUtils.bind(this, function() 7324 { 7325 var authWin = this.drive.createAuthWin(); 7326 //FIXME This doean't work to set focus back to main window until closing the file is done 7327 authWin.blur(); 7328 window.focus(); 7329 7330 closeFile(mxUtils.bind(this, function() 7331 { 7332 this.stateArg = null; 7333 7334 this.drive.authorize(false, mxUtils.bind(this, function() 7335 { 7336 this.setMode(App.MODE_GOOGLE); 7337 this.hideDialog(); 7338 this.showSplash(); 7339 }), mxUtils.bind(this, function(resp) 7340 { 7341 this.handleError(resp); 7342 }), true, authWin); //Remember is true since add account imply keeping that account 7343 }), mxResources.get('closingFile') + '...'); 7344 })); 7345 btn.className = 'geBtn'; 7346 btn.style.margin = '0px'; 7347 div.appendChild(btn); 7348 this.userPanel.appendChild(div); 7349 } 7350 } 7351 7352 var addUser = mxUtils.bind(this, function(user, logo, logout, label) 7353 { 7354 if (user != null) 7355 { 7356 if (connected) 7357 { 7358 this.userPanel.appendChild(document.createElement('hr')); 7359 } 7360 7361 connected = true; 7362 var userTable = document.createElement('table'); 7363 userTable.style.borderSpacing = '0'; 7364 userTable.style.fontSize = '10pt'; 7365 userTable.style.width = '100%'; 7366 userTable.style.padding = '10px'; 7367 7368 var tbody = document.createElement('tbody'); 7369 var row = document.createElement('tr'); 7370 var td = document.createElement('td'); 7371 td.setAttribute('valig', 'top'); 7372 td.style.width = '40px'; 7373 7374 if (logo != null) 7375 { 7376 var img = document.createElement('img'); 7377 img.setAttribute('width', '40'); 7378 img.setAttribute('height', '40'); 7379 img.setAttribute('border', '0'); 7380 img.setAttribute('src', logo); 7381 img.style.marginRight = '6px'; 7382 7383 td.appendChild(img); 7384 } 7385 7386 row.appendChild(td); 7387 7388 var td = document.createElement('td'); 7389 td.setAttribute('valign', 'middle'); 7390 td.style.whiteSpace = 'nowrap'; 7391 td.style.maxWidth = '0'; 7392 td.style.overflow = 'hidden'; 7393 td.style.textOverflow = 'ellipsis'; 7394 7395 mxUtils.write(td, user.displayName); 7396 7397 if (user.email != null) 7398 { 7399 mxUtils.br(td); 7400 7401 var small = document.createElement('small'); 7402 small.style.color = 'gray'; 7403 mxUtils.write(small, user.email); 7404 td.appendChild(small); 7405 } 7406 7407 if (label != null) 7408 { 7409 var div = document.createElement('div'); 7410 div.style.marginTop = '4px'; 7411 7412 var i = document.createElement('i'); 7413 mxUtils.write(i, label); 7414 div.appendChild(i); 7415 td.appendChild(div); 7416 } 7417 7418 row.appendChild(td); 7419 tbody.appendChild(row); 7420 userTable.appendChild(tbody); 7421 7422 this.userPanel.appendChild(userTable); 7423 var div = document.createElement('div'); 7424 div.style.textAlign = 'center'; 7425 div.style.padding = '10px'; 7426 div.style.whiteSpace = 'nowrap'; 7427 7428 if (logout != null) 7429 { 7430 var btn = mxUtils.button(mxResources.get('signOut'), logout); 7431 btn.className = 'geBtn'; 7432 div.appendChild(btn); 7433 } 7434 7435 this.userPanel.appendChild(div); 7436 } 7437 }); 7438 7439 if (this.dropbox != null) 7440 { 7441 addUser(this.dropbox.getUser(), IMAGE_PATH + '/dropbox-logo.svg', mxUtils.bind(this, function() 7442 { 7443 var file = this.getCurrentFile(); 7444 7445 if (file != null && file.constructor == DropboxFile) 7446 { 7447 var doLogout = mxUtils.bind(this, function() 7448 { 7449 this.dropbox.logout(); 7450 window.location.hash = ''; 7451 }); 7452 7453 if (!file.isModified()) 7454 { 7455 doLogout(); 7456 } 7457 else 7458 { 7459 this.confirm(mxResources.get('allChangesLost'), null, doLogout, 7460 mxResources.get('cancel'), mxResources.get('discardChanges')); 7461 } 7462 } 7463 else 7464 { 7465 this.dropbox.logout(); 7466 } 7467 }), mxResources.get('dropbox')); 7468 } 7469 7470 if (this.oneDrive != null) 7471 { 7472 addUser(this.oneDrive.getUser(), IMAGE_PATH + '/onedrive-logo.svg', this.oneDrive.noLogout? null : mxUtils.bind(this, function() 7473 { 7474 var file = this.getCurrentFile(); 7475 7476 if (file != null && file.constructor == OneDriveFile) 7477 { 7478 var doLogout = mxUtils.bind(this, function() 7479 { 7480 this.oneDrive.logout(); 7481 window.location.hash = ''; 7482 }); 7483 7484 if (!file.isModified()) 7485 { 7486 doLogout(); 7487 } 7488 else 7489 { 7490 this.confirm(mxResources.get('allChangesLost'), null, doLogout, 7491 mxResources.get('cancel'), mxResources.get('discardChanges')); 7492 } 7493 } 7494 else 7495 { 7496 this.oneDrive.logout(); 7497 } 7498 }), mxResources.get('oneDrive')); 7499 } 7500 7501 if (this.gitHub != null) 7502 { 7503 addUser(this.gitHub.getUser(), IMAGE_PATH + '/github-logo.svg', mxUtils.bind(this, function() 7504 { 7505 var file = this.getCurrentFile(); 7506 7507 if (file != null && file.constructor == GitHubFile) 7508 { 7509 var doLogout = mxUtils.bind(this, function() 7510 { 7511 this.gitHub.logout(); 7512 window.location.hash = ''; 7513 }); 7514 7515 if (!file.isModified()) 7516 { 7517 doLogout(); 7518 } 7519 else 7520 { 7521 this.confirm(mxResources.get('allChangesLost'), null, doLogout, 7522 mxResources.get('cancel'), mxResources.get('discardChanges')); 7523 } 7524 } 7525 else 7526 { 7527 this.gitHub.logout(); 7528 } 7529 }), mxResources.get('github')); 7530 } 7531 7532 if (this.gitLab != null) 7533 { 7534 addUser(this.gitLab.getUser(), IMAGE_PATH + '/gitlab-logo.svg', mxUtils.bind(this, function() 7535 { 7536 var file = this.getCurrentFile(); 7537 7538 if (file != null && file.constructor == GitLabFile) 7539 { 7540 var doLogout = mxUtils.bind(this, function() 7541 { 7542 this.gitLab.logout(); 7543 window.location.hash = ''; 7544 }); 7545 7546 if (!file.isModified()) 7547 { 7548 doLogout(); 7549 } 7550 else 7551 { 7552 this.confirm(mxResources.get('allChangesLost'), null, doLogout, 7553 mxResources.get('cancel'), mxResources.get('discardChanges')); 7554 } 7555 } 7556 else 7557 { 7558 this.gitLab.logout(); 7559 } 7560 }), mxResources.get('gitlab')); 7561 } 7562 7563 if (this.notion != null) 7564 { 7565 addUser(this.notion.getUser(), IMAGE_PATH + '/notion-logo.svg', mxUtils.bind(this, function() 7566 { 7567 var file = this.getCurrentFile(); 7568 7569 if (file != null && file.constructor == NotionFile) 7570 { 7571 var doLogout = mxUtils.bind(this, function() 7572 { 7573 this.notion.logout(); 7574 window.location.hash = ''; 7575 }); 7576 7577 if (!file.isModified()) 7578 { 7579 doLogout(); 7580 } 7581 else 7582 { 7583 this.confirm(mxResources.get('allChangesLost'), null, doLogout, 7584 mxResources.get('cancel'), mxResources.get('discardChanges')); 7585 } 7586 } 7587 else 7588 { 7589 this.notion.logout(); 7590 } 7591 }), mxResources.get('notion')); 7592 } 7593 7594 //TODO We have no user info from Trello, how we can create a user? 7595 if (this.trello != null) 7596 { 7597 addUser(this.trello.getUser(), IMAGE_PATH + '/trello-logo.svg', mxUtils.bind(this, function() 7598 { 7599 var file = this.getCurrentFile(); 7600 7601 if (file != null && file.constructor == TrelloFile) 7602 { 7603 var doLogout = mxUtils.bind(this, function() 7604 { 7605 this.trello.logout(); 7606 window.location.hash = ''; 7607 }); 7608 7609 if (!file.isModified()) 7610 { 7611 doLogout(); 7612 } 7613 else 7614 { 7615 this.confirm(mxResources.get('allChangesLost'), null, doLogout, 7616 mxResources.get('cancel'), mxResources.get('discardChanges')); 7617 } 7618 } 7619 else 7620 { 7621 this.trello.logout(); 7622 } 7623 }), mxResources.get('trello')); 7624 } 7625 7626 if (!connected) 7627 { 7628 var div = document.createElement('div'); 7629 div.style.textAlign = 'center'; 7630 div.style.padding = '10px'; 7631 div.innerHTML = mxResources.get('notConnected'); 7632 7633 this.userPanel.appendChild(div); 7634 } 7635 7636 var div = document.createElement('div'); 7637 div.style.textAlign = 'center'; 7638 div.style.padding = '10px'; 7639 div.style.background = Editor.isDarkMode() ? '' : 'whiteSmoke'; 7640 div.style.borderTop = '1px solid #e0e0e0'; 7641 div.style.whiteSpace = 'nowrap'; 7642 7643 if (urlParams['sketch'] == '1') 7644 { 7645 var btn = mxUtils.button(mxResources.get('share'), mxUtils.bind(this, function() 7646 { 7647 this.actions.get('share').funct(); 7648 })); 7649 btn.className = 'geBtn'; 7650 div.appendChild(btn); 7651 this.userPanel.appendChild(div); 7652 7653 if (this.commentsSupported()) 7654 { 7655 btn = mxUtils.button(mxResources.get('comments'), mxUtils.bind(this, function() 7656 { 7657 this.actions.get('comments').funct(); 7658 })); 7659 btn.className = 'geBtn'; 7660 div.appendChild(btn); 7661 this.userPanel.appendChild(div); 7662 } 7663 } 7664 else 7665 { 7666 var btn = mxUtils.button(mxResources.get('close'), mxUtils.bind(this, function() 7667 { 7668 if (!mxEvent.isConsumed(evt) && this.userPanel != null && this.userPanel.parentNode != null) 7669 { 7670 this.userPanel.parentNode.removeChild(this.userPanel); 7671 } 7672 })); 7673 btn.className = 'geBtn'; 7674 div.appendChild(btn); 7675 this.userPanel.appendChild(div); 7676 } 7677 7678 document.body.appendChild(this.userPanel); 7679 } 7680 7681 mxEvent.consume(evt); 7682 })); 7683 7684 mxEvent.addListener(document.body, 'click', mxUtils.bind(this, function(evt) 7685 { 7686 if (!mxEvent.isConsumed(evt) && this.userPanel != null && this.userPanel.parentNode != null) 7687 { 7688 this.userPanel.parentNode.removeChild(this.userPanel); 7689 } 7690 })); 7691 } 7692 7693 var user = null; 7694 7695 if (this.drive != null && this.drive.getUser() != null) 7696 { 7697 user = this.drive.getUser(); 7698 } 7699 else if (this.oneDrive != null && this.oneDrive.getUser() != null) 7700 { 7701 user = this.oneDrive.getUser(); 7702 } 7703 else if (this.dropbox != null && this.dropbox.getUser() != null) 7704 { 7705 user = this.dropbox.getUser(); 7706 } 7707 else if (this.gitHub != null && this.gitHub.getUser() != null) 7708 { 7709 user = this.gitHub.getUser(); 7710 } 7711 else if (this.gitLab != null && this.gitLab.getUser() != null) 7712 { 7713 user = this.gitLab.getUser(); 7714 } 7715 else if (this.notion != null && this.notion.getUser() != null) 7716 { 7717 user = this.notion.getUser(); 7718 } 7719 //TODO Trello no user issue 7720 7721 if (user != null) 7722 { 7723 this.userElement.innerHTML = ''; 7724 7725 if (screen.width > 560) 7726 { 7727 mxUtils.write(this.userElement, user.displayName); 7728 this.userElement.style.display = 'block'; 7729 } 7730 } 7731 else 7732 { 7733 this.userElement.style.display = 'none'; 7734 } 7735 } 7736}; 7737 7738//TODO Use this function to get the currently logged in user 7739App.prototype.getCurrentUser = function() 7740{ 7741 var user = null; 7742 7743 if (this.drive != null && this.drive.getUser() != null) 7744 { 7745 user = this.drive.getUser(); 7746 } 7747 else if (this.oneDrive != null && this.oneDrive.getUser() != null) 7748 { 7749 user = this.oneDrive.getUser(); 7750 } 7751 else if (this.dropbox != null && this.dropbox.getUser() != null) 7752 { 7753 user = this.dropbox.getUser(); 7754 } 7755 else if (this.gitHub != null && this.gitHub.getUser() != null) 7756 { 7757 user = this.gitHub.getUser(); 7758 } 7759 //TODO Trello no user issue 7760 7761 return user; 7762} 7763/** 7764 * Override depends on mxSettings which is not defined in the minified viewer. 7765 */ 7766var editorResetGraph = Editor.prototype.resetGraph; 7767Editor.prototype.resetGraph = function() 7768{ 7769 editorResetGraph.apply(this, arguments); 7770 7771 // Overrides default with persisted value 7772 if (this.graph.defaultPageFormat == null) 7773 { 7774 this.graph.pageFormat = mxSettings.getPageFormat(); 7775 } 7776}; 7777