2016-09-06 14:07:11 +00:00
/ * *
* Copyright ( c ) 2006 - 2012 , JGraph Ltd
* /
/ * *
* Construcs a new sidebar for the given editor .
* /
function Sidebar ( editorUi , container )
{
this . editorUi = editorUi ;
this . container = container ;
this . palettes = new Object ( ) ;
this . taglist = new Object ( ) ;
this . showTooltips = true ;
2016-10-04 20:51:29 +00:00
this . graph = editorUi . createTemporaryGraph ( this . editorUi . editor . graph . getStylesheet ( ) ) ;
2016-09-06 14:07:11 +00:00
this . graph . cellRenderer . antiAlias = false ;
this . graph . foldingEnabled = false ;
2016-10-04 20:51:29 +00:00
2016-09-06 14:07:11 +00:00
// Workaround for blank output in IE11-
if ( ! mxClient . IS _IE && ! mxClient . IS _IE11 )
{
this . graph . container . style . display = 'none' ;
}
document . body . appendChild ( this . graph . container ) ;
this . pointerUpHandler = mxUtils . bind ( this , function ( )
{
this . showTooltips = true ;
} ) ;
mxEvent . addListener ( document , ( mxClient . IS _POINTER ) ? 'pointerup' : 'mouseup' , this . pointerUpHandler ) ;
this . pointerDownHandler = mxUtils . bind ( this , function ( )
{
this . showTooltips = false ;
this . hideTooltip ( ) ;
} ) ;
mxEvent . addListener ( document , ( mxClient . IS _POINTER ) ? 'pointerdown' : 'mousedown' , this . pointerDownHandler ) ;
this . pointerMoveHandler = mxUtils . bind ( this , function ( evt )
{
var src = mxEvent . getSource ( evt ) ;
while ( src != null )
{
if ( src == this . currentElt )
{
return ;
}
src = src . parentNode ;
}
this . hideTooltip ( ) ;
} ) ;
mxEvent . addListener ( document , ( mxClient . IS _POINTER ) ? 'pointermove' : 'mousemove' , this . pointerMoveHandler ) ;
// Handles mouse leaving the window
this . pointerOutHandler = mxUtils . bind ( this , function ( evt )
{
if ( evt . toElement == null && evt . relatedTarget == null )
{
this . hideTooltip ( ) ;
}
} ) ;
mxEvent . addListener ( document , ( mxClient . IS _POINTER ) ? 'pointerout' : 'mouseout' , this . pointerOutHandler ) ;
// Enables tooltips after scroll
mxEvent . addListener ( container , 'scroll' , mxUtils . bind ( this , function ( )
{
this . showTooltips = true ;
} ) ) ;
this . init ( ) ;
// Pre-fetches tooltip image
if ( ! mxClient . IS _SVG )
{
new Image ( ) . src = IMAGE _PATH + '/tooltip.png' ;
}
} ;
/ * *
* Adds all palettes to the sidebar .
* /
Sidebar . prototype . init = function ( )
{
var dir = STENCIL _PATH ;
this . addSearchPalette ( true ) ;
this . addGeneralPalette ( true ) ;
this . addMiscPalette ( false ) ;
this . addAdvancedPalette ( false ) ;
2016-10-27 15:53:53 +00:00
this . addBasicPalette ( dir ) ;
2016-09-06 14:07:11 +00:00
this . addStencilPalette ( 'arrows' , mxResources . get ( 'arrows' ) , dir + '/arrows.xml' ,
';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2' ) ;
this . addUmlPalette ( false ) ;
this . addBpmnPalette ( dir , false ) ;
this . addStencilPalette ( 'flowchart' , 'Flowchart' , dir + '/flowchart.xml' ,
';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2' ) ;
this . addImagePalette ( 'clipart' , mxResources . get ( 'clipart' ) , dir + '/clipart/' , '_128x128.png' ,
[ 'Earth_globe' , 'Empty_Folder' , 'Full_Folder' , 'Gear' , 'Lock' , 'Software' , 'Virus' , 'Email' ,
'Database' , 'Router_Icon' , 'iPad' , 'iMac' , 'Laptop' , 'MacBook' , 'Monitor_Tower' , 'Printer' ,
'Server_Tower' , 'Workstation' , 'Firewall_02' , 'Wireless_Router_N' , 'Credit_Card' ,
'Piggy_Bank' , 'Graph' , 'Safe' , 'Shopping_Cart' , 'Suit1' , 'Suit2' , 'Suit3' , 'Pilot1' ,
'Worker1' , 'Soldier1' , 'Doctor1' , 'Tech1' , 'Security1' , 'Telesales1' ] , null ,
{ 'Wireless_Router_N' : 'wireless router switch wap wifi access point wlan' ,
'Router_Icon' : 'router switch' } ) ;
} ;
/ * *
* Sets the default font size .
* /
Sidebar . prototype . collapsedImage = ( ! mxClient . IS _SVG ) ? IMAGE _PATH + '/collapsed.gif' : '' ;
/ * *
* Sets the default font size .
* /
Sidebar . prototype . expandedImage = ( ! mxClient . IS _SVG ) ? IMAGE _PATH + '/expanded.gif' : '' ;
/ * *
* Sets the default font size .
* /
Sidebar . prototype . tooltipImage = ( ! mxClient . IS _SVG ) ? IMAGE _PATH + '/tooltip.png' : '' ;
/ * *
*
* /
Sidebar . prototype . searchImage = ( ! mxClient . IS _SVG ) ? IMAGE _PATH + '/search.png' : '' ;
/ * *
* Specifies if tooltips should be visible . Default is true .
* /
Sidebar . prototype . enableTooltips = true ;
/ * *
* Specifies the delay for the tooltip . Default is 16 px .
* /
Sidebar . prototype . tooltipBorder = 16 ;
/ * *
* Specifies the delay for the tooltip . Default is 300 ms .
* /
Sidebar . prototype . tooltipDelay = 300 ;
/ * *
* Specifies the delay for the drop target icons . Default is 200 ms .
* /
Sidebar . prototype . dropTargetDelay = 200 ;
/ * *
* Specifies the URL of the gear image .
* /
Sidebar . prototype . gearImage = STENCIL _PATH + '/clipart/Gear_128x128.png' ;
/ * *
* Specifies the width of the thumbnails .
* /
Sidebar . prototype . thumbWidth = 36 ;
/ * *
* Specifies the height of the thumbnails .
* /
Sidebar . prototype . thumbHeight = 36 ;
/ * *
* Specifies the padding for the thumbnails . Default is 3.
* /
Sidebar . prototype . thumbPadding = ( document . documentMode >= 5 ) ? 0 : 1 ;
/ * *
* Specifies the delay for the tooltip . Default is 2 px .
* /
Sidebar . prototype . thumbBorder = 2 ;
/ * *
* Specifies the size of the sidebar titles .
* /
Sidebar . prototype . sidebarTitleSize = 9 ;
/ * *
* Specifies if titles in the sidebar should be enabled .
* /
Sidebar . prototype . sidebarTitles = false ;
/ * *
* Specifies if titles in the tooltips should be enabled .
* /
Sidebar . prototype . tooltipTitles = true ;
/ * *
* Specifies if titles in the tooltips should be enabled .
* /
Sidebar . prototype . maxTooltipWidth = 400 ;
/ * *
* Specifies if titles in the tooltips should be enabled .
* /
Sidebar . prototype . maxTooltipHeight = 400 ;
/ * *
* Specifies if stencil files should be loaded and added to the search index
* when stencil palettes are added . If this is false then the stencil files
* are lazy - loaded when the palette is shown .
* /
Sidebar . prototype . addStencilsToIndex = true ;
/ * *
* Specifies the width for clipart images . Default is 80.
* /
Sidebar . prototype . defaultImageWidth = 80 ;
/ * *
* Specifies the height for clipart images . Default is 80.
* /
Sidebar . prototype . defaultImageHeight = 80 ;
/ * *
* Adds all palettes to the sidebar .
* /
Sidebar . prototype . showTooltip = function ( elt , cells , w , h , title , showLabel )
{
if ( this . enableTooltips && this . showTooltips )
{
if ( this . currentElt != elt )
{
if ( this . thread != null )
{
window . clearTimeout ( this . thread ) ;
this . thread = null ;
}
var show = mxUtils . bind ( this , function ( )
{
// Lazy creation of the DOM nodes and graph instance
if ( this . tooltip == null )
{
this . tooltip = document . createElement ( 'div' ) ;
this . tooltip . className = 'geSidebarTooltip' ;
this . tooltip . style . zIndex = mxPopupMenu . prototype . zIndex - 1 ;
document . body . appendChild ( this . tooltip ) ;
this . graph2 = new Graph ( this . tooltip , null , null , this . editorUi . editor . graph . getStylesheet ( ) ) ;
this . graph2 . resetViewOnRootChange = false ;
this . graph2 . foldingEnabled = false ;
this . graph2 . gridEnabled = false ;
this . graph2 . autoScroll = false ;
this . graph2 . setTooltips ( false ) ;
this . graph2 . setConnectable ( false ) ;
this . graph2 . setEnabled ( false ) ;
if ( ! mxClient . IS _SVG )
{
this . graph2 . view . canvas . style . position = 'relative' ;
}
this . tooltipImage = mxUtils . createImage ( this . tooltipImage ) ;
this . tooltipImage . className = 'geSidebarTooltipImage' ;
this . tooltipImage . style . zIndex = mxPopupMenu . prototype . zIndex - 1 ;
this . tooltipImage . style . position = 'absolute' ;
this . tooltipImage . style . width = '14px' ;
this . tooltipImage . style . height = '27px' ;
document . body . appendChild ( this . tooltipImage ) ;
}
this . graph2 . model . clear ( ) ;
this . graph2 . view . setTranslate ( this . tooltipBorder , this . tooltipBorder ) ;
if ( w > this . maxTooltipWidth || h > this . maxTooltipHeight )
{
this . graph2 . view . scale = Math . round ( Math . min ( this . maxTooltipWidth / w , this . maxTooltipHeight / h ) * 100 ) / 100 ;
}
else
{
this . graph2 . view . scale = 1 ;
}
this . tooltip . style . display = 'block' ;
this . graph2 . labelsVisible = ( showLabel == null || showLabel ) ;
this . graph2 . addCells ( cells ) ;
var bounds = this . graph2 . getGraphBounds ( ) ;
var width = bounds . width + 2 * this . tooltipBorder + 4 ;
var height = bounds . height + 2 * this . tooltipBorder ;
if ( mxClient . IS _QUIRKS )
{
height += 4 ;
this . tooltip . style . overflow = 'hidden' ;
}
else
{
this . tooltip . style . overflow = 'visible' ;
}
this . tooltipImage . style . visibility = 'visible' ;
this . tooltip . style . width = width + 'px' ;
// Adds title for entry
if ( this . tooltipTitles && title != null && title . length > 0 )
{
if ( this . tooltipTitle == null )
{
this . tooltipTitle = document . createElement ( 'div' ) ;
this . tooltipTitle . style . borderTop = '1px solid gray' ;
this . tooltipTitle . style . textAlign = 'center' ;
this . tooltipTitle . style . width = '100%' ;
// Oversize titles are cut-off currently. Should make tooltip wider later.
this . tooltipTitle . style . overflow = 'hidden' ;
if ( mxClient . IS _SVG )
{
this . tooltipTitle . style . paddingTop = '6px' ;
}
else
{
this . tooltipTitle . style . position = 'absolute' ;
this . tooltipTitle . style . paddingTop = '6px' ;
}
this . tooltip . appendChild ( this . tooltipTitle ) ;
}
else
{
this . tooltipTitle . innerHTML = '' ;
}
this . tooltipTitle . style . display = '' ;
mxUtils . write ( this . tooltipTitle , title ) ;
var ddy = this . tooltipTitle . offsetHeight + 10 ;
height += ddy ;
if ( mxClient . IS _SVG )
{
this . tooltipTitle . style . marginTop = ( 2 - ddy ) + 'px' ;
}
else
{
height -= 6 ;
this . tooltipTitle . style . top = ( height - ddy ) + 'px' ;
}
}
else if ( this . tooltipTitle != null && this . tooltipTitle . parentNode != null )
{
this . tooltipTitle . style . display = 'none' ;
}
this . tooltip . style . height = height + 'px' ;
var x0 = - Math . round ( bounds . x - this . tooltipBorder ) ;
var y0 = - Math . round ( bounds . y - this . tooltipBorder ) ;
var b = document . body ;
var d = document . documentElement ;
2016-10-27 15:53:53 +00:00
var bottom = Math . max ( b . clientHeight || 0 , d . clientHeight ) ;
2016-09-06 14:07:11 +00:00
var left = this . container . clientWidth + this . editorUi . splitSize + 3 + this . editorUi . container . offsetLeft ;
var top = Math . min ( bottom - height - 20 /*status bar*/ , Math . max ( 0 , ( this . editorUi . container . offsetTop +
this . container . offsetTop + elt . offsetTop - this . container . scrollTop - height / 2 + 16 ) ) ) ;
if ( mxClient . IS _SVG )
{
if ( x0 != 0 || y0 != 0 )
{
this . graph2 . view . canvas . setAttribute ( 'transform' , 'translate(' + x0 + ',' + y0 + ')' ) ;
}
else
{
this . graph2 . view . canvas . removeAttribute ( 'transform' ) ;
}
}
else
{
this . graph2 . view . drawPane . style . left = x0 + 'px' ;
this . graph2 . view . drawPane . style . top = y0 + 'px' ;
}
2016-10-27 15:53:53 +00:00
2016-09-06 14:07:11 +00:00
// Workaround for ignored position CSS style in IE9
// (changes to relative without the following line)
this . tooltip . style . position = 'absolute' ;
this . tooltip . style . left = left + 'px' ;
this . tooltip . style . top = top + 'px' ;
this . tooltipImage . style . left = ( left - 13 ) + 'px' ;
this . tooltipImage . style . top = ( top + height / 2 - 13 ) + 'px' ;
} ) ;
if ( this . tooltip != null && this . tooltip . style . display != 'none' )
{
show ( ) ;
}
else
{
this . thread = window . setTimeout ( show , this . tooltipDelay ) ;
}
this . currentElt = elt ;
}
}
} ;
/ * *
* Hides the current tooltip .
* /
Sidebar . prototype . hideTooltip = function ( )
{
if ( this . thread != null )
{
window . clearTimeout ( this . thread ) ;
this . thread = null ;
}
if ( this . tooltip != null )
{
this . tooltip . style . display = 'none' ;
this . tooltipImage . style . visibility = 'hidden' ;
this . currentElt = null ;
}
} ;
/ * *
* Hides the current tooltip .
* /
Sidebar . prototype . addEntry = function ( tags , fn )
{
if ( this . taglist != null && tags != null && tags . length > 0 )
{
// Replaces special characters
var tmp = tags . toLowerCase ( ) . replace ( /[\/\,\(\)]/g , ' ' ) . split ( ' ' ) ;
for ( var i = 0 ; i < tmp . length ; i ++ )
{
// Replaces trailing numbers and special characters
tmp [ i ] = tmp [ i ] . replace ( /\.*\d*$/ , '' ) ;
if ( tmp [ i ] . length > 1 )
{
var entry = this . taglist [ tmp [ i ] ] ;
if ( entry == null )
{
entry = { entries : [ ] , dict : new mxDictionary ( ) } ;
this . taglist [ tmp [ i ] ] = entry ;
}
// Ignores duplicates
if ( entry . dict . get ( fn ) == null )
{
entry . dict . put ( fn , fn ) ;
entry . entries . push ( fn ) ;
}
}
}
}
return fn ;
} ;
/ * *
* Adds shape search UI .
* /
Sidebar . prototype . searchEntries = function ( searchTerms , count , page , success , error )
{
if ( this . taglist != null && searchTerms != null )
{
var tmp = searchTerms . toLowerCase ( ) . split ( ' ' ) ;
var dict = new mxDictionary ( ) ;
var max = ( page + 1 ) * count ;
var results = [ ] ;
var index = 0 ;
for ( var i = 0 ; i < tmp . length ; i ++ )
{
if ( tmp [ i ] . length > 0 )
{
var entry = this . taglist [ tmp [ i ] ] ;
var tmpDict = new mxDictionary ( ) ;
if ( entry != null )
{
var arr = entry . entries ;
results = [ ] ;
for ( var j = 0 ; j < arr . length ; j ++ )
{
var entry = arr [ j ] ;
// NOTE Array does not contain duplicates
if ( ( index == 0 ) == ( dict . get ( entry ) == null ) )
{
tmpDict . put ( entry , entry ) ;
results . push ( entry ) ;
if ( i == tmp . length - 1 && results . length == max )
{
2016-09-20 12:35:42 +00:00
success ( results . slice ( page * count , max ) , max , true , tmp ) ;
2016-09-06 14:07:11 +00:00
return ;
}
}
}
}
else
{
results = [ ] ;
}
dict = tmpDict ;
index ++ ;
}
}
var len = results . length ;
2016-09-20 12:35:42 +00:00
success ( results . slice ( page * count , ( page + 1 ) * count ) , len , false , tmp ) ;
2016-09-06 14:07:11 +00:00
}
else
{
2016-09-20 12:35:42 +00:00
success ( [ ] , null , null , tmp ) ;
2016-09-06 14:07:11 +00:00
}
} ;
/ * *
* Adds shape search UI .
* /
Sidebar . prototype . filterTags = function ( tags )
{
if ( tags != null )
{
var arr = tags . split ( ' ' ) ;
var result = [ ] ;
var hash = { } ;
// Ignores tags with leading numbers, strips trailing numbers
for ( var i = 0 ; i < arr . length ; i ++ )
{
// Removes duplicates
if ( hash [ arr [ i ] ] == null )
{
hash [ arr [ i ] ] = '1' ;
result . push ( arr [ i ] ) ;
}
}
return result . join ( ' ' ) ;
}
return null ;
} ;
/ * *
* Adds the general palette to the sidebar .
* /
Sidebar . prototype . cloneCell = function ( cell , value )
{
var clone = cell . clone ( ) ;
if ( value != null )
{
clone . value = value ;
}
return clone ;
} ;
/ * *
* Adds shape search UI .
* /
Sidebar . prototype . addSearchPalette = function ( expand )
{
var elt = document . createElement ( 'div' ) ;
elt . style . visibility = 'hidden' ;
this . container . appendChild ( elt ) ;
var div = document . createElement ( 'div' ) ;
div . className = 'geSidebar' ;
div . style . boxSizing = 'border-box' ;
div . style . overflow = 'hidden' ;
div . style . width = '100%' ;
div . style . padding = '8px' ;
div . style . paddingTop = '14px' ;
div . style . paddingBottom = '0px' ;
if ( ! expand )
{
div . style . display = 'none' ;
}
var inner = document . createElement ( 'div' ) ;
inner . style . whiteSpace = 'nowrap' ;
inner . style . textOverflow = 'clip' ;
inner . style . paddingBottom = '8px' ;
inner . style . cursor = 'default' ;
var input = document . createElement ( 'input' ) ;
input . setAttribute ( 'placeholder' , mxResources . get ( 'searchShapes' ) ) ;
input . setAttribute ( 'type' , 'text' ) ;
input . style . fontSize = '12px' ;
input . style . overflow = 'hidden' ;
input . style . boxSizing = 'border-box' ;
input . style . border = 'solid 1px #d5d5d5' ;
input . style . borderRadius = '4px' ;
input . style . width = '100%' ;
input . style . outline = 'none' ;
input . style . padding = '6px' ;
inner . appendChild ( input ) ;
var cross = document . createElement ( 'img' ) ;
cross . setAttribute ( 'src' , Sidebar . prototype . searchImage ) ;
cross . setAttribute ( 'title' , mxResources . get ( 'search' ) ) ;
cross . style . position = 'relative' ;
cross . style . left = '-18px' ;
if ( mxClient . IS _QUIRKS )
{
input . style . height = '28px' ;
cross . style . top = '-4px' ;
}
else
{
2016-11-04 22:16:17 +00:00
cross . style . top = '1px' ;
2016-09-06 14:07:11 +00:00
}
// Needed to block event transparency in IE
cross . style . background = 'url(\'' + this . editorUi . editor . transparentImage + '\')' ;
var find ;
inner . appendChild ( cross ) ;
div . appendChild ( inner ) ;
var center = document . createElement ( 'center' ) ;
var button = mxUtils . button ( mxResources . get ( 'moreResults' ) , function ( )
{
find ( ) ;
} ) ;
button . style . display = 'none' ;
// Workaround for inherited line-height in quirks mode
button . style . lineHeight = 'normal' ;
button . style . marginTop = '4px' ;
button . style . marginBottom = '8px' ;
center . style . paddingTop = '4px' ;
center . style . paddingBottom = '8px' ;
center . appendChild ( button ) ;
div . appendChild ( center ) ;
var searchTerm = '' ;
var active = false ;
var complete = false ;
var page = 0 ;
var hash = new Object ( ) ;
// Count is dynamically updated below
var count = 12 ;
var clearDiv = mxUtils . bind ( this , function ( )
{
active = false ;
this . currentSearch = null ;
var child = div . firstChild ;
while ( child != null )
{
var next = child . nextSibling ;
if ( child != inner && child != center )
{
child . parentNode . removeChild ( child ) ;
}
child = next ;
}
} ) ;
2016-11-04 22:16:17 +00:00
mxEvent . addListener ( cross , 'click' , function ( )
{
if ( cross . getAttribute ( 'src' ) == Dialog . prototype . closeImage )
{
cross . setAttribute ( 'src' , Sidebar . prototype . searchImage ) ;
cross . setAttribute ( 'title' , mxResources . get ( 'search' ) ) ;
button . style . display = 'none' ;
input . value = '' ;
searchTerm = '' ;
clearDiv ( ) ;
}
input . focus ( ) ;
} ) ;
2016-09-06 14:07:11 +00:00
find = mxUtils . bind ( this , function ( )
{
// Shows 4 rows (minimum 4 results)
count = 4 * Math . max ( 1 , Math . floor ( this . container . clientWidth / ( this . thumbWidth + 10 ) ) ) ;
this . hideTooltip ( ) ;
if ( input . value != '' )
{
if ( center . parentNode != null )
{
if ( searchTerm != input . value )
{
clearDiv ( ) ;
searchTerm = input . value ;
hash = new Object ( ) ;
complete = false ;
page = 0 ;
}
if ( ! active && ! complete )
{
button . setAttribute ( 'disabled' , 'true' ) ;
button . style . display = '' ;
button . style . cursor = 'wait' ;
button . innerHTML = mxResources . get ( 'loading' ) + '...' ;
active = true ;
// Ignores old results
var current = new Object ( ) ;
this . currentSearch = current ;
2016-09-20 12:35:42 +00:00
this . searchEntries ( searchTerm , count , page , mxUtils . bind ( this , function ( results , len , more , terms )
2016-09-06 14:07:11 +00:00
{
if ( this . currentSearch == current )
{
results = ( results != null ) ? results : [ ] ;
active = false ;
page ++ ;
center . parentNode . removeChild ( center ) ;
2016-09-20 12:35:42 +00:00
this . insertSearchHint ( div , searchTerm , count , page , results , len , more , terms ) ;
2016-09-06 14:07:11 +00:00
for ( var i = 0 ; i < results . length ; i ++ )
{
var elt = results [ i ] ( ) ;
// Avoids duplicates in results
if ( hash [ elt . innerHTML ] == null )
{
hash [ elt . innerHTML ] = '1' ;
div . appendChild ( results [ i ] ( ) ) ;
}
}
if ( more )
{
button . removeAttribute ( 'disabled' ) ;
button . innerHTML = mxResources . get ( 'moreResults' ) ;
}
else
{
button . innerHTML = mxResources . get ( 'reset' ) ;
button . style . display = 'none' ;
complete = true ;
}
button . style . cursor = '' ;
div . appendChild ( center ) ;
}
} ) , mxUtils . bind ( this , function ( )
{
// TODO: Error handling
button . style . cursor = '' ;
} ) ) ;
}
}
}
else
{
clearDiv ( ) ;
input . value = '' ;
searchTerm = '' ;
hash = new Object ( ) ;
button . style . display = 'none' ;
complete = false ;
input . focus ( ) ;
}
} ) ;
mxEvent . addListener ( input , 'keydown' , mxUtils . bind ( this , function ( evt )
{
if ( evt . keyCode == 13 /* Enter */ )
{
find ( ) ;
}
} ) ) ;
mxEvent . addListener ( input , 'focus' , function ( )
{
input . style . paddingRight = '' ;
} ) ;
mxEvent . addListener ( input , 'blur' , function ( )
{
input . style . paddingRight = '20px' ;
} ) ;
input . style . paddingRight = '20px' ;
mxEvent . addListener ( input , 'keyup' , mxUtils . bind ( this , function ( evt )
{
2016-11-04 22:16:17 +00:00
if ( input . value == '' )
{
cross . setAttribute ( 'src' , Sidebar . prototype . searchImage ) ;
cross . setAttribute ( 'title' , mxResources . get ( 'search' ) ) ;
}
else
{
cross . setAttribute ( 'src' , Dialog . prototype . closeImage ) ;
cross . setAttribute ( 'title' , mxResources . get ( 'reset' ) ) ;
}
2016-09-06 14:07:11 +00:00
if ( input . value == '' )
{
complete = true ;
button . style . display = 'none' ;
}
else if ( input . value != searchTerm )
{
button . style . display = 'none' ;
complete = false ;
}
else if ( ! active )
{
if ( complete )
{
button . style . display = 'none' ;
}
else
{
button . style . display = '' ;
}
}
} ) ) ;
// Workaround for blocked text selection in Editor
mxEvent . addListener ( input , 'mousedown' , function ( evt )
{
if ( evt . stopPropagation )
{
evt . stopPropagation ( ) ;
}
evt . cancelBubble = true ;
} ) ;
// Workaround for blocked text selection in Editor
mxEvent . addListener ( input , 'selectstart' , function ( evt )
{
if ( evt . stopPropagation )
{
evt . stopPropagation ( ) ;
}
evt . cancelBubble = true ;
} ) ;
var outer = document . createElement ( 'div' ) ;
outer . appendChild ( div ) ;
this . container . appendChild ( outer ) ;
// Keeps references to the DOM nodes
this . palettes [ 'search' ] = [ elt , outer ] ;
} ;
2016-09-20 12:35:42 +00:00
/ * *
* Adds the general palette to the sidebar .
* /
Sidebar . prototype . insertSearchHint = function ( div , searchTerm , count , page , results , len , more , terms )
{
if ( results . length == 0 && page == 1 )
{
var err = document . createElement ( 'div' ) ;
err . className = 'geTitle' ;
err . style . cssText = 'background-color:transparent;border-color:transparent;' +
'color:gray;padding:6px 0px 0px 0px !important;margin:4px 8px 4px 8px;' +
'text-align:center;cursor:default !important' ;
mxUtils . write ( err , mxResources . get ( 'noResultsFor' , [ searchTerm ] ) ) ;
div . appendChild ( err ) ;
}
} ;
2016-09-06 14:07:11 +00:00
/ * *
* Adds the general palette to the sidebar .
* /
Sidebar . prototype . addGeneralPalette = function ( expand )
{
var fns = [
this . createVertexTemplateEntry ( 'whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Rectangle' , null , null , 'rect rectangle box' ) ,
this . createVertexTemplateEntry ( 'rounded=1;whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Rounded Rectangle' , null , null , 'rounded rect rectangle box' ) ,
2016-09-20 12:35:42 +00:00
this . createVertexTemplateEntry ( 'ellipse;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Ellipse' , null , null , 'oval ellipse state' ) ,
2016-09-06 14:07:11 +00:00
// Explicit strokecolor/fillcolor=none is a workaround to maintain transparent background regardless of current style
this . createVertexTemplateEntry ( 'text;html=1;strokeColor=none;fillColor=none;align=center;verticalAlign=middle;whiteSpace=wrap;' ,
40 , 20 , 'Text' , 'Text' , null , null , 'text textbox textarea label' ) ,
this . createVertexTemplateEntry ( 'shape=ext;double=1;whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Double Rectangle' , null , null , 'rect rectangle box double' ) ,
this . createVertexTemplateEntry ( 'shape=ext;double=1;rounded=1;whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Double Rounded Rectangle' , null , null , 'rounded rect rectangle box double' ) ,
2016-09-20 12:35:42 +00:00
this . createVertexTemplateEntry ( 'ellipse;shape=doubleEllipse;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Double Ellipse' , null , null , 'oval ellipse start end state double' ) ,
2016-09-06 14:07:11 +00:00
this . createVertexTemplateEntry ( 'rhombus;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Diamond' , null , null , 'diamond rhombus if condition decision conditional question test' ) ,
this . createVertexTemplateEntry ( 'shape=parallelogram;whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Parallelogram' ) ,
this . createVertexTemplateEntry ( 'triangle;whiteSpace=wrap;html=1;' , 60 , 80 , '' , 'Triangle' , null , null , 'triangle logic inverter buffer' ) ,
this . createVertexTemplateEntry ( 'shape=cylinder;whiteSpace=wrap;html=1;' , 60 , 80 , '' , 'Cylinder' , null , null , 'cylinder data database' ) ,
this . createVertexTemplateEntry ( 'shape=hexagon;perimeter=hexagonPerimeter;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Hexagon' , null , null , 'hexagon preparation' ) ,
this . createVertexTemplateEntry ( 'shape=process;whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Process' , null , null , 'process task' ) ,
this . createVertexTemplateEntry ( 'ellipse;shape=cloud;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Cloud' , null , null , 'cloud network' ) ,
this . createVertexTemplateEntry ( 'shape=document;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Document' ) ,
this . createVertexTemplateEntry ( 'shape=internalStorage;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Internal Storage' ) ,
this . createVertexTemplateEntry ( 'shape=cube;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Cube' ) ,
this . createVertexTemplateEntry ( 'shape=step;whiteSpace=wrap;html=1;' , 120 , 80 , '' , 'Step' ) ,
this . createVertexTemplateEntry ( 'shape=trapezoid;whiteSpace=wrap;html=1;' , 120 , 60 , '' , 'Trapezoid' ) ,
this . createVertexTemplateEntry ( 'shape=tape;whiteSpace=wrap;html=1;' , 120 , 100 , '' , 'Tape' ) ,
this . createVertexTemplateEntry ( 'shape=note;whiteSpace=wrap;html=1;' , 80 , 100 , '' , 'Note' ) ,
this . createVertexTemplateEntry ( 'shape=card;whiteSpace=wrap;html=1;' , 80 , 100 , '' , 'Card' ) ,
this . createEdgeTemplateEntry ( 'endArrow=classic;html=1;' , 50 , 50 , '' , 'Connection' ) ,
this . createEdgeTemplateEntry ( 'endArrow=classic;startArrow=classic;html=1;' , 50 , 50 , '' , 'Connection' )
2016-09-20 12:35:42 +00:00
] ;
2016-09-06 14:07:11 +00:00
this . addPaletteFunctions ( 'general' , mxResources . get ( 'general' ) , ( expand != null ) ? expand : true , fns ) ;
} ;
2016-10-27 15:53:53 +00:00
/ * *
* Adds the general palette to the sidebar .
* /
Sidebar . prototype . addBasicPalette = function ( dir )
{
this . addStencilPalette ( 'basic' , mxResources . get ( 'basic' ) , dir + '/basic.xml' ,
';whiteSpace=wrap;html=1;fillColor=#ffffff;strokeColor=#000000;strokeWidth=2' ,
null , null , null , null , [
this . createVertexTemplateEntry ( 'whiteSpace=wrap;html=1;aspect=fixed;' , 80 , 80 , '' , 'Square' , null , null , 'square' ) ,
this . createVertexTemplateEntry ( 'ellipse;whiteSpace=wrap;html=1;aspect=fixed;' , 80 , 80 , '' , 'Circle' , null , null , 'circle' ) ,
this . createVertexTemplateEntry ( 'shape=ext;double=1;whiteSpace=wrap;html=1;aspect=fixed;' , 80 , 80 , '' , 'Double Square' , null , null , 'double square' ) ,
this . createVertexTemplateEntry ( 'ellipse;shape=doubleEllipse;whiteSpace=wrap;html=1;aspect=fixed;' , 80 , 80 , '' , 'Double Circle' , null , null , 'double circle' )
] ) ;
} ;
2016-09-06 14:07:11 +00:00
/ * *
* Adds the general palette to the sidebar .
* /
Sidebar . prototype . addMiscPalette = function ( expand )
{
var fns = [
this . createVertexTemplateEntry ( 'text;html=1;fontSize=24;fontStyle=1;verticalAlign=middle;align=center;' , 100 , 40 , 'Title' , 'Title' , null , null , 'text heading title' ) ,
this . createVertexTemplateEntry ( 'text;html=1;spacing=5;spacingTop=-20;whiteSpace=wrap;overflow=hidden;' , 190 , 120 ,
'<h1>Heading</h1><p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.</p>' ,
'Textbox' , null , null , 'text textbox textarea' ) ,
this . createVertexTemplateEntry ( 'text;html=1;whiteSpace=wrap;verticalAlign=middle;overflow=hidden;' , 100 , 80 ,
'<ul><li>Value 1</li><li>Value 2</li><li>Value 3</li></ul>' , 'Unordered List' ) ,
this . createVertexTemplateEntry ( 'text;html=1;whiteSpace=wrap;verticalAlign=middle;overflow=hidden;' , 100 , 80 ,
'<ol><li>Value 1</li><li>Value 2</li><li>Value 3</li></ol>' , 'Ordered List' ) ,
this . createVertexTemplateEntry ( 'text;html=1;fillColor=#ffffff;overflow=fill;rounded=0;' , 280 , 160 ,
'<table border="1" width="100%" height="100%" cellpadding="4" style="width:100%;height:100%;border-collapse:collapse;">' +
'<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>' +
'<tr style="border:1px solid #98bf21;"><td>Value 1</td><td>Value 2</td><td>Value 3</td></tr>' +
'<tr style="background-color:#EAF2D3;border:1px solid #98bf21;"><td>Value 4</td><td>Value 5</td><td>Value 6</td></tr>' +
'<tr style="border:1px solid #98bf21;"><td>Value 7</td><td>Value 8</td><td>Value 9</td></tr>' +
'<tr style="background-color:#EAF2D3;border:1px solid #98bf21;"><td>Value 10</td><td>Value 11</td><td>Value 12</td></tr></table>' , 'Table 1' ) ,
this . createVertexTemplateEntry ( 'text;html=1;strokeColor=#c0c0c0;overflow=fill;' , 180 , 140 ,
'<table border="0" width="100%" height="100%" style="width:100%;height:100%;border-collapse:collapse;">' +
'<tr><td align="center">Value 1</td><td align="center">Value 2</td><td align="center">Value 3</td></tr>' +
'<tr><td align="center">Value 4</td><td align="center">Value 5</td><td align="center">Value 6</td></tr>' +
'<tr><td align="center">Value 7</td><td align="center">Value 8</td><td align="center">Value 9</td></tr></table>' , 'Table 2' ) ,
this . createVertexTemplateEntry ( 'text;html=1;overflow=fill;' , 180 , 140 ,
'<table border="1" width="100%" height="100%" style="width:100%;height:100%;border-collapse:collapse;">' +
'<tr><td align="center">Value 1</td><td align="center">Value 2</td><td align="center">Value 3</td></tr>' +
'<tr><td align="center">Value 4</td><td align="center">Value 5</td><td align="center">Value 6</td></tr>' +
'<tr><td align="center">Value 7</td><td align="center">Value 8</td><td align="center">Value 9</td></tr></table>' , 'Table 3' ) ,
this . createVertexTemplateEntry ( 'text;html=1;overflow=fill;' , 160 , 140 ,
'<table border="1" width="100%" height="100%" cellpadding="4" style="width:100%;height:100%;border-collapse:collapse;">' +
'<tr><th align="center"><b>Title</b></th></tr>' +
'<tr><td align="center">Section 1.1\nSection 1.2\nSection 1.3</td></tr>' +
'<tr><td align="center">Section 2.1\nSection 2.2\nSection 2.3</td></tr></table>' , 'Table 4' ) ,
this . addEntry ( 'link hyperlink' , mxUtils . bind ( this , function ( )
{
var cell = new mxCell ( 'Link' , new mxGeometry ( 0 , 0 , 60 , 40 ) , 'text;html=1;whiteSpace=wrap;align=center;verticalAlign=middle;fontColor=#0000EE;fontStyle=4;' ) ;
cell . vertex = true ;
this . graph . setLinkForCell ( cell , 'https://www.draw.io' ) ;
return this . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Link' ) ;
} ) ) ,
this . addEntry ( 'timestamp date time text label' , mxUtils . bind ( this , function ( )
{
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;' ) ;
cell . vertex = true ;
this . graph . setAttributeForCell ( cell , 'placeholders' , '1' ) ;
return this . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Timestamp' ) ;
} ) ) ,
this . addEntry ( 'variable placeholder metadata hello world text label' , mxUtils . bind ( this , function ( )
{
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;' ) ;
cell . vertex = true ;
this . graph . setAttributeForCell ( cell , 'placeholders' , '1' ) ;
this . graph . setAttributeForCell ( cell , 'name' , 'Variable' ) ;
return this . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Variable' ) ;
} ) ) ,
this . createVertexTemplateEntry ( 'shape=umlActor;verticalLabelPosition=bottom;labelBackgroundColor=#ffffff;verticalAlign=top;html=1;' , 30 , 60 , 'Actor' , 'Actor' , false , null , 'user person human stickman' ) ,
this . createVertexTemplateEntry ( 'html=1;whiteSpace=wrap;comic=1;strokeWidth=2;fontFamily=Comic Sans MS;fontStyle=1;' , 120 , 60 , 'RECTANGLE' , 'Comic Rectangle' , true , null , 'comic rectangle rect box text retro' ) ,
this . createVertexTemplateEntry ( 'rhombus;html=1;align=center;whiteSpace=wrap;comic=1;strokeWidth=2;fontFamily=Comic Sans MS;fontStyle=1;' , 100 , 100 , 'DIAMOND' , 'Comic Diamond' , true , null , 'comic diamond rhombus if condition decision conditional question test retro' ) ,
this . createEdgeTemplateEntry ( 'edgeStyle=segmentEdgeStyle;rounded=0;comic=1;strokeWidth=2;endArrow=blockThin;html=1;fontFamily=Comic Sans MS;fontStyle=1;' , 50 , 50 , '' , 'Comic Arrow 1' ) ,
this . createEdgeTemplateEntry ( 'rounded=0;comic=1;strokeWidth=2;endArrow=blockThin;html=1;fontFamily=Comic Sans MS;fontStyle=1;' , 50 , 50 , '' , 'Comic Arrow 2' ) ,
this . createVertexTemplateEntry ( 'html=1;whiteSpace=wrap;aspect=fixed;shape=isoRectangle;' , 150 , 90 , '' , 'Isometric Square' , true , null , 'rectangle rect box iso isometric' ) ,
this . createVertexTemplateEntry ( 'html=1;whiteSpace=wrap;aspect=fixed;shape=isoCube;' , 90 , 100 , '' , 'Isometric Cube' , true , null , 'cube box iso isometric' ) ,
this . createEdgeTemplateEntry ( 'edgeStyle=isometricEdgeStyle;endArrow=none;html=1;' , 50 , 100 , '' , 'Isometric Edge 1' ) ,
this . createEdgeTemplateEntry ( 'edgeStyle=isometricEdgeStyle;endArrow=none;html=1;elbow=vertical;' , 50 , 100 , '' , 'Isometric Edge 2' ) ,
2016-10-04 20:51:29 +00:00
this . createVertexTemplateEntry ( 'line;strokeWidth=2;html=1;' , 160 , 10 , '' , 'Horizontal Line' ) ,
this . createVertexTemplateEntry ( 'line;strokeWidth=2;direction=south;html=1;' , 10 , 160 , '' , 'Vertical Line' ) ,
2016-11-10 10:28:55 +00:00
this . createVertexTemplateEntry ( 'line;strokeWidth=4;html=1;perimeter=backbonePerimeter;points=[];outlineConnect=0;' , 160 , 10 , '' , 'Horizontal Backbone' , false , null , 'backbone bus network' ) ,
this . createVertexTemplateEntry ( 'line;strokeWidth=4;direction=south;html=1;perimeter=backbonePerimeter;points=[];outlineConnect=0;' , 10 , 160 , '' , 'Vertical Backbone' , false , null , 'backbone bus network' ) ,
2016-09-06 14:07:11 +00:00
this . createVertexTemplateEntry ( 'shape=curlyBracket;whiteSpace=wrap;html=1;rounded=1;' , 20 , 120 , '' , 'Curly Bracket' ) ,
this . createVertexTemplateEntry ( 'shape=image;html=1;verticalLabelPosition=bottom;labelBackgroundColor=#ffffff;verticalAlign=top;imageAspect=1;aspect=fixed;image=' + this . gearImage , 52 , 61 , '' , 'Image (Fixed Aspect)' , false , null , 'fixed image icon symbol' ) ,
this . createVertexTemplateEntry ( 'shape=image;html=1;verticalLabelPosition=bottom;labelBackgroundColor=#ffffff;verticalAlign=top;imageAspect=0;image=' + this . gearImage , 50 , 60 , '' , 'Image (Variable Aspect)' , false , null , 'strechted image icon symbol' ) ,
this . createVertexTemplateEntry ( 'icon;html=1;image=' + this . gearImage , 60 , 60 , 'Icon' , 'Icon' , false , null , 'icon image symbol' ) ,
this . createVertexTemplateEntry ( 'label;whiteSpace=wrap;html=1;image=' + this . gearImage , 140 , 60 , 'Label' , 'Label 1' , null , null , 'label image icon symbol' ) ,
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' ) ,
this . createEdgeTemplateEntry ( 'shape=flexArrow;endArrow=classic;html=1;fillColor=#ffffff;' , 50 , 50 , '' , 'Arrow' ) ,
this . createEdgeTemplateEntry ( 'shape=flexArrow;endArrow=classic;startArrow=classic;html=1;fillColor=#ffffff;' , 50 , 50 , '' , 'Arrow' ) ,
this . createEdgeTemplateEntry ( 'endArrow=none;html=1;dashed=1;dashPattern=1 4;' , 50 , 50 , '' , 'Dotted Line' ) ,
this . createEdgeTemplateEntry ( 'endArrow=none;dashed=1;html=1;' , 50 , 50 , '' , 'Dashed Line' ) ,
this . createEdgeTemplateEntry ( 'endArrow=none;html=1;' , 50 , 50 , '' , 'Line' ) ,
this . createEdgeTemplateEntry ( 'edgeStyle=segmentEdgeStyle;endArrow=classic;html=1;' , 50 , 50 , '' , 'Manual Line' ) ,
this . createEdgeTemplateEntry ( 'edgeStyle=elbowEdgeStyle;elbow=horizontal;endArrow=classic;html=1;' , 50 , 50 , '' , 'Horizontal Elbow' ) ,
this . createEdgeTemplateEntry ( 'edgeStyle=elbowEdgeStyle;elbow=vertical;endArrow=classic;html=1;' , 50 , 50 , '' , 'Vertical Elbow' ) ,
this . addEntry ( 'curve' , mxUtils . bind ( this , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 50 , 50 ) , 'curved=1;endArrow=classic;html=1;' ) ;
cell . geometry . setTerminalPoint ( new mxPoint ( 0 , 50 ) , true ) ;
cell . geometry . setTerminalPoint ( new mxPoint ( 50 , 0 ) , false ) ;
cell . geometry . points = [ new mxPoint ( 50 , 50 ) , new mxPoint ( 0 , 0 ) ] ;
cell . geometry . relative = true ;
cell . edge = true ;
return this . createEdgeTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Curve' ) ;
} ) ) ,
this . createEdgeTemplateEntry ( 'shape=link;html=1;' , 50 , 50 , '' , 'Link' )
2016-09-20 12:35:42 +00:00
] ;
2016-09-06 14:07:11 +00:00
this . addPaletteFunctions ( 'misc' , mxResources . get ( 'misc' ) , ( expand != null ) ? expand : true , fns ) ;
} ;
/ * *
* Adds the container palette to the sidebar .
* /
Sidebar . prototype . addAdvancedPalette = function ( expand )
{
this . addPaletteFunctions ( 'advanced' , mxResources . get ( 'advanced' ) , ( expand != null ) ? expand : false , this . createAdvancedShapes ( ) ) ;
} ;
/ * *
* Adds the container palette to the sidebar .
* /
Sidebar . prototype . createAdvancedShapes = function ( )
{
// Avoids having to bind all functions to "this"
var sb = this ;
// Reusable cells
var field = new mxCell ( 'List Item' , new mxGeometry ( 0 , 0 , 60 , 26 ) , 'text;html=1;strokeColor=none;fillColor=none;align=left;verticalAlign=top;spacingLeft=4;spacingRight=4;whiteSpace=wrap;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;' ) ;
field . vertex = true ;
return [
this . createVertexTemplateEntry ( 'shape=xor;whiteSpace=wrap;html=1;' , 60 , 80 , '' , 'Or' , null , null , 'logic or' ) ,
this . createVertexTemplateEntry ( 'shape=or;whiteSpace=wrap;html=1;' , 60 , 80 , '' , 'And' , null , null , 'logic and' ) ,
this . createVertexTemplateEntry ( 'shape=dataStorage;whiteSpace=wrap;html=1;' , 100 , 80 , '' , 'Data Storage' ) ,
this . createVertexTemplateEntry ( 'shape=tapeData;whiteSpace=wrap;html=1;perimeter=ellipsePerimeter;' , 80 , 80 , '' , 'Tape Data' ) ,
this . createVertexTemplateEntry ( 'shape=manualInput;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Manual Input' ) ,
this . createVertexTemplateEntry ( 'shape=loopLimit;whiteSpace=wrap;html=1;' , 100 , 80 , '' , 'Loop Limit' ) ,
this . createVertexTemplateEntry ( 'shape=offPageConnector;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Off Page Connector' ) ,
this . createVertexTemplateEntry ( 'shape=delay;whiteSpace=wrap;html=1;' , 80 , 40 , '' , 'Delay' ) ,
this . createVertexTemplateEntry ( 'shape=display;whiteSpace=wrap;html=1;' , 80 , 40 , '' , 'Display' ) ,
this . createVertexTemplateEntry ( 'shape=singleArrow;direction=west;whiteSpace=wrap;html=1;' , 100 , 60 , '' , 'Arrow Left' ) ,
this . createVertexTemplateEntry ( 'shape=singleArrow;whiteSpace=wrap;html=1;' , 100 , 60 , '' , 'Arrow Right' ) ,
this . createVertexTemplateEntry ( 'shape=singleArrow;direction=north;whiteSpace=wrap;html=1;' , 60 , 100 , '' , 'Arrow Up' ) ,
this . createVertexTemplateEntry ( 'shape=singleArrow;direction=south;whiteSpace=wrap;html=1;' , 60 , 100 , '' , 'Arrow Down' ) ,
this . createVertexTemplateEntry ( 'shape=doubleArrow;whiteSpace=wrap;html=1;' , 100 , 60 , '' , 'Double Arrow' ) ,
this . createVertexTemplateEntry ( 'shape=doubleArrow;direction=south;whiteSpace=wrap;html=1;' , 60 , 100 , '' , 'Double Arrow Vertical' , null , null , 'double arrow' ) ,
this . createVertexTemplateEntry ( 'shape=actor;whiteSpace=wrap;html=1;' , 40 , 60 , '' , 'User' , null , null , 'user person human' ) ,
this . createVertexTemplateEntry ( 'shape=cross;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Cross' ) ,
this . createVertexTemplateEntry ( 'shape=corner;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Corner' ) ,
this . createVertexTemplateEntry ( 'shape=tee;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Tee' ) ,
this . createVertexTemplateEntry ( 'shape=datastore;whiteSpace=wrap;html=1;' , 60 , 60 , '' , 'Data Store' , null , null , 'data store cylinder database' ) ,
this . createVertexTemplateEntry ( 'shape=orEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Or' , null , null , 'or circle oval ellipse' ) ,
this . createVertexTemplateEntry ( 'shape=sumEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Sum' , null , null , 'sum circle oval ellipse' ) ,
this . createVertexTemplateEntry ( 'shape=lineEllipse;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Ellipse with horizontal divider' , null , null , 'circle oval ellipse' ) ,
this . createVertexTemplateEntry ( 'shape=lineEllipse;line=vertical;perimeter=ellipsePerimeter;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Ellipse with vertical divider' , null , null , 'circle oval ellipse' ) ,
this . createVertexTemplateEntry ( 'shape=sortShape;perimeter=rhombusPerimeter;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Sort' , null , null , 'sort' ) ,
this . createVertexTemplateEntry ( 'shape=collate;whiteSpace=wrap;html=1;' , 80 , 80 , '' , 'Collate' , null , null , 'collate' ) ,
this . createVertexTemplateEntry ( 'shape=switch;whiteSpace=wrap;html=1;' , 60 , 60 , '' , 'Switch' , null , null , 'switch router' ) ,
this . createVertexTemplateEntry ( 'shape=dimension;whiteSpace=wrap;html=1;align=center;points=[];verticalAlign=bottom;spacingBottom=-5;labelBackgroundColor=#ffffff' , 100 , 40 , 'Label' , 'Horizontal Dimension' , null , null , 'horizontal dimension measure' ) ,
this . createVertexTemplateEntry ( 'shape=dimension;direction=north;whiteSpace=wrap;html=1;align=right;points=[];verticalAlign=middle;labelBackgroundColor=#ffffff' , 40 , 100 , 'Label' , 'Vertical Dimension' , null , null , 'vertical dimension measure' ) ,
this . createVertexTemplateEntry ( 'swimlane;whiteSpace=wrap;html=1;' , 200 , 200 , 'Container' , 'Container' , null , null , 'container swimlane lane pool' ) ,
this . addEntry ( 'list' , function ( )
{
var cell = new mxCell ( 'List' , new mxGeometry ( 0 , 0 , 140 , 110 ) ,
'swimlane;html=1;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=none;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=#ffffff;' ) ;
cell . vertex = true ;
cell . insert ( sb . cloneCell ( field , 'Item 1' ) ) ;
cell . insert ( sb . cloneCell ( field , 'Item 2' ) ) ;
cell . insert ( sb . cloneCell ( field , 'Item 3' ) ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'List' ) ;
} ) ,
this . addEntry ( 'list item entry value' , function ( )
{
return sb . createVertexTemplateFromCells ( [ sb . cloneCell ( field , 'List Item' ) ] , field . geometry . width , field . geometry . height , 'List Item' ) ;
} )
] ;
} ;
/ * *
* Adds the general palette to the sidebar .
* /
Sidebar . prototype . addUmlPalette = function ( expand )
{
// Avoids having to bind all functions to "this"
var sb = this ;
// Reusable cells
var field = new mxCell ( '+ field: type' , new mxGeometry ( 0 , 0 , 100 , 26 ) , 'text;html=1;strokeColor=none;fillColor=none;align=left;verticalAlign=top;spacingLeft=4;spacingRight=4;whiteSpace=wrap;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;' ) ;
field . vertex = true ;
var divider = new mxCell ( '' , new mxGeometry ( 0 , 0 , 40 , 8 ) , 'line;html=1;strokeWidth=1;fillColor=none;align=left;verticalAlign=middle;spacingTop=-1;spacingLeft=3;spacingRight=3;rotatable=0;labelPosition=right;points=[];portConstraint=eastwest;' ) ;
divider . vertex = true ;
// Default tags
var dt = 'uml static class ' ;
var fns = [
this . createVertexTemplateEntry ( 'html=1;' , 110 , 50 , 'Object' , 'Object' , null , null , dt + 'object instance' ) ,
this . createVertexTemplateEntry ( 'html=1;' , 110 , 50 , '«interface»<br><b>Name</b>' , 'Interface' , null , null , dt + 'interface object instance annotated annotation' ) ,
this . addEntry ( dt + 'object instance' , function ( )
{
var cell = new mxCell ( 'Classname' , new mxGeometry ( 0 , 0 , 160 , 90 ) ,
'swimlane;html=1;fontStyle=1;align=center;verticalAlign=top;childLayout=stackLayout;horizontal=1;startSize=26;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=#ffffff;' ) ;
cell . vertex = true ;
cell . insert ( field . clone ( ) ) ;
cell . insert ( divider . clone ( ) ) ;
cell . insert ( sb . cloneCell ( field , '+ method(type): type' ) ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Class' ) ;
} ) ,
this . addEntry ( dt + 'section subsection' , function ( )
{
var cell = new mxCell ( 'Classname' , new mxGeometry ( 0 , 0 , 140 , 110 ) ,
'swimlane;html=1;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=none;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=#ffffff;' ) ;
cell . vertex = true ;
cell . insert ( field . clone ( ) ) ;
cell . insert ( field . clone ( ) ) ;
cell . insert ( field . clone ( ) ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Class 2' ) ;
} ) ,
this . addEntry ( dt + 'item member method function variable field attribute label' , function ( )
{
return sb . createVertexTemplateFromCells ( [ sb . cloneCell ( field , '+ item: attribute' ) ] , field . geometry . width , field . geometry . height , 'Item 1' ) ;
} ) ,
this . addEntry ( dt + 'item member method function variable field attribute label' , function ( )
{
var cell = new mxCell ( 'item: attribute' , new mxGeometry ( 0 , 0 , 120 , field . geometry . height ) , 'label;html=1;fontStyle=0;strokeColor=none;fillColor=none;align=left;verticalAlign=top;overflow=hidden;' +
'spacingLeft=28;spacingRight=4;whiteSpace=wrap;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;imageWidth=16;imageHeight=16;image=' + sb . gearImage ) ;
cell . vertex = true ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Item 2' ) ;
} ) ,
this . addEntry ( dt + 'divider hline line separator' , function ( )
{
return sb . createVertexTemplateFromCells ( [ divider . clone ( ) ] , divider . geometry . width , divider . geometry . height , 'Divider' ) ;
} ) ,
this . addEntry ( dt + 'spacer space gap separator' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 20 , 14 ) , 'text;html=1;strokeColor=none;fillColor=none;align=left;verticalAlign=middle;spacingTop=-1;spacingLeft=4;spacingRight=4;rotatable=0;labelPosition=right;points=[];portConstraint=eastwest;' ) ;
cell . vertex = true ;
return sb . createVertexTemplateFromCells ( [ cell . clone ( ) ] , cell . geometry . width , cell . geometry . height , 'Spacer' ) ;
} ) ,
this . createVertexTemplateEntry ( 'text;html=1;align=center;fontStyle=1;verticalAlign=middle;spacingLeft=3;spacingRight=3;strokeColor=none;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;' ,
80 , 26 , 'Title' , 'Title' , null , null , dt + 'title label' ) ,
this . addEntry ( dt + 'component' , function ( )
{
var cell = new mxCell ( '«Annotation»<br/><b>Component</b>' , new mxGeometry ( 0 , 0 , 180 , 90 ) , 'html=1;' ) ;
cell . vertex = true ;
var symbol = new mxCell ( '' , new mxGeometry ( 1 , 0 , 20 , 20 ) , 'shape=component;jettyWidth=8;jettyHeight=4;' ) ;
symbol . vertex = true ;
symbol . geometry . relative = true ;
symbol . geometry . offset = new mxPoint ( - 27 , 7 ) ;
cell . insert ( symbol ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Component' ) ;
} ) ,
this . addEntry ( dt + 'component' , function ( )
{
var cell = new mxCell ( '<p style="margin:0px;margin-top:6px;text-align:center;"><b>Component</b></p>' +
'<hr/><p style="margin:0px;margin-left:8px;">+ Attribute1: Type<br/>+ Attribute2: Type</p>' , new mxGeometry ( 0 , 0 , 180 , 90 ) ,
'align=left;overflow=fill;html=1;' ) ;
cell . vertex = true ;
var symbol = new mxCell ( '' , new mxGeometry ( 1 , 0 , 20 , 20 ) , 'shape=component;jettyWidth=8;jettyHeight=4;' ) ;
symbol . vertex = true ;
symbol . geometry . relative = true ;
symbol . geometry . offset = new mxPoint ( - 24 , 4 ) ;
cell . insert ( symbol ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Component with Attributes' ) ;
} ) ,
this . createVertexTemplateEntry ( 'verticalAlign=top;align=left;spacingTop=8;spacingLeft=2;spacingRight=12;shape=cube;size=10;direction=south;fontStyle=4;html=1;' ,
180 , 120 , 'Block' , 'Block' , null , null , dt + 'block' ) ,
this . createVertexTemplateEntry ( 'shape=component;align=left;spacingLeft=36;' , 120 , 60 , 'Module' , 'Module' , null , null , dt + 'module' ) ,
this . createVertexTemplateEntry ( 'shape=folder;fontStyle=1;spacingTop=10;tabWidth=40;tabHeight=14;tabPosition=left;html=1;' , 70 , 50 ,
'package' , 'Package' , null , null , dt + 'package' ) ,
this . createVertexTemplateEntry ( 'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;' ,
160 , 90 , '<p style="margin:0px;margin-top:4px;text-align:center;text-decoration:underline;"><b>Object:Type</b></p><hr/>' +
'<p style="margin:0px;margin-left:8px;">field1 = value1<br/>field2 = value2<br>field3 = value3</p>' , 'Object' ,
null , null , dt + 'object instance' ) ,
this . createVertexTemplateEntry ( 'verticalAlign=top;align=left;overflow=fill;html=1;' , 180 , 90 ,
'<div style="box-sizing:border-box;width:100%;background:#e4e4e4;padding:2px;">Tablename</div>' +
'<table style="width:100%;font-size:1em;" cellpadding="2" cellspacing="0">' +
'<tr><td>PK</td><td>uniqueId</td></tr><tr><td>FK1</td><td>' +
'foreignKey</td></tr><tr><td></td><td>fieldname</td></tr></table>' , 'Entity' , null , null , 'er entity table' ) ,
this . addEntry ( dt + 'object instance' , function ( )
{
var cell = new mxCell ( '<p style="margin:0px;margin-top:4px;text-align:center;">' +
'<b>Class</b></p>' +
'<hr size="1"/><div style="height:2px;"></div>' , new mxGeometry ( 0 , 0 , 140 , 60 ) ,
'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;' ) ;
cell . vertex = true ;
return sb . createVertexTemplateFromCells ( [ cell . clone ( ) ] , cell . geometry . width , cell . geometry . height , 'Class 3' ) ;
} ) ,
this . addEntry ( dt + 'object instance' , function ( )
{
var cell = new mxCell ( '<p style="margin:0px;margin-top:4px;text-align:center;">' +
'<b>Class</b></p>' +
'<hr size="1"/><div style="height:2px;"></div><hr size="1"/><div style="height:2px;"></div>' , new mxGeometry ( 0 , 0 , 140 , 60 ) ,
'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;' ) ;
cell . vertex = true ;
return sb . createVertexTemplateFromCells ( [ cell . clone ( ) ] , cell . geometry . width , cell . geometry . height , 'Class 4' ) ;
} ) ,
this . addEntry ( dt + 'object instance' , function ( )
{
var cell = new mxCell ( '<p style="margin:0px;margin-top:4px;text-align:center;">' +
'<b>Class</b></p>' +
'<hr size="1"/><p style="margin:0px;margin-left:4px;">+ field: Type</p><hr size="1"/>' +
'<p style="margin:0px;margin-left:4px;">+ method(): Type</p>' , new mxGeometry ( 0 , 0 , 160 , 90 ) ,
'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;' ) ;
cell . vertex = true ;
return sb . createVertexTemplateFromCells ( [ cell . clone ( ) ] , cell . geometry . width , cell . geometry . height , 'Class 5' ) ;
} ) ,
this . addEntry ( dt + 'object instance' , function ( )
{
var cell = new mxCell ( '<p style="margin:0px;margin-top:4px;text-align:center;">' +
'<i><<Interface>></i><br/><b>Interface</b></p>' +
'<hr size="1"/><p style="margin:0px;margin-left:4px;">+ field1: Type<br/>' +
'+ field2: Type</p>' +
'<hr size="1"/><p style="margin:0px;margin-left:4px;">' +
'+ method1(Type): Type<br/>' +
'+ method2(Type, Type): Type</p>' , new mxGeometry ( 0 , 0 , 190 , 140 ) ,
'verticalAlign=top;align=left;overflow=fill;fontSize=12;fontFamily=Helvetica;html=1;' ) ;
cell . vertex = true ;
return sb . createVertexTemplateFromCells ( [ cell . clone ( ) ] , cell . geometry . width , cell . geometry . height , 'Interface 2' ) ;
} ) ,
this . createVertexTemplateEntry ( 'shape=lollipop;direction=south;html=1;' , 30 , 10 , '' , 'Provided Interface' , null , null , dt + 'provided interface' ) ,
this . createVertexTemplateEntry ( 'shape=requires;direction=north;html=1;' , 30 , 20 , '' , 'Required Interface' , null , null , dt + 'required interface' ) ,
this . createVertexTemplateEntry ( 'shape=umlBoundary;whiteSpace=wrap;html=1;' , 100 , 80 , 'Boundary Object' , 'Boundary Object' , null , null , 'uml boundary object' ) ,
this . createVertexTemplateEntry ( 'ellipse;shape=umlEntity;whiteSpace=wrap;html=1;' , 80 , 80 , 'Entity Object' , 'Entity Object' , null , null , 'uml entity object' ) ,
this . createVertexTemplateEntry ( 'ellipse;shape=umlControl;whiteSpace=wrap;html=1;' , 70 , 80 , 'Control Object' , 'Control Object' , null , null , 'uml control object' ) ,
this . createVertexTemplateEntry ( 'shape=umlActor;verticalLabelPosition=bottom;labelBackgroundColor=#ffffff;verticalAlign=top;html=1;' , 30 , 60 , 'Actor' , 'Actor' , false , null , 'uml actor' ) ,
this . createVertexTemplateEntry ( 'ellipse;whiteSpace=wrap;html=1;' , 140 , 70 , 'Use Case' , 'Use Case' , null , null , 'uml use case usecase' ) ,
this . addEntry ( 'uml activity state start' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 30 , 30 ) ,
'ellipse;html=1;shape=startState;fillColor=#000000;strokeColor=#ff0000;' ) ;
cell . vertex = true ;
var edge = new mxCell ( '' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 15 , 90 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
cell . insertEdge ( edge , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge ] , 30 , 90 , 'Start' ) ;
} ) ,
this . addEntry ( 'uml activity state' , function ( )
{
var cell = new mxCell ( 'Activity' , new mxGeometry ( 0 , 0 , 120 , 40 ) ,
'rounded=1;whiteSpace=wrap;html=1;arcSize=40;fillColor=#ffffc0;strokeColor=#ff0000;' ) ;
cell . vertex = true ;
var edge = new mxCell ( '' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 60 , 100 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
cell . insertEdge ( edge , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge ] , 120 , 100 , 'Activity' ) ;
} ) ,
this . addEntry ( 'uml activity composite state' , function ( )
{
var cell = new mxCell ( 'Composite State' , new mxGeometry ( 0 , 0 , 160 , 60 ) ,
'swimlane;html=1;fontStyle=1;align=center;verticalAlign=middle;childLayout=stackLayout;horizontal=1;startSize=30;horizontalStack=0;resizeParent=0;resizeLast=1;container=0;collapsible=0;rounded=1;arcSize=30;strokeColor=#ff0000;fillColor=#ffffc0;swimlaneFillColor=#ffffc0;' ) ;
cell . vertex = true ;
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;' ) ;
cell1 . vertex = true ;
cell . insert ( cell1 ) ;
var edge = new mxCell ( '' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 80 , 120 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
cell . insertEdge ( edge , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge ] , 160 , 120 , 'Composite State' ) ;
} ) ,
this . addEntry ( 'uml activity condition' , function ( )
{
var cell = new mxCell ( 'Condition' , new mxGeometry ( 0 , 0 , 80 , 40 ) , 'rhombus;whiteSpace=wrap;html=1;fillColor=#ffffc0;strokeColor=#ff0000;' ) ;
cell . vertex = true ;
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;' ) ;
edge1 . geometry . setTerminalPoint ( new mxPoint ( 180 , 20 ) , false ) ;
edge1 . geometry . relative = true ;
edge1 . geometry . x = - 1 ;
edge1 . edge = true ;
cell . insertEdge ( edge1 , true ) ;
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;' ) ;
edge2 . geometry . setTerminalPoint ( new mxPoint ( 40 , 100 ) , false ) ;
edge2 . geometry . relative = true ;
edge2 . geometry . x = - 1 ;
edge2 . edge = true ;
cell . insertEdge ( edge2 , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge1 , edge2 ] , 180 , 100 , 'Condition' ) ;
} ) ,
this . addEntry ( 'uml activity fork join' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 200 , 10 ) , 'shape=line;html=1;strokeWidth=6;strokeColor=#ff0000;' ) ;
cell . vertex = true ;
var edge = new mxCell ( '' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'edgeStyle=orthogonalEdgeStyle;html=1;verticalAlign=bottom;endArrow=open;endSize=8;strokeColor=#ff0000;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 100 , 80 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
cell . insertEdge ( edge , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge ] , 200 , 80 , 'Fork/Join' ) ;
} ) ,
this . createVertexTemplateEntry ( 'ellipse;html=1;shape=endState;fillColor=#000000;strokeColor=#ff0000;' , 30 , 30 , '' , 'End' , null , null , 'uml activity state end' ) ,
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' ) ,
this . createVertexTemplateEntry ( 'shape=umlLifeline;participant=umlActor;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;labelBackgroundColor=#ffffff;outlineConnect=0;' ,
20 , 300 , '' , 'Actor Lifeline' , null , null , 'uml sequence participant lifeline actor' ) ,
this . createVertexTemplateEntry ( 'shape=umlLifeline;participant=umlBoundary;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;labelBackgroundColor=#ffffff;outlineConnect=0;' ,
50 , 300 , '' , 'Boundary Lifeline' , null , null , 'uml sequence participant lifeline boundary' ) ,
this . createVertexTemplateEntry ( 'shape=umlLifeline;participant=umlEntity;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;labelBackgroundColor=#ffffff;outlineConnect=0;' ,
40 , 300 , '' , 'Entity Lifeline' , null , null , 'uml sequence participant lifeline entity' ) ,
this . createVertexTemplateEntry ( 'shape=umlLifeline;participant=umlControl;perimeter=lifelinePerimeter;whiteSpace=wrap;html=1;container=1;collapsible=0;recursiveResize=0;verticalAlign=top;spacingTop=36;labelBackgroundColor=#ffffff;outlineConnect=0;' ,
40 , 300 , '' , 'Control Lifeline' , null , null , 'uml sequence participant lifeline control' ) ,
this . createVertexTemplateEntry ( 'shape=umlFrame;whiteSpace=wrap;html=1;' , 300 , 200 , 'frame' , 'Frame' , null , null , 'uml sequence frame' ) ,
this . createVertexTemplateEntry ( 'shape=umlDestroy;whiteSpace=wrap;html=1;strokeWidth=3;' , 30 , 30 , '' , 'Destruction' , null , null , 'uml sequence destruction destroy' ) ,
this . createVertexTemplateEntry ( 'shape=note;whiteSpace=wrap;html=1;size=14;verticalAlign=top;align=left;spacingTop=-6;' , 100 , 70 , 'Note' , 'Note' , null , null , 'uml note' ) ,
this . addEntry ( 'uml sequence invoke invocation call activation' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 10 , 80 ) , 'html=1;points=[];perimeter=orthogonalPerimeter;' ) ;
cell . vertex = true ;
var edge = new mxCell ( 'dispatch' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'html=1;verticalAlign=bottom;startArrow=oval;endArrow=block;startSize=8;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( - 60 , 0 ) , true ) ;
edge . geometry . relative = true ;
edge . edge = true ;
cell . insertEdge ( edge , false ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge ] , 10 , 80 , 'Found Message' ) ;
} ) ,
this . addEntry ( 'uml sequence invoke call delegation synchronous invocation activation' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 10 , 80 ) , 'html=1;points=[];perimeter=orthogonalPerimeter;' ) ;
cell . vertex = true ;
var edge1 = new mxCell ( 'dispatch' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'html=1;verticalAlign=bottom;endArrow=block;entryX=0;entryY=0;' ) ;
edge1 . geometry . setTerminalPoint ( new mxPoint ( - 70 , 0 ) , true ) ;
edge1 . geometry . relative = true ;
edge1 . edge = true ;
cell . insertEdge ( edge1 , false ) ;
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;' ) ;
edge2 . geometry . setTerminalPoint ( new mxPoint ( - 70 , 76 ) , false ) ;
edge2 . geometry . relative = true ;
edge2 . edge = true ;
cell . insertEdge ( edge2 , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge1 , edge2 ] , 10 , 80 , 'Synchronous Invocation' ) ;
} ) ,
this . addEntry ( 'uml sequence self call recursion delegation activation' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 20 , 10 , 40 ) , 'html=1;points=[];perimeter=orthogonalPerimeter;' ) ;
cell . vertex = true ;
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;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 5 , 0 ) , true ) ;
edge . geometry . points = [ new mxPoint ( 30 , 0 ) ] ;
edge . geometry . relative = true ;
edge . edge = true ;
cell . insertEdge ( edge , false ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge ] , 10 , 60 , 'Self Call' ) ;
} ) ,
this . addEntry ( 'uml sequence invoke call delegation callback activation' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 10 , 60 ) , 'html=1;points=[];perimeter=orthogonalPerimeter;' ) ;
cell . vertex = true ;
var edge1 = new mxCell ( 'callback' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'html=1;verticalAlign=bottom;endArrow=block;entryX=1;entryY=0;' ) ;
edge1 . geometry . setTerminalPoint ( new mxPoint ( 70 , 0 ) , true ) ;
edge1 . geometry . relative = true ;
edge1 . edge = true ;
cell . insertEdge ( edge1 , false ) ;
var edge2 = new mxCell ( 'return' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'html=1;verticalAlign=bottom;endArrow=open;dashed=1;endSize=8;exitX=1;exitY=0.95;' ) ;
edge2 . geometry . setTerminalPoint ( new mxPoint ( 70 , 57 ) , false ) ;
edge2 . geometry . relative = true ;
edge2 . edge = true ;
cell . insertEdge ( edge2 , true ) ;
return sb . createVertexTemplateFromCells ( [ cell , edge1 , edge2 ] , 10 , 60 , 'Callback' ) ;
} ) ,
this . createVertexTemplateEntry ( 'html=1;points=[];perimeter=orthogonalPerimeter;' , 10 , 80 , '' , 'Activation' , null , null , 'uml sequence activation' ) ,
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' ) ,
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' ) ,
this . createEdgeTemplateEntry ( 'html=1;verticalAlign=bottom;endArrow=block;' , 80 , 0 , 'dispatch' , 'Message' , null , 'uml sequence message call invoke dispatch' ) ,
this . addEntry ( 'uml sequence return message' , function ( )
{
var edge = new mxCell ( 'return' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'html=1;verticalAlign=bottom;endArrow=open;dashed=1;endSize=8;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 80 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 80 , 0 , 'Return' ) ;
} ) ,
this . addEntry ( 'uml relation' , function ( )
{
var edge = new mxCell ( 'name' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'endArrow=block;endFill=1;html=1;edgeStyle=orthogonalEdgeStyle;align=left;verticalAlign=top;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 160 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . geometry . x = - 1 ;
edge . edge = true ;
var cell = new mxCell ( '1' , new mxGeometry ( - 1 , 0 , 0 , 0 ) , 'resizable=0;html=1;align=left;verticalAlign=bottom;labelBackgroundColor=#ffffff;fontSize=10;' ) ;
cell . geometry . relative = true ;
cell . setConnectable ( false ) ;
cell . vertex = true ;
edge . insert ( cell ) ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 160 , 0 , 'Relation 1' ) ;
} ) ,
this . addEntry ( 'uml association' , function ( )
{
var edge = new mxCell ( '' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'endArrow=none;html=1;edgeStyle=orthogonalEdgeStyle;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 160 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
var cell1 = new mxCell ( 'parent' , new mxGeometry ( - 1 , 0 , 0 , 0 ) , 'resizable=0;html=1;align=left;verticalAlign=bottom;labelBackgroundColor=#ffffff;fontSize=10;' ) ;
cell1 . geometry . relative = true ;
cell1 . setConnectable ( false ) ;
cell1 . vertex = true ;
edge . insert ( cell1 ) ;
var cell2 = new mxCell ( 'child' , new mxGeometry ( 1 , 0 , 0 , 0 ) , 'resizable=0;html=1;align=right;verticalAlign=bottom;labelBackgroundColor=#ffffff;fontSize=10;' ) ;
cell2 . geometry . relative = true ;
cell2 . setConnectable ( false ) ;
cell2 . vertex = true ;
edge . insert ( cell2 ) ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 160 , 0 , 'Association 1' ) ;
} ) ,
this . addEntry ( 'uml aggregation' , function ( )
{
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;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 160 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . geometry . x = - 1 ;
edge . geometry . y = 3 ;
edge . edge = true ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 160 , 0 , 'Aggregation' ) ;
} ) ,
this . addEntry ( 'uml composition' , function ( )
{
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;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 160 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . geometry . x = - 1 ;
edge . geometry . y = 3 ;
edge . edge = true ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 160 , 0 , 'Composition' ) ;
} ) ,
this . addEntry ( 'uml relation' , function ( )
{
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;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 160 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
var cell1 = new mxCell ( '0..n' , new mxGeometry ( - 1 , 0 , 0 , 0 ) , 'resizable=0;html=1;align=left;verticalAlign=top;labelBackgroundColor=#ffffff;fontSize=10;' ) ;
cell1 . geometry . relative = true ;
cell1 . setConnectable ( false ) ;
cell1 . vertex = true ;
edge . insert ( cell1 ) ;
var cell2 = new mxCell ( '1' , new mxGeometry ( 1 , 0 , 0 , 0 ) , 'resizable=0;html=1;align=right;verticalAlign=top;labelBackgroundColor=#ffffff;fontSize=10;' ) ;
cell2 . geometry . relative = true ;
cell2 . setConnectable ( false ) ;
cell2 . vertex = true ;
edge . insert ( cell2 ) ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 160 , 0 , 'Relation 2' ) ;
} ) ,
this . createEdgeTemplateEntry ( 'endArrow=open;endSize=12;dashed=1;html=1;' , 160 , 0 , 'Use' , 'Dependency' , null , 'uml dependency use' ) ,
this . createEdgeTemplateEntry ( 'endArrow=block;endSize=16;endFill=0;html=1;' , 160 , 0 , 'Extends' , 'Generalization' , null , 'uml generalization extend' ) ,
this . createEdgeTemplateEntry ( 'endArrow=block;startArrow=block;endFill=1;startFill=1;html=1;' , 160 , 0 , '' , 'Association 2' , null , 'uml association' ) ,
this . createEdgeTemplateEntry ( 'endArrow=open;startArrow=circlePlus;endFill=0;startFill=0;endSize=8;html=1;' , 160 , 0 , '' , 'Inner Class' , null , 'inner class' ) ,
this . createEdgeTemplateEntry ( 'endArrow=open;startArrow=cross;endFill=0;startFill=0;endSize=8;startSize=10;html=1;' , 160 , 0 , '' , 'Terminate' , null , 'terminate' )
] ;
this . addPaletteFunctions ( 'uml' , mxResources . get ( 'uml' ) , expand || false , fns ) ;
} ;
/ * *
* Adds the BPMN library to the sidebar .
* /
Sidebar . prototype . addBpmnPalette = function ( dir , expand )
{
// Avoids having to bind all functions to "this"
var sb = this ;
var fns =
[
this . createVertexTemplateEntry ( 'shape=ext;rounded=1;html=1;whiteSpace=wrap;' , 120 , 80 , 'Task' , 'Process' , null , null , 'bpmn task process' ) ,
this . createVertexTemplateEntry ( 'shape=ext;rounded=1;html=1;whiteSpace=wrap;double=1;' , 120 , 80 , 'Transaction' , 'Transaction' , null , null , 'bpmn transaction' ) ,
this . createVertexTemplateEntry ( 'shape=ext;rounded=1;html=1;whiteSpace=wrap;dashed=1;dashPattern=1 4;' , 120 , 80 , 'Event\nSub-Process' , 'Event Sub-Process' , null , null , 'bpmn event subprocess sub process sub-process' ) ,
this . createVertexTemplateEntry ( 'shape=ext;rounded=1;html=1;whiteSpace=wrap;strokeWidth=3;' , 120 , 80 , 'Call Activity' , 'Call Activity' , null , null , 'bpmn call activity' ) ,
this . addEntry ( 'bpmn subprocess sub process sub-process' , function ( )
{
var cell = new mxCell ( 'Sub-Process' , new mxGeometry ( 0 , 0 , 120 , 80 ) , 'html=1;whiteSpace=wrap;rounded=1;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 0.5 , 1 , 14 , 14 ) , 'html=1;shape=plus;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( - 7 , - 14 ) ;
cell . insert ( cell1 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Sub-Process' ) ;
} ) ,
this . addEntry ( this . getTagsForStencil ( 'mxgraph.bpmn' , 'loop' , 'subprocess sub process sub-process looped' ) . join ( ' ' ) , function ( )
{
var cell = new mxCell ( 'Looped\nSub-Process' , new mxGeometry ( 0 , 0 , 120 , 80 ) , 'html=1;whiteSpace=wrap;rounded=1' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 0.5 , 1 , 14 , 14 ) , 'html=1;shape=mxgraph.bpmn.loop;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( - 15 , - 14 ) ;
cell . insert ( cell1 ) ;
var cell2 = new mxCell ( '' , new mxGeometry ( 0.5 , 1 , 14 , 14 ) , 'html=1;shape=plus;' ) ;
cell2 . vertex = true ;
cell2 . geometry . relative = true ;
cell2 . geometry . offset = new mxPoint ( 1 , - 14 ) ;
cell . insert ( cell2 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Looped Sub-Process' ) ;
} ) ,
this . addEntry ( 'bpmn receive task' , function ( )
{
var cell = new mxCell ( 'Receive' , new mxGeometry ( 0 , 0 , 120 , 80 ) , 'html=1;whiteSpace=wrap;rounded=1;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 0 , 0 , 20 , 14 ) , 'html=1;shape=message;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( 7 , 7 ) ;
cell . insert ( cell1 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Receive Task' ) ;
} ) ,
this . addEntry ( this . getTagsForStencil ( 'mxgraph.bpmn' , 'user_task' ) . join ( ' ' ) , function ( )
{
var cell = new mxCell ( 'User' , new mxGeometry ( 0 , 0 , 120 , 80 ) , 'html=1;whiteSpace=wrap;rounded=1;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 0 , 0 , 14 , 14 ) , 'html=1;shape=mxgraph.bpmn.user_task;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( 7 , 7 ) ;
cell . insert ( cell1 ) ;
var cell2 = new mxCell ( '' , new mxGeometry ( 0.5 , 1 , 14 , 14 ) , 'html=1;shape=plus;' ) ;
cell2 . vertex = true ;
cell2 . geometry . relative = true ;
cell2 . geometry . offset = new mxPoint ( - 7 , - 14 ) ;
cell . insert ( cell2 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'User Task' ) ;
} ) ,
this . addEntry ( this . getTagsForStencil ( 'mxgraph.bpmn' , 'timer_start' , 'attached' ) . join ( ' ' ) , function ( )
{
var cell = new mxCell ( 'Process' , new mxGeometry ( 0 , 0 , 120 , 80 ) , 'html=1;whiteSpace=wrap;rounded=1;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 1 , 1 , 30 , 30 ) , 'shape=mxgraph.bpmn.timer_start;perimeter=ellipsePerimeter;html=1;verticalLabelPosition=bottom;labelBackgroundColor=#ffffff;verticalAlign=top;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( - 40 , - 15 ) ;
cell . insert ( cell1 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , 120 , 95 , 'Attached Timer Event 1' ) ;
} ) ,
this . addEntry ( this . getTagsForStencil ( 'mxgraph.bpmn' , 'timer_start' , 'attached' ) . join ( ' ' ) , function ( )
{
var cell = new mxCell ( 'Process' , new mxGeometry ( 0 , 0 , 120 , 80 ) , 'html=1;whiteSpace=wrap;rounded=1;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 1 , 0 , 30 , 30 ) , 'shape=mxgraph.bpmn.timer_start;perimeter=ellipsePerimeter;html=1;labelPosition=right;labelBackgroundColor=#ffffff;align=left;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( - 15 , 10 ) ;
cell . insert ( cell1 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , 135 , 80 , 'Attached Timer Event 2' ) ;
} ) ,
this . createVertexTemplateEntry ( 'swimlane;html=1;horizontal=0;startSize=20;' , 320 , 240 , 'Pool' , 'Pool' , null , null , 'bpmn pool' ) ,
this . createVertexTemplateEntry ( 'swimlane;html=1;horizontal=0;swimlaneFillColor=white;swimlaneLine=0;' , 300 , 120 , 'Lane' , 'Lane' , null , null , 'bpmn lane' ) ,
this . createVertexTemplateEntry ( 'shape=hexagon;html=1;whiteSpace=wrap;perimeter=hexagonPerimeter;' , 60 , 50 , '' , 'Conversation' , null , null , 'bpmn conversation' ) ,
this . createVertexTemplateEntry ( 'shape=hexagon;html=1;whiteSpace=wrap;perimeter=hexagonPerimeter;strokeWidth=4' , 60 , 50 , '' , 'Call Conversation' , null , null , 'bpmn call conversation' ) ,
this . addEntry ( 'bpmn subconversation sub conversation sub-conversation' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 60 , 50 ) , 'shape=hexagon;whiteSpace=wrap;html=1;perimeter=hexagonPerimeter;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 0.5 , 1 , 14 , 14 ) , 'html=1;shape=plus;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( - 7 , - 14 ) ;
cell . insert ( cell1 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Sub-Conversation' ) ;
} ) ,
this . addEntry ( 'bpmn data object' , function ( )
{
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 40 , 60 ) , 'shape=note;whiteSpace=wrap;size=16;html=1;' ) ;
cell . vertex = true ;
var cell1 = new mxCell ( '' , new mxGeometry ( 0 , 0 , 14 , 14 ) , 'html=1;shape=singleArrow;arrowWidth=0.4;arrowSize=0.4;' ) ;
cell1 . vertex = true ;
cell1 . geometry . relative = true ;
cell1 . geometry . offset = new mxPoint ( 2 , 2 ) ;
cell . insert ( cell1 ) ;
var cell2 = new mxCell ( '' , new mxGeometry ( 0.5 , 1 , 14 , 14 ) , 'html=1;whiteSpace=wrap;shape=parallelMarker;' ) ;
cell2 . vertex = true ;
cell2 . geometry . relative = true ;
cell2 . geometry . offset = new mxPoint ( - 7 , - 14 ) ;
cell . insert ( cell2 ) ;
return sb . createVertexTemplateFromCells ( [ cell ] , cell . geometry . width , cell . geometry . height , 'Data Object' ) ;
} ) ,
this . createVertexTemplateEntry ( 'shape=datastore;whiteSpace=wrap;html=1;' , 60 , 60 , '' , 'Data Store' , null , null , 'bpmn data store' ) ,
this . createVertexTemplateEntry ( 'shape=plus;html=1;' , 14 , 14 , '' , 'Sub-Process Marker' , null , null , 'bpmn subprocess sub process sub-process marker' ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.loop;html=1;' , 14 , 14 , '' , 'Loop Marker' , null , null , 'bpmn loop marker' ) ,
this . createVertexTemplateEntry ( 'shape=parallelMarker;html=1;' , 14 , 14 , '' , 'Parallel MI Marker' , null , null , 'bpmn parallel mi marker' ) ,
this . createVertexTemplateEntry ( 'shape=parallelMarker;direction=south;html=1;' , 14 , 14 , '' , 'Sequential MI Marker' , null , null , 'bpmn sequential mi marker' ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.ad_hoc;fillColor=#000000;html=1;' , 14 , 14 , '' , 'Ad Hoc Marker' , null , null , 'bpmn ad hoc marker' ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.compensation;html=1;' , 14 , 14 , '' , 'Compensation Marker' , null , null , 'bpmn compensation marker' ) ,
this . createVertexTemplateEntry ( 'shape=message;whiteSpace=wrap;html=1;fillColor=#000000;strokeColor=#ffffff;strokeWidth=2;' , 40 , 30 , '' , 'Send Task' , null , null , 'bpmn send task' ) ,
this . createVertexTemplateEntry ( 'shape=message;whiteSpace=wrap;html=1;' , 40 , 30 , '' , 'Receive Task' , null , null , 'bpmn receive task' ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.user_task;html=1;' , 14 , 14 , '' , 'User Task' , null , null , this . getTagsForStencil ( 'mxgraph.bpmn' , 'user_task' ) . join ( ' ' ) ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.manual_task;html=1;' , 14 , 14 , '' , 'Manual Task' , null , null , this . getTagsForStencil ( 'mxgraph.bpmn' , 'user_task' ) . join ( ' ' ) ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.business_rule_task;html=1;' , 14 , 14 , '' , 'Business Rule Task' , null , null , this . getTagsForStencil ( 'mxgraph.bpmn' , 'business_rule_task' ) . join ( ' ' ) ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.service_task;html=1;' , 14 , 14 , '' , 'Service Task' , null , null , this . getTagsForStencil ( 'mxgraph.bpmn' , 'service_task' ) . join ( ' ' ) ) ,
this . createVertexTemplateEntry ( 'shape=mxgraph.bpmn.script_task;html=1;' , 14 , 14 , '' , 'Script Task' , null , null , this . getTagsForStencil ( 'mxgraph.bpmn' , 'script_task' ) . join ( ' ' ) ) ,
this . createEdgeTemplateEntry ( 'endArrow=block;endFill=1;endSize=6;html=1;' , 100 , 0 , '' , 'Sequence Flow' , null , 'bpmn sequence flow' ) ,
this . createEdgeTemplateEntry ( 'startArrow=dash;startSize=8;endArrow=block;endFill=1;endSize=6;html=1;' , 100 , 0 , '' , 'Default Flow' , null , 'bpmn default flow' ) ,
this . createEdgeTemplateEntry ( 'startArrow=diamondThin;startFill=0;startSize=14;endArrow=block;endFill=1;endSize=6;html=1;' , 100 , 0 , '' , 'Conditional Flow' , null , 'bpmn conditional flow' ) ,
this . createEdgeTemplateEntry ( 'startArrow=oval;startFill=0;startSize=7;endArrow=block;endFill=0;endSize=10;dashed=1;html=1;' , 100 , 0 , '' , 'Message Flow 1' , null , 'bpmn message flow' ) ,
this . addEntry ( 'bpmn message flow' , function ( )
{
var edge = new mxCell ( '' , new mxGeometry ( 0 , 0 , 0 , 0 ) , 'startArrow=oval;startFill=0;startSize=7;endArrow=block;endFill=0;endSize=10;dashed=1;html=1;' ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 0 , 0 ) , true ) ;
edge . geometry . setTerminalPoint ( new mxPoint ( 100 , 0 ) , false ) ;
edge . geometry . relative = true ;
edge . edge = true ;
var cell = new mxCell ( '' , new mxGeometry ( 0 , 0 , 20 , 14 ) , 'shape=message;html=1;' ) ;
cell . geometry . relative = true ;
cell . setConnectable ( false ) ;
cell . vertex = true ;
cell . geometry . offset = new mxPoint ( - 10 , - 7 ) ;
edge . insert ( cell ) ;
return sb . createEdgeTemplateFromCells ( [ edge ] , 100 , 0 , 'Message Flow 2' ) ;
} ) ,
this . createEdgeTemplateEntry ( 'shape=link;html=1;' , 100 , 0 , '' , 'Link' , null , 'bpmn link' )
] ;
this . addPaletteFunctions ( 'bpmn' , 'BPMN ' + mxResources . get ( 'general' ) , false , fns ) ;
} ;
/ * *
* Creates and returns the given title element .
* /
Sidebar . prototype . createTitle = function ( label )
{
var elt = document . createElement ( 'a' ) ;
elt . setAttribute ( 'href' , 'javascript:void(0);' ) ;
elt . setAttribute ( 'title' , mxResources . get ( 'sidebarTooltip' ) ) ;
elt . className = 'geTitle' ;
mxUtils . write ( elt , label ) ;
return elt ;
} ;
/ * *
* Creates a thumbnail for the given cells .
* /
Sidebar . prototype . createThumb = function ( cells , width , height , parent , title , showLabel , showTitle , realWidth , realHeight )
{
this . graph . labelsVisible = ( showLabel == null || showLabel ) ;
var fo = mxClient . NO _FO ;
mxClient . NO _FO = Editor . prototype . originalNoForeignObject ;
2016-09-12 11:14:09 +00:00
this . graph . view . scaleAndTranslate ( 1 , 0 , 0 ) ;
this . graph . addCells ( cells ) ;
var bounds = this . graph . getGraphBounds ( ) ;
var s = Math . floor ( Math . min ( ( width - 2 * this . thumbBorder ) / bounds . width , ( height - 2 * this . thumbBorder )
/ bounds.height) * 100) / 100 ;
this . graph . view . scaleAndTranslate ( s , Math . floor ( ( width - bounds . width * s ) / 2 / s - bounds . x ) ,
Math . floor ( ( height - bounds . height * s ) / 2 / s - bounds . y ) ) ;
2016-09-06 14:07:11 +00:00
var node = null ;
// For supporting HTML labels in IE9 standards mode the container is cloned instead
if ( this . graph . dialect == mxConstants . DIALECT _SVG && ! mxClient . NO _FO )
{
node = this . graph . view . getCanvas ( ) . ownerSVGElement . cloneNode ( true ) ;
}
// LATER: Check if deep clone can be used for quirks if container in DOM
else
{
node = this . graph . container . cloneNode ( false ) ;
node . innerHTML = this . graph . container . innerHTML ;
}
this . graph . getModel ( ) . clear ( ) ;
mxClient . NO _FO = fo ;
// Catch-all event handling
if ( mxClient . IS _IE6 )
{
parent . style . backgroundImage = 'url(' + this . editorUi . editor . transparentImage + ')' ;
}
node . style . position = 'relative' ;
node . style . overflow = 'hidden' ;
node . style . cursor = 'move' ;
node . style . left = this . thumbBorder + 'px' ;
node . style . top = this . thumbBorder + 'px' ;
node . style . width = width + 'px' ;
node . style . height = height + 'px' ;
node . style . visibility = '' ;
node . style . minWidth = '' ;
node . style . minHeight = '' ;
parent . appendChild ( node ) ;
// Adds title for sidebar entries
if ( this . sidebarTitles && title != null && showTitle != false )
{
var border = ( mxClient . IS _QUIRKS ) ? 2 * this . thumbPadding + 2 : 0 ;
parent . style . height = ( this . thumbHeight + border + this . sidebarTitleSize + 8 ) + 'px' ;
var div = document . createElement ( 'div' ) ;
div . style . fontSize = this . sidebarTitleSize + 'px' ;
div . style . color = '#303030' ;
div . style . textAlign = 'center' ;
div . style . whiteSpace = 'nowrap' ;
if ( mxClient . IS _IE )
{
div . style . height = ( this . sidebarTitleSize + 12 ) + 'px' ;
}
div . style . paddingTop = '4px' ;
mxUtils . write ( div , title ) ;
parent . appendChild ( div ) ;
}
return bounds ;
} ;
/ * *
* Creates and returns a new palette item for the given image .
* /
Sidebar . prototype . createItem = function ( cells , title , showLabel , showTitle , width , height , allowCellsInserted )
{
var elt = document . createElement ( 'a' ) ;
elt . setAttribute ( 'href' , 'javascript:void(0);' ) ;
elt . className = 'geItem' ;
elt . style . overflow = 'hidden' ;
var border = ( mxClient . IS _QUIRKS ) ? 8 + 2 * this . thumbPadding : 2 * this . thumbBorder ;
elt . style . width = ( this . thumbWidth + border ) + 'px' ;
elt . style . height = ( this . thumbHeight + border ) + 'px' ;
elt . style . padding = this . thumbPadding + 'px' ;
// Blocks default click action
mxEvent . addListener ( elt , 'click' , function ( evt )
{
mxEvent . consume ( evt ) ;
} ) ;
this . createThumb ( cells , this . thumbWidth , this . thumbHeight , elt , title , showLabel , showTitle , width , height ) ;
var bounds = new mxRectangle ( 0 , 0 , width , height ) ;
if ( cells . length > 1 || cells [ 0 ] . vertex )
{
var ds = this . createDragSource ( elt , this . createDropHandler ( cells , true , allowCellsInserted ,
bounds ) , this . createDragPreview ( width , height ) , cells , bounds ) ;
this . addClickHandler ( elt , ds , cells ) ;
// Uses guides for vertices only if enabled in graph
ds . isGuidesEnabled = mxUtils . bind ( this , function ( )
{
return this . editorUi . editor . graph . graphHandler . guidesEnabled ;
} ) ;
}
else if ( cells [ 0 ] != null && cells [ 0 ] . edge )
{
var ds = this . createDragSource ( elt , this . createDropHandler ( cells , false , allowCellsInserted ,
bounds ) , this . createDragPreview ( width , height ) , cells , bounds ) ;
this . addClickHandler ( elt , ds , cells ) ;
}
// Shows a tooltip with the rendered cell
if ( ! mxClient . IS _IOS )
{
mxEvent . addGestureListeners ( elt , null , mxUtils . bind ( this , function ( evt )
{
if ( mxEvent . isMouseEvent ( evt ) )
{
this . showTooltip ( elt , cells , bounds . width , bounds . height , title , showLabel ) ;
}
} ) ) ;
}
return elt ;
} ;
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . updateShapes = function ( source , targets )
{
var graph = this . editorUi . editor . graph ;
var sourceCellStyle = graph . getCellStyle ( source ) ;
var result = [ ] ;
graph . model . beginUpdate ( ) ;
try
{
var cellStyle = graph . getModel ( ) . getStyle ( source ) ;
// Lists the styles to carry over from the existing shape
var styles = [ 'shadow' , 'dashed' , 'dashPattern' , 'fontFamily' , 'fontSize' , 'fontColor' , 'align' , 'startFill' ,
'startSize' , 'endFill' , 'endSize' , 'strokeColor' , 'strokeWidth' , 'fillColor' , 'gradientColor' ,
'html' , 'part' , 'noEdgeStyle' , 'edgeStyle' , 'elbow' , 'childLayout' ] ;
for ( var i = 0 ; i < targets . length ; i ++ )
{
var targetCell = targets [ i ] ;
if ( ( graph . getModel ( ) . isVertex ( targetCell ) == graph . getModel ( ) . isVertex ( source ) ) ||
( graph . getModel ( ) . isEdge ( targetCell ) == graph . getModel ( ) . isEdge ( source ) ) )
{
var state = graph . view . getState ( targetCell ) ;
var style = ( state != null ) ? state . style : graph . getCellStyle ( targets [ i ] ) ;
graph . getModel ( ) . setStyle ( targetCell , cellStyle ) ;
// Removes all children of composite cells
if ( state != null && mxUtils . getValue ( state . style , 'composite' , '0' ) == '1' )
{
var childCount = graph . model . getChildCount ( targetCell ) ;
for ( var j = childCount ; j >= 0 ; j -- )
{
graph . model . remove ( graph . model . getChildAt ( targetCell , j ) ) ;
}
}
if ( style != null )
{
// Replaces the participant style in the lifeline shape with the target shape
if ( style [ mxConstants . STYLE _SHAPE ] == 'umlLifeline' &&
sourceCellStyle [ mxConstants . STYLE _SHAPE ] != 'umlLifeline' )
{
graph . setCellStyles ( mxConstants . STYLE _SHAPE , 'umlLifeline' , [ targetCell ] ) ;
graph . setCellStyles ( 'participant' , sourceCellStyle [ mxConstants . STYLE _SHAPE ] , [ targetCell ] ) ;
}
for ( var j = 0 ; j < styles . length ; j ++ )
{
var value = style [ styles [ j ] ] ;
if ( value != null )
{
graph . setCellStyles ( styles [ j ] , value , [ targetCell ] ) ;
}
}
}
result . push ( targetCell ) ;
}
}
}
finally
{
graph . model . endUpdate ( ) ;
}
return result ;
} ;
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . createDropHandler = function ( cells , allowSplit , allowCellsInserted , bounds )
{
allowCellsInserted = ( allowCellsInserted != null ) ? allowCellsInserted : true ;
return mxUtils . bind ( this , function ( graph , evt , target , x , y )
{
if ( graph . isEnabled ( ) )
{
cells = graph . getImportableCells ( cells ) ;
if ( cells . length > 0 )
{
graph . stopEditing ( ) ;
2016-11-10 10:28:55 +00:00
// Holding alt while mouse is released ignores drop target
var validDropTarget = ( target != null && ! mxEvent . isAltDown ( evt ) ) ?
graph . isValidDropTarget ( target , cells , evt ) : false ;
2016-09-06 14:07:11 +00:00
var select = null ;
if ( target != null && ! validDropTarget )
{
target = null ;
}
if ( ! graph . isCellLocked ( target || graph . getDefaultParent ( ) ) )
{
graph . model . beginUpdate ( ) ;
try
{
x = Math . round ( x ) ;
y = Math . round ( y ) ;
// Splits the target edge or inserts into target group
if ( allowSplit && graph . isSplitTarget ( target , cells , evt ) )
{
var clones = graph . cloneCells ( cells ) ;
graph . splitEdge ( target , clones , null ,
x - bounds . width / 2 , y - bounds . height / 2 ) ;
select = clones ;
}
else if ( cells . length > 0 )
{
select = graph . importCells ( cells , x , y , target ) ;
}
// Executes parent layout hooks for position/order
if ( graph . layoutManager != null )
{
var layout = graph . layoutManager . getLayout ( target ) ;
if ( layout != null )
{
var s = graph . view . scale ;
var tr = graph . view . translate ;
var tx = ( x + tr . x ) * s ;
var ty = ( y + tr . y ) * s ;
for ( var i = 0 ; i < select . length ; i ++ )
{
layout . moveCell ( select [ i ] , tx , ty ) ;
}
}
}
if ( allowCellsInserted )
{
graph . fireEvent ( new mxEventObject ( 'cellsInserted' , 'cells' , select ) ) ;
}
}
finally
{
graph . model . endUpdate ( ) ;
}
if ( select != null && select . length > 0 )
{
graph . scrollCellToVisible ( select [ 0 ] ) ;
graph . setSelectionCells ( select ) ;
}
}
}
mxEvent . consume ( evt ) ;
}
} ) ;
} ;
/ * *
* Creates and returns a preview element for the given width and height .
* /
Sidebar . prototype . createDragPreview = function ( width , height )
{
var elt = document . createElement ( 'div' ) ;
elt . style . border = '1px dashed black' ;
elt . style . width = width + 'px' ;
elt . style . height = height + 'px' ;
return elt ;
} ;
/ * *
* Creates a drag source for the given element .
* /
Sidebar . prototype . dropAndConnect = function ( source , targets , direction , dropCellIndex )
{
var geo = this . getDropAndConnectGeometry ( source , targets [ dropCellIndex ] , direction , targets ) ;
if ( geo != null )
{
var graph = this . editorUi . editor . graph ;
// Targets without the new edge for selection
var tmp = [ ] ;
graph . model . beginUpdate ( ) ;
try
{
var sourceGeo = graph . getCellGeometry ( source ) ;
var geo2 = graph . getCellGeometry ( targets [ dropCellIndex ] ) ;
// Handles special case where target should be ignored for stack layouts
var targetParent = graph . model . getParent ( source ) ;
var validLayout = true ;
// Ignores parent if it has a stack layout
if ( graph . layoutManager != null )
{
var layout = graph . layoutManager . getLayout ( targetParent ) ;
// LATER: Use parent of parent if valid layout
if ( layout != null && layout . constructor == mxStackLayout )
{
validLayout = false ;
var tmp = graph . view . getState ( targetParent ) ;
// Offsets by parent position
if ( tmp != null )
{
var offset = new mxPoint ( ( tmp . x / graph . view . scale - graph . view . translate . x ) ,
( tmp . y / graph . view . scale - graph . view . translate . y ) ) ;
geo . x += offset . x ;
geo . y += offset . y ;
var pt = geo . getTerminalPoint ( false ) ;
if ( pt != null )
{
pt . x += offset . x ;
pt . y += offset . y ;
}
}
}
}
var dx = geo2 . x ;
var dy = geo2 . y ;
// Ignores geometry of edges
if ( graph . model . isEdge ( targets [ dropCellIndex ] ) )
{
dx = 0 ;
dy = 0 ;
}
var useParent = graph . model . isEdge ( source ) || ( sourceGeo != null && ! sourceGeo . relative && validLayout ) ;
targets = graph . importCells ( targets , ( geo . x - ( useParent ? dx : 0 ) ) ,
( geo . y - ( useParent ? dy : 0 ) ) , ( useParent ) ? targetParent : null ) ;
tmp = targets ;
if ( graph . model . isEdge ( source ) )
{
// Adds new terminal to edge
// LATER: Push new terminal out radially from edge start point
graph . model . setTerminal ( source , targets [ dropCellIndex ] , direction == mxConstants . DIRECTION _NORTH ) ;
}
else if ( graph . model . isEdge ( targets [ dropCellIndex ] ) )
{
// Adds new outgoing connection to vertex and clears points
graph . model . setTerminal ( targets [ dropCellIndex ] , source , true ) ;
var geo3 = graph . getCellGeometry ( targets [ dropCellIndex ] ) ;
geo3 . points = null ;
if ( geo3 . getTerminalPoint ( false ) != null )
{
geo3 . setTerminalPoint ( geo . getTerminalPoint ( false ) , false ) ;
}
else if ( useParent && graph . model . isVertex ( targetParent ) )
{
// Adds parent offset to other nodes
var tmpState = graph . view . getState ( targetParent ) ;
var offset = new mxPoint ( ( tmpState . x / graph . view . scale - graph . view . translate . x ) ,
( tmpState . y / graph . view . scale - graph . view . translate . y ) ) ;
graph . cellsMoved ( targets , offset . x , offset . y , null , null , true ) ;
}
}
else
{
geo2 = graph . getCellGeometry ( targets [ dropCellIndex ] ) ;
dx = geo . x - Math . round ( geo2 . x ) ;
dy = geo . y - Math . round ( geo2 . y ) ;
geo . x = Math . round ( geo2 . x ) ;
geo . y = Math . round ( geo2 . y ) ;
graph . model . setGeometry ( targets [ dropCellIndex ] , geo ) ;
graph . cellsMoved ( targets , dx , dy , null , null , true ) ;
tmp = targets . slice ( ) ;
targets . push ( graph . insertEdge ( null , null , '' , source , targets [ dropCellIndex ] ,
graph . createCurrentEdgeStyle ( ) ) ) ;
}
graph . fireEvent ( new mxEventObject ( 'cellsInserted' , 'cells' , targets ) ) ;
}
finally
{
graph . model . endUpdate ( ) ;
}
graph . setSelectionCells ( tmp ) ;
}
} ;
/ * *
* Creates a drag source for the given element .
* /
Sidebar . prototype . getDropAndConnectGeometry = function ( source , target , direction , targets )
{
var graph = this . editorUi . editor . graph ;
var view = graph . view ;
var keepSize = targets . length > 1 ;
var geo = graph . getCellGeometry ( source ) ;
var geo2 = graph . getCellGeometry ( target ) ;
if ( geo != null && geo2 != null )
{
geo2 = geo2 . clone ( ) ;
if ( graph . model . isEdge ( source ) )
{
var state = graph . view . getState ( source ) ;
var pts = state . absolutePoints ;
var p0 = pts [ 0 ] ;
var pe = pts [ pts . length - 1 ] ;
if ( direction == mxConstants . DIRECTION _NORTH )
{
geo2 . x = p0 . x / view . scale - view . translate . x - geo2 . width / 2 ;
geo2 . y = p0 . y / view . scale - view . translate . y - geo2 . height / 2 ;
}
else
{
geo2 . x = pe . x / view . scale - view . translate . x - geo2 . width / 2 ;
geo2 . y = pe . y / view . scale - view . translate . y - geo2 . height / 2 ;
}
}
else
{
if ( geo . relative )
{
var state = graph . view . getState ( source ) ;
geo = geo . clone ( ) ;
geo . x = ( state . x - view . translate . x ) / view . scale ;
geo . y = ( state . y - view . translate . y ) / view . scale ;
}
var length = graph . defaultEdgeLength ;
// Maintains edge length
if ( graph . model . isEdge ( target ) && geo2 . getTerminalPoint ( true ) != null && geo2 . getTerminalPoint ( false ) != null )
{
var p0 = geo2 . getTerminalPoint ( true ) ;
var pe = geo2 . getTerminalPoint ( false ) ;
var dx = pe . x - p0 . x ;
var dy = pe . y - p0 . y ;
length = Math . sqrt ( dx * dx + dy * dy ) ;
geo2 . x = geo . getCenterX ( ) ;
geo2 . y = geo . getCenterY ( ) ;
geo2 . width = 1 ;
geo2 . height = 1 ;
if ( direction == mxConstants . DIRECTION _NORTH )
{
geo2 . height = length
geo2 . y = geo . y - length ;
geo2 . setTerminalPoint ( new mxPoint ( geo2 . x , geo2 . y ) , false ) ;
}
else if ( direction == mxConstants . DIRECTION _EAST )
{
geo2 . width = length
geo2 . x = geo . x + geo . width ;
geo2 . setTerminalPoint ( new mxPoint ( geo2 . x + geo2 . width , geo2 . y ) , false ) ;
}
else if ( direction == mxConstants . DIRECTION _SOUTH )
{
geo2 . height = length
geo2 . y = geo . y + geo . height ;
geo2 . setTerminalPoint ( new mxPoint ( geo2 . x , geo2 . y + geo2 . height ) , false ) ;
}
else if ( direction == mxConstants . DIRECTION _WEST )
{
geo2 . width = length
geo2 . x = geo . x - length ;
geo2 . setTerminalPoint ( new mxPoint ( geo2 . x , geo2 . y ) , false ) ;
}
}
else
{
// Try match size or ignore if width or height < 45 which
// is considered special enough to be ignored here
if ( ! keepSize && geo2 . width > 45 && geo2 . height > 45 &&
geo . width > 45 && geo . height > 45 )
{
geo2 . width = geo2 . width * ( geo . height / geo2 . height ) ;
geo2 . height = geo . height ;
}
geo2 . x = geo . x + geo . width / 2 - geo2 . width / 2 ;
geo2 . y = geo . y + geo . height / 2 - geo2 . height / 2 ;
if ( direction == mxConstants . DIRECTION _NORTH )
{
geo2 . y = geo2 . y - geo . height / 2 - geo2 . height / 2 - length ;
}
else if ( direction == mxConstants . DIRECTION _EAST )
{
geo2 . x = geo2 . x + geo . width / 2 + geo2 . width / 2 + length ;
}
else if ( direction == mxConstants . DIRECTION _SOUTH )
{
geo2 . y = geo2 . y + geo . height / 2 + geo2 . height / 2 + length ;
}
else if ( direction == mxConstants . DIRECTION _WEST )
{
geo2 . x = geo2 . x - geo . width / 2 - geo2 . width / 2 - length ;
}
// Adds offset to match cells without connecting edge
if ( graph . model . isEdge ( target ) && geo2 . getTerminalPoint ( true ) != null && target . getTerminal ( false ) != null )
{
var targetGeo = graph . getCellGeometry ( target . getTerminal ( false ) ) ;
if ( targetGeo != null )
{
if ( direction == mxConstants . DIRECTION _NORTH )
{
geo2 . x -= targetGeo . getCenterX ( ) ;
geo2 . y -= targetGeo . getCenterY ( ) + targetGeo . height / 2 ;
}
else if ( direction == mxConstants . DIRECTION _EAST )
{
geo2 . x -= targetGeo . getCenterX ( ) - targetGeo . width / 2 ;
geo2 . y -= targetGeo . getCenterY ( ) ;
}
else if ( direction == mxConstants . DIRECTION _SOUTH )
{
geo2 . x -= targetGeo . getCenterX ( ) ;
geo2 . y -= targetGeo . getCenterY ( ) - targetGeo . height / 2 ;
}
else if ( direction == mxConstants . DIRECTION _WEST )
{
geo2 . x -= targetGeo . getCenterX ( ) + targetGeo . width / 2 ;
geo2 . y -= targetGeo . getCenterY ( ) ;
}
}
}
}
}
}
return geo2 ;
} ;
/ * *
* Creates a drag source for the given element .
* /
Sidebar . prototype . createDragSource = function ( elt , dropHandler , preview , cells , bounds )
{
// Checks if the cells contain any vertices
var ui = this . editorUi ;
var graph = ui . editor . graph ;
var freeSourceEdge = null ;
var firstVertex = null ;
var sidebar = this ;
for ( var i = 0 ; i < cells . length ; i ++ )
{
if ( firstVertex == null && this . editorUi . editor . graph . model . isVertex ( cells [ i ] ) )
{
firstVertex = i ;
}
else if ( freeSourceEdge == null && this . editorUi . editor . graph . model . isEdge ( cells [ i ] ) &&
this . editorUi . editor . graph . model . getTerminal ( cells [ i ] , true ) == null )
{
freeSourceEdge = i ;
}
if ( firstVertex != null && freeSourceEdge != null )
{
break ;
}
}
var dragSource = mxUtils . makeDraggable ( elt , this . editorUi . editor . graph , mxUtils . bind ( this , function ( graph , evt , target , x , y )
{
if ( this . updateThread != null )
{
window . clearTimeout ( this . updateThread ) ;
}
if ( cells != null && currentStyleTarget != null && activeArrow == styleTarget )
{
var tmp = graph . isCellSelected ( currentStyleTarget . cell ) ? graph . getSelectionCells ( ) : [ currentStyleTarget . cell ] ;
var updatedCells = this . updateShapes ( ( graph . model . isEdge ( currentStyleTarget . cell ) ) ? cells [ 0 ] : cells [ firstVertex ] , tmp ) ;
graph . setSelectionCells ( updatedCells ) ;
}
else if ( cells != null && activeArrow != null && currentTargetState != null && activeArrow != styleTarget )
{
var index = ( graph . model . isEdge ( currentTargetState . cell ) || freeSourceEdge == null ) ? firstVertex : freeSourceEdge ;
this . dropAndConnect ( currentTargetState . cell , cells , direction , index ) ;
}
else
{
dropHandler . apply ( this , arguments ) ;
}
if ( this . editorUi . hoverIcons != null )
{
this . editorUi . hoverIcons . update ( graph . view . getState ( graph . getSelectionCell ( ) ) ) ;
}
} ) ,
preview , 0 , 0 , this . editorUi . editor . graph . autoscroll , true , true ) ;
// Stops dragging if cancel is pressed
this . editorUi . editor . graph . addListener ( mxEvent . ESCAPE , function ( sender , evt )
{
if ( dragSource . isActive ( ) )
{
dragSource . reset ( ) ;
}
} ) ;
// Overrides mouseDown to ignore popup triggers
var mouseDown = dragSource . mouseDown ;
dragSource . mouseDown = function ( evt )
{
if ( ! mxEvent . isPopupTrigger ( evt ) && ! mxEvent . isMultiTouchEvent ( evt ) )
{
graph . stopEditing ( ) ;
mouseDown . apply ( this , arguments ) ;
}
} ;
// Workaround for event redirection via image tag in quirks and IE8
function createArrow ( img , tooltip )
{
var arrow = null ;
if ( mxClient . IS _IE && ! mxClient . IS _SVG )
{
// Workaround for PNG images in IE6
if ( mxClient . IS _IE6 && document . compatMode != 'CSS1Compat' )
{
arrow = document . createElement ( mxClient . VML _PREFIX + ':image' ) ;
arrow . setAttribute ( 'src' , img . src ) ;
arrow . style . borderStyle = 'none' ;
}
else
{
arrow = document . createElement ( 'div' ) ;
arrow . style . backgroundImage = 'url(' + img . src + ')' ;
arrow . style . backgroundPosition = 'center' ;
arrow . style . backgroundRepeat = 'no-repeat' ;
}
arrow . style . width = ( img . width + 4 ) + 'px' ;
arrow . style . height = ( img . height + 4 ) + 'px' ;
arrow . style . display = ( mxClient . IS _QUIRKS ) ? 'inline' : 'inline-block' ;
}
else
{
arrow = mxUtils . createImage ( img . src ) ;
arrow . style . width = img . width + 'px' ;
arrow . style . height = img . height + 'px' ;
}
if ( tooltip != null )
{
arrow . setAttribute ( 'title' , tooltip ) ;
}
mxUtils . setOpacity ( arrow , ( img == this . refreshTarget ) ? 30 : 20 ) ;
arrow . style . position = 'absolute' ;
arrow . style . cursor = 'crosshair' ;
return arrow ;
} ;
var currentTargetState = null ;
var currentStateHandle = null ;
var currentStyleTarget = null ;
var activeTarget = false ;
var arrowUp = createArrow ( this . triangleUp , mxResources . get ( 'connect' ) ) ;
var arrowRight = createArrow ( this . triangleRight , mxResources . get ( 'connect' ) ) ;
var arrowDown = createArrow ( this . triangleDown , mxResources . get ( 'connect' ) ) ;
var arrowLeft = createArrow ( this . triangleLeft , mxResources . get ( 'connect' ) ) ;
var styleTarget = createArrow ( this . refreshTarget , mxResources . get ( 'replace' ) ) ;
// Workaround for actual parentNode not being updated in old IE
var styleTargetParent = null ;
var roundSource = createArrow ( this . roundDrop ) ;
var roundTarget = createArrow ( this . roundDrop ) ;
var direction = mxConstants . DIRECTION _NORTH ;
var activeArrow = null ;
function checkArrow ( x , y , bounds , arrow )
{
if ( arrow . parentNode != null )
{
if ( mxUtils . contains ( bounds , x , y ) )
{
mxUtils . setOpacity ( arrow , 100 ) ;
activeArrow = arrow ;
}
else
{
mxUtils . setOpacity ( arrow , ( arrow == styleTarget ) ? 30 : 20 ) ;
}
}
return bounds ;
} ;
// Hides guides and preview if target is active
var dsCreatePreviewElement = dragSource . createPreviewElement ;
// Stores initial size of preview element
dragSource . createPreviewElement = function ( graph )
{
var elt = dsCreatePreviewElement . apply ( this , arguments ) ;
// Pass-through events required to tooltip on replace shape
if ( mxClient . IS _SVG )
{
elt . style . pointerEvents = 'none' ;
}
this . previewElementWidth = elt . style . width ;
this . previewElementHeight = elt . style . height ;
return elt ;
} ;
// Shows/hides hover icons
var dragEnter = dragSource . dragEnter ;
dragSource . dragEnter = function ( graph , evt )
{
if ( ui . hoverIcons != null )
{
ui . hoverIcons . setDisplay ( 'none' ) ;
}
dragEnter . apply ( this , arguments ) ;
} ;
var dragExit = dragSource . dragExit ;
dragSource . dragExit = function ( graph , evt )
{
if ( ui . hoverIcons != null )
{
ui . hoverIcons . setDisplay ( '' ) ;
}
dragExit . apply ( this , arguments ) ;
} ;
dragSource . dragOver = function ( graph , evt )
{
mxDragSource . prototype . dragOver . apply ( this , arguments ) ;
if ( this . currentGuide != null && activeArrow != null )
{
this . currentGuide . hide ( ) ;
}
if ( this . previewElement != null )
{
var view = graph . view ;
if ( currentStyleTarget != null && activeArrow == styleTarget )
{
this . previewElement . style . display = ( graph . model . isEdge ( currentStyleTarget . cell ) ) ? 'none' : '' ;
this . previewElement . style . left = currentStyleTarget . x + 'px' ;
this . previewElement . style . top = currentStyleTarget . y + 'px' ;
this . previewElement . style . width = currentStyleTarget . width + 'px' ;
this . previewElement . style . height = currentStyleTarget . height + 'px' ;
}
else if ( currentTargetState != null && activeArrow != null )
{
var index = ( graph . model . isEdge ( currentTargetState . cell ) || freeSourceEdge == null ) ? firstVertex : freeSourceEdge ;
var geo = sidebar . getDropAndConnectGeometry ( currentTargetState . cell , cells [ index ] , direction , cells ) ;
var geo2 = ( ! graph . model . isEdge ( currentTargetState . cell ) ) ? graph . getCellGeometry ( currentTargetState . cell ) : null ;
var geo3 = graph . getCellGeometry ( cells [ index ] ) ;
var parent = graph . model . getParent ( currentTargetState . cell ) ;
var dx = view . translate . x * view . scale ;
var dy = view . translate . y * view . scale ;
if ( geo2 != null && ! geo2 . relative && graph . model . isVertex ( parent ) )
{
var pState = view . getState ( parent ) ;
dx = pState . x ;
dy = pState . y ;
}
var dx2 = geo3 . x ;
var dy2 = geo3 . y ;
// Ignores geometry of edges
if ( graph . model . isEdge ( cells [ index ] ) )
{
dx2 = 0 ;
dy2 = 0 ;
}
// Shows preview at drop location
this . previewElement . style . left = ( ( geo . x - dx2 ) * view . scale + dx ) + 'px' ;
this . previewElement . style . top = ( ( geo . y - dy2 ) * view . scale + dy ) + 'px' ;
if ( cells . length == 1 )
{
this . previewElement . style . width = ( geo . width * view . scale ) + 'px' ;
this . previewElement . style . height = ( geo . height * view . scale ) + 'px' ;
}
this . previewElement . style . display = '' ;
}
else if ( dragSource . currentHighlight . state != null &&
graph . model . isEdge ( dragSource . currentHighlight . state . cell ) )
{
// Centers drop cells when splitting edges
this . previewElement . style . left = Math . round ( parseInt ( this . previewElement . style . left ) -
bounds . width * view . scale / 2 ) + 'px' ;
this . previewElement . style . top = Math . round ( parseInt ( this . previewElement . style . top ) -
bounds . height * view . scale / 2 ) + 'px' ;
}
else
{
this . previewElement . style . width = this . previewElementWidth ;
this . previewElement . style . height = this . previewElementHeight ;
this . previewElement . style . display = '' ;
}
}
} ;
var startTime = new Date ( ) . getTime ( ) ;
var timeOnTarget = 0 ;
var prev = null ;
// Gets source cell style to compare shape below
var sourceCellStyle = this . editorUi . editor . graph . getCellStyle ( cells [ 0 ] ) ;
// Allows drop into cell only if target is a valid root
dragSource . getDropTarget = mxUtils . bind ( this , function ( graph , x , y , evt )
{
// Alt means no targets at all
// LATER: Show preview where result will go
var cell = ( ! mxEvent . isAltDown ( evt ) && cells != null ) ? graph . getCellAt ( x , y ) : null ;
// Uses connectable parent vertex if one exists
if ( cell != null && ! this . graph . isCellConnectable ( cell ) )
{
var parent = this . graph . getModel ( ) . getParent ( cell ) ;
if ( this . graph . getModel ( ) . isVertex ( parent ) && this . graph . isCellConnectable ( parent ) )
{
cell = parent ;
}
}
// Ignores locked cells
if ( graph . isCellLocked ( cell ) )
{
cell = null ;
}
var state = graph . view . getState ( cell ) ;
activeArrow = null ;
var bbox = null ;
// Time on target
if ( prev != state )
{
prev = state ;
startTime = new Date ( ) . getTime ( ) ;
timeOnTarget = 0 ;
if ( this . updateThread != null )
{
window . clearTimeout ( this . updateThread ) ;
}
if ( state != null )
{
this . updateThread = window . setTimeout ( function ( )
{
if ( activeArrow == null )
{
prev = state ;
dragSource . getDropTarget ( graph , x , y , evt ) ;
}
} , this . dropTargetDelay + 10 ) ;
}
}
else
{
timeOnTarget = new Date ( ) . getTime ( ) - startTime ;
}
// Shift means disabled, delayed on cells with children, shows after this.dropTargetDelay, hides after 2500ms
if ( timeOnTarget < 2500 && state != null && ! mxEvent . isShiftDown ( evt ) &&
// If shape is equal or target has no stroke then add long delay except for images
( ( ( mxUtils . getValue ( state . style , mxConstants . STYLE _SHAPE ) != mxUtils . getValue ( sourceCellStyle , mxConstants . STYLE _SHAPE ) &&
mxUtils . getValue ( state . style , mxConstants . STYLE _STROKECOLOR , mxConstants . NONE ) != mxConstants . NONE ) ||
mxUtils . getValue ( sourceCellStyle , mxConstants . STYLE _SHAPE ) == 'image' ) ||
timeOnTarget > 1500 || graph . model . isEdge ( state . cell ) ) && ( timeOnTarget > this . dropTargetDelay ) &&
( ( graph . model . isVertex ( state . cell ) && firstVertex != null ) ||
( graph . model . isEdge ( state . cell ) && graph . model . isEdge ( cells [ 0 ] ) ) ) )
{
currentStyleTarget = state ;
var tmp = ( graph . model . isEdge ( state . cell ) ) ? graph . view . getPoint ( state ) :
new mxPoint ( state . getCenterX ( ) , state . getCenterY ( ) ) ;
tmp = new mxRectangle ( tmp . x - this . refreshTarget . width / 2 , tmp . y - this . refreshTarget . height / 2 ,
this . refreshTarget . width , this . refreshTarget . height ) ;
styleTarget . style . left = Math . floor ( tmp . x ) + 'px' ;
styleTarget . style . top = Math . floor ( tmp . y ) + 'px' ;
if ( styleTargetParent == null )
{
graph . container . appendChild ( styleTarget ) ;
styleTargetParent = styleTarget . parentNode ;
}
checkArrow ( x , y , tmp , styleTarget ) ;
}
// Does not reset on ignored edges
else if ( currentStyleTarget == null || ! mxUtils . contains ( currentStyleTarget , x , y ) ||
( timeOnTarget > 1500 && ! mxEvent . isShiftDown ( evt ) ) )
{
currentStyleTarget = null ;
if ( styleTargetParent != null )
{
styleTarget . parentNode . removeChild ( styleTarget ) ;
styleTargetParent = null ;
}
}
else if ( currentStyleTarget != null && styleTargetParent != null )
{
// Sets active Arrow as side effect
var tmp = ( graph . model . isEdge ( currentStyleTarget . cell ) ) ? graph . view . getPoint ( currentStyleTarget ) : new mxPoint ( currentStyleTarget . getCenterX ( ) , currentStyleTarget . getCenterY ( ) ) ;
tmp = new mxRectangle ( tmp . x - this . refreshTarget . width / 2 , tmp . y - this . refreshTarget . height / 2 ,
this . refreshTarget . width , this . refreshTarget . height ) ;
checkArrow ( x , y , tmp , styleTarget ) ;
}
// Checks if inside bounds
if ( activeTarget && currentTargetState != null && ! mxEvent . isAltDown ( evt ) && activeArrow == null )
{
// LATER: Use hit-detection for edges
bbox = mxRectangle . fromRectangle ( currentTargetState ) ;
if ( graph . model . isEdge ( currentTargetState . cell ) )
{
var pts = currentTargetState . absolutePoints ;
if ( roundSource . parentNode != null )
{
var p0 = pts [ 0 ] ;
bbox . add ( checkArrow ( x , y , new mxRectangle ( p0 . x - this . roundDrop . width / 2 ,
p0 . y - this . roundDrop . height / 2 , this . roundDrop . width , this . roundDrop . height ) , roundSource ) ) ;
}
if ( roundTarget . parentNode != null )
{
var pe = pts [ pts . length - 1 ] ;
bbox . add ( checkArrow ( x , y , new mxRectangle ( pe . x - this . roundDrop . width / 2 ,
pe . y - this . roundDrop . height / 2 ,
this . roundDrop . width , this . roundDrop . height ) , roundTarget ) ) ;
}
}
else
{
var bds = mxRectangle . fromRectangle ( currentTargetState ) ;
// Uses outer bounding box to take rotation into account
if ( currentTargetState . shape != null && currentTargetState . shape . boundingBox != null )
{
bds = mxRectangle . fromRectangle ( currentTargetState . shape . boundingBox ) ;
}
bds . grow ( this . graph . tolerance ) ;
bds . grow ( HoverIcons . prototype . arrowSpacing ) ;
var handler = this . graph . selectionCellsHandler . getHandler ( currentTargetState . cell ) ;
if ( handler != null )
{
bds . x -= handler . horizontalOffset / 2 ;
bds . y -= handler . verticalOffset / 2 ;
bds . width += handler . horizontalOffset ;
bds . height += handler . verticalOffset ;
// Adds bounding box of rotation handle to avoid overlap
if ( handler . rotationShape != null && handler . rotationShape . node != null &&
handler . rotationShape . node . style . visibility != 'hidden' &&
handler . rotationShape . node . style . display != 'none' &&
handler . rotationShape . boundingBox != null )
{
bds . add ( handler . rotationShape . boundingBox ) ;
}
}
bbox . add ( checkArrow ( x , y , new mxRectangle ( currentTargetState . getCenterX ( ) - this . triangleUp . width / 2 ,
bds . y - this . triangleUp . height , this . triangleUp . width , this . triangleUp . height ) , arrowUp ) ) ;
bbox . add ( checkArrow ( x , y , new mxRectangle ( bds . x + bds . width ,
currentTargetState . getCenterY ( ) - this . triangleRight . height / 2 ,
this . triangleRight . width , this . triangleRight . height ) , arrowRight ) ) ;
bbox . add ( checkArrow ( x , y , new mxRectangle ( currentTargetState . getCenterX ( ) - this . triangleDown . width / 2 ,
bds . y + bds . height , this . triangleDown . width , this . triangleDown . height ) , arrowDown ) ) ;
bbox . add ( checkArrow ( x , y , new mxRectangle ( bds . x - this . triangleLeft . width ,
currentTargetState . getCenterY ( ) - this . triangleLeft . height / 2 ,
this . triangleLeft . width , this . triangleLeft . height ) , arrowLeft ) ) ;
}
// Adds tolerance
if ( bbox != null )
{
bbox . grow ( 10 ) ;
}
}
direction = mxConstants . DIRECTION _NORTH ;
if ( activeArrow == arrowRight )
{
direction = mxConstants . DIRECTION _EAST ;
}
else if ( activeArrow == arrowDown || activeArrow == roundTarget )
{
direction = mxConstants . DIRECTION _SOUTH ;
}
else if ( activeArrow == arrowLeft )
{
direction = mxConstants . DIRECTION _WEST ;
}
if ( currentStyleTarget != null && activeArrow == styleTarget )
{
state = currentStyleTarget ;
}
var validTarget = ( firstVertex == null || graph . isCellConnectable ( cells [ firstVertex ] ) ) &&
( ( graph . model . isEdge ( cell ) && firstVertex != null ) ||
( graph . model . isVertex ( cell ) && graph . isCellConnectable ( cell ) ) ) ;
// Drop arrows shown after this.dropTargetDelay, hidden after 5 secs, switches arrows after 500ms
if ( ( currentTargetState != null && timeOnTarget >= 5000 ) ||
( currentTargetState != state &&
( bbox == null || ! mxUtils . contains ( bbox , x , y ) ||
( timeOnTarget > 500 && activeArrow == null && validTarget ) ) ) )
{
activeTarget = false ;
currentTargetState = ( ( timeOnTarget < 5000 && timeOnTarget > this . dropTargetDelay ) || graph . model . isEdge ( cell ) ) ? state : null ;
if ( currentTargetState != null && validTarget )
{
var elts = [ roundSource , roundTarget , arrowUp , arrowRight , arrowDown , arrowLeft ] ;
for ( var i = 0 ; i < elts . length ; i ++ )
{
if ( elts [ i ] . parentNode != null )
{
elts [ i ] . parentNode . removeChild ( elts [ i ] ) ;
}
}
if ( graph . model . isEdge ( cell ) )
{
var pts = state . absolutePoints ;
if ( pts != null )
{
var p0 = pts [ 0 ] ;
var pe = pts [ pts . length - 1 ] ;
var tol = graph . tolerance ;
var box = new mxRectangle ( x - tol , y - tol , 2 * tol , 2 * tol ) ;
roundSource . style . left = Math . floor ( p0 . x - this . roundDrop . width / 2 ) + 'px' ;
roundSource . style . top = Math . floor ( p0 . y - this . roundDrop . height / 2 ) + 'px' ;
roundTarget . style . left = Math . floor ( pe . x - this . roundDrop . width / 2 ) + 'px' ;
roundTarget . style . top = Math . floor ( pe . y - this . roundDrop . height / 2 ) + 'px' ;
if ( graph . model . getTerminal ( cell , true ) == null )
{
graph . container . appendChild ( roundSource ) ;
}
if ( graph . model . getTerminal ( cell , false ) == null )
{
graph . container . appendChild ( roundTarget ) ;
}
}
}
else
{
var bds = mxRectangle . fromRectangle ( state ) ;
// Uses outer bounding box to take rotation into account
if ( state . shape != null && state . shape . boundingBox != null )
{
bds = mxRectangle . fromRectangle ( state . shape . boundingBox ) ;
}
bds . grow ( this . graph . tolerance ) ;
bds . grow ( HoverIcons . prototype . arrowSpacing ) ;
var handler = this . graph . selectionCellsHandler . getHandler ( state . cell ) ;
if ( handler != null )
{
bds . x -= handler . horizontalOffset / 2 ;
bds . y -= handler . verticalOffset / 2 ;
bds . width += handler . horizontalOffset ;
bds . height += handler . verticalOffset ;
// Adds bounding box of rotation handle to avoid overlap
if ( handler . rotationShape != null && handler . rotationShape . node != null &&
handler . rotationShape . node . style . visibility != 'hidden' &&
handler . rotationShape . node . style . display != 'none' &&
handler . rotationShape . boundingBox != null )
{
bds . add ( handler . rotationShape . boundingBox ) ;
}
}
arrowUp . style . left = Math . floor ( state . getCenterX ( ) - this . triangleUp . width / 2 ) + 'px' ;
arrowUp . style . top = Math . floor ( bds . y - this . triangleUp . height ) + 'px' ;
arrowRight . style . left = Math . floor ( bds . x + bds . width ) + 'px' ;
arrowRight . style . top = Math . floor ( state . getCenterY ( ) - this . triangleRight . height / 2 ) + 'px' ;
arrowDown . style . left = arrowUp . style . left
arrowDown . style . top = Math . floor ( bds . y + bds . height ) + 'px' ;
arrowLeft . style . left = Math . floor ( bds . x - this . triangleLeft . width ) + 'px' ;
arrowLeft . style . top = arrowRight . style . top ;
if ( state . style [ 'portConstraint' ] != 'eastwest' )
{
graph . container . appendChild ( arrowUp ) ;
graph . container . appendChild ( arrowDown ) ;
}
graph . container . appendChild ( arrowRight ) ;
graph . container . appendChild ( arrowLeft ) ;
}
// Hides handle for cell under mouse
if ( state != null )
{
currentStateHandle = graph . selectionCellsHandler . getHandler ( state . cell ) ;
if ( currentStateHandle != null && currentStateHandle . setHandlesVisible != null )
{
currentStateHandle . setHandlesVisible ( false ) ;
}
}
activeTarget = true ;
}
else
{
var elts = [ roundSource , roundTarget , arrowUp , arrowRight , arrowDown , arrowLeft ] ;
for ( var i = 0 ; i < elts . length ; i ++ )
{
if ( elts [ i ] . parentNode != null )
{
elts [ i ] . parentNode . removeChild ( elts [ i ] ) ;
}
}
}
}
if ( ! activeTarget && currentStateHandle != null )
{
currentStateHandle . setHandlesVisible ( true ) ;
}
// Handles drop target
var target = ( ( ! mxEvent . isAltDown ( evt ) || mxEvent . isShiftDown ( evt ) ) &&
! ( currentStyleTarget != null && activeArrow == styleTarget ) ) ?
mxDragSource . prototype . getDropTarget . apply ( this , arguments ) : null ;
var model = graph . getModel ( ) ;
if ( target != null )
{
if ( activeArrow != null || ! graph . isSplitTarget ( target , cells , evt ) )
{
// Selects parent group as drop target
while ( target != null && ! graph . isValidDropTarget ( target , cells , evt ) && model . isVertex ( model . getParent ( target ) ) )
{
target = model . getParent ( target ) ;
}
if ( graph . view . currentRoot == target || ( ! graph . isValidRoot ( target ) &&
graph . getModel ( ) . getChildCount ( target ) == 0 ) ||
graph . isCellLocked ( target ) || model . isEdge ( target ) )
{
target = null ;
}
}
}
return target ;
} ) ;
dragSource . stopDrag = function ( )
{
mxDragSource . prototype . stopDrag . apply ( this , arguments ) ;
var elts = [ roundSource , roundTarget , styleTarget , arrowUp , arrowRight , arrowDown , arrowLeft ] ;
for ( var i = 0 ; i < elts . length ; i ++ )
{
if ( elts [ i ] . parentNode != null )
{
elts [ i ] . parentNode . removeChild ( elts [ i ] ) ;
}
}
if ( currentTargetState != null && currentStateHandle != null )
{
currentStateHandle . reset ( ) ;
}
currentStateHandle = null ;
currentTargetState = null ;
currentStyleTarget = null ;
styleTargetParent = null ;
activeArrow = null ;
} ;
return dragSource ;
} ;
/ * *
* Adds a handler for inserting the cell with a single click .
* /
Sidebar . prototype . itemClicked = function ( cells , ds , evt , elt )
{
var graph = this . editorUi . editor . graph ;
// Alt+Click inserts and connects
if ( mxEvent . isAltDown ( evt ) )
{
if ( graph . getSelectionCount ( ) == 1 && graph . model . isVertex ( graph . getSelectionCell ( ) ) )
{
var firstVertex = null ;
for ( var i = 0 ; i < cells . length && firstVertex == null ; i ++ )
{
if ( graph . model . isVertex ( cells [ i ] ) )
{
firstVertex = i ;
}
}
if ( firstVertex != null )
{
this . dropAndConnect ( graph . getSelectionCell ( ) , cells , ( mxEvent . isMetaDown ( evt ) || mxEvent . isControlDown ( evt ) ) ?
( mxEvent . isShiftDown ( evt ) ? mxConstants . DIRECTION _WEST : mxConstants . DIRECTION _NORTH ) :
( mxEvent . isShiftDown ( evt ) ? mxConstants . DIRECTION _EAST : mxConstants . DIRECTION _SOUTH ) , firstVertex ) ;
graph . scrollCellToVisible ( graph . getSelectionCell ( ) ) ;
}
}
}
// Shift+Click updates shape
else if ( mxEvent . isShiftDown ( evt ) )
{
if ( ! graph . isSelectionEmpty ( ) )
{
this . updateShapes ( cells [ 0 ] , graph . getSelectionCells ( ) ) ;
graph . scrollCellToVisible ( graph . getSelectionCell ( ) ) ;
}
}
else
{
2016-11-04 22:16:17 +00:00
var pt = graph . getFreeInsertPoint ( ) ;
2016-09-06 14:07:11 +00:00
ds . drop ( graph , evt , null , pt . x , pt . y ) ;
if ( this . editorUi . hoverIcons != null && mxEvent . isTouchEvent ( evt ) )
{
this . editorUi . hoverIcons . update ( graph . view . getState ( graph . getSelectionCell ( ) ) ) ;
}
}
} ;
/ * *
* Adds a handler for inserting the cell with a single click .
* /
Sidebar . prototype . addClickHandler = function ( elt , ds , cells )
{
var graph = this . editorUi . editor . graph ;
var oldMouseUp = ds . mouseUp ;
var first = null ;
mxEvent . addGestureListeners ( elt , function ( evt )
{
first = new mxPoint ( mxEvent . getClientX ( evt ) , mxEvent . getClientY ( evt ) ) ;
} ) ;
ds . mouseUp = mxUtils . bind ( this , function ( evt )
{
if ( ! mxEvent . isPopupTrigger ( evt ) && this . currentGraph == null && first != null )
{
var tol = graph . tolerance ;
if ( Math . abs ( first . x - mxEvent . getClientX ( evt ) ) <= tol &&
Math . abs ( first . y - mxEvent . getClientY ( evt ) ) <= tol )
{
this . itemClicked ( cells , ds , evt , elt ) ;
}
}
oldMouseUp . apply ( ds , arguments ) ;
first = null ;
// Blocks tooltips on this element after single click
this . currentElt = elt ;
} ) ;
} ;
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . createVertexTemplateEntry = function ( style , width , height , value , title , showLabel , showTitle , tags )
{
tags = ( tags != null && tags . length > 0 ) ? tags : title . toLowerCase ( ) ;
return this . addEntry ( tags , mxUtils . bind ( this , function ( )
{
return this . createVertexTemplate ( style , width , height , value , title , showLabel , showTitle ) ;
} ) ) ;
}
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . createVertexTemplate = function ( style , width , height , value , title , showLabel , showTitle , allowCellsInserted )
{
var cells = [ new mxCell ( ( value != null ) ? value : '' , new mxGeometry ( 0 , 0 , width , height ) , style ) ] ;
cells [ 0 ] . vertex = true ;
return this . createVertexTemplateFromCells ( cells , width , height , title , showLabel , showTitle , allowCellsInserted ) ;
} ;
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . createVertexTemplateFromCells = function ( cells , width , height , title , showLabel , showTitle , allowCellsInserted )
{
return this . createItem ( cells , title , showLabel , showTitle , width , height , allowCellsInserted ) ;
} ;
/ * *
*
* /
Sidebar . prototype . createEdgeTemplateEntry = function ( style , width , height , value , title , showLabel , tags , allowCellsInserted )
{
tags = ( tags != null && tags . length > 0 ) ? tags : title . toLowerCase ( ) ;
return this . addEntry ( tags , mxUtils . bind ( this , function ( )
{
return this . createEdgeTemplate ( style , width , height , value , title , showLabel , allowCellsInserted ) ;
} ) ) ;
} ;
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . createEdgeTemplate = function ( style , width , height , value , title , showLabel , allowCellsInserted )
{
var cell = new mxCell ( ( value != null ) ? value : '' , new mxGeometry ( 0 , 0 , width , height ) , style ) ;
cell . geometry . setTerminalPoint ( new mxPoint ( 0 , height ) , true ) ;
cell . geometry . setTerminalPoint ( new mxPoint ( width , 0 ) , false ) ;
cell . geometry . relative = true ;
cell . edge = true ;
return this . createEdgeTemplateFromCells ( [ cell ] , width , height , title , showLabel , allowCellsInserted ) ;
} ;
/ * *
* Creates a drop handler for inserting the given cells .
* /
Sidebar . prototype . createEdgeTemplateFromCells = function ( cells , width , height , title , showLabel , allowCellsInserted )
{
return this . createItem ( cells , title , showLabel , true , width , height , allowCellsInserted ) ;
} ;
/ * *
* Adds the given palette .
* /
Sidebar . prototype . addPaletteFunctions = function ( id , title , expanded , fns )
{
this . addPalette ( id , title , expanded , mxUtils . bind ( this , function ( content )
{
for ( var i = 0 ; i < fns . length ; i ++ )
{
content . appendChild ( fns [ i ] ( content ) ) ;
}
} ) ) ;
} ;
/ * *
* Adds the given palette .
* /
Sidebar . prototype . addPalette = function ( id , title , expanded , onInit )
{
var elt = this . createTitle ( title ) ;
this . container . appendChild ( elt ) ;
var div = document . createElement ( 'div' ) ;
div . className = 'geSidebar' ;
// Disables built-in pan and zoom in IE10 and later
if ( mxClient . IS _POINTER )
{
div . style . touchAction = 'none' ;
}
// Shows tooltip if mouse over background
mxEvent . addListener ( div , 'mousemove' , mxUtils . bind ( this , function ( evt )
{
if ( mxEvent . getSource ( evt ) == div )
{
div . setAttribute ( 'title' , mxResources . get ( 'sidebarTooltip' ) ) ;
}
else
{
div . removeAttribute ( 'title' ) ;
}
} ) ) ;
if ( expanded )
{
onInit ( div ) ;
onInit = null ;
}
else
{
div . style . display = 'none' ;
}
this . addFoldingHandler ( elt , div , onInit ) ;
var outer = document . createElement ( 'div' ) ;
outer . appendChild ( div ) ;
this . container . appendChild ( outer ) ;
// Keeps references to the DOM nodes
if ( id != null )
{
this . palettes [ id ] = [ elt , outer ] ;
}
return div ;
} ;
/ * *
* Create the given title element .
* /
Sidebar . prototype . addFoldingHandler = function ( title , content , funct )
{
var initialized = false ;
// Avoids mixed content warning in IE6-8
if ( ! mxClient . IS _IE || document . documentMode >= 8 )
{
title . style . backgroundImage = ( content . style . display == 'none' ) ?
'url(\'' + this . collapsedImage + '\')' : 'url(\'' + this . expandedImage + '\')' ;
}
title . style . backgroundRepeat = 'no-repeat' ;
title . style . backgroundPosition = '0% 50%' ;
mxEvent . addListener ( title , 'click' , mxUtils . bind ( this , function ( evt )
{
if ( content . style . display == 'none' )
{
if ( ! initialized )
{
initialized = true ;
if ( funct != null )
{
// Wait cursor does not show up on Mac
title . style . cursor = 'wait' ;
var prev = title . innerHTML ;
title . innerHTML = mxResources . get ( 'loading' ) + '...' ;
window . setTimeout ( function ( )
{
var fo = mxClient . NO _FO ;
mxClient . NO _FO = Editor . prototype . originalNoForeignObject ;
funct ( content ) ;
mxClient . NO _FO = fo ;
content . style . display = 'block' ;
title . style . cursor = '' ;
title . innerHTML = prev ;
} , 0 ) ;
}
else
{
content . style . display = 'block' ;
}
}
else
{
content . style . display = 'block' ;
}
title . style . backgroundImage = 'url(\'' + this . expandedImage + '\')' ;
}
else
{
title . style . backgroundImage = 'url(\'' + this . collapsedImage + '\')' ;
content . style . display = 'none' ;
}
mxEvent . consume ( evt ) ;
} ) ) ;
} ;
/ * *
* Removes the palette for the given ID .
* /
Sidebar . prototype . removePalette = function ( id )
{
var elts = this . palettes [ id ] ;
if ( elts != null )
{
this . palettes [ id ] = null ;
for ( var i = 0 ; i < elts . length ; i ++ )
{
this . container . removeChild ( elts [ i ] ) ;
}
return true ;
}
return false ;
} ;
/ * *
* Adds the given image palette .
* /
Sidebar . prototype . addImagePalette = function ( id , title , prefix , postfix , items , titles , tags )
{
var showTitles = titles != null ;
var fns = [ ] ;
for ( var i = 0 ; i < items . length ; i ++ )
{
( mxUtils . bind ( this , function ( item , title , tmpTags )
{
if ( tmpTags == null )
{
var slash = item . lastIndexOf ( '/' ) ;
var dot = item . lastIndexOf ( '.' ) ;
tmpTags = item . substring ( ( slash >= 0 ) ? slash + 1 : 0 , ( dot >= 0 ) ? dot : item . length ) . replace ( /[-_]/g , ' ' ) ;
}
fns . push ( this . createVertexTemplateEntry ( 'image;html=1;labelBackgroundColor=#ffffff;image=' + prefix + item + postfix ,
this . defaultImageWidth , this . defaultImageHeight , '' , title , title != null , null , this . filterTags ( tmpTags ) ) ) ;
} ) ) ( items [ i ] , ( titles != null ) ? titles [ i ] : null , ( tags != null ) ? tags [ items [ i ] ] : null ) ;
}
this . addPaletteFunctions ( id , title , false , fns ) ;
} ;
/ * *
* Creates the array of tags for the given stencil . Duplicates are allowed and will be filtered out later .
* /
Sidebar . prototype . getTagsForStencil = function ( packageName , stencilName , moreTags )
{
var tags = packageName . split ( '.' ) ;
for ( var i = 1 ; i < tags . length ; i ++ )
{
tags [ i ] = tags [ i ] . replace ( /_/g , ' ' )
}
tags . push ( stencilName . replace ( /_/g , ' ' ) ) ;
if ( moreTags != null )
{
tags . push ( moreTags ) ;
}
return tags . slice ( 1 , tags . length ) ;
} ;
/ * *
* Adds the given stencil palette .
* /
2016-10-27 15:53:53 +00:00
Sidebar . prototype . addStencilPalette = function ( id , title , stencilFile , style , ignore , onInit , scale , tags , customFns )
2016-09-06 14:07:11 +00:00
{
scale = ( scale != null ) ? scale : 1 ;
2016-10-27 15:53:53 +00:00
2016-09-06 14:07:11 +00:00
if ( this . addStencilsToIndex )
{
// LATER: Handle asynchronous loading dependency
var fns = [ ] ;
2016-10-27 15:53:53 +00:00
if ( customFns != null )
{
for ( var i = 0 ; i < customFns . length ; i ++ )
{
fns . push ( customFns [ i ] ) ;
}
}
2016-09-06 14:07:11 +00:00
mxStencilRegistry . loadStencilSet ( stencilFile , mxUtils . bind ( this , function ( packageName , stencilName , displayName , w , h )
{
if ( ignore == null || mxUtils . indexOf ( ignore , stencilName ) < 0 )
{
var tmp = this . getTagsForStencil ( packageName , stencilName ) ;
var tmpTags = ( tags != null ) ? tags [ stencilName ] : null ;
if ( tmpTags != null )
{
tmp . push ( tmpTags ) ;
}
fns . push ( this . createVertexTemplateEntry ( 'shape=' + packageName + stencilName . toLowerCase ( ) + style ,
Math . round ( w * scale ) , Math . round ( h * scale ) , '' , stencilName . replace ( /_/g , ' ' ) , null , null ,
this . filterTags ( tmp . join ( ' ' ) ) ) ) ;
}
} ) , true , true ) ;
this . addPaletteFunctions ( id , title , false , fns ) ;
}
else
{
this . addPalette ( id , title , false , mxUtils . bind ( this , function ( content )
{
if ( style == null )
{
style = '' ;
}
if ( onInit != null )
{
onInit . call ( this , content ) ;
}
2016-10-27 15:53:53 +00:00
if ( customFns != null )
{
for ( var i = 0 ; i < customFns . length ; i ++ )
{
customFns [ i ] ( content ) ;
}
}
2016-09-06 14:07:11 +00:00
mxStencilRegistry . loadStencilSet ( stencilFile , mxUtils . bind ( this , function ( packageName , stencilName , displayName , w , h )
{
if ( ignore == null || mxUtils . indexOf ( ignore , stencilName ) < 0 )
{
content . appendChild ( this . createVertexTemplate ( 'shape=' + packageName + stencilName . toLowerCase ( ) + style ,
Math . round ( w * scale ) , Math . round ( h * scale ) , '' , stencilName . replace ( /_/g , ' ' ) , true ) ) ;
}
} ) , true ) ;
} ) ) ;
}
} ;
/ * *
* Adds the given stencil palette .
* /
Sidebar . prototype . destroy = function ( )
{
if ( this . graph != null )
{
if ( this . graph . container != null && this . graph . container . parentNode != null )
{
this . graph . container . parentNode . removeChild ( this . graph . container ) ;
}
this . graph . destroy ( ) ;
this . graph = null ;
}
if ( this . pointerUpHandler != null )
{
mxEvent . removeListener ( document , ( mxClient . IS _POINTER ) ? 'pointerup' : 'mouseup' , this . pointerUpHandler ) ;
this . pointerUpHandler = null ;
}
if ( this . pointerDownHandler != null )
{
mxEvent . removeListener ( document , ( mxClient . IS _POINTER ) ? 'pointerdown' : 'mousedown' , this . pointerDownHandler ) ;
this . pointerDownHandler = null ;
}
if ( this . pointerMoveHandler != null )
{
mxEvent . removeListener ( document , ( mxClient . IS _POINTER ) ? 'pointermove' : 'mousemove' , this . pointerMoveHandler ) ;
this . pointerMoveHandler = null ;
}
if ( this . pointerOutHandler != null )
{
mxEvent . removeListener ( document , ( mxClient . IS _POINTER ) ? 'pointerout' : 'mouseout' , this . pointerOutHandler ) ;
this . pointerOutHandler = null ;
}
} ;