1/** 2 * Copyright (c) 2006-2012, JGraph Ltd 3 */ 4/** 5 * Construcs a new sidebar for the given editor. 6 */ 7function Sidebar(editorUi, container) 8{ 9 this.editorUi = editorUi; 10 this.container = container; 11 this.palettes = new Object(); 12 this.taglist = new Object(); 13 this.lastCreated = 0; 14 this.showTooltips = true; 15 this.graph = editorUi.createTemporaryGraph(this.editorUi.editor.graph.getStylesheet()); 16 this.graph.cellRenderer.minSvgStrokeWidth = this.minThumbStrokeWidth; 17 this.graph.cellRenderer.antiAlias = this.thumbAntiAlias; 18 this.graph.container.style.visibility = 'hidden'; 19 this.graph.foldingEnabled = false; 20 21 document.body.appendChild(this.graph.container); 22 23 this.pointerUpHandler = mxUtils.bind(this, function() 24 { 25 if (this.tooltipCloseImage == null || this.tooltipCloseImage.style.display == 'none') 26 { 27 this.showTooltips = true; 28 this.hideTooltip(); 29 } 30 }); 31 32 mxEvent.addListener(document, (mxClient.IS_POINTER) ? 'pointerup' : 'mouseup', this.pointerUpHandler); 33 34 this.pointerDownHandler = mxUtils.bind(this, function() 35 { 36 if (this.tooltipCloseImage == null || this.tooltipCloseImage.style.display == 'none') 37 { 38 this.showTooltips = false; 39 this.hideTooltip(); 40 } 41 }); 42 43 mxEvent.addListener(document, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', this.pointerDownHandler); 44 45 this.pointerMoveHandler = mxUtils.bind(this, function(evt) 46 { 47 if (Date.now() - this.lastCreated > 300 && (this.tooltipCloseImage == null || 48 this.tooltipCloseImage.style.display == 'none')) 49 { 50 var src = mxEvent.getSource(evt); 51 52 while (src != null) 53 { 54 if (src == this.currentElt) 55 { 56 return; 57 } 58 59 src = src.parentNode; 60 } 61 62 this.hideTooltip(); 63 } 64 }); 65 66 mxEvent.addListener(document, (mxClient.IS_POINTER) ? 'pointermove' : 'mousemove', this.pointerMoveHandler); 67 68 // Handles mouse leaving the window 69 this.pointerOutHandler = mxUtils.bind(this, function(evt) 70 { 71 if (evt.toElement == null && evt.relatedTarget == null) 72 { 73 this.hideTooltip(); 74 } 75 }); 76 77 mxEvent.addListener(document, (mxClient.IS_POINTER) ? 'pointerout' : 'mouseout', this.pointerOutHandler); 78 79 // Enables tooltips after scroll 80 mxEvent.addListener(container, 'scroll', mxUtils.bind(this, function() 81 { 82 this.showTooltips = true; 83 this.hideTooltip(); 84 })); 85 86 this.init(); 87}; 88 89/** 90 * Adds all palettes to the sidebar. 91 */ 92Sidebar.prototype.init = function() 93{ 94 var dir = STENCIL_PATH; 95 96 this.addSearchPalette(true); 97 this.addGeneralPalette(true); 98 this.addMiscPalette(false); 99 this.addAdvancedPalette(false); 100 this.addBasicPalette(dir); 101 102 this.setCurrentSearchEntryLibrary('arrows'); 103 this.addStencilPalette('arrows', mxResources.get('arrows'), dir + '/arrows.xml', 104 ';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2'); 105 this.setCurrentSearchEntryLibrary(); 106 107 this.addUmlPalette(false); 108 this.addBpmnPalette(dir, false); 109 110 this.setCurrentSearchEntryLibrary('flowchart'); 111 this.addStencilPalette('flowchart', 'Flowchart', dir + '/flowchart.xml', 112 ';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2'); 113 this.setCurrentSearchEntryLibrary(); 114 115 this.setCurrentSearchEntryLibrary('clipart'); 116 this.addImagePalette('clipart', mxResources.get('clipart'), dir + '/clipart/', '_128x128.png', 117 ['Earth_globe', 'Empty_Folder', 'Full_Folder', 'Gear', 'Lock', 'Software', 'Virus', 'Email', 118 'Database', 'Router_Icon', 'iPad', 'iMac', 'Laptop', 'MacBook', 'Monitor_Tower', 'Printer', 119 'Server_Tower', 'Workstation', 'Firewall_02', 'Wireless_Router_N', 'Credit_Card', 120 'Piggy_Bank', 'Graph', 'Safe', 'Shopping_Cart', 'Suit1', 'Suit2', 'Suit3', 'Pilot1', 121 'Worker1', 'Soldier1', 'Doctor1', 'Tech1', 'Security1', 'Telesales1'], null, 122 {'Wireless_Router_N': 'wireless router switch wap wifi access point wlan', 123 'Router_Icon': 'router switch'}); 124 this.setCurrentSearchEntryLibrary(); 125}; 126 127/** 128 * Sets the default font size. 129 */ 130Sidebar.prototype.collapsedImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/collapsed.gif' : ''; 131 132/** 133 * Sets the default font size. 134 */ 135Sidebar.prototype.expandedImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/expanded.gif' : ''; 136 137/** 138 * 139 */ 140Sidebar.prototype.searchImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/search.png' : ''; 141 142/** 143 * Specifies if tooltips should be visible. Default is true. 144 */ 145Sidebar.prototype.enableTooltips = true; 146 147/** 148 * Specifies the delay for the tooltip. Default is 16 px. 149 */ 150Sidebar.prototype.tooltipBorder = 16; 151 152/** 153 * Specifies the delay for the tooltip. Default is 300 ms. 154 */ 155Sidebar.prototype.tooltipDelay = 300; 156 157/** 158 * Specifies the delay for the drop target icons. Default is 200 ms. 159 */ 160Sidebar.prototype.dropTargetDelay = 200; 161 162/** 163 * Specifies the URL of the gear image. 164 */ 165Sidebar.prototype.gearImage = STENCIL_PATH + '/clipart/Gear_128x128.png'; 166 167/** 168 * Specifies the width of the thumbnails. 169 */ 170Sidebar.prototype.thumbWidth = 42; 171 172/** 173 * Specifies the height of the thumbnails. 174 */ 175Sidebar.prototype.thumbHeight = 42; 176 177/** 178 * Specifies the width of the thumbnails. 179 */ 180Sidebar.prototype.minThumbStrokeWidth = 1; 181 182/** 183 * Specifies the width of the thumbnails. 184 */ 185Sidebar.prototype.thumbAntiAlias = false; 186 187/** 188 * Specifies the padding for the thumbnails. Default is 3. 189 */ 190Sidebar.prototype.thumbPadding = (document.documentMode >= 5) ? 2 : 3; 191 192/** 193 * Specifies the delay for the tooltip. Default is 2 px. 194 */ 195Sidebar.prototype.thumbBorder = 2; 196 197/* 198 * Experimental smaller sidebar entries 199 */ 200if (urlParams['sidebar-entries'] != 'large') 201{ 202 Sidebar.prototype.thumbPadding = (document.documentMode >= 5) ? 0 : 1; 203 Sidebar.prototype.thumbBorder = 1; 204 Sidebar.prototype.thumbWidth = 32; 205 Sidebar.prototype.thumbHeight = 30; 206 Sidebar.prototype.minThumbStrokeWidth = 1.3; 207 Sidebar.prototype.thumbAntiAlias = true; 208} 209 210/** 211 * Specifies the size of the sidebar titles. 212 */ 213Sidebar.prototype.sidebarTitleSize = 8; 214 215/** 216 * Specifies if titles in the sidebar should be enabled. 217 */ 218Sidebar.prototype.sidebarTitles = false; 219 220/** 221 * Specifies if titles in the tooltips should be enabled. 222 */ 223Sidebar.prototype.tooltipTitles = true; 224 225/** 226 * Specifies if titles in the tooltips should be enabled. 227 */ 228Sidebar.prototype.maxTooltipWidth = 400; 229 230/** 231 * Specifies if titles in the tooltips should be enabled. 232 */ 233Sidebar.prototype.maxTooltipHeight = 400; 234 235/** 236 * Specifies if stencil files should be loaded and added to the search index 237 * when stencil palettes are added. If this is false then the stencil files 238 * are lazy-loaded when the palette is shown. 239 */ 240Sidebar.prototype.addStencilsToIndex = true; 241 242/** 243 * Specifies the width for clipart images. Default is 80. 244 */ 245Sidebar.prototype.defaultImageWidth = 80; 246 247/** 248 * Specifies the height for clipart images. Default is 80. 249 */ 250Sidebar.prototype.defaultImageHeight = 80; 251 252/** 253 * Specifies the height for clipart images. Default is 80. 254 */ 255Sidebar.prototype.tooltipMouseDown = null; 256 257/** 258 * Adds all palettes to the sidebar. 259 */ 260Sidebar.prototype.getTooltipOffset = function(elt, bounds) 261{ 262 var b = document.body; 263 var d = document.documentElement; 264 var bottom = Math.max(b.clientHeight || 0, d.clientHeight); 265 var width = bounds.width + 2 * this.tooltipBorder + 4; 266 var height = bounds.height + 2 * this.tooltipBorder; 267 268 return new mxPoint(this.container.offsetWidth + this.editorUi.splitSize + 10 + this.editorUi.container.offsetLeft, 269 Math.min(bottom - height - 20 /*status bar*/, Math.max(0, (this.editorUi.container.offsetTop + 270 this.container.offsetTop + elt.offsetTop - this.container.scrollTop - height / 2 + 16)))); 271}; 272 273/** 274 * Adds all palettes to the sidebar. 275 */ 276Sidebar.prototype.createTooltip = function(elt, cells, w, h, title, showLabel, off, maxSize, mouseDown, closable, applyAllStyles) 277{ 278 applyAllStyles = (applyAllStyles != null) ? applyAllStyles : true; 279 this.tooltipMouseDown = mouseDown; 280 281 // Lazy creation of the DOM nodes and graph instance 282 if (this.tooltip == null) 283 { 284 this.tooltip = document.createElement('div'); 285 this.tooltip.className = 'geSidebarTooltip'; 286 this.tooltip.style.userSelect = 'none'; 287 this.tooltip.style.zIndex = mxPopupMenu.prototype.zIndex - 1; 288 document.body.appendChild(this.tooltip); 289 290 mxEvent.addMouseWheelListener(mxUtils.bind(this, function(evt) 291 { 292 this.hideTooltip(); 293 }), this.tooltip); 294 295 this.graph2 = new Graph(this.tooltip, null, null, this.editorUi.editor.graph.getStylesheet()); 296 this.graph2.resetViewOnRootChange = false; 297 this.graph2.foldingEnabled = false; 298 this.graph2.gridEnabled = false; 299 this.graph2.autoScroll = false; 300 this.graph2.setTooltips(false); 301 this.graph2.setConnectable(false); 302 this.graph2.setPanning(false); 303 this.graph2.setEnabled(false); 304 305 // Blocks all links 306 this.graph2.openLink = mxUtils.bind(this, function() 307 { 308 this.hideTooltip(); 309 }); 310 311 mxEvent.addGestureListeners(this.tooltip, mxUtils.bind(this, function(evt) 312 { 313 if (this.tooltipMouseDown != null) 314 { 315 this.tooltipMouseDown(evt); 316 } 317 318 window.setTimeout(mxUtils.bind(this, function() 319 { 320 if (this.tooltipCloseImage == null || this.tooltipCloseImage.style.display == 'none') 321 { 322 this.hideTooltip(); 323 } 324 }), 0); 325 }), null, mxUtils.bind(this, function(evt) 326 { 327 this.hideTooltip(); 328 })); 329 330 if (!mxClient.IS_SVG) 331 { 332 this.graph2.view.canvas.style.position = 'relative'; 333 } 334 335 var close = document.createElement('img'); 336 close.setAttribute('src', Dialog.prototype.closeImage); 337 close.setAttribute('title', mxResources.get('close')); 338 close.style.position = 'absolute'; 339 close.style.cursor = 'default'; 340 close.style.padding = '8px'; 341 close.style.right = '2px'; 342 close.style.top = '2px'; 343 this.tooltip.appendChild(close); 344 this.tooltipCloseImage = close; 345 346 mxEvent.addListener(close, 'click', mxUtils.bind(this, function(evt) 347 { 348 this.hideTooltip(); 349 mxEvent.consume(evt); 350 })); 351 } 352 353 this.tooltipCloseImage.style.display = (closable) ? '' : 'none'; 354 this.graph2.model.clear(); 355 this.graph2.view.setTranslate(this.tooltipBorder, this.tooltipBorder); 356 357 if (!maxSize && (w > this.maxTooltipWidth || h > this.maxTooltipHeight)) 358 { 359 this.graph2.view.scale = Math.round(Math.min(this.maxTooltipWidth / w, this.maxTooltipHeight / h) * 100) / 100; 360 } 361 else 362 { 363 this.graph2.view.scale = 1; 364 } 365 366 this.tooltip.style.display = 'block'; 367 this.graph2.labelsVisible = (showLabel == null || showLabel); 368 var fo = mxClient.NO_FO; 369 mxClient.NO_FO = Editor.prototype.originalNoForeignObject; 370 371 // Applies current style for preview 372 var temp = this.graph2.cloneCells(cells); 373 this.editorUi.insertHandler(temp, null, this.graph2.model, 374 (!applyAllStyles) ? this.editorUi.editor.graph.defaultVertexStyle : null, 375 (!applyAllStyles) ? this.editorUi.editor.graph.defaultEdgeStyle : null, 376 applyAllStyles, true); 377 this.graph2.addCells(temp); 378 379 mxClient.NO_FO = fo; 380 var bounds = this.graph2.getGraphBounds(); 381 382 // Maximum size applied with transform for faster repaint 383 if (maxSize && w > 0 && h > 0 && (bounds.width > w || bounds.height > h)) 384 { 385 var s = Math.round(Math.min(w / bounds.width, h / bounds.height) * 100) / 100; 386 387 if (!mxClient.NO_FO) 388 { 389 this.graph2.view.getDrawPane().ownerSVGElement.style.transform = 'scale(' + s + ')'; 390 this.graph2.view.getDrawPane().ownerSVGElement.style.transformOrigin = '0 0'; 391 bounds.width *= s; 392 bounds.height *= s; 393 } 394 else 395 { 396 this.graph2.view.setScale(Math.round(Math.min(this.maxTooltipWidth / bounds.width, this.maxTooltipHeight / bounds.height) * 100) / 100); 397 bounds = this.graph2.getGraphBounds(); 398 } 399 } 400 else if (!mxClient.NO_FO) 401 { 402 this.graph2.view.getDrawPane().ownerSVGElement.style.transform = ''; 403 } 404 405 var width = bounds.width + 2 * this.tooltipBorder + 4; 406 var height = bounds.height + 2 * this.tooltipBorder; 407 408 this.tooltip.style.overflow = 'visible'; 409 this.tooltip.style.width = width + 'px'; 410 var w2 = width; 411 412 // Adds title for entry 413 if (this.tooltipTitles && title != null && title.length > 0) 414 { 415 if (this.tooltipTitle == null) 416 { 417 this.tooltipTitle = document.createElement('div'); 418 this.tooltipTitle.style.borderTop = '1px solid gray'; 419 this.tooltipTitle.style.textAlign = 'center'; 420 this.tooltipTitle.style.width = '100%'; 421 this.tooltipTitle.style.overflow = 'hidden'; 422 this.tooltipTitle.style.position = 'absolute'; 423 this.tooltipTitle.style.paddingTop = '6px'; 424 this.tooltipTitle.style.bottom = '6px'; 425 426 this.tooltip.appendChild(this.tooltipTitle); 427 } 428 else 429 { 430 this.tooltipTitle.innerHTML = ''; 431 } 432 433 this.tooltipTitle.style.display = ''; 434 mxUtils.write(this.tooltipTitle, title); 435 436 // Allows for wider labels 437 w2 = Math.min(this.maxTooltipWidth, Math.max(width, this.tooltipTitle.scrollWidth + 4)); 438 var ddy = this.tooltipTitle.offsetHeight + 10; 439 height += ddy; 440 441 if (mxClient.IS_SVG) 442 { 443 this.tooltipTitle.style.marginTop = (2 - ddy) + 'px'; 444 } 445 else 446 { 447 height -= 6; 448 this.tooltipTitle.style.top = (height - ddy) + 'px'; 449 } 450 } 451 else if (this.tooltipTitle != null && this.tooltipTitle.parentNode != null) 452 { 453 this.tooltipTitle.style.display = 'none'; 454 } 455 456 // Updates width if label is wider 457 if (w2 > width) 458 { 459 this.tooltip.style.width = w2 + 'px'; 460 } 461 462 this.tooltip.style.height = height + 'px'; 463 var x0 = -Math.round(bounds.x - this.tooltipBorder) + 464 ((w2 > width) ? (w2 - width) / 2 : 0); 465 var y0 = -Math.round(bounds.y - this.tooltipBorder); 466 off = (off != null) ? off : this.getTooltipOffset(elt, bounds); 467 var left = off.x; 468 var top = off.y; 469 470 if (mxClient.IS_SVG) 471 { 472 if (x0 != 0 || y0 != 0) 473 { 474 this.graph2.view.canvas.setAttribute('transform', 'translate(' + x0 + ',' + y0 + ')'); 475 } 476 else 477 { 478 this.graph2.view.canvas.removeAttribute('transform'); 479 } 480 } 481 else 482 { 483 this.graph2.view.drawPane.style.left = x0 + 'px'; 484 this.graph2.view.drawPane.style.top = y0 + 'px'; 485 } 486 487 // Workaround for ignored position CSS style in IE9 488 // (changes to relative without the following line) 489 this.tooltip.style.position = 'absolute'; 490 this.tooltip.style.left = left + 'px'; 491 this.tooltip.style.top = top + 'px'; 492 493 mxUtils.fit(this.tooltip); 494 this.lastCreated = Date.now(); 495}; 496 497/** 498 * Adds all palettes to the sidebar. 499 */ 500Sidebar.prototype.showTooltip = function(elt, cells, w, h, title, showLabel) 501{ 502 if (this.enableTooltips && this.showTooltips) 503 { 504 if (this.currentElt != elt) 505 { 506 if (this.thread != null) 507 { 508 window.clearTimeout(this.thread); 509 this.thread = null; 510 } 511 512 var show = mxUtils.bind(this, function() 513 { 514 this.createTooltip(elt, cells, w, h, title, showLabel); 515 }); 516 517 if (this.tooltip != null && this.tooltip.style.display != 'none') 518 { 519 show(); 520 } 521 else 522 { 523 this.thread = window.setTimeout(show, this.tooltipDelay); 524 } 525 526 this.currentElt = elt; 527 } 528 } 529}; 530 531/** 532 * Hides the current tooltip. 533 */ 534Sidebar.prototype.hideTooltip = function() 535{ 536 if (this.thread != null) 537 { 538 window.clearTimeout(this.thread); 539 this.thread = null; 540 } 541 542 if (this.tooltip != null) 543 { 544 this.tooltip.style.display = 'none'; 545 this.currentElt = null; 546 } 547 548 this.tooltipMouseDown = null; 549}; 550 551/** 552 * Hides the current tooltip. 553 */ 554Sidebar.prototype.addDataEntry = function(tags, width, height, title, data) 555{ 556 return this.addEntry(tags, mxUtils.bind(this, function() 557 { 558 return this.createVertexTemplateFromData(data, width, height, title); 559 })); 560}; 561 562/** 563 * Adds the give entries to the search index. 564 */ 565Sidebar.prototype.addEntries = function(images) 566{ 567 for (var i = 0; i < images.length; i++) 568 { 569 (mxUtils.bind(this, function(img) 570 { 571 var data = img.data; 572 var tags = (img.title != null) ? img.title : ''; 573 574 if (img.tags != null) 575 { 576 tags += ' ' + img.tags; 577 } 578 579 if (data != null && tags.length > 0) 580 { 581 this.addEntry(tags, mxUtils.bind(this, function() 582 { 583 data = this.editorUi.convertDataUri(data); 584 var s = 'shape=image;verticalLabelPosition=bottom;verticalAlign=top;imageAspect=0;'; 585 586 if (img.aspect == 'fixed') 587 { 588 s += 'aspect=fixed;' 589 } 590 591 return this.createVertexTemplate(s + 'image=' + 592 data, img.w, img.h, '', img.title || '', false, false, true) 593 })); 594 } 595 else if (img.xml != null && tags.length > 0) 596 { 597 this.addEntry(tags, mxUtils.bind(this, function() 598 { 599 var cells = this.editorUi.stringToCells(Graph.decompress(img.xml)); 600 601 return this.createVertexTemplateFromCells( 602 cells, img.w, img.h, img.title || '', true, false, true); 603 })); 604 } 605 }))(images[i]); 606 } 607}; 608 609/** 610 * Hides the current tooltip. 611 */ 612Sidebar.prototype.setCurrentSearchEntryLibrary = function(id, lib) 613{ 614 this.currentSearchEntryLibrary = (id != null) ? {id: id, lib: lib} : null; 615}; 616 617/** 618 * Hides the current tooltip. 619 */ 620Sidebar.prototype.addEntry = function(tags, fn) 621{ 622 if (this.taglist != null && tags != null && tags.length > 0) 623 { 624 if (this.currentSearchEntryLibrary != null) 625 { 626 fn.parentLibraries = [this.currentSearchEntryLibrary]; 627 } 628 629 // Replaces special characters 630 var tmp = tags.toLowerCase().replace(/[\/\,\(\)]/g, ' ').split(' '); 631 var tagList = []; 632 var hash = {}; 633 634 // Finds unique tags 635 for (var i = 0; i < tmp.length; i++) 636 { 637 if (hash[tmp[i]] == null) 638 { 639 hash[tmp[i]] = true; 640 tagList.push(tmp[i]); 641 } 642 643 // Adds additional entry with removed trailing numbers 644 var normalized = tmp[i].replace(/\.*\d*$/, ''); 645 646 if (normalized != tmp[i]) 647 { 648 if (hash[normalized] == null) 649 { 650 hash[normalized] = true; 651 tagList.push(normalized); 652 } 653 } 654 } 655 656 for (var i = 0; i < tagList.length; i++) 657 { 658 this.addEntryForTag(tagList[i], fn); 659 } 660 } 661 662 return fn; 663}; 664 665/** 666 * Hides the current tooltip. 667 */ 668Sidebar.prototype.addEntryForTag = function(tag, fn) 669{ 670 if (tag != null && tag.length > 1) 671 { 672 var entry = this.taglist[tag]; 673 674 if (typeof entry !== 'object') 675 { 676 entry = {entries: []}; 677 this.taglist[tag] = entry; 678 } 679 680 entry.entries.push(fn); 681 } 682}; 683 684/** 685 * Adds shape search UI. 686 */ 687Sidebar.prototype.searchEntries = function(searchTerms, count, page, success, error) 688{ 689 if (this.taglist != null && searchTerms != null) 690 { 691 var tmp = searchTerms.toLowerCase().split(' '); 692 var dict = new mxDictionary(); 693 var max = (page + 1) * count; 694 var results = []; 695 var index = 0; 696 697 for (var i = 0; i < tmp.length; i++) 698 { 699 if (tmp[i].length > 0) 700 { 701 var entry = this.taglist[tmp[i]]; 702 var tmpDict = new mxDictionary(); 703 704 if (entry != null) 705 { 706 var arr = entry.entries; 707 results = []; 708 709 for (var j = 0; j < arr.length; j++) 710 { 711 var entry = arr[j]; 712 713 // NOTE Array does not contain duplicates 714 if ((index == 0) == (dict.get(entry) == null)) 715 { 716 tmpDict.put(entry, entry); 717 results.push(entry); 718 719 if (i == tmp.length - 1 && results.length == max) 720 { 721 success(results.slice(page * count, max), max, true, tmp); 722 723 return; 724 } 725 } 726 } 727 } 728 else 729 { 730 results = []; 731 } 732 733 dict = tmpDict; 734 index++; 735 } 736 } 737 738 var len = results.length; 739 success(results.slice(page * count, (page + 1) * count), len, false, tmp); 740 } 741 else 742 { 743 success([], null, null, tmp); 744 } 745}; 746 747/** 748 * Adds shape search UI. 749 */ 750Sidebar.prototype.filterTags = function(tags) 751{ 752 if (tags != null) 753 { 754 var arr = tags.split(' '); 755 var result = []; 756 var hash = {}; 757 758 // Ignores tags with leading numbers, strips trailing numbers 759 for (var i = 0; i < arr.length; i++) 760 { 761 // Removes duplicates 762 if (hash[arr[i]] == null) 763 { 764 hash[arr[i]] = '1'; 765 result.push(arr[i]); 766 } 767 } 768 769 return result.join(' '); 770 } 771 772 return null; 773}; 774 775/** 776 * Adds the general palette to the sidebar. 777 */ 778Sidebar.prototype.cloneCell = function(cell, value) 779{ 780 var clone = cell.clone(); 781 782 if (value != null) 783 { 784 clone.value = value; 785 } 786 787 return clone; 788}; 789 790/** 791 * Adds shape search UI. 792 */ 793Sidebar.prototype.showPopupMenuForEntry = function(elt, libs, evt) 794{ 795 // Hook for subclassers 796}; 797 798/** 799 * Adds shape search UI. 800 */ 801Sidebar.prototype.addSearchPalette = function(expand) 802{ 803 var elt = document.createElement('div'); 804 elt.style.visibility = 'hidden'; 805 this.container.appendChild(elt); 806 807 var div = document.createElement('div'); 808 div.className = 'geSidebar'; 809 div.style.boxSizing = 'border-box'; 810 div.style.overflow = 'hidden'; 811 div.style.width = '100%'; 812 div.style.padding = '8px'; 813 div.style.paddingTop = '14px'; 814 div.style.paddingBottom = '0px'; 815 816 if (!expand) 817 { 818 div.style.display = 'none'; 819 } 820 821 var inner = document.createElement('div'); 822 inner.style.whiteSpace = 'nowrap'; 823 inner.style.textOverflow = 'clip'; 824 inner.style.paddingBottom = '8px'; 825 inner.style.cursor = 'default'; 826 827 var input = document.createElement('input'); 828 input.setAttribute('placeholder', mxResources.get('searchShapes')); 829 input.setAttribute('type', 'text'); 830 input.style.fontSize = '12px'; 831 input.style.overflow = 'hidden'; 832 input.style.boxSizing = 'border-box'; 833 input.style.border = 'solid 1px #d5d5d5'; 834 input.style.borderRadius = '4px'; 835 input.style.width = '100%'; 836 input.style.outline = 'none'; 837 input.style.padding = '6px'; 838 input.style.paddingRight = '20px'; 839 inner.appendChild(input); 840 841 var cross = document.createElement('img'); 842 cross.setAttribute('src', Sidebar.prototype.searchImage); 843 cross.setAttribute('title', mxResources.get('search')); 844 cross.style.position = 'relative'; 845 cross.style.left = '-18px'; 846 cross.style.top = '1px'; 847 848 // Needed to block event transparency in IE 849 cross.style.background = 'url(\'' + this.editorUi.editor.transparentImage + '\')'; 850 851 var find; 852 853 inner.appendChild(cross); 854 div.appendChild(inner); 855 856 var center = document.createElement('center'); 857 var button = mxUtils.button(mxResources.get('moreResults'), function() 858 { 859 find(); 860 }); 861 button.style.display = 'none'; 862 863 // Workaround for inherited line-height in quirks mode 864 button.style.lineHeight = 'normal'; 865 button.style.fontSize = '12px'; 866 button.style.padding = '6px 12px 6px 12px'; 867 button.style.marginTop = '4px'; 868 button.style.marginBottom = '8px'; 869 center.style.paddingTop = '4px'; 870 center.style.paddingBottom = '4px'; 871 872 center.appendChild(button); 873 div.appendChild(center); 874 875 var searchTerm = ''; 876 var active = false; 877 var complete = false; 878 var page = 0; 879 var hash = new Object(); 880 881 // Count is dynamically updated below 882 var count = 12; 883 884 var clearDiv = mxUtils.bind(this, function() 885 { 886 active = false; 887 this.currentSearch = null; 888 var child = div.firstChild; 889 890 while (child != null) 891 { 892 var next = child.nextSibling; 893 894 if (child != inner && child != center) 895 { 896 child.parentNode.removeChild(child); 897 } 898 899 child = next; 900 } 901 }); 902 903 mxEvent.addListener(cross, 'click', function() 904 { 905 if (cross.getAttribute('src') == Dialog.prototype.closeImage) 906 { 907 cross.setAttribute('src', Sidebar.prototype.searchImage); 908 cross.setAttribute('title', mxResources.get('search')); 909 button.style.display = 'none'; 910 input.value = ''; 911 searchTerm = ''; 912 clearDiv(); 913 } 914 915 input.focus(); 916 }); 917 918 find = mxUtils.bind(this, function() 919 { 920 // Shows 4 rows (minimum 4 results) 921 count = 4 * Math.max(1, Math.floor(this.container.clientWidth / (this.thumbWidth + 10))); 922 this.hideTooltip(); 923 924 if (input.value != '') 925 { 926 if (center.parentNode != null) 927 { 928 if (searchTerm != input.value) 929 { 930 clearDiv(); 931 searchTerm = input.value; 932 hash = new Object(); 933 complete = false; 934 page = 0; 935 } 936 937 if (!active && !complete) 938 { 939 button.setAttribute('disabled', 'true'); 940 button.style.display = ''; 941 button.style.cursor = 'wait'; 942 button.innerHTML = mxResources.get('loading') + '...'; 943 active = true; 944 945 // Ignores old results 946 var current = new Object(); 947 this.currentSearch = current; 948 949 this.searchEntries(searchTerm, count, page, mxUtils.bind(this, function(results, len, more, terms) 950 { 951 if (this.currentSearch == current) 952 { 953 results = (results != null) ? results : []; 954 active = false; 955 page++; 956 this.insertSearchHint(div, searchTerm, count, page, results, len, more, terms); 957 958 // Allows to repeat the search 959 if (results.length == 0 && page == 1) 960 { 961 searchTerm = ''; 962 } 963 964 if (center.parentNode != null) 965 { 966 center.parentNode.removeChild(center); 967 } 968 969 for (var i = 0; i < results.length; i++) 970 { 971 (mxUtils.bind(this, function(result) 972 { 973 try 974 { 975 var elt = result(); 976 977 // Avoids duplicates in results 978 if (hash[elt.innerHTML] == null) 979 { 980 hash[elt.innerHTML] = (result.parentLibraries != null) ? result.parentLibraries.slice() : []; 981 div.appendChild(elt); 982 } 983 else if (result.parentLibraries != null) 984 { 985 hash[elt.innerHTML] = hash[elt.innerHTML].concat(result.parentLibraries); 986 } 987 988 mxEvent.addGestureListeners(elt, null, null, mxUtils.bind(this, function(evt) 989 { 990 var libs = hash[elt.innerHTML]; 991 992 if (mxEvent.isPopupTrigger(evt)) 993 { 994 this.showPopupMenuForEntry(elt, libs, evt); 995 } 996 })); 997 998 // Disables the built-in context menu 999 mxEvent.disableContextMenu(elt); 1000 } 1001 catch (e) 1002 { 1003 // ignore 1004 } 1005 }))(results[i]); 1006 } 1007 1008 if (more) 1009 { 1010 button.removeAttribute('disabled'); 1011 button.innerHTML = mxResources.get('moreResults'); 1012 } 1013 else 1014 { 1015 button.innerHTML = mxResources.get('reset'); 1016 button.style.display = 'none'; 1017 complete = true; 1018 } 1019 1020 button.style.cursor = ''; 1021 div.appendChild(center); 1022 } 1023 }), mxUtils.bind(this, function() 1024 { 1025 // TODO: Error handling 1026 button.style.cursor = ''; 1027 })); 1028 } 1029 } 1030 } 1031 else 1032 { 1033 clearDiv(); 1034 input.value = ''; 1035 searchTerm = ''; 1036 hash = new Object(); 1037 button.style.display = 'none'; 1038 complete = false; 1039 input.focus(); 1040 } 1041 }); 1042 1043 this.searchShapes = function(value) 1044 { 1045 input.value = value; 1046 find(); 1047 }; 1048 1049 mxEvent.addListener(input, 'keydown', mxUtils.bind(this, function(evt) 1050 { 1051 if (evt.keyCode == 13 /* Enter */) 1052 { 1053 find(); 1054 mxEvent.consume(evt); 1055 } 1056 })); 1057 1058 mxEvent.addListener(input, 'keyup', mxUtils.bind(this, function(evt) 1059 { 1060 if (input.value == '') 1061 { 1062 cross.setAttribute('src', Sidebar.prototype.searchImage); 1063 cross.setAttribute('title', mxResources.get('search')); 1064 } 1065 else 1066 { 1067 cross.setAttribute('src', Dialog.prototype.closeImage); 1068 cross.setAttribute('title', mxResources.get('reset')); 1069 } 1070 1071 if (input.value == '') 1072 { 1073 complete = true; 1074 button.style.display = 'none'; 1075 } 1076 else if (input.value != searchTerm) 1077 { 1078 button.style.display = 'none'; 1079 complete = false; 1080 } 1081 else if (!active) 1082 { 1083 if (complete) 1084 { 1085 button.style.display = 'none'; 1086 } 1087 else 1088 { 1089 button.style.display = ''; 1090 } 1091 } 1092 })); 1093 1094 // Workaround for blocked text selection in Editor 1095 mxEvent.addListener(input, 'mousedown', function(evt) 1096 { 1097 if (evt.stopPropagation) 1098 { 1099 evt.stopPropagation(); 1100 } 1101 1102 evt.cancelBubble = true; 1103 }); 1104 1105 // Workaround for blocked text selection in Editor 1106 mxEvent.addListener(input, 'selectstart', function(evt) 1107 { 1108 if (evt.stopPropagation) 1109 { 1110 evt.stopPropagation(); 1111 } 1112 1113 evt.cancelBubble = true; 1114 }); 1115 1116 var outer = document.createElement('div'); 1117 outer.appendChild(div); 1118 this.container.appendChild(outer); 1119 1120 // Keeps references to the DOM nodes 1121 this.palettes['search'] = [elt, outer]; 1122}; 1123 1124/** 1125 * Adds the general palette to the sidebar. 1126 */ 1127Sidebar.prototype.insertSearchHint = function(div, searchTerm, count, page, results, len, more, terms) 1128{ 1129 if (results.length == 0 && page == 1) 1130 { 1131 var err = document.createElement('div'); 1132 err.className = 'geTitle'; 1133 err.style.cssText = 'background-color:transparent;border-color:transparent;' + 1134 'color:gray;padding:6px 0px 0px 0px !important;margin:4px 8px 4px 8px;' + 1135 'text-align:center;cursor:default !important'; 1136 1137 mxUtils.write(err, mxResources.get('noResultsFor', [searchTerm])); 1138 div.appendChild(err); 1139 } 1140}; 1141 1142/** 1143 * Adds the general palette to the sidebar. 1144 */ 1145Sidebar.prototype.addGeneralPalette = function(expand) 1146{ 1147 var lineTags = 'line lines connector connectors connection connections arrow arrows '; 1148 this.setCurrentSearchEntryLibrary('general', 'general'); 1149 var sb = this; 1150 1151 var temp = parseInt(this.editorUi.editor.graph.defaultVertexStyle['fontSize']); 1152 var fontSize = !isNaN(temp) ? 'fontSize=' + Math.min(16, temp) + ';' : ''; 1153 1154 // Reusable cells 1155 var field = new mxCell('List Item', new mxGeometry(0, 0, 80, 30), 1156 'text;strokeColor=none;fillColor=none;align=left;verticalAlign=middle;' + 1157 'spacingLeft=4;spacingRight=4;overflow=hidden;points=[[0,0.5],[1,0.5]];' + 1158 'portConstraint=eastwest;rotatable=0;' + fontSize); 1159 field.vertex = true; 1160 1161 var fns = [ 1162 this.createVertexTemplateEntry('rounded=0;whiteSpace=wrap;html=1;', 120, 60, '', 'Rectangle', null, null, 'rect rectangle box'), 1163 this.createVertexTemplateEntry('rounded=1;whiteSpace=wrap;html=1;', 120, 60, '', 'Rounded Rectangle', null, null, 'rounded rect rectangle box'), 1164 // Explicit strokecolor/fillcolor=none is a workaround to maintain transparent background regardless of current style 1165 this.createVertexTemplateEntry('text;html=1;strokeColor=none;fillColor=none;align=center;verticalAlign=middle;whiteSpace=wrap;rounded=0;', 1166 60, 30, 'Text', 'Text', null, null, 'text textbox textarea label'), 1167 this.createVertexTemplateEntry('text;html=1;strokeColor=none;fillColor=none;spacing=5;spacingTop=-20;whiteSpace=wrap;overflow=hidden;rounded=0;', 190, 120, 1168 '<h1>Heading</h1><p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.</p>', 1169 'Textbox', null, null, 'text textbox textarea'), 1170 this.createVertexTemplateEntry('ellipse;whiteSpace=wrap;html=1;', 120, 80, '', 'Ellipse', null, null, 'oval ellipse state'), 1171 this.createVertexTemplateEntry('whiteSpace=wrap;html=1;aspect=fixed;', 80, 80, '', 'Square', null, null, 'square'), 1172 this.createVertexTemplateEntry('ellipse;whiteSpace=wrap;html=1;aspect=fixed;', 80, 80, '', 'Circle', null, null, 'circle'), 1173 this.createVertexTemplateEntry('shape=process;whiteSpace=wrap;html=1;backgroundOutline=1;', 120, 60, '', 'Process', null, null, 'process task'), 1174 this.createVertexTemplateEntry('rhombus;whiteSpace=wrap;html=1;', 80, 80, '', 'Diamond', null, null, 'diamond rhombus if condition decision conditional question test'), 1175 this.createVertexTemplateEntry('shape=parallelogram;perimeter=parallelogramPerimeter;whiteSpace=wrap;html=1;fixedSize=1;', 120, 60, '', 'Parallelogram'), 1176 this.createVertexTemplateEntry('shape=hexagon;perimeter=hexagonPerimeter2;whiteSpace=wrap;html=1;fixedSize=1;', 120, 80, '', 'Hexagon', null, null, 'hexagon preparation'), 1177 this.createVertexTemplateEntry('triangle;whiteSpace=wrap;html=1;', 60, 80, '', 'Triangle', null, null, 'triangle logic inverter buffer'), 1178 this.createVertexTemplateEntry('shape=cylinder3;whiteSpace=wrap;html=1;boundedLbl=1;backgroundOutline=1;size=15;', 60, 80, '', 'Cylinder', null, null, 'cylinder data database'), 1179 this.createVertexTemplateEntry('ellipse;shape=cloud;whiteSpace=wrap;html=1;', 120, 80, '', 'Cloud', null, null, 'cloud network'), 1180 this.createVertexTemplateEntry('shape=document;whiteSpace=wrap;html=1;boundedLbl=1;', 120, 80, '', 'Document'), 1181 this.createVertexTemplateEntry('shape=internalStorage;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Internal Storage'), 1182 this.createVertexTemplateEntry('shape=cube;whiteSpace=wrap;html=1;boundedLbl=1;backgroundOutline=1;darkOpacity=0.05;darkOpacity2=0.1;', 120, 80, '', 'Cube'), 1183 this.createVertexTemplateEntry('shape=step;perimeter=stepPerimeter;whiteSpace=wrap;html=1;fixedSize=1;', 120, 80, '', 'Step'), 1184 this.createVertexTemplateEntry('shape=trapezoid;perimeter=trapezoidPerimeter;whiteSpace=wrap;html=1;fixedSize=1;', 120, 60, '', 'Trapezoid'), 1185 this.createVertexTemplateEntry('shape=tape;whiteSpace=wrap;html=1;', 120, 100, '', 'Tape'), 1186 this.createVertexTemplateEntry('shape=note;whiteSpace=wrap;html=1;backgroundOutline=1;darkOpacity=0.05;', 80, 100, '', 'Note'), 1187 this.createVertexTemplateEntry('shape=card;whiteSpace=wrap;html=1;', 80, 100, '', 'Card'), 1188 this.createVertexTemplateEntry('shape=callout;whiteSpace=wrap;html=1;perimeter=calloutPerimeter;', 120, 80, '', 'Callout', null, null, 'bubble chat thought speech message'), 1189 this.createVertexTemplateEntry('shape=umlActor;verticalLabelPosition=bottom;verticalAlign=top;html=1;outlineConnect=0;', 30, 60, 'Actor', 'Actor', false, null, 'user person human stickman'), 1190 this.createVertexTemplateEntry('shape=xor;whiteSpace=wrap;html=1;', 60, 80, '', 'Or', null, null, 'logic or'), 1191 this.createVertexTemplateEntry('shape=or;whiteSpace=wrap;html=1;', 60, 80, '', 'And', null, null, 'logic and'), 1192 this.createVertexTemplateEntry('shape=dataStorage;whiteSpace=wrap;html=1;fixedSize=1;', 100, 80, '', 'Data Storage'), 1193 this.createVertexTemplateEntry('swimlane;startSize=0;', 200, 200, '', 'Container', null, null, 'container swimlane lane pool group'), 1194 this.createVertexTemplateEntry('swimlane;', 200, 200, 'Vertical Container', 'Container', null, null, 'container swimlane lane pool group'), 1195 this.createVertexTemplateEntry('swimlane;horizontal=0;', 200, 200, 'Horizontal Container', 'Horizontal Container', null, null, 'container swimlane lane pool group'), 1196 this.addEntry('list group erd table', function() 1197 { 1198 var cell = new mxCell('List', new mxGeometry(0, 0, 140, 120), 1199 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=30;horizontalStack=0;' + 1200 'resizeParent=1;resizeParentMax=0;resizeLast=0;collapsible=1;marginBottom=0;'); 1201 cell.vertex = true; 1202 cell.insert(sb.cloneCell(field, 'Item 1')); 1203 cell.insert(sb.cloneCell(field, 'Item 2')); 1204 cell.insert(sb.cloneCell(field, 'Item 3')); 1205 1206 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'List'); 1207 }), 1208 this.addEntry('list item entry value group erd table', function() 1209 { 1210 return sb.createVertexTemplateFromCells([sb.cloneCell(field, 'List Item')], field.geometry.width, field.geometry.height, 'List Item'); 1211 }), 1212 this.addEntry('curve', mxUtils.bind(this, function() 1213 { 1214 var cell = new mxCell('', new mxGeometry(0, 0, 50, 50), 'curved=1;endArrow=classic;html=1;'); 1215 cell.geometry.setTerminalPoint(new mxPoint(0, 50), true); 1216 cell.geometry.setTerminalPoint(new mxPoint(50, 0), false); 1217 cell.geometry.points = [new mxPoint(50, 50), new mxPoint(0, 0)]; 1218 cell.geometry.relative = true; 1219 cell.edge = true; 1220 1221 return this.createEdgeTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Curve'); 1222 })), 1223 this.createEdgeTemplateEntry('shape=flexArrow;endArrow=classic;startArrow=classic;html=1;', 100, 100, '', 'Bidirectional Arrow', null, lineTags + 'bidirectional'), 1224 this.createEdgeTemplateEntry('shape=flexArrow;endArrow=classic;html=1;', 50, 50, '', 'Arrow', null, lineTags + 'directional directed'), 1225 this.createEdgeTemplateEntry('endArrow=none;dashed=1;html=1;', 50, 50, '', 'Dashed Line', null, lineTags + 'dashed undirected no'), 1226 this.createEdgeTemplateEntry('endArrow=none;dashed=1;html=1;dashPattern=1 3;strokeWidth=2;', 50, 50, '', 'Dotted Line', null, lineTags + 'dotted undirected no'), 1227 this.createEdgeTemplateEntry('endArrow=none;html=1;', 50, 50, '', 'Line', null, lineTags + 'simple undirected plain blank no'), 1228 this.createEdgeTemplateEntry('endArrow=classic;startArrow=classic;html=1;', 50, 50, '', 'Bidirectional Connector', null, lineTags + 'bidirectional'), 1229 this.createEdgeTemplateEntry('endArrow=classic;html=1;', 50, 50, '', 'Directional Connector', null, lineTags + 'directional directed'), 1230 this.createEdgeTemplateEntry('shape=link;html=1;', 100, 0, '', 'Link', null, lineTags + 'link'), 1231 this.addEntry(lineTags + 'edge title', mxUtils.bind(this, function() 1232 { 1233 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'endArrow=classic;html=1;'); 1234 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1235 edge.geometry.setTerminalPoint(new mxPoint(100, 0), false); 1236 edge.geometry.relative = true; 1237 edge.edge = true; 1238 1239 var cell0 = new mxCell('Label', new mxGeometry(0, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=center;verticalAlign=middle;'); 1240 cell0.geometry.relative = true; 1241 cell0.setConnectable(false); 1242 cell0.vertex = true; 1243 edge.insert(cell0); 1244 1245 return this.createEdgeTemplateFromCells([edge], 100, 0, 'Connector with Label'); 1246 })), 1247 this.addEntry(lineTags + 'edge title multiplicity', mxUtils.bind(this, function() 1248 { 1249 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'endArrow=classic;html=1;'); 1250 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1251 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 1252 edge.geometry.relative = true; 1253 edge.edge = true; 1254 1255 var cell0 = new mxCell('Label', new mxGeometry(0, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=center;verticalAlign=middle;'); 1256 cell0.geometry.relative = true; 1257 cell0.setConnectable(false); 1258 cell0.vertex = true; 1259 edge.insert(cell0); 1260 1261 var cell1 = new mxCell('Source', new mxGeometry(-1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=left;verticalAlign=bottom;'); 1262 cell1.geometry.relative = true; 1263 cell1.setConnectable(false); 1264 cell1.vertex = true; 1265 edge.insert(cell1); 1266 1267 return this.createEdgeTemplateFromCells([edge], 160, 0, 'Connector with 2 Labels'); 1268 })), 1269 this.addEntry(lineTags + 'edge title multiplicity', mxUtils.bind(this, function() 1270 { 1271 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'endArrow=classic;html=1;'); 1272 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1273 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 1274 edge.geometry.relative = true; 1275 edge.edge = true; 1276 1277 var cell0 = new mxCell('Label', new mxGeometry(0, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=center;verticalAlign=middle;'); 1278 cell0.geometry.relative = true; 1279 cell0.setConnectable(false); 1280 cell0.vertex = true; 1281 edge.insert(cell0); 1282 1283 var cell1 = new mxCell('Source', new mxGeometry(-1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=left;verticalAlign=bottom;'); 1284 cell1.geometry.relative = true; 1285 cell1.setConnectable(false); 1286 cell1.vertex = true; 1287 edge.insert(cell1); 1288 1289 var cell2 = new mxCell('Target', new mxGeometry(1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=right;verticalAlign=bottom;'); 1290 cell2.geometry.relative = true; 1291 cell2.setConnectable(false); 1292 cell2.vertex = true; 1293 edge.insert(cell2); 1294 1295 return this.createEdgeTemplateFromCells([edge], 160, 0, 'Connector with 3 Labels'); 1296 })), 1297 this.addEntry(lineTags + 'edge shape symbol message mail email', mxUtils.bind(this, function() 1298 { 1299 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'endArrow=classic;html=1;'); 1300 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1301 edge.geometry.setTerminalPoint(new mxPoint(100, 0), false); 1302 edge.geometry.relative = true; 1303 edge.edge = true; 1304 1305 var cell = new mxCell('', new mxGeometry(0, 0, 20, 14), 'shape=message;html=1;outlineConnect=0;'); 1306 cell.geometry.relative = true; 1307 cell.vertex = true; 1308 cell.geometry.offset = new mxPoint(-10, -7); 1309 edge.insert(cell); 1310 1311 return this.createEdgeTemplateFromCells([edge], 100, 0, 'Connector with Symbol'); 1312 })) 1313 ]; 1314 1315 this.addPaletteFunctions('general', mxResources.get('general'), (expand != null) ? expand : true, fns); 1316 this.setCurrentSearchEntryLibrary(); 1317}; 1318 1319/** 1320 * Adds the general palette to the sidebar. 1321 */ 1322Sidebar.prototype.addMiscPalette = function(expand) 1323{ 1324 var sb = this; 1325 var lineTags = 'line lines connector connectors connection connections arrow arrows ' 1326 this.setCurrentSearchEntryLibrary('general', 'misc'); 1327 1328 var fns = [ 1329 this.createVertexTemplateEntry('text;strokeColor=none;fillColor=none;html=1;fontSize=24;fontStyle=1;verticalAlign=middle;align=center;', 100, 40, 'Title', 'Title', null, null, 'text heading title'), 1330 this.createVertexTemplateEntry('text;strokeColor=none;fillColor=none;html=1;whiteSpace=wrap;verticalAlign=middle;overflow=hidden;', 100, 80, 1331 '<ul><li>Value 1</li><li>Value 2</li><li>Value 3</li></ul>', 'Unordered List'), 1332 this.createVertexTemplateEntry('text;strokeColor=none;fillColor=none;html=1;whiteSpace=wrap;verticalAlign=middle;overflow=hidden;', 100, 80, 1333 '<ol><li>Value 1</li><li>Value 2</li><li>Value 3</li></ol>', 'Ordered List'), 1334 this.addDataEntry('table', 180, 120, 'Table 1', '7ZjBTuMwEIafJtdVnFDoXhtYLnBZeAHTTGtLjieyB9Lu0zNO3FYVVM0upMASKZU8Y49jf3+tX06SF9Xq2sla3WIJJsmvkrxwiNS1qlUBxiRZqsskv0yyLOVfkv060Cva3rSWDiz1Kci6gidpHqHLdAlPaxMTXsk6NEk+hNRMUcXLvBTcbJQmuKvlPPQ3vAnOeZKO7vSfkEo5nqMlqS24WDNHY2TtdTtZN0JpU97INT7S5jWbaBaXB45gdXCLbSru7xqwAnJrHtLoklQcMe0wpAr0Um3KIptU+i6x3NbuiHEjQnsdYN4fIC+ZtDS/YU7SLvuwfMmqdFjfS7cEiokatSVwV09Mw8fcQhtToMGA3KINLyKsY6eBxab2AYmwioGLYLaTtlAmM34YU5H+mCQT3k3BsdjF/IThjgq0nhwLHeYA6akB31vA7N8EPHsH/c6G1c/aMHqn35ulQUa5MNhwqHRZgn3lPyD6cs+Pcj8fCPtkxP4C+2of+QlUOB9VOKTC1hxOIMPF6CGDech6X6ZTWMr0O5+qiw+zlJ8j9k9gKSIdZfgMniLEaCpDm8r0hKYi/uKm/v+dK3H8ij/YQRr4hv81wX+AsXzrm/oRHQZ0Fg53nyPbvr2vlc8='), 1335 this.addDataEntry('table', 180, 120, 'Table 2', '7ZhRb5swEMc/Da+TDSFJX0u7vrQv7bR3NzjBkvEhcynJPv3OYDeLUlS0JURTkUDyne+M+f05nXCUZOXuwYqqeIJc6ii5j5LMAmA3KneZ1DqKmcqj5C6KY0Z3FH/vmeXtLKuElQaHJMRdwpvQW9l5OkeNe+0ddSEqN0Tx6ly3BZa0zTtOw6ZQKF8qsXLzDb0E+WoUFl/UL+diZK/AoFBGWp+zAq1FVat2sS6iUDp/FHvYYnhMsFz0tjSPlF/7aAvNn6bfv7Qod70MWpcH8CChlGj3FNKoHAsfsew4sUKqTRHSPDwm6s6xec89IKWBp/ox4WQ4YdoyKqGf5QqF2QyBfQozt1D9EHYj0TsqUAalvX8jGgHaWmmdgQaniQHjHoRQ+Ukt1yH3FRChDOA9mPdFWyjpLV2EKWPf0iilt8nI5gebLhduMQNTo6Uvwa0hRY2NrHGogPHfCTg7g36zE/1+tsOY8bMKaYyLPgj5zxoBMV1raMgsVJ5L88HHwIcKkHwqwPxC/NNe/uctpP+S/+6Y/QhyzHvlSCY5QkI8nh6Lqb1crL3sj2Uao9sse8tr9oXKa3G1bnPTyz+d+I/fbTjr1WM+6XGFdsP51G8u3W+WI/YbfnoAEAps8YUKjH9+QnCxijo9IAgKLCcFrtFz+v/4byZBRmg6ZB4OQNu5o/PR3w=='), 1336 this.addDataEntry('table title', 180, 150, 'Table with Title 1', '7VjBbqMwEP0arisMpeleQ7u9tJemP+CGCbZkbGRPQtKv37FxGnWTKJVa0q6CBJLnecbg9xg9mSQvm/W95a14NBWoJL9L8tIag/2oWZegVJKlskry2yTLUrqT7M+RWRZm05Zb0PiRgqwvWHG1hB555i8KetThRkXUCd76IYbZfOqQW5zJV4/lKQFzo5FLDZYAFmKleOtkSL8NGUKq6oFvzBK3C22j6YKqZ/FpvporWWsaz2kbfslpa6Qf3a0IcDEpvjpYhPXR7Qco7v0eTANoN5TSyQpFzLjpKUoFyFpsy4oIctcD9Vvtjk0aREIPk5vvkXuUV3pllFw9wRy5rgPFAhsVd9oJiTBr+dyndvSxHCS4sqZ95rYGjMC/pHlsIZUqjTJeJ220fxCaNk4qWGxrXwyiaWJgIzFviwZSiildRFOZ/iqSgnZTUsx2MV0+3WJptENLX4dfA7jDDhx+VMDssICbSHGU6ZSeV18g59Wwcmrts3dyflopQ8wulOkoFLKqQH+mj/KTfXQ9EO3FSPse7ev3lJ9BhetRhWMqsOx8MkxGSxnaUiZntJSbS+6qybdZyu+R9h9gKSwdZfgJnsLYaCpDmwpjZ3QVtn+qv6DGYqeP/IN10sAn/v+T+G9wlos+qp/QYUBroXD36zLMvfuz+Rc='), 1337 this.addDataEntry('table title', 180, 150, 'Table with Title 2', '7VjBbqMwEP0arivAJdtcod1e0ktT7d0NBiwZG9nTQvbrd2xMsy1B7WoTglaVQJp5nhnwe/gdCEhWd3eaNtW9ypkIyG1AMq0U9FHdZUyIIA55HpCbII5DvIP4x8Rq5FbDhmom4TMNcd/wQsUz65FH+iRYjxrYC4+aijY2BLdKUgNUw5b/shgJEdgpCZRLphGIXC4EbQx35TeuouIi39C9eoZh0JClBe9Y/qBa47u1ajc4zPjWAodv/cvYZSp4KTHe4S7tE9NGcRvdviAwzPA7YxpYN8mOgzw1d0zVDPQeS1qeQ+UrrnsGw4rxshraEg9S0wPla++BbAw838e5JyPuJ2nHVwZOxQPbAZWlU6CCWvidthUHtm3ozpa2+C0d5T/XqnmkumTggWOkFVyITAllZZRK2geBanyDYMXQ+6QAVD2o5Yl5HepISVK8kKYs/JYECe4mwzw65HjZcg2ZkgY0fjx2BqMGWmbgswLGxwXce4q9TB/pSU4g59VIzuikekppqw96/rVU4TupFFJbCNViWvE8Z/JfDhL58CBdnYn3ZMT7TxfG4cL5H1ysr00NPpDLctN3rmYSqHsrzp8H5VyCrf4H33sv5rJ8bzWj730fyXlaPZfte6uL+d71pO8tnP95fW9CoAv43vrL987te+sZfS8KR3qSZR+8k56r9cWML4omnW/hAszrfBMKnd/5MD38wXBrb35w/AY='), 1338 this.addDataEntry('crossfunctional cross-functional cross functional flowchart swimlane table', 400, 400, 'Cross-Functional Flowchart', '5ZjfbpswFMafhstNGJK2u1xY05tOmtYn8OAstmRsZB8C6dPvmD9NN6CNqoStTCKSz/GxcX6f+ZAJ4iSv7ywvxFeTgQri2yBOrDHYtvI6AaWCKJRZEH8JoiikXxBtJ3pZ0xsW3ILGUwZE7YA9VyW0Gbq5cx+2pU5RGs198VaZKhXcYlvs8KC6Yid44ZvIf/jUJhVSZff8YErss320saa6lxocdYS+1Kgy188zDukWD/LRT7jqShQvnGzmbkospKV1cg/fwbWFPgt1wXXWBYWRGsHe7olAP/NPo/GhWzWjmCu509ROwZdSoqMAFqGeJNmkOox3YHJAe6CSSmYo2opV2NIOBcidwD+S3LWJ3dPYozDU6LQZ1yke6PQ5RWMpxYaiVDJXXHs9hLHykf48yfiMTrOO9YYuWlkSflwHaxqXUMyOMV2+3GJitEPLZYMBuMMKHM4o1ynSROPSHHoF2hGvKcWiMyi1Gij1TXAHrymVGq0h7Z6jCz8OJyGNx5HWk0if6J0d6XoSabQgpOxqTqZXk0zjBTGNbuZkej1p0i/u0/dp0mfw5bEdfzljvhmo8/aNPj/J63/Hjj8tGOS8JszCBaOc13sZG6DszffFF9p/a75j8lzOfNnwoPmedjqbOPv9Dftlw7PgglDObMDD09qCWF7UgSk8fqZq+n77ivUL'), 1339 this.createVertexTemplateEntry('text;html=1;strokeColor=#c0c0c0;fillColor=#ffffff;overflow=fill;rounded=0;', 280, 160, 1340 '<table border="1" width="100%" height="100%" cellpadding="4" style="width:100%;height:100%;border-collapse:collapse;">' + 1341 '<tr style="background-color:#A7C942;color:#ffffff;border:1px solid #98bf21;"><th align="left">Title 1</th><th align="left">Title 2</th><th align="left">Title 3</th></tr>' + 1342 '<tr style="border:1px solid #98bf21;"><td>Value 1</td><td>Value 2</td><td>Value 3</td></tr>' + 1343 '<tr style="background-color:#EAF2D3;border:1px solid #98bf21;"><td>Value 4</td><td>Value 5</td><td>Value 6</td></tr>' + 1344 '<tr style="border:1px solid #98bf21;"><td>Value 7</td><td>Value 8</td><td>Value 9</td></tr>' + 1345 '<tr style="background-color:#EAF2D3;border:1px solid #98bf21;"><td>Value 10</td><td>Value 11</td><td>Value 12</td></tr></table>', 'HTML Table 1'), 1346 this.createVertexTemplateEntry('text;html=1;strokeColor=#c0c0c0;fillColor=none;overflow=fill;', 180, 140, 1347 '<table border="0" width="100%" height="100%" style="width:100%;height:100%;border-collapse:collapse;">' + 1348 '<tr><td align="center">Value 1</td><td align="center">Value 2</td><td align="center">Value 3</td></tr>' + 1349 '<tr><td align="center">Value 4</td><td align="center">Value 5</td><td align="center">Value 6</td></tr>' + 1350 '<tr><td align="center">Value 7</td><td align="center">Value 8</td><td align="center">Value 9</td></tr></table>', 'HTML Table 2'), 1351 this.createVertexTemplateEntry('text;html=1;strokeColor=none;fillColor=none;overflow=fill;', 180, 140, 1352 '<table border="1" width="100%" height="100%" style="width:100%;height:100%;border-collapse:collapse;">' + 1353 '<tr><td align="center">Value 1</td><td align="center">Value 2</td><td align="center">Value 3</td></tr>' + 1354 '<tr><td align="center">Value 4</td><td align="center">Value 5</td><td align="center">Value 6</td></tr>' + 1355 '<tr><td align="center">Value 7</td><td align="center">Value 8</td><td align="center">Value 9</td></tr></table>', 'HTML Table 3'), 1356 this.createVertexTemplateEntry('text;html=1;strokeColor=none;fillColor=none;overflow=fill;', 160, 140, 1357 '<table border="1" width="100%" height="100%" cellpadding="4" style="width:100%;height:100%;border-collapse:collapse;">' + 1358 '<tr><th align="center"><b>Title</b></th></tr>' + 1359 '<tr><td align="center">Section 1.1\nSection 1.2\nSection 1.3</td></tr>' + 1360 '<tr><td align="center">Section 2.1\nSection 2.2\nSection 2.3</td></tr></table>', 'HTML Table 4'), 1361 this.addEntry('link hyperlink', mxUtils.bind(this, function() 1362 { 1363 var cell = new mxCell('Link', new mxGeometry(0, 0, 60, 40), 'text;html=1;strokeColor=none;fillColor=none;whiteSpace=wrap;align=center;verticalAlign=middle;fontColor=#0000EE;fontStyle=4;'); 1364 cell.vertex = true; 1365 this.graph.setLinkForCell(cell, 'https://www.draw.io'); 1366 1367 return this.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Link'); 1368 })), 1369 this.addEntry('timestamp date time text label', mxUtils.bind(this, function() 1370 { 1371 var cell = new mxCell('%date{ddd mmm dd yyyy HH:MM:ss}%', new mxGeometry(0, 0, 160, 20), 'text;html=1;strokeColor=none;fillColor=none;align=center;verticalAlign=middle;whiteSpace=wrap;overflow=hidden;'); 1372 cell.vertex = true; 1373 this.graph.setAttributeForCell(cell, 'placeholders', '1'); 1374 1375 return this.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Timestamp'); 1376 })), 1377 this.addEntry('variable placeholder metadata hello world text label', mxUtils.bind(this, function() 1378 { 1379 var cell = new mxCell('%name% Text', new mxGeometry(0, 0, 80, 20), 'text;html=1;strokeColor=none;fillColor=none;align=center;verticalAlign=middle;whiteSpace=wrap;overflow=hidden;'); 1380 cell.vertex = true; 1381 this.graph.setAttributeForCell(cell, 'placeholders', '1'); 1382 this.graph.setAttributeForCell(cell, 'name', 'Variable'); 1383 1384 return this.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Variable'); 1385 })), 1386 this.createVertexTemplateEntry('shape=ext;double=1;rounded=0;whiteSpace=wrap;html=1;', 120, 80, '', 'Double Rectangle', null, null, 'rect rectangle box double'), 1387 this.createVertexTemplateEntry('shape=ext;double=1;rounded=1;whiteSpace=wrap;html=1;', 120, 80, '', 'Double Rounded Rectangle', null, null, 'rounded rect rectangle box double'), 1388 this.createVertexTemplateEntry('ellipse;shape=doubleEllipse;whiteSpace=wrap;html=1;', 100, 60, '', 'Double Ellipse', null, null, 'oval ellipse start end state double'), 1389 this.createVertexTemplateEntry('shape=ext;double=1;whiteSpace=wrap;html=1;aspect=fixed;', 80, 80, '', 'Double Square', null, null, 'double square'), 1390 this.createVertexTemplateEntry('ellipse;shape=doubleEllipse;whiteSpace=wrap;html=1;aspect=fixed;', 80, 80, '', 'Double Circle', null, null, 'double circle'), 1391 this.createVertexTemplateEntry('rounded=1;whiteSpace=wrap;html=1;strokeWidth=2;fillWeight=4;hachureGap=8;hachureAngle=45;fillColor=#1ba1e2;sketch=1;', 120, 60, '', 'Rectangle Sketch', true, null, 'rectangle rect box text sketch comic retro'), 1392 this.createVertexTemplateEntry('ellipse;whiteSpace=wrap;html=1;strokeWidth=2;fillWeight=2;hachureGap=8;fillColor=#990000;fillStyle=dots;sketch=1;', 120, 60, '', 'Ellipse Sketch', true, null, 'ellipse oval sketch comic retro'), 1393 this.createVertexTemplateEntry('rhombus;whiteSpace=wrap;html=1;strokeWidth=2;fillWeight=-1;hachureGap=8;fillStyle=cross-hatch;fillColor=#006600;sketch=1;', 120, 60, '', 'Diamond Sketch', true, null, 'diamond sketch comic retro'), 1394 this.createVertexTemplateEntry('html=1;whiteSpace=wrap;shape=isoCube2;backgroundOutline=1;isoAngle=15;', 90, 100, '', 'Isometric Cube', true, null, 'cube box iso isometric'), 1395 this.createVertexTemplateEntry('html=1;whiteSpace=wrap;aspect=fixed;shape=isoRectangle;', 150, 90, '', 'Isometric Square', true, null, 'rectangle rect box iso isometric'), 1396 this.createEdgeTemplateEntry('edgeStyle=isometricEdgeStyle;endArrow=none;html=1;', 50, 100, '', 'Isometric Edge 1'), 1397 this.createEdgeTemplateEntry('edgeStyle=isometricEdgeStyle;endArrow=none;html=1;elbow=vertical;', 50, 100, '', 'Isometric Edge 2'), 1398 this.createVertexTemplateEntry('shape=curlyBracket;whiteSpace=wrap;html=1;rounded=1;', 20, 120, '', 'Left Curly Bracket'), 1399 this.createVertexTemplateEntry('shape=curlyBracket;whiteSpace=wrap;html=1;rounded=1;flipH=1;', 20, 120, '', 'Right Curly Bracket'), 1400 this.createVertexTemplateEntry('line;strokeWidth=2;html=1;', 160, 10, '', 'Horizontal Line'), 1401 this.createVertexTemplateEntry('line;strokeWidth=2;direction=south;html=1;', 10, 160, '', 'Vertical Line'), 1402 this.createVertexTemplateEntry('line;strokeWidth=4;html=1;perimeter=backbonePerimeter;points=[];outlineConnect=0;', 160, 10, '', 'Horizontal Backbone', false, null, 'backbone bus network'), 1403 this.createVertexTemplateEntry('line;strokeWidth=4;direction=south;html=1;perimeter=backbonePerimeter;points=[];outlineConnect=0;', 10, 160, '', 'Vertical Backbone', false, null, 'backbone bus network'), 1404 this.createVertexTemplateEntry('shape=crossbar;whiteSpace=wrap;html=1;rounded=1;', 120, 20, '', 'Horizontal Crossbar', false, null, 'crossbar distance measure dimension unit'), 1405 this.createVertexTemplateEntry('shape=crossbar;whiteSpace=wrap;html=1;rounded=1;direction=south;', 20, 120, '', 'Vertical Crossbar', false, null, 'crossbar distance measure dimension unit'), 1406 this.createVertexTemplateEntry('shape=image;html=1;verticalLabelPosition=bottom;verticalAlign=top;imageAspect=1;aspect=fixed;image=' + this.gearImage, 52, 61, '', 'Image (Fixed Aspect)', false, null, 'fixed image icon symbol'), 1407 this.createVertexTemplateEntry('shape=image;html=1;verticalLabelPosition=bottom;verticalAlign=top;imageAspect=0;image=' + this.gearImage, 50, 60, '', 'Image (Variable Aspect)', false, null, 'strechted image icon symbol'), 1408 this.createVertexTemplateEntry('icon;html=1;image=' + this.gearImage, 60, 60, 'Icon', 'Icon', false, null, 'icon image symbol'), 1409 this.createVertexTemplateEntry('label;whiteSpace=wrap;html=1;image=' + this.gearImage, 140, 60, 'Label', 'Label 1', null, null, 'label image icon symbol'), 1410 this.createVertexTemplateEntry('label;whiteSpace=wrap;html=1;align=center;verticalAlign=bottom;spacingLeft=0;spacingBottom=4;imageAlign=center;imageVerticalAlign=top;image=' + this.gearImage, 120, 80, 'Label', 'Label 2', null, null, 'label image icon symbol'), 1411 this.addEntry('shape group container', function() 1412 { 1413 var cell = new mxCell('Label', new mxGeometry(0, 0, 160, 70), 1414 'html=1;whiteSpace=wrap;container=1;recursiveResize=0;collapsible=0;'); 1415 cell.vertex = true; 1416 1417 var symbol = new mxCell('', new mxGeometry(20, 20, 20, 30), 'triangle;html=1;whiteSpace=wrap;'); 1418 symbol.vertex = true; 1419 cell.insert(symbol); 1420 1421 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Shape Group'); 1422 }), 1423 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;left=0;right=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1424 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;bottom=0;top=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1425 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;bottom=0;right=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1426 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;bottom=1;right=1;left=1;top=0;fillColor=none;routingCenterX=-0.5;', 120, 60, '', 'Partial Rectangle'), 1427 this.createVertexTemplateEntry('shape=waypoint;sketch=0;fillStyle=solid;size=6;pointerEvents=1;points=[];fillColor=none;resizable=0;rotatable=0;perimeter=centerPerimeter;snapToPoint=1;', 40, 40, '', 'Waypoint'), 1428 this.createEdgeTemplateEntry('edgeStyle=segmentEdgeStyle;endArrow=classic;html=1;', 50, 50, '', 'Manual Line', null, lineTags + 'manual'), 1429 this.createEdgeTemplateEntry('shape=filledEdge;rounded=0;fixDash=1;endArrow=none;strokeWidth=10;fillColor=#ffffff;edgeStyle=orthogonalEdgeStyle;', 60, 40, '', 'Filled Edge'), 1430 this.createEdgeTemplateEntry('edgeStyle=elbowEdgeStyle;elbow=horizontal;endArrow=classic;html=1;', 50, 50, '', 'Horizontal Elbow', null, lineTags + 'elbow horizontal'), 1431 this.createEdgeTemplateEntry('edgeStyle=elbowEdgeStyle;elbow=vertical;endArrow=classic;html=1;', 50, 50, '', 'Vertical Elbow', null, lineTags + 'elbow vertical') 1432 ]; 1433 1434 this.addPaletteFunctions('misc', mxResources.get('misc'), (expand != null) ? expand : true, fns); 1435 this.setCurrentSearchEntryLibrary(); 1436}; 1437/** 1438 * Adds the container palette to the sidebar. 1439 */ 1440Sidebar.prototype.addAdvancedPalette = function(expand) 1441{ 1442 this.setCurrentSearchEntryLibrary('general', 'advanced'); 1443 this.addPaletteFunctions('advanced', mxResources.get('advanced'), (expand != null) ? expand : false, this.createAdvancedShapes()); 1444 this.setCurrentSearchEntryLibrary(); 1445}; 1446 1447/** 1448 * Adds the general palette to the sidebar. 1449 */ 1450Sidebar.prototype.addBasicPalette = function(dir) 1451{ 1452 this.setCurrentSearchEntryLibrary('basic'); 1453 this.addStencilPalette('basic', mxResources.get('basic'), dir + '/basic.xml', 1454 ';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2', 1455 null, null, null, null, [ 1456 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;top=0;bottom=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1457 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;right=0;top=0;bottom=0;fillColor=none;routingCenterX=-0.5;', 120, 60, '', 'Partial Rectangle'), 1458 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;bottom=0;right=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1459 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;top=0;left=0;fillColor=none;', 120, 60, '', 'Partial Rectangle') 1460 ]); 1461 this.setCurrentSearchEntryLibrary(); 1462}; 1463 1464/** 1465 * Adds the container palette to the sidebar. 1466 */ 1467Sidebar.prototype.createAdvancedShapes = function() 1468{ 1469 // Avoids having to bind all functions to "this" 1470 var sb = this; 1471 1472 // Reusable cells 1473 var field = new mxCell('List Item', new mxGeometry(0, 0, 60, 26), 'text;strokeColor=none;fillColor=none;align=left;verticalAlign=top;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;'); 1474 field.vertex = true; 1475 1476 return [ 1477 this.createVertexTemplateEntry('shape=tapeData;whiteSpace=wrap;html=1;perimeter=ellipsePerimeter;', 80, 80, '', 'Tape Data'), 1478 this.createVertexTemplateEntry('shape=manualInput;whiteSpace=wrap;html=1;', 80, 80, '', 'Manual Input'), 1479 this.createVertexTemplateEntry('shape=loopLimit;whiteSpace=wrap;html=1;', 100, 80, '', 'Loop Limit'), 1480 this.createVertexTemplateEntry('shape=offPageConnector;whiteSpace=wrap;html=1;', 80, 80, '', 'Off Page Connector'), 1481 this.createVertexTemplateEntry('shape=delay;whiteSpace=wrap;html=1;', 80, 40, '', 'Delay'), 1482 this.createVertexTemplateEntry('shape=display;whiteSpace=wrap;html=1;', 80, 40, '', 'Display'), 1483 this.createVertexTemplateEntry('shape=singleArrow;direction=west;whiteSpace=wrap;html=1;', 100, 60, '', 'Arrow Left'), 1484 this.createVertexTemplateEntry('shape=singleArrow;whiteSpace=wrap;html=1;', 100, 60, '', 'Arrow Right'), 1485 this.createVertexTemplateEntry('shape=singleArrow;direction=north;whiteSpace=wrap;html=1;', 60, 100, '', 'Arrow Up'), 1486 this.createVertexTemplateEntry('shape=singleArrow;direction=south;whiteSpace=wrap;html=1;', 60, 100, '', 'Arrow Down'), 1487 this.createVertexTemplateEntry('shape=doubleArrow;whiteSpace=wrap;html=1;', 100, 60, '', 'Double Arrow'), 1488 this.createVertexTemplateEntry('shape=doubleArrow;direction=south;whiteSpace=wrap;html=1;', 60, 100, '', 'Double Arrow Vertical', null, null, 'double arrow'), 1489 this.createVertexTemplateEntry('shape=actor;whiteSpace=wrap;html=1;', 40, 60, '', 'User', null, null, 'user person human'), 1490 this.createVertexTemplateEntry('shape=cross;whiteSpace=wrap;html=1;', 80, 80, '', 'Cross'), 1491 this.createVertexTemplateEntry('shape=corner;whiteSpace=wrap;html=1;', 80, 80, '', 'Corner'), 1492 this.createVertexTemplateEntry('shape=tee;whiteSpace=wrap;html=1;', 80, 80, '', 'Tee'), 1493 this.createVertexTemplateEntry('shape=datastore;whiteSpace=wrap;html=1;', 60, 60, '', 'Data Store', null, null, 'data store cylinder database'), 1494 this.createVertexTemplateEntry('shape=orEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Or', null, null, 'or circle oval ellipse'), 1495 this.createVertexTemplateEntry('shape=sumEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Sum', null, null, 'sum circle oval ellipse'), 1496 this.createVertexTemplateEntry('shape=lineEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Ellipse with horizontal divider', null, null, 'circle oval ellipse'), 1497 this.createVertexTemplateEntry('shape=lineEllipse;line=vertical;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Ellipse with vertical divider', null, null, 'circle oval ellipse'), 1498 this.createVertexTemplateEntry('shape=sortShape;perimeter=rhombusPerimeter;whiteSpace=wrap;html=1;', 80, 80, '', 'Sort', null, null, 'sort'), 1499 this.createVertexTemplateEntry('shape=collate;whiteSpace=wrap;html=1;', 80, 80, '', 'Collate', null, null, 'collate'), 1500 this.createVertexTemplateEntry('shape=switch;whiteSpace=wrap;html=1;', 60, 60, '', 'Switch', null, null, 'switch router'), 1501 this.addEntry('process bar', function() 1502 { 1503 return sb.createVertexTemplateFromData('zZXRaoMwFIafJpcDjbNrb2233rRQ8AkyPdPQaCRJV+3T7yTG2rUVBoOtgpDzn/xJzncCIdGyateKNeVW5iBI9EqipZLS9KOqXYIQhAY8J9GKUBrgT+jbRDZ02aBhCmrzEwPtDZ9MHKBXdkpmoDWKCVN9VptO+Kw+8kqwGqMkK7nIN6yTB7uTNizbD1FSSsVPsjYMC1qFKHxwIZZSSIVxLZ1/nJNar5+oQPMT7IYCrqUta1ENzuqGaeOFTArBGs3f3Vmtoo2Se7ja1h00kSoHK4bBIKUNy3hdoPYU0mF91i9mT8EEL2ocZ3gKa00ayWujLZY4IfHKFonVDLsRGgXuQ90zBmWgneyTk3yT1iArMKrDKUeem9L3ajHrbSXwohxsQd/ggOleKM7ese048J2/fwuim1uQGmhQCW8vQMkacP3GCQgBFMftHEsr7cYYe95CnmKTPMFbYD8CQ++DGQy+/M5X4ku5wHYmdIktfvk9tecpavThqS3m/0YtnqIWPTy1cD77K2wYjo+Ay317I74A', 296, 100, 'Process Bar'); 1504 }), 1505 this.createVertexTemplateEntry('swimlane;', 200, 200, 'Container', 'Container', null, null, 'container swimlane lane pool group'), 1506 this.addEntry('list group erd table', function() 1507 { 1508 var cell = new mxCell('List', new mxGeometry(0, 0, 140, 110), 1509 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=none;horizontalStack=0;' + 1510 'resizeParent=1;resizeParentMax=0;resizeLast=0;collapsible=1;marginBottom=0;'); 1511 cell.vertex = true; 1512 cell.insert(sb.cloneCell(field, 'Item 1')); 1513 cell.insert(sb.cloneCell(field, 'Item 2')); 1514 cell.insert(sb.cloneCell(field, 'Item 3')); 1515 1516 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'List'); 1517 }), 1518 this.addEntry('list item entry value group erd table', function() 1519 { 1520 return sb.createVertexTemplateFromCells([sb.cloneCell(field, 'List Item')], field.geometry.width, field.geometry.height, 'List Item'); 1521 }) 1522 ]; 1523}; 1524 1525/** 1526 * Adds the container palette to the sidebar. 1527 */ 1528Sidebar.prototype.createAdvancedShapes = function() 1529{ 1530 // Avoids having to bind all functions to "this" 1531 var sb = this; 1532 1533 // Reusable cells 1534 var field = new mxCell('List Item', new mxGeometry(0, 0, 60, 26), 'text;strokeColor=none;fillColor=none;align=left;verticalAlign=top;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;'); 1535 field.vertex = true; 1536 1537 return [ 1538 this.createVertexTemplateEntry('shape=tapeData;whiteSpace=wrap;html=1;perimeter=ellipsePerimeter;', 80, 80, '', 'Tape Data'), 1539 this.createVertexTemplateEntry('shape=manualInput;whiteSpace=wrap;html=1;', 80, 80, '', 'Manual Input'), 1540 this.createVertexTemplateEntry('shape=loopLimit;whiteSpace=wrap;html=1;', 100, 80, '', 'Loop Limit'), 1541 this.createVertexTemplateEntry('shape=offPageConnector;whiteSpace=wrap;html=1;', 80, 80, '', 'Off Page Connector'), 1542 this.createVertexTemplateEntry('shape=delay;whiteSpace=wrap;html=1;', 80, 40, '', 'Delay'), 1543 this.createVertexTemplateEntry('shape=display;whiteSpace=wrap;html=1;', 80, 40, '', 'Display'), 1544 this.createVertexTemplateEntry('shape=singleArrow;direction=west;whiteSpace=wrap;html=1;', 100, 60, '', 'Arrow Left'), 1545 this.createVertexTemplateEntry('shape=singleArrow;whiteSpace=wrap;html=1;', 100, 60, '', 'Arrow Right'), 1546 this.createVertexTemplateEntry('shape=singleArrow;direction=north;whiteSpace=wrap;html=1;', 60, 100, '', 'Arrow Up'), 1547 this.createVertexTemplateEntry('shape=singleArrow;direction=south;whiteSpace=wrap;html=1;', 60, 100, '', 'Arrow Down'), 1548 this.createVertexTemplateEntry('shape=doubleArrow;whiteSpace=wrap;html=1;', 100, 60, '', 'Double Arrow'), 1549 this.createVertexTemplateEntry('shape=doubleArrow;direction=south;whiteSpace=wrap;html=1;', 60, 100, '', 'Double Arrow Vertical', null, null, 'double arrow'), 1550 this.createVertexTemplateEntry('shape=actor;whiteSpace=wrap;html=1;', 40, 60, '', 'User', null, null, 'user person human'), 1551 this.createVertexTemplateEntry('shape=cross;whiteSpace=wrap;html=1;', 80, 80, '', 'Cross'), 1552 this.createVertexTemplateEntry('shape=corner;whiteSpace=wrap;html=1;', 80, 80, '', 'Corner'), 1553 this.createVertexTemplateEntry('shape=tee;whiteSpace=wrap;html=1;', 80, 80, '', 'Tee'), 1554 this.createVertexTemplateEntry('shape=datastore;whiteSpace=wrap;html=1;', 60, 60, '', 'Data Store', null, null, 'data store cylinder database'), 1555 this.createVertexTemplateEntry('shape=orEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Or', null, null, 'or circle oval ellipse'), 1556 this.createVertexTemplateEntry('shape=sumEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Sum', null, null, 'sum circle oval ellipse'), 1557 this.createVertexTemplateEntry('shape=lineEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Ellipse with horizontal divider', null, null, 'circle oval ellipse'), 1558 this.createVertexTemplateEntry('shape=lineEllipse;line=vertical;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;backgroundOutline=1;', 80, 80, '', 'Ellipse with vertical divider', null, null, 'circle oval ellipse'), 1559 this.createVertexTemplateEntry('shape=sortShape;perimeter=rhombusPerimeter;whiteSpace=wrap;html=1;', 80, 80, '', 'Sort', null, null, 'sort'), 1560 this.createVertexTemplateEntry('shape=collate;whiteSpace=wrap;html=1;', 80, 80, '', 'Collate', null, null, 'collate'), 1561 this.createVertexTemplateEntry('shape=switch;whiteSpace=wrap;html=1;', 60, 60, '', 'Switch', null, null, 'switch router'), 1562 this.addEntry('process bar', function() 1563 { 1564 return sb.createVertexTemplateFromData('zZXRaoMwFIafJpcDjbNrb2233rRQ8AkyPdPQaCRJV+3T7yTG2rUVBoOtgpDzn/xJzncCIdGyateKNeVW5iBI9EqipZLS9KOqXYIQhAY8J9GKUBrgT+jbRDZ02aBhCmrzEwPtDZ9MHKBXdkpmoDWKCVN9VptO+Kw+8kqwGqMkK7nIN6yTB7uTNizbD1FSSsVPsjYMC1qFKHxwIZZSSIVxLZ1/nJNar5+oQPMT7IYCrqUta1ENzuqGaeOFTArBGs3f3Vmtoo2Se7ja1h00kSoHK4bBIKUNy3hdoPYU0mF91i9mT8EEL2ocZ3gKa00ayWujLZY4IfHKFonVDLsRGgXuQ90zBmWgneyTk3yT1iArMKrDKUeem9L3ajHrbSXwohxsQd/ggOleKM7ese048J2/fwuim1uQGmhQCW8vQMkacP3GCQgBFMftHEsr7cYYe95CnmKTPMFbYD8CQ++DGQy+/M5X4ku5wHYmdIktfvk9tecpavThqS3m/0YtnqIWPTy1cD77K2wYjo+Ay317I74A', 296, 100, 'Process Bar'); 1565 }), 1566 this.createVertexTemplateEntry('swimlane;', 200, 200, 'Container', 'Container', null, null, 'container swimlane lane pool group'), 1567 this.addEntry('list group erd table', function() 1568 { 1569 var cell = new mxCell('List', new mxGeometry(0, 0, 140, 110), 1570 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=none;horizontalStack=0;' + 1571 'resizeParent=1;resizeParentMax=0;resizeLast=0;collapsible=1;marginBottom=0;'); 1572 cell.vertex = true; 1573 cell.insert(sb.cloneCell(field, 'Item 1')); 1574 cell.insert(sb.cloneCell(field, 'Item 2')); 1575 cell.insert(sb.cloneCell(field, 'Item 3')); 1576 1577 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'List'); 1578 }), 1579 this.addEntry('list item entry value group erd table', function() 1580 { 1581 return sb.createVertexTemplateFromCells([sb.cloneCell(field, 'List Item')], field.geometry.width, field.geometry.height, 'List Item'); 1582 }) 1583 ]; 1584}; 1585 1586/** 1587 * Adds the general palette to the sidebar. 1588 */ 1589Sidebar.prototype.addBasicPalette = function(dir) 1590{ 1591 this.setCurrentSearchEntryLibrary('basic'); 1592 this.addStencilPalette('basic', mxResources.get('basic'), dir + '/basic.xml', 1593 ';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2', 1594 null, null, null, null, [ 1595 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;top=0;bottom=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1596 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;right=0;top=0;bottom=0;fillColor=none;routingCenterX=-0.5;', 120, 60, '', 'Partial Rectangle'), 1597 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;bottom=0;right=0;fillColor=none;', 120, 60, '', 'Partial Rectangle'), 1598 this.createVertexTemplateEntry('shape=partialRectangle;whiteSpace=wrap;html=1;top=0;left=0;fillColor=none;', 120, 60, '', 'Partial Rectangle') 1599 ]); 1600 this.setCurrentSearchEntryLibrary(); 1601}; 1602 1603/** 1604 * Adds the general palette to the sidebar. 1605 */ 1606Sidebar.prototype.addUmlPalette = function(expand) 1607{ 1608 // Avoids having to bind all functions to "this" 1609 var sb = this; 1610 1611 // Reusable cells 1612 var field = new mxCell('+ field: type', new mxGeometry(0, 0, 100, 26), 'text;strokeColor=none;fillColor=none;align=left;verticalAlign=top;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;'); 1613 field.vertex = true; 1614 1615 var divider = new mxCell('', new mxGeometry(0, 0, 40, 8), 'line;strokeWidth=1;fillColor=none;align=left;verticalAlign=middle;spacingTop=-1;spacingLeft=3;spacingRight=3;rotatable=0;labelPosition=right;points=[];portConstraint=eastwest;'); 1616 divider.vertex = true; 1617 1618 // Default tags 1619 var dt = 'uml static class '; 1620 this.setCurrentSearchEntryLibrary('uml'); 1621 1622 var fns = [ 1623 this.createVertexTemplateEntry('html=1;', 110, 50, 'Object', 'Object', null, null, dt + 'object instance'), 1624 this.createVertexTemplateEntry('html=1;', 110, 50, '«interface»<br><b>Name</b>', 'Interface', null, null, dt + 'interface object instance annotated annotation'), 1625 this.addEntry(dt + 'object instance', function() 1626 { 1627 var cell = new mxCell('Classname', new mxGeometry(0, 0, 160, 90), 1628 'swimlane;fontStyle=1;align=center;verticalAlign=top;childLayout=stackLayout;horizontal=1;startSize=26;horizontalStack=0;resizeParent=1;resizeParentMax=0;resizeLast=0;collapsible=1;marginBottom=0;'); 1629 cell.vertex = true; 1630 cell.insert(field.clone()); 1631 cell.insert(divider.clone()); 1632 cell.insert(sb.cloneCell(field, '+ method(type): type')); 1633 1634 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Class'); 1635 }), 1636 this.addEntry(dt + 'section subsection', function() 1637 { 1638 var cell = new mxCell('Classname', new mxGeometry(0, 0, 140, 110), 1639 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=none;horizontalStack=0;resizeParent=1;resizeParentMax=0;resizeLast=0;collapsible=1;marginBottom=0;'); 1640 cell.vertex = true; 1641 cell.insert(field.clone()); 1642 cell.insert(field.clone()); 1643 cell.insert(field.clone()); 1644 1645 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Class 2'); 1646 }), 1647 this.addEntry(dt + 'item member method function variable field attribute label', function() 1648 { 1649 return sb.createVertexTemplateFromCells([sb.cloneCell(field, '+ item: attribute')], field.geometry.width, field.geometry.height, 'Item 1'); 1650 }), 1651 this.addEntry(dt + 'item member method function variable field attribute label', function() 1652 { 1653 var cell = new mxCell('item: attribute', new mxGeometry(0, 0, 120, field.geometry.height), 'label;fontStyle=0;strokeColor=none;fillColor=none;align=left;verticalAlign=top;overflow=hidden;' + 1654 'spacingLeft=28;spacingRight=4;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;imageWidth=16;imageHeight=16;image=' + sb.gearImage); 1655 cell.vertex = true; 1656 1657 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Item 2'); 1658 }), 1659 this.addEntry(dt + 'divider hline line separator', function() 1660 { 1661 return sb.createVertexTemplateFromCells([divider.clone()], divider.geometry.width, divider.geometry.height, 'Divider'); 1662 }), 1663 this.addEntry(dt + 'spacer space gap separator', function() 1664 { 1665 var cell = new mxCell('', new mxGeometry(0, 0, 20, 14), 'text;strokeColor=none;fillColor=none;align=left;verticalAlign=middle;spacingTop=-1;spacingLeft=4;spacingRight=4;rotatable=0;labelPosition=right;points=[];portConstraint=eastwest;'); 1666 cell.vertex = true; 1667 1668 return sb.createVertexTemplateFromCells([cell.clone()], cell.geometry.width, cell.geometry.height, 'Spacer'); 1669 }), 1670 this.createVertexTemplateEntry('text;align=center;fontStyle=1;verticalAlign=middle;spacingLeft=3;spacingRight=3;strokeColor=none;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;', 1671 80, 26, 'Title', 'Title', null, null, dt + 'title label'), 1672 this.addEntry(dt + 'component', function() 1673 { 1674 var cell = new mxCell('«Annotation»<br/><b>Component</b>', new mxGeometry(0, 0, 180, 90), 'html=1;dropTarget=0;'); 1675 cell.vertex = true; 1676 1677 var symbol = new mxCell('', new mxGeometry(1, 0, 20, 20), 'shape=module;jettyWidth=8;jettyHeight=4;'); 1678 symbol.vertex = true; 1679 symbol.geometry.relative = true; 1680 symbol.geometry.offset = new mxPoint(-27, 7); 1681 cell.insert(symbol); 1682 1683 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Component'); 1684 }), 1685 this.addEntry(dt + 'component', function() 1686 { 1687 var cell = new mxCell('<p style="margin:0px;margin-top:6px;text-align:center;"><b>Component</b></p>' + 1688 '<hr/><p style="margin:0px;margin-left:8px;">+ Attribute1: Type<br/>+ Attribute2: Type</p>', new mxGeometry(0, 0, 180, 90), 1689 'align=left;overflow=fill;html=1;dropTarget=0;'); 1690 cell.vertex = true; 1691 1692 var symbol = new mxCell('', new mxGeometry(1, 0, 20, 20), 'shape=component;jettyWidth=8;jettyHeight=4;'); 1693 symbol.vertex = true; 1694 symbol.geometry.relative = true; 1695 symbol.geometry.offset = new mxPoint(-24, 4); 1696 cell.insert(symbol); 1697 1698 return sb.createVertexTemplateFromCells([cell], cell.geometry.width, cell.geometry.height, 'Component with Attributes'); 1699 }), 1700 this.createVertexTemplateEntry('verticalAlign=top;align=left;spacingTop=8;spacingLeft=2;spacingRight=12;shape=cube;size=10;direction=south;fontStyle=4;html=1;', 1701 180, 120, 'Block', 'Block', null, null, dt + 'block'), 1702 this.createVertexTemplateEntry('shape=module;align=left;spacingLeft=20;align=center;verticalAlign=top;', 100, 50, 'Module', 'Module', null, null, dt + 'module component'), 1703 this.createVertexTemplateEntry('shape=folder;fontStyle=1;spacingTop=10;tabWidth=40;tabHeight=14;tabPosition=left;html=1;', 70, 50, 1704 'package', 'Package', null, null, dt + 'package'), 1705 this.createVertexTemplateEntry('verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;', 1706 160, 90, '<p style="margin:0px;margin-top:4px;text-align:center;text-decoration:underline;"><b>Object:Type</b></p><hr/>' + 1707 '<p style="margin:0px;margin-left:8px;">field1 = value1<br/>field2 = value2<br>field3 = value3</p>', 'Object', 1708 null, null, dt + 'object instance'), 1709 this.createVertexTemplateEntry('verticalAlign=top;align=left;overflow=fill;html=1;',180, 90, 1710 '<div style="box-sizing:border-box;width:100%;background:#e4e4e4;padding:2px;">Tablename</div>' + 1711 '<table style="width:100%;font-size:1em;" cellpadding="2" cellspacing="0">' + 1712 '<tr><td>PK</td><td>uniqueId</td></tr><tr><td>FK1</td><td>' + 1713 'foreignKey</td></tr><tr><td></td><td>fieldname</td></tr></table>', 'Entity', null, null, 'er entity table'), 1714 this.addEntry(dt + 'object instance', function() 1715 { 1716 var cell = new mxCell('<p style="margin:0px;margin-top:4px;text-align:center;">' + 1717 '<b>Class</b></p>' + 1718 '<hr size="1"/><div style="height:2px;"></div>', new mxGeometry(0, 0, 140, 60), 1719 'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;'); 1720 cell.vertex = true; 1721 1722 return sb.createVertexTemplateFromCells([cell.clone()], cell.geometry.width, cell.geometry.height, 'Class 3'); 1723 }), 1724 this.addEntry(dt + 'object instance', function() 1725 { 1726 var cell = new mxCell('<p style="margin:0px;margin-top:4px;text-align:center;">' + 1727 '<b>Class</b></p>' + 1728 '<hr size="1"/><div style="height:2px;"></div><hr size="1"/><div style="height:2px;"></div>', new mxGeometry(0, 0, 140, 60), 1729 'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;'); 1730 cell.vertex = true; 1731 1732 return sb.createVertexTemplateFromCells([cell.clone()], cell.geometry.width, cell.geometry.height, 'Class 4'); 1733 }), 1734 this.addEntry(dt + 'object instance', function() 1735 { 1736 var cell = new mxCell('<p style="margin:0px;margin-top:4px;text-align:center;">' + 1737 '<b>Class</b></p>' + 1738 '<hr size="1"/><p style="margin:0px;margin-left:4px;">+ field: Type</p><hr size="1"/>' + 1739 '<p style="margin:0px;margin-left:4px;">+ method(): Type</p>', new mxGeometry(0, 0, 160, 90), 1740 'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;'); 1741 cell.vertex = true; 1742 1743 return sb.createVertexTemplateFromCells([cell.clone()], cell.geometry.width, cell.geometry.height, 'Class 5'); 1744 }), 1745 this.addEntry(dt + 'object instance', function() 1746 { 1747 var cell = new mxCell('<p style="margin:0px;margin-top:4px;text-align:center;">' + 1748 '<i><<Interface>></i><br/><b>Interface</b></p>' + 1749 '<hr size="1"/><p style="margin:0px;margin-left:4px;">+ field1: Type<br/>' + 1750 '+ field2: Type</p>' + 1751 '<hr size="1"/><p style="margin:0px;margin-left:4px;">' + 1752 '+ method1(Type): Type<br/>' + 1753 '+ method2(Type, Type): Type</p>', new mxGeometry(0, 0, 190, 140), 1754 'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;'); 1755 cell.vertex = true; 1756 1757 return sb.createVertexTemplateFromCells([cell.clone()], cell.geometry.width, cell.geometry.height, 'Interface 2'); 1758 }), 1759 this.createVertexTemplateEntry('shape=providedRequiredInterface;html=1;verticalLabelPosition=bottom;sketch=0;', 20, 20, '', 'Provided/Required Interface', null, null, 'uml provided required interface lollipop notation'), 1760 this.createVertexTemplateEntry('shape=requiredInterface;html=1;verticalLabelPosition=bottom;sketch=0;', 10, 20, '', 'Required Interface', null, null, 'uml required interface lollipop notation'), 1761 this.addEntry('uml lollipop notation provided required interface', function() 1762 { 1763 return sb.createVertexTemplateFromData('zVRNT8MwDP01uaLSMu6sfFxAmrQDcAytaQJZXLnu2u7XkzQZXTUmuIA4VIqf/ZzkvdQiyzf9HclaPWAJRmQ3IssJkcNq0+dgjEgTXYrsWqRp4j6R3p7Ino/ZpJYEln9CSANhK00LAQlAw4OJAGFrS/D1iciWSKywQivNPWLtwHMHvgHzsNY7z5Ato4MUb0zMgi2viLBzoUULAbnVxsSWzTtwofYBtlTACkhvgIHWtSy0rWKSJVXAJ5Lh4FBWMNMicAJ0cSzPWBW1uQN0fWlwJQRGst7OW8kmhNVn3Sd1hdp1TJMhVCzmhHipUDO54RYHm07Q6NHXfmV/65eS5jXXVJhj15yCNDz54GyxD58PwjL2v/SmMuE7POqSVdxj5vm/cK6PG4X/5deNvPjeSEfQdeOV75Rm8K/dZzo3LOaGSaMr69aF0wbIA00NhZfpVff+JSwJGr2TL2Nnr3jtbzDeabEUi2v/Tlo22kKO1gbq0Z8ZDwzE0J+cNidM2ROinF18CR6KeivQleI59pVrM8knfV04Dc1gx+FM/QA=', 1764 40, 10, 'Lollipop Notation'); 1765 }), 1766 this.createVertexTemplateEntry('shape=umlBoundary;whiteSpace=wrap;html=1;', 100, 80, 'Boundary Object', 'Boundary Object', null, null, 'uml boundary object'), 1767 this.createVertexTemplateEntry('ellipse;shape=umlEntity;whiteSpace=wrap;html=1;', 80, 80, 'Entity Object', 'Entity Object', null, null, 'uml entity object'), 1768 this.createVertexTemplateEntry('ellipse;shape=umlControl;whiteSpace=wrap;html=1;', 70, 80, 'Control Object', 'Control Object', null, null, 'uml control object'), 1769 this.createVertexTemplateEntry('shape=umlActor;verticalLabelPosition=bottom;verticalAlign=top;html=1;', 30, 60, 'Actor', 'Actor', false, null, 'uml actor'), 1770 this.createVertexTemplateEntry('ellipse;whiteSpace=wrap;html=1;', 140, 70, 'Use Case', 'Use Case', null, null, 'uml use case usecase'), 1771 this.addEntry('uml activity state start', function() 1772 { 1773 var cell = new mxCell('', new mxGeometry(0, 0, 30, 30), 1774 'ellipse;html=1;shape=startState;fillColor=#000000;strokeColor=#ff0000;'); 1775 cell.vertex = true; 1776 1777 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;'); 1778 edge.geometry.setTerminalPoint(new mxPoint(15, 90), false); 1779 edge.geometry.relative = true; 1780 edge.edge = true; 1781 1782 cell.insertEdge(edge, true); 1783 1784 return sb.createVertexTemplateFromCells([cell, edge], 30, 90, 'Start'); 1785 }), 1786 this.addEntry('uml activity state', function() 1787 { 1788 var cell = new mxCell('Activity', new mxGeometry(0, 0, 120, 40), 1789 'rounded=1;whiteSpace=wrap;html=1;arcSize=40;fontColor=#000000;fillColor=#ffffc0;strokeColor=#ff0000;'); 1790 cell.vertex = true; 1791 1792 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;'); 1793 edge.geometry.setTerminalPoint(new mxPoint(60, 100), false); 1794 edge.geometry.relative = true; 1795 edge.edge = true; 1796 1797 cell.insertEdge(edge, true); 1798 1799 return sb.createVertexTemplateFromCells([cell, edge], 120, 100, 'Activity'); 1800 }), 1801 this.addEntry('uml activity composite state', function() 1802 { 1803 var cell = new mxCell('Composite State', new mxGeometry(0, 0, 160, 60), 1804 'swimlane;fontStyle=1;align=center;verticalAlign=middle;childLayout=stackLayout;horizontal=1;startSize=30;horizontalStack=0;resizeParent=0;resizeLast=1;container=0;fontColor=#000000;collapsible=0;rounded=1;arcSize=30;strokeColor=#ff0000;fillColor=#ffffc0;swimlaneFillColor=#ffffc0;dropTarget=0;'); 1805 cell.vertex = true; 1806 1807 var cell1 = new mxCell('Subtitle', new mxGeometry(0, 0, 200, 26), 'text;html=1;strokeColor=none;fillColor=none;align=center;verticalAlign=middle;spacingLeft=4;spacingRight=4;whiteSpace=wrap;overflow=hidden;rotatable=0;fontColor=#000000;'); 1808 cell1.vertex = true; 1809 cell.insert(cell1); 1810 1811 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;'); 1812 edge.geometry.setTerminalPoint(new mxPoint(80, 120), false); 1813 edge.geometry.relative = true; 1814 edge.edge = true; 1815 1816 cell.insertEdge(edge, true); 1817 1818 return sb.createVertexTemplateFromCells([cell, edge], 160, 120, 'Composite State'); 1819 }), 1820 this.addEntry('uml activity condition', function() 1821 { 1822 var cell = new mxCell('Condition', new mxGeometry(0, 0, 80, 40), 'rhombus;whiteSpace=wrap;html=1;fillColor=#ffffc0;strokeColor=#ff0000;'); 1823 cell.vertex = true; 1824 1825 var edge1 = new mxCell('no', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;align=left;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;'); 1826 edge1.geometry.setTerminalPoint(new mxPoint(180, 20), false); 1827 edge1.geometry.relative = true; 1828 edge1.geometry.x = -1; 1829 edge1.edge = true; 1830 1831 cell.insertEdge(edge1, true); 1832 1833 var edge2 = new mxCell('yes', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;align=left;verticalAlign=top;endArrow=open;endSize=8;strokeColor=#ff0000;'); 1834 edge2.geometry.setTerminalPoint(new mxPoint(40, 100), false); 1835 edge2.geometry.relative = true; 1836 edge2.geometry.x = -1; 1837 edge2.edge = true; 1838 1839 cell.insertEdge(edge2, true); 1840 1841 return sb.createVertexTemplateFromCells([cell, edge1, edge2], 180, 100, 'Condition'); 1842 }), 1843 this.addEntry('uml activity fork join', function() 1844 { 1845 var cell = new mxCell('', new mxGeometry(0, 0, 200, 10), 'shape=line;html=1;strokeWidth=6;strokeColor=#ff0000;'); 1846 cell.vertex = true; 1847 1848 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;'); 1849 edge.geometry.setTerminalPoint(new mxPoint(100, 80), false); 1850 edge.geometry.relative = true; 1851 edge.edge = true; 1852 1853 cell.insertEdge(edge, true); 1854 1855 return sb.createVertexTemplateFromCells([cell, edge], 200, 80, 'Fork/Join'); 1856 }), 1857 this.createVertexTemplateEntry('ellipse;html=1;shape=endState;fillColor=#000000;strokeColor=#ff0000;', 30, 30, '', 'End', null, null, 'uml activity state end'), 1858 this.createVertexTemplateEntry('shape=umlLifeline;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;outlineConnect=0;', 100, 300, ':Object', 'Lifeline', null, null, 'uml sequence participant lifeline'), 1859 this.createVertexTemplateEntry('shape=umlLifeline;participant=umlActor;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;outlineConnect=0;', 1860 20, 300, '', 'Actor Lifeline', null, null, 'uml sequence participant lifeline actor'), 1861 this.createVertexTemplateEntry('shape=umlLifeline;participant=umlBoundary;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;outlineConnect=0;', 1862 50, 300, '', 'Boundary Lifeline', null, null, 'uml sequence participant lifeline boundary'), 1863 this.createVertexTemplateEntry('shape=umlLifeline;participant=umlEntity;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;outlineConnect=0;', 1864 40, 300, '', 'Entity Lifeline', null, null, 'uml sequence participant lifeline entity'), 1865 this.createVertexTemplateEntry('shape=umlLifeline;participant=umlControl;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;outlineConnect=0;', 1866 40, 300, '', 'Control Lifeline', null, null, 'uml sequence participant lifeline control'), 1867 this.createVertexTemplateEntry('shape=umlFrame;whiteSpace=wrap;html=1;', 300, 200, 'frame', 'Frame', null, null, 'uml sequence frame'), 1868 this.createVertexTemplateEntry('shape=umlDestroy;whiteSpace=wrap;html=1;strokeWidth=3;', 30, 30, '', 'Destruction', null, null, 'uml sequence destruction destroy'), 1869 this.addEntry('uml sequence invoke invocation call activation', function() 1870 { 1871 var cell = new mxCell('', new mxGeometry(0, 0, 10, 80), 'html=1;points=[];perimeter=orthogonalPerimeter;'); 1872 cell.vertex = true; 1873 1874 var edge = new mxCell('dispatch', new mxGeometry(0, 0, 0, 0), 'html=1;verticalAlign=bottom;startArrow=oval;endArrow=block;startSize=8;'); 1875 edge.geometry.setTerminalPoint(new mxPoint(-60, 0), true); 1876 edge.geometry.relative = true; 1877 edge.edge = true; 1878 1879 cell.insertEdge(edge, false); 1880 1881 return sb.createVertexTemplateFromCells([cell, edge], 10, 80, 'Found Message'); 1882 }), 1883 this.addEntry('uml sequence invoke call delegation synchronous invocation activation', function() 1884 { 1885 var cell = new mxCell('', new mxGeometry(0, 0, 10, 80), 'html=1;points=[];perimeter=orthogonalPerimeter;'); 1886 cell.vertex = true; 1887 1888 var edge1 = new mxCell('dispatch', new mxGeometry(0, 0, 0, 0), 'html=1;verticalAlign=bottom;endArrow=block;entryX=0;entryY=0;'); 1889 edge1.geometry.setTerminalPoint(new mxPoint(-70, 0), true); 1890 edge1.geometry.relative = true; 1891 edge1.edge = true; 1892 1893 cell.insertEdge(edge1, false); 1894 1895 var edge2 = new mxCell('return', new mxGeometry(0, 0, 0, 0), 'html=1;verticalAlign=bottom;endArrow=open;dashed=1;endSize=8;exitX=0;exitY=0.95;'); 1896 edge2.geometry.setTerminalPoint(new mxPoint(-70, 76), false); 1897 edge2.geometry.relative = true; 1898 edge2.edge = true; 1899 1900 cell.insertEdge(edge2, true); 1901 1902 return sb.createVertexTemplateFromCells([cell, edge1, edge2], 10, 80, 'Synchronous Invocation'); 1903 }), 1904 this.addEntry('uml sequence self call recursion delegation activation', function() 1905 { 1906 var cell = new mxCell('', new mxGeometry(-5, 20, 10, 40), 'html=1;points=[];perimeter=orthogonalPerimeter;'); 1907 cell.vertex = true; 1908 1909 var edge = new mxCell('self call', new mxGeometry(0, 0, 0, 0), 'edgeStyle=orthogonalEdgeStyle;html=1;align=left;spacingLeft=2;endArrow=block;rounded=0;entryX=1;entryY=0;'); 1910 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1911 edge.geometry.points = [new mxPoint(30, 0)]; 1912 edge.geometry.relative = true; 1913 edge.edge = true; 1914 1915 cell.insertEdge(edge, false); 1916 1917 return sb.createVertexTemplateFromCells([cell, edge], 10, 60, 'Self Call'); 1918 }), 1919 this.addEntry('uml sequence invoke call delegation callback activation', function() 1920 { 1921 // TODO: Check if more entries should be converted to compressed XML 1922 return sb.createVertexTemplateFromData('xZRNT8MwDIZ/Ta6oaymD47rBTkiTuMAxW6wmIm0q19s6fj1OE3V0Y2iCA4dK8euP2I+riGxedUuUjX52CqzIHkU2R+conKpuDtaKNDFKZAuRpgl/In264J303qSRCDVdk5CGhJ20WwhKEFo62ChoqritxURkReNMTa2X80LkC68AmgoIkEWHpF3pamlXR7WIFwASdBeb7KXY4RIc5+KBQ/ZGkY4RYY5Egyl1zLqLmmyDXQ6Zx4n5EIf+HkB2BmAjrV3LzftPIPw4hgNn1pQ1a2tH5Cp2QK1miG7vNeu4iJe4pdeY2BtvbCQDGlAljMCQxBJotJ8rWCFYSWY3LvUdmZi68rvkkLiU6QnL1m1xAzHoBOdw61WEb88II9AW67/ydQ2wq1Cy1aAGvOrFfPh6997qDA3g+dxzv3nIL6MPU/8T+kMw8+m4QPgdfrEJNo8PSQj/+s58Ag==', 1923 10, 60, 'Callback'); 1924 }), 1925 this.createVertexTemplateEntry('html=1;points=[];perimeter=orthogonalPerimeter;', 10, 80, '', 'Activation', null, null, 'uml sequence activation'), 1926 1927 this.createEdgeTemplateEntry('html=1;verticalAlign=bottom;startArrow=oval;startFill=1;endArrow=block;startSize=8;', 60, 0, 'dispatch', 'Found Message 1', null, 'uml sequence message call invoke dispatch'), 1928 this.createEdgeTemplateEntry('html=1;verticalAlign=bottom;startArrow=circle;startFill=1;endArrow=open;startSize=6;endSize=8;', 80, 0, 'dispatch', 'Found Message 2', null, 'uml sequence message call invoke dispatch'), 1929 this.createEdgeTemplateEntry('html=1;verticalAlign=bottom;endArrow=block;', 80, 0, 'dispatch', 'Message', null, 'uml sequence message call invoke dispatch'), 1930 this.addEntry('uml sequence return message', function() 1931 { 1932 var edge = new mxCell('return', new mxGeometry(0, 0, 0, 0), 'html=1;verticalAlign=bottom;endArrow=open;dashed=1;endSize=8;'); 1933 edge.geometry.setTerminalPoint(new mxPoint(80, 0), true); 1934 edge.geometry.setTerminalPoint(new mxPoint(0, 0), false); 1935 edge.geometry.relative = true; 1936 edge.edge = true; 1937 1938 return sb.createEdgeTemplateFromCells([edge], 80, 0, 'Return'); 1939 }), 1940 this.addEntry('uml relation', function() 1941 { 1942 var edge = new mxCell('name', new mxGeometry(0, 0, 0, 0), 'endArrow=block;endFill=1;html=1;edgeStyle=orthogonalEdgeStyle;align=left;verticalAlign=top;'); 1943 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1944 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 1945 edge.geometry.relative = true; 1946 edge.geometry.x = -1; 1947 edge.edge = true; 1948 1949 var cell = new mxCell('1', new mxGeometry(-1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=left;verticalAlign=bottom;'); 1950 cell.geometry.relative = true; 1951 cell.setConnectable(false); 1952 cell.vertex = true; 1953 edge.insert(cell); 1954 1955 return sb.createEdgeTemplateFromCells([edge], 160, 0, 'Relation 1'); 1956 }), 1957 this.addEntry('uml association', function() 1958 { 1959 var edge = new mxCell('', new mxGeometry(0, 0, 0, 0), 'endArrow=none;html=1;edgeStyle=orthogonalEdgeStyle;'); 1960 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1961 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 1962 edge.geometry.relative = true; 1963 edge.edge = true; 1964 1965 var cell1 = new mxCell('parent', new mxGeometry(-1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=left;verticalAlign=bottom;'); 1966 cell1.geometry.relative = true; 1967 cell1.setConnectable(false); 1968 cell1.vertex = true; 1969 edge.insert(cell1); 1970 1971 var cell2 = new mxCell('child', new mxGeometry(1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=right;verticalAlign=bottom;'); 1972 cell2.geometry.relative = true; 1973 cell2.setConnectable(false); 1974 cell2.vertex = true; 1975 edge.insert(cell2); 1976 1977 return sb.createEdgeTemplateFromCells([edge], 160, 0, 'Association 1'); 1978 }), 1979 this.addEntry('uml aggregation', function() 1980 { 1981 var edge = new mxCell('1', new mxGeometry(0, 0, 0, 0), 'endArrow=open;html=1;endSize=12;startArrow=diamondThin;startSize=14;startFill=0;edgeStyle=orthogonalEdgeStyle;align=left;verticalAlign=bottom;'); 1982 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1983 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 1984 edge.geometry.relative = true; 1985 edge.geometry.x = -1; 1986 edge.geometry.y = 3; 1987 edge.edge = true; 1988 1989 return sb.createEdgeTemplateFromCells([edge], 160, 0, 'Aggregation 1'); 1990 }), 1991 this.addEntry('uml composition', function() 1992 { 1993 var edge = new mxCell('1', new mxGeometry(0, 0, 0, 0), 'endArrow=open;html=1;endSize=12;startArrow=diamondThin;startSize=14;startFill=1;edgeStyle=orthogonalEdgeStyle;align=left;verticalAlign=bottom;'); 1994 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 1995 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 1996 edge.geometry.relative = true; 1997 edge.geometry.x = -1; 1998 edge.geometry.y = 3; 1999 edge.edge = true; 2000 2001 return sb.createEdgeTemplateFromCells([edge], 160, 0, 'Composition 1'); 2002 }), 2003 this.addEntry('uml relation', function() 2004 { 2005 var edge = new mxCell('Relation', new mxGeometry(0, 0, 0, 0), 'endArrow=open;html=1;endSize=12;startArrow=diamondThin;startSize=14;startFill=0;edgeStyle=orthogonalEdgeStyle;'); 2006 edge.geometry.setTerminalPoint(new mxPoint(0, 0), true); 2007 edge.geometry.setTerminalPoint(new mxPoint(160, 0), false); 2008 edge.geometry.relative = true; 2009 edge.edge = true; 2010 2011 var cell1 = new mxCell('0..n', new mxGeometry(-1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=left;verticalAlign=top;'); 2012 cell1.geometry.relative = true; 2013 cell1.setConnectable(false); 2014 cell1.vertex = true; 2015 edge.insert(cell1); 2016 2017 var cell2 = new mxCell('1', new mxGeometry(1, 0, 0, 0), 'edgeLabel;resizable=0;html=1;align=right;verticalAlign=top;'); 2018 cell2.geometry.relative = true; 2019 cell2.setConnectable(false); 2020 cell2.vertex = true; 2021 edge.insert(cell2); 2022 2023 return sb.createEdgeTemplateFromCells([edge], 160, 0, 'Relation 2'); 2024 }), 2025 this.createEdgeTemplateEntry('endArrow=open;endSize=12;dashed=1;html=1;', 160, 0, 'Use', 'Dependency', null, 'uml dependency use'), 2026 this.createEdgeTemplateEntry('endArrow=block;endSize=16;endFill=0;html=1;', 160, 0, 'Extends', 'Generalization', null, 'uml generalization extend'), 2027 this.createEdgeTemplateEntry('endArrow=block;startArrow=block;endFill=1;startFill=1;html=1;', 160, 0, '', 'Association 2', null, 'uml association'), 2028 this.createEdgeTemplateEntry('endArrow=open;startArrow=circlePlus;endFill=0;startFill=0;endSize=8;html=1;', 160, 0, '', 'Inner Class', null, 'uml inner class'), 2029 this.createEdgeTemplateEntry('endArrow=open;startArrow=cross;endFill=0;startFill=0;endSize=8;startSize=10;html=1;', 160, 0, '', 'Terminate', null, 'uml terminate'), 2030 this.createEdgeTemplateEntry('endArrow=block;dashed=1;endFill=0;endSize=12;html=1;', 160, 0, '', 'Implementation', null, 'uml realization implementation'), 2031 this.createEdgeTemplateEntry('endArrow=diamondThin;endFill=0;endSize=24;html=1;', 160, 0, '', 'Aggregation 2', null, 'uml aggregation'), 2032 this.createEdgeTemplateEntry('endArrow=diamondThin;endFill=1;endSize=24;html=1;', 160, 0, '', 'Composition 2', null, 'uml composition'), 2033 this.createEdgeTemplateEntry('endArrow=open;endFill=1;endSize=12;html=1;', 160, 0, '', 'Association 3', null, 'uml association') 2034 ]; 2035 2036 this.addPaletteFunctions('uml', mxResources.get('uml'), expand || false, fns); 2037 this.setCurrentSearchEntryLibrary(); 2038}; 2039 2040/** 2041 * Creates and returns the given title element. 2042 */ 2043Sidebar.prototype.createTitle = function(label) 2044{ 2045 var elt = document.createElement('a'); 2046 elt.setAttribute('title', mxResources.get('sidebarTooltip')); 2047 elt.className = 'geTitle'; 2048 mxUtils.write(elt, label); 2049 2050 return elt; 2051}; 2052 2053/** 2054 * Creates a thumbnail for the given cells. 2055 */ 2056Sidebar.prototype.createThumb = function(cells, width, height, parent, title, showLabel, showTitle) 2057{ 2058 this.graph.labelsVisible = (showLabel == null || showLabel); 2059 var fo = mxClient.NO_FO; 2060 mxClient.NO_FO = Editor.prototype.originalNoForeignObject; 2061 this.graph.view.scaleAndTranslate(1, 0, 0); 2062 this.graph.addCells(cells); 2063 2064 var bounds = this.graph.getGraphBounds(); 2065 var s = Math.floor(Math.min((width - 2 * this.thumbBorder) / bounds.width, 2066 (height - 2 * this.thumbBorder) / bounds.height) * 100) / 100; 2067 this.graph.view.scaleAndTranslate(s, Math.floor((width - bounds.width * s) / 2 / s - bounds.x), 2068 Math.floor((height - bounds.height * s) / 2 / s - bounds.y)); 2069 var node = null; 2070 2071 // For supporting HTML labels in IE9 standards mode the container is cloned instead 2072 if (this.graph.dialect == mxConstants.DIALECT_SVG && !mxClient.NO_FO && 2073 this.graph.view.getCanvas().ownerSVGElement != null) 2074 { 2075 node = this.graph.view.getCanvas().ownerSVGElement.cloneNode(true); 2076 } 2077 // LATER: Check if deep clone can be used for quirks if container in DOM 2078 else 2079 { 2080 node = this.graph.container.cloneNode(false); 2081 node.innerHTML = this.graph.container.innerHTML; 2082 } 2083 2084 this.graph.getModel().clear(); 2085 mxClient.NO_FO = fo; 2086 2087 node.style.position = 'relative'; 2088 node.style.overflow = 'hidden'; 2089 node.style.left = this.thumbBorder + 'px'; 2090 node.style.top = this.thumbBorder + 'px'; 2091 node.style.width = width + 'px'; 2092 node.style.height = height + 'px'; 2093 node.style.visibility = ''; 2094 node.style.minWidth = ''; 2095 node.style.minHeight = ''; 2096 2097 parent.appendChild(node); 2098 2099 // Adds title for sidebar entries 2100 if (this.sidebarTitles && title != null && showTitle != false) 2101 { 2102 var border = 0; 2103 parent.style.height = (this.thumbHeight + border + this.sidebarTitleSize + 8) + 'px'; 2104 2105 var div = document.createElement('div'); 2106 div.style.color = Editor.isDarkMode() ? '#A0A0A0' : '#303030'; 2107 div.style.fontSize = this.sidebarTitleSize + 'px'; 2108 div.style.textAlign = 'center'; 2109 div.style.whiteSpace = 'nowrap'; 2110 div.style.overflow = 'hidden'; 2111 div.style.textOverflow = 'ellipsis'; 2112 2113 if (mxClient.IS_IE) 2114 { 2115 div.style.height = (this.sidebarTitleSize + 12) + 'px'; 2116 } 2117 2118 div.style.paddingTop = '4px'; 2119 mxUtils.write(div, title); 2120 parent.appendChild(div); 2121 } 2122 2123 return bounds; 2124}; 2125 2126/** 2127 * Returns a function that creates a title. 2128 */ 2129Sidebar.prototype.createSection = function(title) 2130{ 2131 return mxUtils.bind(this, function() 2132 { 2133 var elt = document.createElement('div'); 2134 elt.setAttribute('title', title); 2135 elt.style.textOverflow = 'ellipsis'; 2136 elt.style.whiteSpace = 'nowrap'; 2137 elt.style.textAlign = 'center'; 2138 elt.style.overflow = 'hidden'; 2139 elt.style.width = '100%'; 2140 elt.style.padding = '14px 0'; 2141 2142 mxUtils.write(elt, title); 2143 2144 return elt; 2145 }); 2146}; 2147 2148/** 2149 * Creates and returns a new palette item for the given image. 2150 */ 2151Sidebar.prototype.createItem = function(cells, title, showLabel, showTitle, width, height, allowCellsInserted, showTooltip) 2152{ 2153 showTooltip = (showTooltip != null) ? showTooltip : true; 2154 2155 var elt = document.createElement('a'); 2156 elt.className = 'geItem'; 2157 elt.style.overflow = 'hidden'; 2158 var border = 2 * this.thumbBorder; 2159 elt.style.width = (this.thumbWidth + border) + 'px'; 2160 elt.style.height = (this.thumbHeight + border) + 'px'; 2161 elt.style.padding = this.thumbPadding + 'px'; 2162 2163 // Blocks default click action 2164 mxEvent.addListener(elt, 'click', function(evt) 2165 { 2166 mxEvent.consume(evt); 2167 }); 2168 2169 // Applies default styles 2170 var originalCells = cells; 2171 cells = this.graph.cloneCells(cells); 2172 this.editorUi.insertHandler(originalCells, null, this.graph.model, 2173 this.editorUi.editor.graph.defaultVertexStyle, 2174 this.editorUi.editor.graph.defaultEdgeStyle, 2175 true, true); 2176 2177 this.createThumb(originalCells, this.thumbWidth, this.thumbHeight, 2178 elt, title, showLabel, showTitle, width, height); 2179 var bounds = new mxRectangle(0, 0, width, height); 2180 2181 if (cells.length > 1 || cells[0].vertex) 2182 { 2183 var ds = this.createDragSource(elt, this.createDropHandler(cells, true, allowCellsInserted, 2184 bounds), this.createDragPreview(width, height), cells, bounds); 2185 this.addClickHandler(elt, ds, cells); 2186 2187 // Uses guides for vertices only if enabled in graph 2188 ds.isGuidesEnabled = mxUtils.bind(this, function() 2189 { 2190 return this.editorUi.editor.graph.graphHandler.guidesEnabled; 2191 }); 2192 } 2193 else if (cells[0] != null && cells[0].edge) 2194 { 2195 var ds = this.createDragSource(elt, this.createDropHandler(cells, false, allowCellsInserted, 2196 bounds), this.createDragPreview(width, height), cells, bounds); 2197 this.addClickHandler(elt, ds, cells); 2198 } 2199 2200 // Shows a tooltip with the rendered cell 2201 if (!mxClient.IS_IOS && showTooltip) 2202 { 2203 mxEvent.addGestureListeners(elt, null, mxUtils.bind(this, function(evt) 2204 { 2205 if (mxEvent.isMouseEvent(evt)) 2206 { 2207 this.showTooltip(elt, cells, bounds.width, bounds.height, title, showLabel); 2208 } 2209 })); 2210 } 2211 2212 return elt; 2213}; 2214 2215/** 2216 * Creates a drop handler for inserting the given cells. 2217 */ 2218Sidebar.prototype.updateShapes = function(source, targets) 2219{ 2220 var graph = this.editorUi.editor.graph; 2221 var sourceCellStyle = graph.getCellStyle(source); 2222 var result = []; 2223 2224 graph.model.beginUpdate(); 2225 try 2226 { 2227 var cellStyle = graph.getModel().getStyle(source); 2228 2229 // Lists the styles to carry over from the existing shape 2230 var styles = ['shadow', 'dashed', 'dashPattern', 'fontFamily', 'fontSize', 'fontColor', 'align', 'startFill', 2231 'startSize', 'endFill', 'endSize', 'strokeColor', 'strokeWidth', 'fillColor', 'gradientColor', 2232 'html', 'part', 'noEdgeStyle', 'edgeStyle', 'elbow', 'childLayout', 'recursiveResize', 2233 'container', 'collapsible', 'connectable', 'comic', 'sketch', 'fillWeight', 'hachureGap', 2234 'hachureAngle', 'jiggle', 'disableMultiStroke', 'disableMultiStrokeFill', 2235 'fillStyle', 'curveFitting', 'simplification', 'sketchStyle']; 2236 2237 for (var i = 0; i < targets.length; i++) 2238 { 2239 var targetCell = targets[i]; 2240 2241 if ((graph.getModel().isVertex(targetCell) == graph.getModel().isVertex(source)) || 2242 (graph.getModel().isEdge(targetCell) == graph.getModel().isEdge(source))) 2243 { 2244 var style = graph.getCurrentCellStyle(targets[i]); 2245 graph.getModel().setStyle(targetCell, cellStyle); 2246 2247 // Removes all children of composite cells 2248 if (mxUtils.getValue(style, 'composite', '0') == '1') 2249 { 2250 var childCount = graph.model.getChildCount(targetCell); 2251 2252 for (var j = childCount; j >= 0; j--) 2253 { 2254 graph.model.remove(graph.model.getChildAt(targetCell, j)); 2255 } 2256 } 2257 2258 // Replaces the participant style in the lifeline shape with the target shape 2259 if (style[mxConstants.STYLE_SHAPE] == 'umlLifeline' && 2260 sourceCellStyle[mxConstants.STYLE_SHAPE] != 'umlLifeline') 2261 { 2262 graph.setCellStyles(mxConstants.STYLE_SHAPE, 'umlLifeline', [targetCell]); 2263 graph.setCellStyles('participant', sourceCellStyle[mxConstants.STYLE_SHAPE], [targetCell]); 2264 } 2265 2266 for (var j = 0; j < styles.length; j++) 2267 { 2268 var value = style[styles[j]]; 2269 2270 if (value != null) 2271 { 2272 graph.setCellStyles(styles[j], value, [targetCell]); 2273 } 2274 } 2275 2276 result.push(targetCell); 2277 } 2278 } 2279 } 2280 finally 2281 { 2282 graph.model.endUpdate(); 2283 } 2284 2285 return result; 2286}; 2287 2288/** 2289 * Creates a drop handler for inserting the given cells. 2290 */ 2291Sidebar.prototype.createDropHandler = function(cells, allowSplit, allowCellsInserted, bounds) 2292{ 2293 allowCellsInserted = (allowCellsInserted != null) ? allowCellsInserted : true; 2294 2295 return mxUtils.bind(this, function(graph, evt, target, x, y, force) 2296 { 2297 var elt = (force) ? null : ((mxEvent.isTouchEvent(evt) || mxEvent.isPenEvent(evt)) ? 2298 document.elementFromPoint(mxEvent.getClientX(evt), mxEvent.getClientY(evt)) : 2299 mxEvent.getSource(evt)); 2300 2301 while (elt != null && elt != this.container) 2302 { 2303 elt = elt.parentNode; 2304 } 2305 2306 if (elt == null && graph.isEnabled()) 2307 { 2308 cells = graph.getImportableCells(cells); 2309 2310 if (cells.length > 0) 2311 { 2312 graph.stopEditing(); 2313 2314 // Holding alt while mouse is released ignores drop target 2315 var validDropTarget = (target != null && !mxEvent.isAltDown(evt)) ? 2316 graph.isValidDropTarget(target, cells, evt) : false; 2317 2318 var select = null; 2319 2320 if (target != null && !validDropTarget) 2321 { 2322 target = null; 2323 } 2324 2325 if (!graph.isCellLocked(target || graph.getDefaultParent())) 2326 { 2327 graph.model.beginUpdate(); 2328 try 2329 { 2330 x = Math.round(x); 2331 y = Math.round(y); 2332 2333 // Splits the target edge or inserts into target group 2334 if (allowSplit && graph.isSplitTarget(target, cells, evt)) 2335 { 2336 var s = graph.view.scale; 2337 var tr = graph.view.translate; 2338 var tx = (x + tr.x) * s; 2339 var ty = (y + tr.y) * s; 2340 2341 var clones = graph.cloneCells(cells); 2342 graph.splitEdge(target, clones, null, 2343 x - bounds.width / 2, y - bounds.height / 2, 2344 tx, ty); 2345 select = clones; 2346 } 2347 else if (cells.length > 0) 2348 { 2349 select = graph.importCells(cells, x, y, target); 2350 } 2351 2352 // Executes parent layout hooks for position/order 2353 if (graph.layoutManager != null) 2354 { 2355 var layout = graph.layoutManager.getLayout(target); 2356 2357 if (layout != null) 2358 { 2359 var s = graph.view.scale; 2360 var tr = graph.view.translate; 2361 var tx = (x + tr.x) * s; 2362 var ty = (y + tr.y) * s; 2363 2364 for (var i = 0; i < select.length; i++) 2365 { 2366 layout.moveCell(select[i], tx, ty); 2367 } 2368 } 2369 } 2370 2371 if (allowCellsInserted && (evt == null || !mxEvent.isShiftDown(evt))) 2372 { 2373 graph.fireEvent(new mxEventObject('cellsInserted', 'cells', select)); 2374 } 2375 } 2376 catch (e) 2377 { 2378 this.editorUi.handleError(e); 2379 } 2380 finally 2381 { 2382 graph.model.endUpdate(); 2383 } 2384 2385 if (select != null && select.length > 0) 2386 { 2387 graph.scrollCellToVisible(select[0]); 2388 graph.setSelectionCells(select); 2389 } 2390 2391 if (graph.editAfterInsert && evt != null && mxEvent.isMouseEvent(evt) && 2392 select != null && select.length == 1) 2393 { 2394 window.setTimeout(function() 2395 { 2396 graph.startEditing(select[0]); 2397 }, 0); 2398 } 2399 } 2400 } 2401 2402 mxEvent.consume(evt); 2403 } 2404 }); 2405}; 2406 2407/** 2408 * Creates and returns a preview element for the given width and height. 2409 */ 2410Sidebar.prototype.createDragPreview = function(width, height) 2411{ 2412 var elt = document.createElement('div'); 2413 elt.className = 'geDragPreview'; 2414 elt.style.width = width + 'px'; 2415 elt.style.height = height + 'px'; 2416 2417 return elt; 2418}; 2419 2420/** 2421 * Creates a drag source for the given element. 2422 */ 2423Sidebar.prototype.dropAndConnect = function(source, targets, direction, dropCellIndex, evt) 2424{ 2425 var geo = this.getDropAndConnectGeometry(source, targets[dropCellIndex], direction, targets); 2426 2427 // Targets without the new edge for selection 2428 var tmp = []; 2429 2430 if (geo != null) 2431 { 2432 var graph = this.editorUi.editor.graph; 2433 var editingCell = null; 2434 2435 graph.model.beginUpdate(); 2436 try 2437 { 2438 var sourceGeo = graph.getCellGeometry(source); 2439 var geo2 = graph.getCellGeometry(targets[dropCellIndex]); 2440 2441 // Handles special case where target should be ignored for stack layouts 2442 var targetParent = graph.model.getParent(source); 2443 var validLayout = true; 2444 2445 // Ignores parent if it has a stack layout or if it is a table or row 2446 if (graph.layoutManager != null) 2447 { 2448 var layout = graph.layoutManager.getLayout(targetParent); 2449 2450 // LATER: Use parent of parent if valid layout 2451 if (layout != null && layout.constructor == mxStackLayout) 2452 { 2453 validLayout = false; 2454 } 2455 } 2456 2457 // Checks if another container is at the drop location 2458 var tmp = (graph.model.isEdge(source)) ? null : graph.view.getState(targetParent); 2459 var dx = 0; 2460 var dy = 0; 2461 2462 // Offsets by parent position 2463 if (tmp != null) 2464 { 2465 var offset = tmp.origin; 2466 dx = offset.x; 2467 dy = offset.y; 2468 2469 var pt = geo.getTerminalPoint(false); 2470 2471 if (pt != null) 2472 { 2473 pt.x += offset.x; 2474 pt.y += offset.y; 2475 } 2476 } 2477 2478 var useParent = !graph.isTableRow(source) && !graph.isTableCell(source) && 2479 (graph.model.isEdge(source) || (sourceGeo != null && 2480 !sourceGeo.relative && validLayout)); 2481 2482 var tempTarget = graph.getCellAt((geo.x + dx + graph.view.translate.x) * graph.view.scale, 2483 (geo.y + dy + graph.view.translate.y) * graph.view.scale, null, null, null, function(state, x, y) 2484 { 2485 return !graph.isContainer(state.cell); 2486 }); 2487 2488 if (tempTarget != null && tempTarget != targetParent) 2489 { 2490 tmp = graph.view.getState(tempTarget); 2491 2492 // Offsets by new parent position 2493 if (tmp != null) 2494 { 2495 var offset = tmp.origin; 2496 targetParent = tempTarget; 2497 useParent = true; 2498 2499 if (!graph.model.isEdge(source)) 2500 { 2501 geo.x -= offset.x - dx; 2502 geo.y -= offset.y - dy; 2503 } 2504 } 2505 } 2506 else if (!validLayout || graph.isTableRow(source) || graph.isTableCell(source)) 2507 { 2508 geo.x += dx; 2509 geo.y += dy; 2510 } 2511 2512 dx = geo2.x; 2513 dy = geo2.y; 2514 2515 // Ignores geometry of edges 2516 if (graph.model.isEdge(targets[dropCellIndex])) 2517 { 2518 dx = 0; 2519 dy = 0; 2520 } 2521 2522 targets = graph.importCells(targets, (geo.x - (useParent ? dx : 0)), 2523 (geo.y - (useParent ? dy : 0)), (useParent) ? targetParent : null); 2524 tmp = targets; 2525 2526 if (graph.model.isEdge(source)) 2527 { 2528 // Adds new terminal to edge 2529 // LATER: Push new terminal out radially from edge start point 2530 graph.model.setTerminal(source, targets[dropCellIndex], 2531 direction == mxConstants.DIRECTION_NORTH); 2532 } 2533 else if (graph.model.isEdge(targets[dropCellIndex])) 2534 { 2535 // Adds new outgoing connection to vertex and clears points 2536 graph.model.setTerminal(targets[dropCellIndex], source, true); 2537 var geo3 = graph.getCellGeometry(targets[dropCellIndex]); 2538 geo3.points = null; 2539 2540 if (geo3.getTerminalPoint(false) != null) 2541 { 2542 geo3.setTerminalPoint(geo.getTerminalPoint(false), false); 2543 } 2544 else if (useParent && graph.model.isVertex(targetParent)) 2545 { 2546 // Adds parent offset to other nodes 2547 var tmpState = graph.view.getState(targetParent); 2548 var offset = (tmpState.cell != graph.view.currentRoot) ? 2549 tmpState.origin : new mxPoint(0, 0); 2550 2551 graph.cellsMoved(targets, offset.x, offset.y, null, null, true); 2552 } 2553 } 2554 else 2555 { 2556 geo2 = graph.getCellGeometry(targets[dropCellIndex]); 2557 dx = geo.x - Math.round(geo2.x); 2558 dy = geo.y - Math.round(geo2.y); 2559 geo.x = Math.round(geo2.x); 2560 geo.y = Math.round(geo2.y); 2561 graph.model.setGeometry(targets[dropCellIndex], geo); 2562 graph.cellsMoved(targets, dx, dy, null, null, true); 2563 tmp = targets.slice(); 2564 editingCell = (tmp.length == 1) ? tmp[0] : null; 2565 targets.push(graph.insertEdge(null, null, '', source, targets[dropCellIndex], 2566 graph.createCurrentEdgeStyle())); 2567 } 2568 2569 if (evt == null || !mxEvent.isShiftDown(evt)) 2570 { 2571 graph.fireEvent(new mxEventObject('cellsInserted', 'cells', targets)); 2572 } 2573 } 2574 catch (e) 2575 { 2576 this.editorUi.handleError(e); 2577 } 2578 finally 2579 { 2580 graph.model.endUpdate(); 2581 } 2582 2583 if (graph.editAfterInsert && evt != null && mxEvent.isMouseEvent(evt) && 2584 editingCell != null) 2585 { 2586 window.setTimeout(function() 2587 { 2588 graph.startEditing(editingCell); 2589 }, 0); 2590 } 2591 } 2592 2593 return tmp; 2594}; 2595 2596/** 2597 * Creates a drag source for the given element. 2598 */ 2599Sidebar.prototype.getDropAndConnectGeometry = function(source, target, direction, targets) 2600{ 2601 var graph = this.editorUi.editor.graph; 2602 var view = graph.view; 2603 var keepSize = targets.length > 1; 2604 var geo = graph.getCellGeometry(source); 2605 var geo2 = graph.getCellGeometry(target); 2606 2607 if (geo != null && geo2 != null) 2608 { 2609 geo2 = geo2.clone(); 2610 2611 if (graph.model.isEdge(source)) 2612 { 2613 var state = graph.view.getState(source); 2614 var pts = state.absolutePoints; 2615 var p0 = pts[0]; 2616 var pe = pts[pts.length - 1]; 2617 2618 if (direction == mxConstants.DIRECTION_NORTH) 2619 { 2620 geo2.x = p0.x / view.scale - view.translate.x - geo2.width / 2; 2621 geo2.y = p0.y / view.scale - view.translate.y - geo2.height / 2; 2622 } 2623 else 2624 { 2625 geo2.x = pe.x / view.scale - view.translate.x - geo2.width / 2; 2626 geo2.y = pe.y / view.scale - view.translate.y - geo2.height / 2; 2627 } 2628 } 2629 else 2630 { 2631 if (geo.relative) 2632 { 2633 var state = graph.view.getState(source); 2634 geo = geo.clone(); 2635 geo.x = (state.x - view.translate.x) / view.scale; 2636 geo.y = (state.y - view.translate.y) / view.scale; 2637 } 2638 2639 var length = graph.defaultEdgeLength; 2640 2641 // Maintains edge length 2642 if (graph.model.isEdge(target) && geo2.getTerminalPoint(true) != null && 2643 geo2.getTerminalPoint(false) != null) 2644 { 2645 var p0 = geo2.getTerminalPoint(true); 2646 var pe = geo2.getTerminalPoint(false); 2647 var dx = pe.x - p0.x; 2648 var dy = pe.y - p0.y; 2649 2650 length = Math.sqrt(dx * dx + dy * dy); 2651 2652 geo2.x = geo.getCenterX(); 2653 geo2.y = geo.getCenterY(); 2654 geo2.width = 1; 2655 geo2.height = 1; 2656 2657 if (direction == mxConstants.DIRECTION_NORTH) 2658 { 2659 geo2.height = length 2660 geo2.y = geo.y - length; 2661 geo2.setTerminalPoint(new mxPoint(geo2.x, geo2.y), false); 2662 } 2663 else if (direction == mxConstants.DIRECTION_EAST) 2664 { 2665 geo2.width = length 2666 geo2.x = geo.x + geo.width; 2667 geo2.setTerminalPoint(new mxPoint(geo2.x + geo2.width, geo2.y), false); 2668 } 2669 else if (direction == mxConstants.DIRECTION_SOUTH) 2670 { 2671 geo2.height = length 2672 geo2.y = geo.y + geo.height; 2673 geo2.setTerminalPoint(new mxPoint(geo2.x, geo2.y + geo2.height), false); 2674 } 2675 else if (direction == mxConstants.DIRECTION_WEST) 2676 { 2677 geo2.width = length 2678 geo2.x = geo.x - length; 2679 geo2.setTerminalPoint(new mxPoint(geo2.x, geo2.y), false); 2680 } 2681 } 2682 else 2683 { 2684 // Try match size or ignore if width or height < 45 which 2685 // is considered special enough to be ignored here 2686 if (!keepSize && geo2.width > 45 && geo2.height > 45 && 2687 geo.width > 45 && geo.height > 45) 2688 { 2689 geo2.width = geo2.width * (geo.height / geo2.height); 2690 geo2.height = geo.height; 2691 } 2692 2693 geo2.x = geo.x + geo.width / 2 - geo2.width / 2; 2694 geo2.y = geo.y + geo.height / 2 - geo2.height / 2; 2695 2696 if (direction == mxConstants.DIRECTION_NORTH) 2697 { 2698 geo2.y = geo2.y - geo.height / 2 - geo2.height / 2 - length; 2699 } 2700 else if (direction == mxConstants.DIRECTION_EAST) 2701 { 2702 geo2.x = geo2.x + geo.width / 2 + geo2.width / 2 + length; 2703 } 2704 else if (direction == mxConstants.DIRECTION_SOUTH) 2705 { 2706 geo2.y = geo2.y + geo.height / 2 + geo2.height / 2 + length; 2707 } 2708 else if (direction == mxConstants.DIRECTION_WEST) 2709 { 2710 geo2.x = geo2.x - geo.width / 2 - geo2.width / 2 - length; 2711 } 2712 2713 // Adds offset to match cells without connecting edge 2714 if (graph.model.isEdge(target) && geo2.getTerminalPoint(true) != null && 2715 target.getTerminal(false) != null) 2716 { 2717 var targetGeo = graph.getCellGeometry(target.getTerminal(false)); 2718 2719 if (targetGeo != null) 2720 { 2721 if (direction == mxConstants.DIRECTION_NORTH) 2722 { 2723 geo2.x -= targetGeo.getCenterX(); 2724 geo2.y -= targetGeo.getCenterY() + targetGeo.height / 2; 2725 } 2726 else if (direction == mxConstants.DIRECTION_EAST) 2727 { 2728 geo2.x -= targetGeo.getCenterX() - targetGeo.width / 2; 2729 geo2.y -= targetGeo.getCenterY(); 2730 } 2731 else if (direction == mxConstants.DIRECTION_SOUTH) 2732 { 2733 geo2.x -= targetGeo.getCenterX(); 2734 geo2.y -= targetGeo.getCenterY() - targetGeo.height / 2; 2735 } 2736 else if (direction == mxConstants.DIRECTION_WEST) 2737 { 2738 geo2.x -= targetGeo.getCenterX() + targetGeo.width / 2; 2739 geo2.y -= targetGeo.getCenterY(); 2740 } 2741 } 2742 } 2743 } 2744 } 2745 } 2746 2747 return geo2; 2748}; 2749 2750/** 2751 * Limits drop style to non-transparent source shapes. 2752 */ 2753Sidebar.prototype.isDropStyleEnabled = function(cells, firstVertex) 2754{ 2755 var result = true; 2756 2757 if (firstVertex != null && cells.length == 1) 2758 { 2759 var vstyle = this.graph.getCellStyle(cells[firstVertex]); 2760 2761 if (vstyle != null) 2762 { 2763 result = mxUtils.getValue(vstyle, mxConstants.STYLE_STROKECOLOR, mxConstants.NONE) != mxConstants.NONE || 2764 mxUtils.getValue(vstyle, mxConstants.STYLE_FILLCOLOR, mxConstants.NONE) != mxConstants.NONE; 2765 } 2766 } 2767 2768 return result; 2769}; 2770 2771/** 2772 * Ignores swimlanes as drop style targets. 2773 */ 2774Sidebar.prototype.isDropStyleTargetIgnored = function(state) 2775{ 2776 return this.graph.isSwimlane(state.cell) || this.graph.isTableCell(state.cell) || 2777 this.graph.isTableRow(state.cell) || this.graph.isTable(state.cell); 2778}; 2779 2780/** 2781 * Creates a drag source for the given element. 2782 */ 2783Sidebar.prototype.createDragSource = function(elt, dropHandler, preview, cells, bounds) 2784{ 2785 // Checks if the cells contain any vertices 2786 var ui = this.editorUi; 2787 var graph = ui.editor.graph; 2788 var freeSourceEdge = null; 2789 var firstVertex = null; 2790 var sidebar = this; 2791 2792 for (var i = 0; i < cells.length; i++) 2793 { 2794 if (firstVertex == null && graph.model.isVertex(cells[i])) 2795 { 2796 firstVertex = i; 2797 } 2798 else if (freeSourceEdge == null && graph.model.isEdge(cells[i]) && 2799 graph.model.getTerminal(cells[i], true) == null) 2800 { 2801 freeSourceEdge = i; 2802 } 2803 2804 if (firstVertex != null && freeSourceEdge != null) 2805 { 2806 break; 2807 } 2808 } 2809 2810 var dropStyleEnabled = this.isDropStyleEnabled(cells, firstVertex); 2811 2812 var dragSource = mxUtils.makeDraggable(elt, graph, mxUtils.bind(this, function(graph, evt, target, x, y) 2813 { 2814 if (this.updateThread != null) 2815 { 2816 window.clearTimeout(this.updateThread); 2817 } 2818 2819 if (cells != null && currentStyleTarget != null && activeArrow == styleTarget) 2820 { 2821 var tmp = graph.isCellSelected(currentStyleTarget.cell) ? graph.getSelectionCells() : [currentStyleTarget.cell]; 2822 var updatedCells = this.updateShapes((graph.model.isEdge(currentStyleTarget.cell)) ? cells[0] : cells[firstVertex], tmp); 2823 graph.setSelectionCells(updatedCells); 2824 } 2825 else if (cells != null && activeArrow != null && currentTargetState != null && activeArrow != styleTarget) 2826 { 2827 var index = (graph.model.isEdge(currentTargetState.cell) || freeSourceEdge == null) ? firstVertex : freeSourceEdge; 2828 graph.setSelectionCells(this.dropAndConnect(currentTargetState.cell, cells, direction, index, evt)); 2829 } 2830 else 2831 { 2832 dropHandler.apply(this, arguments); 2833 } 2834 2835 if (this.editorUi.hoverIcons != null) 2836 { 2837 this.editorUi.hoverIcons.update(graph.view.getState(graph.getSelectionCell())); 2838 } 2839 }), preview, 0, 0, graph.autoscroll, true, true); 2840 2841 // Stops dragging if cancel is pressed 2842 graph.addListener(mxEvent.ESCAPE, function(sender, evt) 2843 { 2844 if (dragSource.isActive()) 2845 { 2846 dragSource.reset(); 2847 } 2848 }); 2849 2850 // Overrides mouseDown to ignore popup triggers 2851 var mouseDown = dragSource.mouseDown; 2852 2853 dragSource.mouseDown = function(evt) 2854 { 2855 if (!mxEvent.isPopupTrigger(evt) && !mxEvent.isMultiTouchEvent(evt) && 2856 !graph.isCellLocked(graph.getDefaultParent())) 2857 { 2858 graph.stopEditing(); 2859 mouseDown.apply(this, arguments); 2860 } 2861 }; 2862 2863 // Workaround for event redirection via image tag in quirks and IE8 2864 function createArrow(img, tooltip) 2865 { 2866 var arrow = null; 2867 arrow = mxUtils.createImage(img.src); 2868 arrow.style.width = img.width + 'px'; 2869 arrow.style.height = img.height + 'px'; 2870 2871 if (tooltip != null) 2872 { 2873 arrow.setAttribute('title', tooltip); 2874 } 2875 2876 mxUtils.setOpacity(arrow, (img == this.refreshTarget) ? 30 : 20); 2877 arrow.style.position = 'absolute'; 2878 arrow.style.cursor = 'crosshair'; 2879 2880 return arrow; 2881 }; 2882 2883 var currentTargetState = null; 2884 var currentStateHandle = null; 2885 var currentStyleTarget = null; 2886 var activeTarget = false; 2887 2888 var arrowUp = createArrow(this.triangleUp, mxResources.get('connect')); 2889 var arrowRight = createArrow(this.triangleRight, mxResources.get('connect')); 2890 var arrowDown = createArrow(this.triangleDown, mxResources.get('connect')); 2891 var arrowLeft = createArrow(this.triangleLeft, mxResources.get('connect')); 2892 var styleTarget = createArrow(this.refreshTarget, mxResources.get('replace')); 2893 // Workaround for actual parentNode not being updated in old IE 2894 var styleTargetParent = null; 2895 var roundSource = createArrow(this.roundDrop); 2896 var roundTarget = createArrow(this.roundDrop); 2897 var direction = mxConstants.DIRECTION_NORTH; 2898 var activeArrow = null; 2899 2900 function checkArrow(x, y, bounds, arrow) 2901 { 2902 if (arrow.parentNode != null) 2903 { 2904 if (mxUtils.contains(bounds, x, y)) 2905 { 2906 mxUtils.setOpacity(arrow, 100); 2907 activeArrow = arrow; 2908 } 2909 else 2910 { 2911 mxUtils.setOpacity(arrow, (arrow == styleTarget) ? 30 : 20); 2912 } 2913 } 2914 2915 return bounds; 2916 }; 2917 2918 // Hides guides and preview if target is active 2919 var dsCreatePreviewElement = dragSource.createPreviewElement; 2920 2921 // Stores initial size of preview element 2922 dragSource.createPreviewElement = function(graph) 2923 { 2924 var elt = dsCreatePreviewElement.apply(this, arguments); 2925 2926 // Pass-through events required to tooltip on replace shape 2927 if (mxClient.IS_SVG) 2928 { 2929 elt.style.pointerEvents = 'none'; 2930 } 2931 2932 this.previewElementWidth = elt.style.width; 2933 this.previewElementHeight = elt.style.height; 2934 2935 return elt; 2936 }; 2937 2938 // Shows/hides hover icons 2939 var dragEnter = dragSource.dragEnter; 2940 dragSource.dragEnter = function(graph, evt) 2941 { 2942 if (ui.hoverIcons != null) 2943 { 2944 ui.hoverIcons.setDisplay('none'); 2945 } 2946 2947 dragEnter.apply(this, arguments); 2948 }; 2949 2950 var dragExit = dragSource.dragExit; 2951 dragSource.dragExit = function(graph, evt) 2952 { 2953 if (ui.hoverIcons != null) 2954 { 2955 ui.hoverIcons.setDisplay(''); 2956 } 2957 2958 dragExit.apply(this, arguments); 2959 }; 2960 2961 dragSource.dragOver = function(graph, evt) 2962 { 2963 mxDragSource.prototype.dragOver.apply(this, arguments); 2964 2965 if (this.currentGuide != null && activeArrow != null) 2966 { 2967 this.currentGuide.hide(); 2968 } 2969 2970 if (this.previewElement != null) 2971 { 2972 var view = graph.view; 2973 2974 if (currentStyleTarget != null && activeArrow == styleTarget) 2975 { 2976 this.previewElement.style.display = (graph.model.isEdge(currentStyleTarget.cell)) ? 'none' : ''; 2977 2978 this.previewElement.style.left = currentStyleTarget.x + 'px'; 2979 this.previewElement.style.top = currentStyleTarget.y + 'px'; 2980 this.previewElement.style.width = currentStyleTarget.width + 'px'; 2981 this.previewElement.style.height = currentStyleTarget.height + 'px'; 2982 } 2983 else if (currentTargetState != null && activeArrow != null) 2984 { 2985 if (dragSource.currentHighlight != null && dragSource.currentHighlight.state != null) 2986 { 2987 dragSource.currentHighlight.hide(); 2988 } 2989 2990 var index = (graph.model.isEdge(currentTargetState.cell) || freeSourceEdge == null) ? firstVertex : freeSourceEdge; 2991 var geo = sidebar.getDropAndConnectGeometry(currentTargetState.cell, cells[index], direction, cells); 2992 var geo2 = (!graph.model.isEdge(currentTargetState.cell)) ? graph.getCellGeometry(currentTargetState.cell) : null; 2993 var geo3 = graph.getCellGeometry(cells[index]); 2994 var parent = graph.model.getParent(currentTargetState.cell); 2995 var dx = view.translate.x * view.scale; 2996 var dy = view.translate.y * view.scale; 2997 2998 if (geo2 != null && !geo2.relative && graph.model.isVertex(parent) && parent != view.currentRoot) 2999 { 3000 var pState = view.getState(parent); 3001 3002 dx = pState.x; 3003 dy = pState.y; 3004 } 3005 3006 var dx2 = geo3.x; 3007 var dy2 = geo3.y; 3008 3009 // Ignores geometry of edges 3010 if (graph.model.isEdge(cells[index])) 3011 { 3012 dx2 = 0; 3013 dy2 = 0; 3014 } 3015 3016 // Shows preview at drop location 3017 this.previewElement.style.left = ((geo.x - dx2) * view.scale + dx) + 'px'; 3018 this.previewElement.style.top = ((geo.y - dy2) * view.scale + dy) + 'px'; 3019 3020 if (cells.length == 1) 3021 { 3022 this.previewElement.style.width = (geo.width * view.scale) + 'px'; 3023 this.previewElement.style.height = (geo.height * view.scale) + 'px'; 3024 } 3025 3026 this.previewElement.style.display = ''; 3027 } 3028 else if (dragSource.currentHighlight.state != null && 3029 graph.model.isEdge(dragSource.currentHighlight.state.cell)) 3030 { 3031 // Centers drop cells when splitting edges 3032 this.previewElement.style.left = Math.round(parseInt(this.previewElement.style.left) - 3033 bounds.width * view.scale / 2) + 'px'; 3034 this.previewElement.style.top = Math.round(parseInt(this.previewElement.style.top) - 3035 bounds.height * view.scale / 2) + 'px'; 3036 } 3037 else 3038 { 3039 this.previewElement.style.width = this.previewElementWidth; 3040 this.previewElement.style.height = this.previewElementHeight; 3041 this.previewElement.style.display = ''; 3042 } 3043 } 3044 }; 3045 3046 var startTime = new Date().getTime(); 3047 var timeOnTarget = 0; 3048 var prev = null; 3049 3050 // Gets source cell style to compare shape below 3051 var sourceCellStyle = this.editorUi.editor.graph.getCellStyle(cells[0]); 3052 3053 // Allows drop into cell only if target is a valid root 3054 dragSource.getDropTarget = mxUtils.bind(this, function(graph, x, y, evt) 3055 { 3056 // Alt means no targets at all 3057 // LATER: Show preview where result will go 3058 var cell = (!mxEvent.isAltDown(evt) && cells != null) ? 3059 graph.getCellAt(x, y, null, null, null, function(state, x, y) 3060 { 3061 return graph.isContainer(state.cell); 3062 }) : null; 3063 3064 // Uses connectable parent vertex if one exists 3065 if (cell != null && !this.graph.isCellConnectable(cell) && 3066 !this.graph.model.isEdge(cell)) 3067 { 3068 var parent = this.graph.getModel().getParent(cell); 3069 3070 if (this.graph.getModel().isVertex(parent) && 3071 this.graph.isCellConnectable(parent)) 3072 { 3073 cell = parent; 3074 } 3075 } 3076 3077 // Ignores locked cells 3078 if (graph.isCellLocked(cell)) 3079 { 3080 cell = null; 3081 } 3082 3083 var state = graph.view.getState(cell); 3084 activeArrow = null; 3085 var bbox = null; 3086 3087 // Time on target 3088 if (prev != state) 3089 { 3090 startTime = new Date().getTime(); 3091 timeOnTarget = 0; 3092 prev = state; 3093 3094 if (this.updateThread != null) 3095 { 3096 window.clearTimeout(this.updateThread); 3097 } 3098 3099 if (state != null) 3100 { 3101 this.updateThread = window.setTimeout(function() 3102 { 3103 if (activeArrow == null) 3104 { 3105 prev = state; 3106 dragSource.getDropTarget(graph, x, y, evt); 3107 } 3108 }, this.dropTargetDelay + 10); 3109 } 3110 } 3111 else 3112 { 3113 timeOnTarget = new Date().getTime() - startTime; 3114 } 3115 3116 // Shift means disabled, delayed on cells with children, shows after this.dropTargetDelay, hides after 2500ms 3117 if (dropStyleEnabled && (timeOnTarget < 2500) && state != null && !mxEvent.isShiftDown(evt) && 3118 // If shape is equal or target has no stroke, fill and gradient then use longer delay except for images 3119 (((mxUtils.getValue(state.style, mxConstants.STYLE_SHAPE) != mxUtils.getValue(sourceCellStyle, mxConstants.STYLE_SHAPE) && 3120 (mxUtils.getValue(state.style, mxConstants.STYLE_STROKECOLOR, mxConstants.NONE) != mxConstants.NONE || 3121 mxUtils.getValue(state.style, mxConstants.STYLE_FILLCOLOR, mxConstants.NONE) != mxConstants.NONE || 3122 mxUtils.getValue(state.style, mxConstants.STYLE_GRADIENTCOLOR, mxConstants.NONE) != mxConstants.NONE)) || 3123 mxUtils.getValue(sourceCellStyle, mxConstants.STYLE_SHAPE) == 'image') || 3124 timeOnTarget > 1500 || graph.model.isEdge(state.cell)) && (timeOnTarget > this.dropTargetDelay) && 3125 !this.isDropStyleTargetIgnored(state) && ((graph.model.isVertex(state.cell) && firstVertex != null) || 3126 (graph.model.isEdge(state.cell) && graph.model.isEdge(cells[0])))) 3127 { 3128 if (graph.isCellEditable(state.cell)) 3129 { 3130 currentStyleTarget = state; 3131 var tmp = (graph.model.isEdge(state.cell)) ? graph.view.getPoint(state) : 3132 new mxPoint(state.getCenterX(), state.getCenterY()); 3133 tmp = new mxRectangle(tmp.x - this.refreshTarget.width / 2, tmp.y - this.refreshTarget.height / 2, 3134 this.refreshTarget.width, this.refreshTarget.height); 3135 3136 styleTarget.style.left = Math.floor(tmp.x) + 'px'; 3137 styleTarget.style.top = Math.floor(tmp.y) + 'px'; 3138 3139 if (styleTargetParent == null) 3140 { 3141 graph.container.appendChild(styleTarget); 3142 styleTargetParent = styleTarget.parentNode; 3143 } 3144 3145 checkArrow(x, y, tmp, styleTarget); 3146 } 3147 } 3148 // Does not reset on ignored edges 3149 else if (currentStyleTarget == null || !mxUtils.contains(currentStyleTarget, x, y) || 3150 (timeOnTarget > 1500 && !mxEvent.isShiftDown(evt))) 3151 { 3152 currentStyleTarget = null; 3153 3154 if (styleTargetParent != null) 3155 { 3156 styleTarget.parentNode.removeChild(styleTarget); 3157 styleTargetParent = null; 3158 } 3159 } 3160 else if (currentStyleTarget != null && styleTargetParent != null) 3161 { 3162 // Sets active Arrow as side effect 3163 var tmp = (graph.model.isEdge(currentStyleTarget.cell)) ? graph.view.getPoint(currentStyleTarget) : new mxPoint(currentStyleTarget.getCenterX(), currentStyleTarget.getCenterY()); 3164 tmp = new mxRectangle(tmp.x - this.refreshTarget.width / 2, tmp.y - this.refreshTarget.height / 2, 3165 this.refreshTarget.width, this.refreshTarget.height); 3166 checkArrow(x, y, tmp, styleTarget); 3167 } 3168 3169 // Checks if inside bounds 3170 if (activeTarget && currentTargetState != null && !mxEvent.isAltDown(evt) && activeArrow == null) 3171 { 3172 // LATER: Use hit-detection for edges 3173 bbox = mxRectangle.fromRectangle(currentTargetState); 3174 3175 if (graph.model.isEdge(currentTargetState.cell)) 3176 { 3177 var pts = currentTargetState.absolutePoints; 3178 3179 if (roundSource.parentNode != null) 3180 { 3181 var p0 = pts[0]; 3182 bbox.add(checkArrow(x, y, new mxRectangle(p0.x - this.roundDrop.width / 2, 3183 p0.y - this.roundDrop.height / 2, this.roundDrop.width, this.roundDrop.height), roundSource)); 3184 } 3185 3186 if (roundTarget.parentNode != null) 3187 { 3188 var pe = pts[pts.length - 1]; 3189 bbox.add(checkArrow(x, y, new mxRectangle(pe.x - this.roundDrop.width / 2, 3190 pe.y - this.roundDrop.height / 2, 3191 this.roundDrop.width, this.roundDrop.height), roundTarget)); 3192 } 3193 } 3194 else 3195 { 3196 var bds = mxRectangle.fromRectangle(currentTargetState); 3197 3198 // Uses outer bounding box to take rotation into account 3199 if (currentTargetState.shape != null && currentTargetState.shape.boundingBox != null) 3200 { 3201 bds = mxRectangle.fromRectangle(currentTargetState.shape.boundingBox); 3202 } 3203 3204 bds.grow(this.graph.tolerance); 3205 bds.grow(HoverIcons.prototype.arrowSpacing); 3206 3207 var handler = this.graph.selectionCellsHandler.getHandler(currentTargetState.cell); 3208 3209 if (handler != null) 3210 { 3211 bds.x -= handler.horizontalOffset / 2; 3212 bds.y -= handler.verticalOffset / 2; 3213 bds.width += handler.horizontalOffset; 3214 bds.height += handler.verticalOffset; 3215 3216 // Adds bounding box of rotation handle to avoid overlap 3217 if (handler.rotationShape != null && handler.rotationShape.node != null && 3218 handler.rotationShape.node.style.visibility != 'hidden' && 3219 handler.rotationShape.node.style.display != 'none' && 3220 handler.rotationShape.boundingBox != null) 3221 { 3222 bds.add(handler.rotationShape.boundingBox); 3223 } 3224 } 3225 3226 bbox.add(checkArrow(x, y, new mxRectangle(currentTargetState.getCenterX() - this.triangleUp.width / 2, 3227 bds.y - this.triangleUp.height, this.triangleUp.width, this.triangleUp.height), arrowUp)); 3228 bbox.add(checkArrow(x, y, new mxRectangle(bds.x + bds.width, 3229 currentTargetState.getCenterY() - this.triangleRight.height / 2, 3230 this.triangleRight.width, this.triangleRight.height), arrowRight)); 3231 bbox.add(checkArrow(x, y, new mxRectangle(currentTargetState.getCenterX() - this.triangleDown.width / 2, 3232 bds.y + bds.height, this.triangleDown.width, this.triangleDown.height), arrowDown)); 3233 bbox.add(checkArrow(x, y, new mxRectangle(bds.x - this.triangleLeft.width, 3234 currentTargetState.getCenterY() - this.triangleLeft.height / 2, 3235 this.triangleLeft.width, this.triangleLeft.height), arrowLeft)); 3236 } 3237 3238 // Adds tolerance 3239 if (bbox != null) 3240 { 3241 bbox.grow(10); 3242 } 3243 } 3244 3245 direction = mxConstants.DIRECTION_NORTH; 3246 3247 if (activeArrow == arrowRight) 3248 { 3249 direction = mxConstants.DIRECTION_EAST; 3250 } 3251 else if (activeArrow == arrowDown || activeArrow == roundTarget) 3252 { 3253 direction = mxConstants.DIRECTION_SOUTH; 3254 } 3255 else if (activeArrow == arrowLeft) 3256 { 3257 direction = mxConstants.DIRECTION_WEST; 3258 } 3259 3260 if (currentStyleTarget != null && activeArrow == styleTarget) 3261 { 3262 state = currentStyleTarget; 3263 } 3264 3265 var validTarget = (firstVertex == null || graph.isCellConnectable(cells[firstVertex])) && 3266 ((graph.model.isEdge(cell) && firstVertex != null) || 3267 (graph.model.isVertex(cell) && graph.isCellConnectable(cell))); 3268 3269 // Drop arrows shown after this.dropTargetDelay, hidden after 5 secs, switches arrows after 500ms 3270 if ((currentTargetState != null && timeOnTarget >= 5000) || 3271 (currentTargetState != state && 3272 (bbox == null || !mxUtils.contains(bbox, x, y) || 3273 (timeOnTarget > 500 && activeArrow == null && validTarget)))) 3274 { 3275 activeTarget = false; 3276 currentTargetState = ((timeOnTarget < 5000 && timeOnTarget > this.dropTargetDelay) || 3277 graph.model.isEdge(cell)) ? state : null; 3278 3279 if (currentTargetState != null && validTarget) 3280 { 3281 var elts = [roundSource, roundTarget, arrowUp, arrowRight, arrowDown, arrowLeft]; 3282 3283 for (var i = 0; i < elts.length; i++) 3284 { 3285 if (elts[i].parentNode != null) 3286 { 3287 elts[i].parentNode.removeChild(elts[i]); 3288 } 3289 } 3290 3291 if (graph.model.isEdge(cell)) 3292 { 3293 var pts = state.absolutePoints; 3294 3295 if (pts != null) 3296 { 3297 var p0 = pts[0]; 3298 var pe = pts[pts.length - 1]; 3299 var tol = graph.tolerance; 3300 var box = new mxRectangle(x - tol, y - tol, 2 * tol, 2 * tol); 3301 3302 roundSource.style.left = Math.floor(p0.x - this.roundDrop.width / 2) + 'px'; 3303 roundSource.style.top = Math.floor(p0.y - this.roundDrop.height / 2) + 'px'; 3304 3305 roundTarget.style.left = Math.floor(pe.x - this.roundDrop.width / 2) + 'px'; 3306 roundTarget.style.top = Math.floor(pe.y - this.roundDrop.height / 2) + 'px'; 3307 3308 if (graph.model.getTerminal(cell, true) == null) 3309 { 3310 graph.container.appendChild(roundSource); 3311 } 3312 3313 if (graph.model.getTerminal(cell, false) == null) 3314 { 3315 graph.container.appendChild(roundTarget); 3316 } 3317 } 3318 } 3319 else 3320 { 3321 var bds = mxRectangle.fromRectangle(state); 3322 3323 // Uses outer bounding box to take rotation into account 3324 if (state.shape != null && state.shape.boundingBox != null) 3325 { 3326 bds = mxRectangle.fromRectangle(state.shape.boundingBox); 3327 } 3328 3329 bds.grow(this.graph.tolerance); 3330 bds.grow(HoverIcons.prototype.arrowSpacing); 3331 3332 var handler = this.graph.selectionCellsHandler.getHandler(state.cell); 3333 3334 if (handler != null) 3335 { 3336 bds.x -= handler.horizontalOffset / 2; 3337 bds.y -= handler.verticalOffset / 2; 3338 bds.width += handler.horizontalOffset; 3339 bds.height += handler.verticalOffset; 3340 3341 // Adds bounding box of rotation handle to avoid overlap 3342 if (handler.rotationShape != null && handler.rotationShape.node != null && 3343 handler.rotationShape.node.style.visibility != 'hidden' && 3344 handler.rotationShape.node.style.display != 'none' && 3345 handler.rotationShape.boundingBox != null) 3346 { 3347 bds.add(handler.rotationShape.boundingBox); 3348 } 3349 } 3350 3351 arrowUp.style.left = Math.floor(state.getCenterX() - this.triangleUp.width / 2) + 'px'; 3352 arrowUp.style.top = Math.floor(bds.y - this.triangleUp.height) + 'px'; 3353 3354 arrowRight.style.left = Math.floor(bds.x + bds.width) + 'px'; 3355 arrowRight.style.top = Math.floor(state.getCenterY() - this.triangleRight.height / 2) + 'px'; 3356 3357 arrowDown.style.left = arrowUp.style.left 3358 arrowDown.style.top = Math.floor(bds.y + bds.height) + 'px'; 3359 3360 arrowLeft.style.left = Math.floor(bds.x - this.triangleLeft.width) + 'px'; 3361 arrowLeft.style.top = arrowRight.style.top; 3362 3363 if (state.style['portConstraint'] != 'eastwest') 3364 { 3365 graph.container.appendChild(arrowUp); 3366 graph.container.appendChild(arrowDown); 3367 } 3368 3369 graph.container.appendChild(arrowRight); 3370 graph.container.appendChild(arrowLeft); 3371 } 3372 3373 // Hides handle for cell under mouse 3374 if (state != null) 3375 { 3376 currentStateHandle = graph.selectionCellsHandler.getHandler(state.cell); 3377 3378 if (currentStateHandle != null && currentStateHandle.setHandlesVisible != null) 3379 { 3380 currentStateHandle.setHandlesVisible(false); 3381 } 3382 } 3383 3384 activeTarget = true; 3385 } 3386 else 3387 { 3388 var elts = [roundSource, roundTarget, arrowUp, arrowRight, arrowDown, arrowLeft]; 3389 3390 for (var i = 0; i < elts.length; i++) 3391 { 3392 if (elts[i].parentNode != null) 3393 { 3394 elts[i].parentNode.removeChild(elts[i]); 3395 } 3396 } 3397 } 3398 } 3399 3400 if (!activeTarget && currentStateHandle != null) 3401 { 3402 currentStateHandle.setHandlesVisible(true); 3403 } 3404 3405 // Handles drop target 3406 var target = ((!mxEvent.isAltDown(evt) || mxEvent.isShiftDown(evt)) && 3407 !(currentStyleTarget != null && activeArrow == styleTarget)) ? 3408 mxDragSource.prototype.getDropTarget.apply(this, arguments) : null; 3409 var model = graph.getModel(); 3410 3411 if (target != null) 3412 { 3413 if (activeArrow != null || !graph.isSplitTarget(target, cells, evt)) 3414 { 3415 // Selects parent group as drop target 3416 while (target != null && !graph.isValidDropTarget(target, cells, evt) && 3417 model.isVertex(model.getParent(target))) 3418 { 3419 target = model.getParent(target); 3420 } 3421 3422 if (target != null && (graph.view.currentRoot == target || 3423 (!graph.isValidRoot(target) && 3424 graph.getModel().getChildCount(target) == 0) || 3425 graph.isCellLocked(target) || model.isEdge(target) || 3426 !graph.isValidDropTarget(target, cells, evt))) 3427 { 3428 target = null; 3429 } 3430 } 3431 } 3432 3433 if (graph.isCellLocked(target)) 3434 { 3435 target = null; 3436 } 3437 3438 return target; 3439 }); 3440 3441 dragSource.stopDrag = function() 3442 { 3443 mxDragSource.prototype.stopDrag.apply(this, arguments); 3444 3445 var elts = [roundSource, roundTarget, styleTarget, arrowUp, arrowRight, arrowDown, arrowLeft]; 3446 3447 for (var i = 0; i < elts.length; i++) 3448 { 3449 if (elts[i].parentNode != null) 3450 { 3451 elts[i].parentNode.removeChild(elts[i]); 3452 } 3453 } 3454 3455 if (currentTargetState != null && currentStateHandle != null) 3456 { 3457 currentStateHandle.reset(); 3458 } 3459 3460 currentStateHandle = null; 3461 currentTargetState = null; 3462 currentStyleTarget = null; 3463 styleTargetParent = null; 3464 activeArrow = null; 3465 }; 3466 3467 return dragSource; 3468}; 3469 3470/** 3471 * Adds a handler for inserting the cell with a single click. 3472 */ 3473Sidebar.prototype.itemClicked = function(cells, ds, evt, elt) 3474{ 3475 var graph = this.editorUi.editor.graph; 3476 graph.container.focus(); 3477 3478 // Alt+Click inserts and connects 3479 if (mxEvent.isAltDown(evt) && graph.getSelectionCount() == 1 && 3480 graph.model.isVertex(graph.getSelectionCell())) 3481 { 3482 var firstVertex = null; 3483 3484 for (var i = 0; i < cells.length && firstVertex == null; i++) 3485 { 3486 if (graph.model.isVertex(cells[i])) 3487 { 3488 firstVertex = i; 3489 } 3490 } 3491 3492 if (firstVertex != null) 3493 { 3494 graph.setSelectionCells(this.dropAndConnect(graph.getSelectionCell(), cells, 3495 (mxEvent.isMetaDown(evt) || mxEvent.isControlDown(evt)) ? 3496 (mxEvent.isShiftDown(evt) ? mxConstants.DIRECTION_WEST : mxConstants.DIRECTION_NORTH) : 3497 (mxEvent.isShiftDown(evt) ? mxConstants.DIRECTION_EAST : mxConstants.DIRECTION_SOUTH), 3498 firstVertex, evt)); 3499 graph.scrollCellToVisible(graph.getSelectionCell()); 3500 } 3501 } 3502 // Shift+Click updates shape 3503 else if (mxEvent.isShiftDown(evt) && !graph.isSelectionEmpty()) 3504 { 3505 var temp = graph.getEditableCells(graph.getSelectionCells()); 3506 this.updateShapes(cells[0], temp); 3507 graph.scrollCellToVisible(temp); 3508 } 3509 else 3510 { 3511 var pt = (mxEvent.isAltDown(evt)) ? graph.getFreeInsertPoint() : 3512 graph.getCenterInsertPoint(graph.getBoundingBoxFromGeometry(cells, true)); 3513 ds.drop(graph, evt, null, pt.x, pt.y, true); 3514 } 3515}; 3516 3517/** 3518 * Adds a handler for inserting the cell with a single click. 3519 */ 3520Sidebar.prototype.addClickHandler = function(elt, ds, cells) 3521{ 3522 var graph = this.editorUi.editor.graph; 3523 var oldMouseDown = ds.mouseDown; 3524 var oldMouseMove = ds.mouseMove; 3525 var oldMouseUp = ds.mouseUp; 3526 var tol = graph.tolerance; 3527 var first = null; 3528 var sb = this; 3529 3530 ds.mouseDown =function(evt) 3531 { 3532 oldMouseDown.apply(this, arguments); 3533 first = new mxPoint(mxEvent.getClientX(evt), mxEvent.getClientY(evt)); 3534 3535 if (this.dragElement != null) 3536 { 3537 this.dragElement.style.display = 'none'; 3538 mxUtils.setOpacity(elt, 50); 3539 } 3540 }; 3541 3542 ds.mouseMove = function(evt) 3543 { 3544 if (this.dragElement != null && this.dragElement.style.display == 'none' && 3545 first != null && (Math.abs(first.x - mxEvent.getClientX(evt)) > tol || 3546 Math.abs(first.y - mxEvent.getClientY(evt)) > tol)) 3547 { 3548 this.dragElement.style.display = ''; 3549 mxUtils.setOpacity(elt, 100); 3550 } 3551 3552 oldMouseMove.apply(this, arguments); 3553 }; 3554 3555 ds.mouseUp = function(evt) 3556 { 3557 try 3558 { 3559 if (!mxEvent.isPopupTrigger(evt) && this.currentGraph == null && 3560 this.dragElement != null && this.dragElement.style.display == 'none') 3561 { 3562 sb.itemClicked(cells, ds, evt, elt); 3563 } 3564 3565 oldMouseUp.apply(ds, arguments); 3566 mxUtils.setOpacity(elt, 100); 3567 first = null; 3568 3569 // Blocks tooltips on this element after single click 3570 sb.currentElt = elt; 3571 } 3572 catch (e) 3573 { 3574 ds.reset(); 3575 sb.editorUi.handleError(e); 3576 } 3577 }; 3578}; 3579 3580/** 3581 * Creates a drop handler for inserting the given cells. 3582 */ 3583Sidebar.prototype.createVertexTemplateEntry = function(style, width, height, value, title, showLabel, showTitle, tags) 3584{ 3585 if (tags != null && title != null) 3586 { 3587 tags += ' ' + title; 3588 } 3589 3590 tags = (tags != null && tags.length > 0) ? tags : ((title != null) ? title.toLowerCase() : ''); 3591 3592 return this.addEntry(tags, mxUtils.bind(this, function() 3593 { 3594 return this.createVertexTemplate(style, width, height, value, title, showLabel, showTitle); 3595 })); 3596} 3597 3598/** 3599 * Creates a drop handler for inserting the given cells. 3600 */ 3601Sidebar.prototype.createVertexTemplate = function(style, width, height, value, title, showLabel, showTitle, allowCellsInserted, showTooltip) 3602{ 3603 var cells = [new mxCell((value != null) ? value : '', new mxGeometry(0, 0, width, height), style)]; 3604 cells[0].vertex = true; 3605 3606 return this.createVertexTemplateFromCells(cells, width, height, title, showLabel, showTitle, allowCellsInserted, showTooltip); 3607}; 3608 3609/** 3610 * Creates a drop handler for inserting the given cells. 3611 */ 3612Sidebar.prototype.createVertexTemplateFromData = function(data, width, height, title, showLabel, showTitle, allowCellsInserted, showTooltip) 3613{ 3614 var doc = mxUtils.parseXml(Graph.decompress(data)); 3615 var codec = new mxCodec(doc); 3616 3617 var model = new mxGraphModel(); 3618 codec.decode(doc.documentElement, model); 3619 3620 var cells = this.graph.cloneCells(model.root.getChildAt(0).children); 3621 3622 return this.createVertexTemplateFromCells(cells, width, height, title, showLabel, showTitle, allowCellsInserted, showTooltip); 3623}; 3624 3625/** 3626 * Creates a drop handler for inserting the given cells. 3627 */ 3628Sidebar.prototype.createVertexTemplateFromCells = function(cells, width, height, title, showLabel, showTitle, allowCellsInserted, showTooltip) 3629{ 3630 // Use this line to convert calls to this function with lots of boilerplate code for creating cells 3631 //console.trace('xml', Graph.compress(mxUtils.getXml(this.graph.encodeCells(cells))), cells); 3632 return this.createItem(cells, title, showLabel, showTitle, width, height, allowCellsInserted, showTooltip); 3633}; 3634 3635/** 3636 * 3637 */ 3638Sidebar.prototype.createEdgeTemplateEntry = function(style, width, height, value, title, showLabel, tags, allowCellsInserted, showTooltip) 3639{ 3640 tags = (tags != null && tags.length > 0) ? tags : title.toLowerCase(); 3641 3642 return this.addEntry(tags, mxUtils.bind(this, function() 3643 { 3644 return this.createEdgeTemplate(style, width, height, value, title, showLabel, allowCellsInserted, showTooltip); 3645 })); 3646}; 3647 3648/** 3649 * Creates a drop handler for inserting the given cells. 3650 */ 3651Sidebar.prototype.createEdgeTemplate = function(style, width, height, value, title, showLabel, allowCellsInserted, showTooltip) 3652{ 3653 var cell = new mxCell((value != null) ? value : '', new mxGeometry(0, 0, width, height), style); 3654 cell.geometry.setTerminalPoint(new mxPoint(0, height), true); 3655 cell.geometry.setTerminalPoint(new mxPoint(width, 0), false); 3656 cell.geometry.relative = true; 3657 cell.edge = true; 3658 3659 return this.createEdgeTemplateFromCells([cell], width, height, title, showLabel, allowCellsInserted, showTooltip); 3660}; 3661 3662/** 3663 * Creates a drop handler for inserting the given cells. 3664 */ 3665Sidebar.prototype.createEdgeTemplateFromCells = function(cells, width, height, title, showLabel, allowCellsInserted, showTooltip, showTitle) 3666{ 3667 return this.createItem(cells, title, showLabel, (showTitle != null) ? showTitle : true, width, height, allowCellsInserted, showTooltip); 3668}; 3669 3670/** 3671 * Adds the given palette. 3672 */ 3673Sidebar.prototype.addPaletteFunctions = function(id, title, expanded, fns) 3674{ 3675 this.addPalette(id, title, expanded, mxUtils.bind(this, function(content) 3676 { 3677 for (var i = 0; i < fns.length; i++) 3678 { 3679 content.appendChild(fns[i](content)); 3680 } 3681 })); 3682}; 3683 3684/** 3685 * Adds the given palette. 3686 */ 3687Sidebar.prototype.addPalette = function(id, title, expanded, onInit) 3688{ 3689 var elt = this.createTitle(title); 3690 this.container.appendChild(elt); 3691 3692 var div = document.createElement('div'); 3693 div.className = 'geSidebar'; 3694 3695 // Disables built-in pan and zoom in IE10 and later 3696 if (mxClient.IS_POINTER) 3697 { 3698 div.style.touchAction = 'none'; 3699 } 3700 3701 if (expanded) 3702 { 3703 onInit(div); 3704 onInit = null; 3705 } 3706 else 3707 { 3708 div.style.display = 'none'; 3709 } 3710 3711 this.addFoldingHandler(elt, div, onInit); 3712 3713 var outer = document.createElement('div'); 3714 outer.appendChild(div); 3715 this.container.appendChild(outer); 3716 3717 // Keeps references to the DOM nodes 3718 if (id != null) 3719 { 3720 this.palettes[id] = [elt, outer]; 3721 } 3722 3723 return div; 3724}; 3725 3726/** 3727 * Create the given title element. 3728 */ 3729Sidebar.prototype.addFoldingHandler = function(title, content, funct) 3730{ 3731 var initialized = false; 3732 3733 // Avoids mixed content warning in IE6-8 3734 if (!mxClient.IS_IE || document.documentMode >= 8) 3735 { 3736 title.style.backgroundImage = (content.style.display == 'none') ? 3737 'url(\'' + this.collapsedImage + '\')' : 'url(\'' + this.expandedImage + '\')'; 3738 } 3739 3740 title.style.backgroundRepeat = 'no-repeat'; 3741 title.style.backgroundPosition = '0% 50%'; 3742 3743 mxEvent.addListener(title, 'click', mxUtils.bind(this, function(evt) 3744 { 3745 if (content.style.display == 'none') 3746 { 3747 if (!initialized) 3748 { 3749 initialized = true; 3750 3751 if (funct != null) 3752 { 3753 // Wait cursor does not show up on Mac 3754 title.style.cursor = 'wait'; 3755 var prev = title.innerHTML; 3756 title.innerHTML = mxResources.get('loading') + '...'; 3757 3758 window.setTimeout(function() 3759 { 3760 content.style.display = 'block'; 3761 title.style.cursor = ''; 3762 title.innerHTML = prev; 3763 3764 var fo = mxClient.NO_FO; 3765 mxClient.NO_FO = Editor.prototype.originalNoForeignObject; 3766 funct(content, title); 3767 mxClient.NO_FO = fo; 3768 }, (mxClient.IS_FF) ? 20 : 0); 3769 } 3770 else 3771 { 3772 content.style.display = 'block'; 3773 } 3774 } 3775 else 3776 { 3777 content.style.display = 'block'; 3778 } 3779 3780 title.style.backgroundImage = 'url(\'' + this.expandedImage + '\')'; 3781 } 3782 else 3783 { 3784 title.style.backgroundImage = 'url(\'' + this.collapsedImage + '\')'; 3785 content.style.display = 'none'; 3786 } 3787 3788 mxEvent.consume(evt); 3789 })); 3790 3791 // Prevents focus 3792 mxEvent.addListener(title, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', 3793 mxUtils.bind(this, function(evt) 3794 { 3795 evt.preventDefault(); 3796 })); 3797}; 3798 3799/** 3800 * Removes the palette for the given ID. 3801 */ 3802Sidebar.prototype.removePalette = function(id) 3803{ 3804 var elts = this.palettes[id]; 3805 3806 if (elts != null) 3807 { 3808 this.palettes[id] = null; 3809 3810 for (var i = 0; i < elts.length; i++) 3811 { 3812 this.container.removeChild(elts[i]); 3813 } 3814 3815 return true; 3816 } 3817 3818 return false; 3819}; 3820 3821/** 3822 * Adds the given image palette. 3823 */ 3824Sidebar.prototype.addImagePalette = function(id, title, prefix, postfix, items, titles, tags) 3825{ 3826 var showTitles = titles != null; 3827 var fns = []; 3828 3829 for (var i = 0; i < items.length; i++) 3830 { 3831 (mxUtils.bind(this, function(item, title, tmpTags) 3832 { 3833 if (tmpTags == null) 3834 { 3835 var slash = item.lastIndexOf('/'); 3836 var dot = item.lastIndexOf('.'); 3837 tmpTags = item.substring((slash >= 0) ? slash + 1 : 0, (dot >= 0) ? dot : item.length).replace(/[-_]/g, ' '); 3838 } 3839 3840 fns.push(this.createVertexTemplateEntry('image;html=1;image=' + prefix + item + postfix, 3841 this.defaultImageWidth, this.defaultImageHeight, '', title, title != null, null, this.filterTags(tmpTags))); 3842 }))(items[i], (titles != null) ? titles[i] : null, (tags != null) ? tags[items[i]] : null); 3843 } 3844 3845 this.addPaletteFunctions(id, title, false, fns); 3846}; 3847 3848/** 3849 * Creates the array of tags for the given stencil. Duplicates are allowed and will be filtered out later. 3850 */ 3851Sidebar.prototype.getTagsForStencil = function(packageName, stencilName, moreTags) 3852{ 3853 var tags = packageName.split('.'); 3854 3855 for (var i = 1; i < tags.length; i++) 3856 { 3857 tags[i] = tags[i].replace(/_/g, ' ') 3858 } 3859 3860 tags.push(stencilName.replace(/_/g, ' ')); 3861 3862 if (moreTags != null) 3863 { 3864 tags.push(moreTags); 3865 } 3866 3867 return tags.slice(1, tags.length); 3868}; 3869 3870/** 3871 * Adds the given stencil palette. 3872 */ 3873Sidebar.prototype.addStencilPalette = function(id, title, stencilFile, style, ignore, onInit, scale, tags, customFns, groupId) 3874{ 3875 scale = (scale != null) ? scale : 1; 3876 3877 if (this.addStencilsToIndex) 3878 { 3879 // LATER: Handle asynchronous loading dependency 3880 var fns = []; 3881 3882 if (customFns != null) 3883 { 3884 for (var i = 0; i < customFns.length; i++) 3885 { 3886 fns.push(customFns[i]); 3887 } 3888 } 3889 3890 mxStencilRegistry.loadStencilSet(stencilFile, mxUtils.bind(this, function(packageName, stencilName, displayName, w, h) 3891 { 3892 if (ignore == null || mxUtils.indexOf(ignore, stencilName) < 0) 3893 { 3894 var tmp = this.getTagsForStencil(packageName, stencilName); 3895 var tmpTags = (tags != null) ? tags[stencilName] : null; 3896 3897 if (tmpTags != null) 3898 { 3899 tmp.push(tmpTags); 3900 } 3901 3902 fns.push(this.createVertexTemplateEntry('shape=' + packageName + stencilName.toLowerCase() + style, 3903 Math.round(w * scale), Math.round(h * scale), '', stencilName.replace(/_/g, ' '), null, null, 3904 this.filterTags(tmp.join(' ')))); 3905 } 3906 }), true, true); 3907 3908 this.addPaletteFunctions(id, title, false, fns); 3909 } 3910 else 3911 { 3912 this.addPalette(id, title, false, mxUtils.bind(this, function(content) 3913 { 3914 if (style == null) 3915 { 3916 style = ''; 3917 } 3918 3919 if (onInit != null) 3920 { 3921 onInit.call(this, content); 3922 } 3923 3924 if (customFns != null) 3925 { 3926 for (var i = 0; i < customFns.length; i++) 3927 { 3928 customFns[i](content); 3929 } 3930 } 3931 3932 mxStencilRegistry.loadStencilSet(stencilFile, mxUtils.bind(this, function(packageName, stencilName, displayName, w, h) 3933 { 3934 if (ignore == null || mxUtils.indexOf(ignore, stencilName) < 0) 3935 { 3936 content.appendChild(this.createVertexTemplate('shape=' + packageName + stencilName.toLowerCase() + style, 3937 Math.round(w * scale), Math.round(h * scale), '', stencilName.replace(/_/g, ' '), true)); 3938 } 3939 }), true); 3940 })); 3941 } 3942}; 3943 3944/** 3945 * Adds the given stencil palette. 3946 */ 3947Sidebar.prototype.destroy = function() 3948{ 3949 if (this.graph != null) 3950 { 3951 if (this.graph.container != null && this.graph.container.parentNode != null) 3952 { 3953 this.graph.container.parentNode.removeChild(this.graph.container); 3954 } 3955 3956 this.graph.destroy(); 3957 this.graph = null; 3958 } 3959 3960 if (this.pointerUpHandler != null) 3961 { 3962 mxEvent.removeListener(document, (mxClient.IS_POINTER) ? 'pointerup' : 'mouseup', this.pointerUpHandler); 3963 this.pointerUpHandler = null; 3964 } 3965 3966 if (this.pointerDownHandler != null) 3967 { 3968 mxEvent.removeListener(document, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown', this.pointerDownHandler); 3969 this.pointerDownHandler = null; 3970 } 3971 3972 if (this.pointerMoveHandler != null) 3973 { 3974 mxEvent.removeListener(document, (mxClient.IS_POINTER) ? 'pointermove' : 'mousemove', this.pointerMoveHandler); 3975 this.pointerMoveHandler = null; 3976 } 3977 3978 if (this.pointerOutHandler != null) 3979 { 3980 mxEvent.removeListener(document, (mxClient.IS_POINTER) ? 'pointerout' : 'mouseout', this.pointerOutHandler); 3981 this.pointerOutHandler = null; 3982 } 3983}; 3984