| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237 |
- /**
- * Copyright (c) 2006-2017, JGraph Ltd
- * Copyright (c) 2006-2017, Gaudenz Alder
- */
- (function()
- {
- if (typeof html4 !== 'undefined')
- {
- /**
- * Enables paste from Lucidchart
- */
- html4.ATTRIBS['span::data-lucid-content'] = 0;
- html4.ATTRIBS['span::data-lucid-type'] = 0;
-
- /**
- * Enables custom fonts in labels.
- */
- html4.ATTRIBS['font::data-font-src'] = 0;
- }
-
- /**
- * Definitions for sketch font styles.
- */
- Editor.sketchFontFamily = 'Architects Daughter';
- Editor.sketchFontSource = 'https%3A%2F%2Ffonts.googleapis.com%2Fcss%3Ffamily%3DArchitects%2BDaughter';
- Editor.sketchFonts = [{'fontFamily': Editor.sketchFontFamily, 'fontUrl': decodeURIComponent(Editor.sketchFontSource)}];
- Editor.sketchDefaultCurveFitting = '1';
- Editor.sketchDefaultJiggle = '2';
-
- /**
- * Icons for new UI style exported from https://fonts.google.com/icons (FFill: 0 Weight: 300 Grade: 0 Optical size: 48).
- */
- Editor.thinCommentImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTEyLjUgMjcuNWgyM3YtMi4yNWgtMjNabTAtNi4zNWgyM3YtMi4zaC0yM1ptMC02LjRoMjNWMTIuNWgtMjNaTTQzIDQyLjEgMzUuOSAzNWgtMjhxLTEuMTUgMC0yLjAyNS0uODc1VDUgMzIuMVY3LjlxMC0xLjE1Ljg3NS0yLjAyNVQ3LjkgNWgzMi4ycTEuMiAwIDIuMDUuODc1UTQzIDYuNzUgNDMgNy45Wk03LjI1IDcuOXYyNC44NUgzNi45bDMuODUgMy44NVY3LjlxMC0uMy0uMTc1LS40NzVUNDAuMSA3LjI1SDcuOXEtLjMgMC0uNDc1LjE3NVQ3LjI1IDcuOVptMCAwdjI4LjdWNy4yNSA3LjlaIi8+PC9zdmc+';
- Editor.thinDesignImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0ibTM2LjUgMTguMzUtNi44NS02Ljg1IDMuMjUtMy4ycS44NS0uODUgMi4wMjUtLjg1IDEuMTc1IDAgMi4wMjUuODVsMi43NSAyLjhxLjg1LjguODUgMnQtLjg1IDJaTTYuODUgNDEuMXYtNi45bDkuNTUtOS41NUw1LjUgMTMuN2w3LjgtNy44NSAxMSAxMSA1LjM1LTUuMzUgNi44NSA2Ljg1LTUuMzUgNS4zNSAxMSAxMS03Ljg1IDcuNzUtMTAuOTUtMTAuOS05LjYgOS41NVptMTEuMi0xOC4wNSA0LjY1LTQuNi00LjA1LTQuMDUtMi4zNSAyLjQtMS42LTEuNiAyLjM1LTIuNC0zLjc1LTMuNzUtNC42IDQuNjVabTE2LjIgMTYuMjUgNC42NS00LjY1LTMuNzUtMy43NS0yLjQgMi40LTEuNi0xLjYgMi40LTIuNC00LjA1LTQtNC42IDQuNjVabS0yNS4xLS40NWgzLjZsMjAuNS0yMC40NS0zLjY1LTMuNjVMOS4xNSAzNS4yWiIvPjwvc3ZnPg==';
- Editor.thinGestureImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI4IDQxLjZxLTIuNSAwLTQuMDc1LTEuNjUtMS41NzUtMS42NS0xLjU3NS00LjM1IDAtMi4zNSAxLjEtNC4xIDEuMS0xLjc1IDIuODI1LTIuOTUgMS43MjUtMS4yIDMuNy0xLjgyNVQzMy42NSAyNnEtLjE1LTMuMDUtMS4yNS00LjQyNVQyOS4yNSAyMC4ycS0yLjMgMC00LjE3NSAxLjV0LTQuNDI1IDQuOTVxLTIuNjUgMy40NS00LjYgNS4xLTEuOTUgMS42NS00LjM1IDEuNjUtMi4xIDAtMy42NS0xLjM3NVE2LjUgMzAuNjUgNi41IDI3LjQ1cTAtMiAxLjItNC41MjUgMS4yLTIuNTI1IDMuNS02LjE3NSAxLjMtMS44NSAxLjk3NS0zLjIyNXQuNjc1LTIuMzc1cTAtLjctLjM1LTEuMDUtLjM1LS4zNS0xLjA1LS4zNS0uNzUgMC0xLjYuNDc1UTEwIDEwLjcgOSAxMS45TDcgOS44cTEuMzUtMS41NSAyLjcyNS0yLjI3NVExMS4xIDYuOCAxMi41IDYuOHExLjkgMCAzLjEgMS4yNSAxLjIgMS4yNSAxLjIgMy4yIDAgMS44LS45NzUgMy41NzVRMTQuODUgMTYuNiAxMy4wNSAxOS4zcS0xLjk1IDIuOTUtMi43NzUgNC43dC0uODI1IDMuNjVxMCAxLjYuNzUgMi4yLjc1LjYgMS43NS42IDEuNDUgMCAyLjgyNS0xLjMgMS4zNzUtMS4zIDMuNTc1LTQuMTUgMi45NS0zLjggNS41MjUtNS43NzUgMi41NzUtMS45NzUgNS42NzUtMS45NzUgMy4xNSAwIDUgMi4zMjVUMzYuNiAyNS45aDQuOXYyLjk1aC00LjlxLS40NSA3LjM1LTMuMiAxMC4wNS0yLjc1IDIuNy01LjQgMi43Wm0uMS0yLjk1cTEuODUgMCAzLjY1LTIuMjUgMS44LTIuMjUgMi03LjUtMi44LjE1LTUuNjI1IDIuMDI1VDI1LjMgMzUuOXEwIDEuMy43NSAyLjAyNS43NS43MjUgMi4wNS43MjVaIi8+PC9zdmc+';
- Editor.thinShapesImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI5Ljc1IDI5Ljc1Wm0tMTMgNS40NXEuNC4wNS43NS4wNUgxOS4wNXY0LjY1cTAgLjI1LjE3NS40MjV0LjQyNS4xNzVIMzkuOXEuMjUgMCAuNDI1LS4xNzV0LjE3NS0uNDI1VjE5LjY1cTAtLjI1LS4xNzUtLjQyNXQtLjQyNS0uMTc1aC00LjY1VjE3LjVxMC0uMzUtLjA1LS43NWg0LjdxMS4xNSAwIDIgLjg3NS44NS44NzUuODUgMi4wMjVWMzkuOXEwIDEuMTUtLjg1IDItLjg1Ljg1LTIgLjg1SDE5LjY1cS0xLjE1IDAtMi4wMjUtLjg1dC0uODc1LTJabTEuNS0zLjk1cS01LjQ1IDAtOS4yMjUtMy44LTMuNzc1LTMuOC0zLjc3NS05LjIgMC01LjQ1IDMuNzc1LTkuMjI1UTEyLjggNS4yNSAxOC4yNSA1LjI1cTUuNCAwIDkuMiAzLjc3NSAzLjggMy43NzUgMy44IDkuMjI1IDAgNS40LTMuOCA5LjItMy44IDMuOC05LjIgMy44Wm0tLjA1LTIuM3E0LjQ1IDAgNy42LTMuMTI1IDMuMTUtMy4xMjUgMy4xNS03LjU3NXQtMy4xMjUtNy42UTIyLjcgNy41IDE4LjI1IDcuNXQtNy42IDMuMTI1UTcuNSAxMy43NSA3LjUgMTguMnQzLjEyNSA3LjZxMy4xMjUgMy4xNSA3LjU3NSAzLjE1Wm0uMDUtMTAuN1oiLz48L3N2Zz4=';
- Editor.thinUndoImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTE0LjQgMzcuNXYtMi4yNWgxMy45cTMuNiAwIDYuMjI1LTIuMzc1UTM3LjE1IDMwLjUgMzcuMTUgMjYuOXEwLTMuNTUtMi42MjUtNS45LTIuNjI1LTIuMzUtNi4yMjUtMi4zNUgxMi45NUwxOSAyNC43bC0xLjYgMS42LTguOC04LjggOC44LTguOCAxLjYgMS42LTYuMDUgNi4wNWgxNS4zcTQuNTUgMCA3Ljg1IDMuMDV0My4zIDcuNXEwIDQuNS0zLjMgNy41NXQtNy44NSAzLjA1WiIvPjwvc3ZnPg==';
- Editor.thinRedoImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTE5Ljc1IDM3LjVxLTQuNTUgMC03Ljg1LTMuMDVUOC42IDI2LjlxMC00LjQ1IDMuMy03LjV0Ny44NS0zLjA1aDE1LjNMMjkgMTAuM2wxLjYtMS42IDguOCA4LjgtOC44IDguOC0xLjYtMS42IDYuMDUtNi4wNUgxOS43cS0zLjYgMC02LjIyNSAyLjM1LTIuNjI1IDIuMzUtMi42MjUgNS45IDAgMy42IDIuNjI1IDUuOTc1UTE2LjEgMzUuMjUgMTkuNyAzNS4yNWgxMy45djIuMjVaIi8+PC9zdmc+';
- Editor.thinDoubleArrowRightImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0ibTEzIDM1LjMtMS42LTEuNiA5Ljc1LTkuNzUtOS43NS05LjcgMS42LTEuNiAxMS4zNSAxMS4zWm0xMi4zIDAtMS42LTEuNiA5Ljc1LTkuNzUtOS43NS05LjcgMS42LTEuNiAxMS4zIDExLjNaIi8+PC9zdmc+';
- Editor.thinNoteImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHZpZXdCb3g9IjAgOTYgOTYwIDk2MCIgd2lkdGg9IjQ4Ij48cGF0aCBkPSJNMjM3LjY5NCA5NTUuOTk5cS0yMy41MjkgMC00MC42MTEtMTcuMDgyLTE3LjA4Mi0xNy4wODItMTcuMDgyLTQwLjYxMVYyNTMuNjk0cTAtMjMuNTI5IDE3LjA4Mi00MC42MTEgMTcuMDgyLTE3LjA4MiA0MC42MTEtMTcuMDgyaDM0Ny41MzdsMTk0Ljc2OCAxOTQuNzY4djUwNy41MzdxMCAyMy41MjktMTcuMDgyIDQwLjYxMS0xNy4wODIgMTcuMDgyLTQwLjYxMSAxNy4wODJIMjM3LjY5NFpNNTYyLjUzOSA0MTEuMjNWMjQxLjM4NUgyMzcuNjk0cS00LjYxNiAwLTguNDYzIDMuODQ2LTMuODQ2IDMuODQ3LTMuODQ2IDguNDYzdjY0NC42MTJxMCA0LjYxNiAzLjg0NiA4LjQ2MyAzLjg0NyAzLjg0NiA4LjQ2MyAzLjg0Nmg0ODQuNjEycTQuNjE2IDAgOC40NjMtMy44NDYgMy44NDYtMy44NDcgMy44NDYtOC40NjNWNDExLjIzSDU2Mi41MzlaTTIyNS4zODUgMjQxLjM4NVY0MTEuMjMgMjQxLjM4NXY2NjkuMjNWMjQxLjM4NVoiLz48L3N2Zz4=';
- Editor.thinTableImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTcgNDFWN2gzNHYzNFptMi4yNS0yMy42aDI5LjVWOS4yNUg5LjI1Wm0xMC42IDEwLjdoOC4zdi04LjRoLTguM1ptMCAxMC42NWg4LjN2LTguNGgtOC4zWk05LjI1IDI4LjFoOC4zNXYtOC40SDkuMjVabTIxLjE1IDBoOC4zNXYtOC40SDMwLjRaTTkuMjUgMzguNzVoOC4zNXYtOC40SDkuMjVabTIxLjE1IDBoOC4zNXYtOC40SDMwLjRaIi8+PC9zdmc+';
- Editor.thinAddCircleImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTIzIDMzLjVoMi4yNXYtOC4yaDguMjVWMjNoLTguMjV2LTguNUgyM1YyM2gtOC41djIuM0gyM1ptMSA5LjVxLTMuOTUgMC03LjQtMS41dC02LjAyNS00LjA3NVE4IDM0Ljg1IDYuNSAzMS40VDUgMjRxMC0zLjk1IDEuNS03LjQyNVE4IDEzLjEgMTAuNTc1IDEwLjU1IDEzLjE1IDggMTYuNiA2LjVUMjQgNXEzLjk1IDAgNy40MjUgMS41UTM0LjkgOCAzNy40NSAxMC41NSA0MCAxMy4xIDQxLjUgMTYuNTc1IDQzIDIwLjA1IDQzIDI0cTAgMy45NS0xLjUgNy40dC00LjA1IDYuMDI1UTM0LjkgNDAgMzEuNDI1IDQxLjUgMjcuOTUgNDMgMjQgNDNabS4wNS0yLjI1cTYuOTUgMCAxMS44MjUtNC45IDQuODc1LTQuOSA0Ljg3NS0xMS45IDAtNi45NS00Ljg3NS0xMS44MjVRMzEgNy4yNSAyNCA3LjI1cS02Ljk1IDAtMTEuODUgNC44NzVRNy4yNSAxNyA3LjI1IDI0cTAgNi45NSA0LjkgMTEuODUgNC45IDQuOSAxMS45IDQuOVpNMjQgMjRaIi8+PC9zdmc+';
- Editor.thinArrowLeftImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI4LjA1IDM1LjMgMTYuNyAyMy45NSAyOC4wNSAxMi42bDEuNiAxLjY1LTkuNyA5LjcgOS43IDkuNzVaIi8+PC9zdmc+';
- Editor.thinArrowRightImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0ibTE4Ljc1IDM1LjMtMS42LTEuNiA5LjctOS43NS05LjctOS43IDEuNi0xLjY1TDMwLjEgMjMuOTVaIi8+PC9zdmc+';
- Editor.thinVerticalDotsImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI0LjA1IDQxLjdxLTEuMjUgMC0yLjEyNS0uODc1dC0uODc1LTIuMDc1cTAtMS4yLjg3NS0yLjEuODc1LS45IDIuMDc1LS45IDEuMjUgMCAyLjEuOS44NS45Ljg1IDIuMSAwIDEuMi0uODUgMi4wNzUtLjg1Ljg3NS0yLjA1Ljg3NVptMC0xNC43NXEtMS4yNSAwLTIuMTI1LS44NzVUMjEuMDUgMjRxMC0xLjI1Ljg3NS0yLjEuODc1LS44NSAyLjA3NS0uODUgMS4yNSAwIDIuMS44NS44NS44NS44NSAyLjA1IDAgMS4yNS0uODUgMi4xMjV0LTIuMDUuODc1Wm0wLTE0LjdxLTEuMjUgMC0yLjEyNS0uODc1VDIxLjA1IDkuMjVxMC0xLjI1Ljg3NS0yLjEyNVQyNCA2LjI1cTEuMjUgMCAyLjEuODc1Ljg1Ljg3NS44NSAyLjEyNXQtLjg1IDIuMTI1cS0uODUuODc1LTIuMDUuODc1WiIvPjwvc3ZnPg==';
- Editor.thinDeleteImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTEzLjkgNDFxLTEuMTUgMC0yLS44NS0uODUtLjg1LS44NS0yLjA1VjEwLjlIOVY4LjY1aDguNTV2LTEuNGgxMi45djEuNEgzOXYyLjI1aC0yLjA1djI3LjJxMCAxLjItLjg1IDIuMDUtLjg1Ljg1LTIgLjg1Wm0yMC44LTMwLjFIMTMuM3YyNy4ycTAgLjMuMTc1LjQ3NXQuNDI1LjE3NWgyMC4ycS4yIDAgLjQtLjJ0LjItLjQ1Wk0xOS4wNSAzNC41aDIuM1YxNS4xaC0yLjNabTcuNiAwaDIuM1YxNS4xaC0yLjNaTTEzLjMgMTAuOXYyNy44NVYzOC4xWiIvPjwvc3ZnPg==';
- Editor.thinLightImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI0IDMwLjc1cTIuOCAwIDQuNzc1LTEuOTc1UTMwLjc1IDI2LjggMzAuNzUgMjRxMC0yLjgtMS45NzUtNC43NzVRMjYuOCAxNy4yNSAyNCAxNy4yNXEtMi44IDAtNC43NzUgMS45NzVRMTcuMjUgMjEuMiAxNy4yNSAyNHEwIDIuOCAxLjk3NSA0Ljc3NVEyMS4yIDMwLjc1IDI0IDMwLjc1Wk0yNCAzM3EtMy43NSAwLTYuMzc1LTIuNjI1VDE1IDI0cTAtMy43NSAyLjYyNS02LjM3NVQyNCAxNXEzLjc1IDAgNi4zNzUgMi42MjVUMzMgMjRxMCAzLjc1LTIuNjI1IDYuMzc1VDI0IDMzWk0zLjY1IDI1LjE1cS0uNSAwLS44MjUtLjMyNVEyLjUgMjQuNSAyLjUgMjRxMC0uNS4zMjUtLjgyNS4zMjUtLjMyNS44MjUtLjMyNWg1LjJxLjUgMCAuODI1LjMyNVExMCAyMy41IDEwIDI0cTAgLjUtLjMyNS44MjUtLjMyNS4zMjUtLjgyNS4zMjVabTM1LjUgMHEtLjUgMC0uODI1LS4zMjVRMzggMjQuNSAzOCAyNHEwLS41LjMyNS0uODI1LjMyNS0uMzI1LjgyNS0uMzI1aDUuMnEuNSAwIC44MjUuMzI1LjMyNS4zMjUuMzI1LjgyNSAwIC41LS4zMjUuODI1LS4zMjUuMzI1LS44MjUuMzI1Wk0yNCAxMHEtLjUgMC0uODI1LS4zMjUtLjMyNS0uMzI1LS4zMjUtLjgyNXYtNS4ycTAtLjUuMzI1LS44MjVRMjMuNSAyLjUgMjQgMi41cS41IDAgLjgyNS4zMjUuMzI1LjMyNS4zMjUuODI1djUuMnEwIC41LS4zMjUuODI1UTI0LjUgMTAgMjQgMTBabTAgMzUuNXEtLjUgMC0uODI1LS4zMjUtLjMyNS0uMzI1LS4zMjUtLjgyNXYtNS4ycTAtLjUuMzI1LS44MjVRMjMuNSAzOCAyNCAzOHEuNSAwIC44MjUuMzI1LjMyNS4zMjUuMzI1LjgyNXY1LjJxMCAuNS0uMzI1LjgyNS0uMzI1LjMyNS0uODI1LjMyNVpNMTIuNSAxNC4xbC0zLTIuOTVxLS4zNS0uMzUtLjMyNS0uODI1UTkuMiA5Ljg1IDkuNSA5LjVxLjM1LS4zNS44LS4zNS40NSAwIC44NS4zNWwyLjk1IDNxLjMuMzUuMy44IDAgLjQ1LS4zLjgtLjMuMy0uNzc1LjMtLjQ3NSAwLS44MjUtLjNabTI0LjM1IDI0LjQtMi45NS0zcS0uMy0uMzUtLjMtLjggMC0uNDUuMzUtLjguMjUtLjM1LjcyNS0uMzV0LjgyNS4zNWwzIDIuOTVxLjM1LjM1LjMyNS44MjUtLjAyNS40NzUtLjMyNS44MjUtLjM1LjM1LS44LjM1LS40NSAwLS44NS0uMzVaTTMzLjkgMTQuMXEtLjM1LS4zNS0uMzUtLjggMC0uNDUuMzUtLjhsMi45NS0zcS4zNS0uMzUuODI1LS4zMjUuNDc1LjAyNS44MjUuMzI1LjM1LjM1LjM1LjggMCAuNDUtLjM1Ljg1bC0zIDIuOTVxLS4zLjMtLjc3NS4zLS40NzUgMC0uODI1LS4zWk05LjUgMzguNXEtLjM1LS4zNS0uMzUtLjggMC0uNDUuMzUtLjg1bDMtMi45NXEuMzUtLjM1LjgtLjM1LjQ1IDAgLjguMzUuMzUuMy4zMjUuNzc1LS4wMjUuNDc1LS4zMjUuODI1bC0yLjk1IDNxLS40LjM1LS44NS4zNS0uNDUgMC0uOC0uMzVaTTI0IDI0WiIvPjwvc3ZnPg==';
- Editor.thinDarkImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI0LjA1IDQxcS03LjEgMC0xMi4wNS00Ljk1UTcuMDUgMzEuMSA3LjA1IDI0cTAtNi44IDQuNi0xMS42NSA0LjYtNC44NSAxMS4zLTUuMjUuMiAwIC40NS4wMjV0LjcuMDI1UTIyLjc1IDguNyAyMiAxMC43MjVxLS43NSAyLjAyNS0uNzUgNC4yNzUgMCA0LjkgMy40NSA4LjM1IDMuNDUgMy40NSA4LjM1IDMuNDUgMi4yIDAgNC4yNzUtLjY3NVQ0MC45IDI0LjJxMCAuMzUuMDI1LjU1LjAyNS4yLjAyNS4zNS0uNCA2LjctNS4yNSAxMS4zUTMwLjg1IDQxIDI0LjA1IDQxWm0wLTIuMjVxNS4xNSAwIDkuMDc1LTMuMTI1UTM3LjA1IDMyLjUgMzguMiAyOC4xcS0xLjIuNS0yLjUuNzI1LTEuMy4yMjUtMi42NS4yMjUtNS44NSAwLTkuOTUtNC4xVDE5IDE1cTAtMS4xNS4yMjUtMi40MjVUMjAgOS43NXEtNC42NSAxLjQtNy42NSA1LjM3NVQ5LjM1IDI0cTAgNi4xNSA0LjI3NSAxMC40NSA0LjI3NSA0LjMgMTAuNDI1IDQuM1ptLS4yNS0xNC41WiIvPjwvc3ZnPg==';
- Editor.thinCommentImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTEyLjUgMjcuNWgyM3YtMi4yNWgtMjNabTAtNi4zNWgyM3YtMi4zaC0yM1ptMC02LjRoMjNWMTIuNWgtMjNaTTQzIDQyLjEgMzUuOSAzNWgtMjhxLTEuMTUgMC0yLjAyNS0uODc1VDUgMzIuMVY3LjlxMC0xLjE1Ljg3NS0yLjAyNVQ3LjkgNWgzMi4ycTEuMiAwIDIuMDUuODc1UTQzIDYuNzUgNDMgNy45Wk03LjI1IDcuOXYyNC44NUgzNi45bDMuODUgMy44NVY3LjlxMC0uMy0uMTc1LS40NzVUNDAuMSA3LjI1SDcuOXEtLjMgMC0uNDc1LjE3NVQ3LjI1IDcuOVptMCAwdjI4LjdWNy4yNSA3LjlaIi8+PC9zdmc+';
- Editor.thinMenuImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTEzLjUgMjYuMTVxLjkgMCAxLjUyNS0uNjI1LjYyNS0uNjI1LjYyNS0xLjUyNSAwLS45LS42MjUtMS41MjUtLjYyNS0uNjI1LTEuNTI1LS42MjUtLjkgMC0xLjUyNS42MjUtLjYyNS42MjUtLjYyNSAxLjUyNSAwIC45LjYyNSAxLjUyNS42MjUuNjI1IDEuNTI1LjYyNVptMTAuNSAwcS45IDAgMS41MjUtLjYyNS42MjUtLjYyNS42MjUtMS41MjUgMC0uOS0uNjI1LTEuNTI1UTI0LjkgMjEuODUgMjQgMjEuODVxLS45IDAtMS41MjUuNjI1LS42MjUuNjI1LS42MjUgMS41MjUgMCAuOS42MjUgMS41MjUuNjI1LjYyNSAxLjUyNS42MjVabTEwLjUgMHEuODUgMCAxLjQ3NS0uNjI1UTM2LjYgMjQuOSAzNi42IDI0cTAtLjktLjYyNS0xLjUyNS0uNjI1LS42MjUtMS41MjUtLjYyNS0uODUgMC0xLjQ3NS42MjUtLjYyNS42MjUtLjYyNSAxLjUyNSAwIC45LjYyNSAxLjUyNS42MjUuNjI1IDEuNTI1LjYyNVpNMjQgNDNxLTMuOTUgMC03LjQtMS41dC02LjAyNS00LjA3NVE4IDM0Ljg1IDYuNSAzMS40VDUgMjRxMC0zLjk1IDEuNS03LjQyNVE4IDEzLjEgMTAuNTc1IDEwLjU1IDEzLjE1IDggMTYuNiA2LjVUMjQgNXEzLjk1IDAgNy40MjUgMS41UTM0LjkgOCAzNy40NSAxMC41NSA0MCAxMy4xIDQxLjUgMTYuNTc1IDQzIDIwLjA1IDQzIDI0cTAgMy45NS0xLjUgNy40dC00LjA1IDYuMDI1UTM0LjkgNDAgMzEuNDI1IDQxLjUgMjcuOTUgNDMgMjQgNDNabTAtMi4yNXE3IDAgMTEuODc1LTQuOVQ0MC43NSAyNHEwLTctNC44NzUtMTEuODc1VDI0IDcuMjVxLTYuOTUgMC0xMS44NSA0Ljg3NVE3LjI1IDE3IDcuMjUgMjRxMCA2Ljk1IDQuOSAxMS44NSA0LjkgNC45IDExLjg1IDQuOVpNMjQgMjRaIi8+PC9zdmc+';
- Editor.thinViewImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTUgMzlWOWgzOHYzMFptMjguNTUtMjAuNmg3LjJ2LTcuMTVoLTcuMlptMCA4Ljk1aDcuMnYtNi43aC03LjJabS0yNi4zIDkuNEgzMS4zdi0yNS41SDcuMjVabTI2LjMgMGg3LjJWMjkuNmgtNy4yWiIvPjwvc3ZnPg==';
- Editor.thinUserAddImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTM2LjYgMjcuNXYtNi4zNWgtNi4zNXYtMi4zaDYuMzVWMTIuNWgyLjN2Ni4zNWg2LjM1djIuM0gzOC45djYuMzVaTTE4IDIzLjM1cS0yLjkgMC00Ljc3NS0xLjg3NVExMS4zNSAxOS42IDExLjM1IDE2LjdxMC0yLjkgMS44NzUtNC43NVQxOCAxMC4xcTIuOSAwIDQuNzc1IDEuODUgMS44NzUgMS44NSAxLjg3NSA0Ljc1dC0xLjg3NSA0Ljc3NVEyMC45IDIzLjM1IDE4IDIzLjM1Wk0zIDM4LjZ2LTMuOHEwLTEuNS44LTIuNzV0Mi4yNS0xLjlxMy40NS0xLjUgNi4yNzUtMi4xNSAyLjgyNS0uNjUgNS42NzUtLjY1IDIuODUgMCA1LjY1LjY1IDIuOC42NSA2LjI1IDIuMTUgMS40NS43IDIuMjc1IDEuOTI1VDMzIDM0Ljh2My44Wm0yLjI1LTIuMjVoMjUuNVYzNC44cTAtLjc1LS41LTEuNDc1LS41LS43MjUtMS4zLTEuMTI1LTMuMi0xLjUtNS42NzUtMi4wNVEyMC44IDI5LjYgMTggMjkuNnEtMi44IDAtNS4zLjU1VDcgMzIuMnEtLjguNC0xLjI3NSAxLjEyNS0uNDc1LjcyNS0uNDc1IDEuNDc1Wk0xOCAyMS4xcTEuODUgMCAzLjEtMS4yNXQxLjI1LTMuMTVxMC0xLjg1LTEuMjUtMy4xVDE4IDEyLjM1cS0xLjg1IDAtMy4xIDEuMjV0LTEuMjUgMy4xcTAgMS45IDEuMjUgMy4xNVQxOCAyMS4xWm0wLTQuNFptMCAxOS42NVoiLz48L3N2Zz4=';
- Editor.thinUserFlashImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTkgMzguNnYtMy44cTAtMS42Ljg1LTIuOC44NS0xLjIgMi4yLTEuODUgMy4yLTEuNCA2LjEyNS0yLjEgMi45MjUtLjcgNS44MjUtLjcgMS40NSAwIDIuOS4xNzV0Mi45LjUyNXYyLjJxLTEuNDUtLjM1LTIuODc1LS41UTI1LjUgMjkuNiAyNCAyOS42cS0yLjc1IDAtNS40LjYtMi42NS42LTUuNiAyLS43NS40LTEuMjUgMS4xMjV0LS41IDEuNDc1djEuNTVIMjkuOHYyLjI1Wm0yLjI1LTIuMjVIMjkuOFptMTIuNzUtMTNxLTIuOSAwLTQuNzc1LTEuODc1UTE3LjM1IDE5LjYgMTcuMzUgMTYuN3EwLTIuOSAxLjg3NS00Ljc1VDI0IDEwLjFxMi45IDAgNC43NzUgMS44NSAxLjg3NSAxLjg1IDEuODc1IDQuNzV0LTEuODc1IDQuNzc1UTI2LjkgMjMuMzUgMjQgMjMuMzVabTAtMi4yNXExLjg1IDAgMy4xLTEuMjV0MS4yNS0zLjE1cTAtMS44NS0xLjI1LTMuMVQyNCAxMi4zNXEtMS44NSAwLTMuMSAxLjI1dC0xLjI1IDMuMXEwIDEuOSAxLjI1IDMuMTVUMjQgMjEuMVptMC00LjRabTEyLjg1IDI4LjA1di03LjhoLTMuNHYtMTAuMWg5LjE1bC0zLjggNy42NWgzLjdaIi8+PC9zdmc+';
- Editor.thinShareImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTExLjkgNDVxLTEuMiAwLTIuMDUtLjg1UTkgNDMuMyA5IDQyLjFWMTguOHEwLTEuMTUuODUtMiAuODUtLjg1IDIuMDUtLjg1aDYuOXYyLjI1aC02LjlxLS4yNSAwLS40NS4ydC0uMi40djIzLjNxMCAuMjUuMi40NXQuNDUuMmgyNC4ycS4yNSAwIC40NS0uMnQuMi0uNDVWMTguOHEwLS4yLS4yLS40dC0uNDUtLjJoLTYuOTV2LTIuMjVoNi45NXExLjIgMCAyLjA1Ljg1Ljg1Ljg1Ljg1IDJ2MjMuM3EwIDEuMi0uODUgMi4wNS0uODUuODUtMi4wNS44NVptMTAuOTUtMTQuNVY4LjFsLTQuNiA0LjU1LTEuNjUtMS42IDcuMzUtNy4zNSA3LjM1IDcuMzUtMS42IDEuNi00LjYtNC41NXYyMi40WiIvPjwvc3ZnPg==';
- Editor.thinTextImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTMuNCA0NC42di05LjI1aDMuNTV2LTIyLjdIMy40VjMuNGg5LjI1djMuNTVoMjIuN1YzLjRoOS4yNXY5LjI1aC0zLjU1djIyLjdoMy41NXY5LjI1aC05LjI1di0zLjU1aC0yMi43djMuNTVabTkuMjUtNS44NWgyMi43di0zLjRoMy40di0yMi43aC0zLjR2LTMuNGgtMjIuN3YzLjRoLTMuNHYyMi43aDMuNFptMy4xNS02LjI1IDcuMzUtMTkuMTVoMS42NWw3LjQ1IDE5LjE1aC0yLjFMMjggMjdoLTcuODVsLTIuMSA1LjVabTQuOTUtNy4zNWg2LjVMMjQuMSAxNi44aC0uM1ptLTE1LjEtMTQuOGg0Ljd2LTQuN2gtNC43Wm0zMiAwaDQuN3YtNC43aC00LjdabTAgMzJoNC43di00LjdoLTQuN1ptLTMyIDBoNC43di00LjdoLTQuN1ptMzItMzJabTAgMjcuM1ptLTI3LjMgMFptMC0yNy4zWiIvPjwvc3ZnPg==';
- Editor.thinRectangleImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTUgMzlWOWgzOHYzMFptMi4yNS0yLjI1aDMzLjV2LTI1LjVINy4yNVptMCAwdi0yNS41IDI1LjVaIi8+PC9zdmc+';
- Editor.thinDataImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI4Ljg1IDM5LjF2LTIuMjVIMzRxMS4yIDAgMi4wMjUtLjgyNVQzNi44NSAzNHYtNC45cTAtMS43NSAxLjA3NS0zLjEyNXQyLjcyNS0xLjgyNXYtLjNxLTEuNjUtLjQ1LTIuNzI1LTEuODI1UTM2Ljg1IDIwLjY1IDM2Ljg1IDE4LjlWMTRxMC0xLjItLjgyNS0yLjAyNVQzNCAxMS4xNWgtNS4xNVY4LjlIMzRxMi4xNSAwIDMuNjI1IDEuNVQzOS4xIDE0djQuOXEwIDEuMjUuODUgMi4wNzUuODUuODI1IDIuMS44MjVoLjg1djQuNGgtLjg1cS0xLjI1IDAtMi4xLjgyNS0uODUuODI1LS44NSAyLjA3NVYzNHEwIDIuMS0xLjUgMy42VDM0IDM5LjFaTTE0IDM5LjFxLTIuMTUgMC0zLjYyNS0xLjVUOC45IDM0di00LjlxMC0xLjI1LS44NS0yLjA3NS0uODUtLjgyNS0yLjEtLjgyNUg1LjF2LTQuNGguODVxMS4yNSAwIDIuMS0uODI1Ljg1LS44MjUuODUtMi4wNzVWMTRxMC0yLjEgMS41LTMuNlQxNCA4LjloNS4xNXYyLjI1SDE0cS0xLjIgMC0yLjAyNS44MjVUMTEuMTUgMTR2NC45cTAgMS43NS0xLjA3NSAzLjEyNVQ3LjM1IDIzLjg1di4zcTEuNjUuNDUgMi43MjUgMS44MjVRMTEuMTUgMjcuMzUgMTEuMTUgMjkuMVYzNHEwIDEuMi44MjUgMi4wMjVUMTQgMzYuODVoNS4xNXYyLjI1WiIvPjwvc3ZnPg==';
- Editor.thinExpandImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTI0IDMwLjEgMTIuNyAxOC43NWwxLjYtMS42IDkuNyA5LjcgOS43LTkuNyAxLjYgMS42NVoiLz48L3N2Zz4=';
- Editor.thinGridImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHZpZXdCb3g9IjAgLTk2MCA5NjAgOTYwIiB3aWR0aD0iNDgiPjxwYXRoIGQ9Ik0yMjMuMjkxLTE1NC41cS0yOS4xMTcgMC00OC45NTQtMTkuODM3VDE1NC41LTIyMy4yOTFxMC0yOS4xMTggMTkuODM3LTQ4LjkxM1ExOTQuMTc0LTI5MiAyMjMuMjkxLTI5MnEyOS4xMTggMCA0OC45MTMgMTkuNzk2UTI5Mi0yNTIuNDA5IDI5Mi0yMjMuMjkxcTAgMjkuMTE3LTE5Ljc5NiA0OC45NTQtMTkuNzk1IDE5LjgzNy00OC45MTMgMTkuODM3Wm0yNTYuOCAwcS0yOS4wNDggMC00OC44ODUtMTkuODM3LTE5LjgzNi0xOS44MzctMTkuODM2LTQ4Ljk1NCAwLTI5LjExOCAxOS43NDUtNDguOTEzUTQ1MC44NjEtMjkyIDQ3OS45MDktMjkydDQ4Ljg4NSAxOS43OTZxMTkuODM2IDE5Ljc5NSAxOS44MzYgNDguOTEzIDAgMjkuMTE3LTE5Ljc0NSA0OC45NTQtMTkuNzQ2IDE5LjgzNy00OC43OTQgMTkuODM3Wm0yNTYuNjE4IDBxLTI5LjExOCAwLTQ4LjkxMy0xOS44MzdRNjY4LTE5NC4xNzQgNjY4LTIyMy4yOTFxMC0yOS4xMTggMTkuNzk2LTQ4LjkxM1E3MDcuNTkxLTI5MiA3MzYuNzA5LTI5MnEyOS4xMTcgMCA0OC45NTQgMTkuNzk2IDE5LjgzNyAxOS43OTUgMTkuODM3IDQ4LjkxMyAwIDI5LjExNy0xOS44MzcgNDguOTU0VDczNi43MDktMTU0LjVaTTIyMy4yOTEtNDExLjM3cS0yOS4xMTcgMC00OC45NTQtMTkuNzQ1LTE5LjgzNy0xOS43NDYtMTkuODM3LTQ4Ljc5NHQxOS44MzctNDguODg1cTE5LjgzNy0xOS44MzYgNDguOTU0LTE5LjgzNiAyOS4xMTggMCA0OC45MTMgMTkuNzQ1UTI5Mi01MDkuMTM5IDI5Mi00ODAuMDkxdC0xOS43OTYgNDguODg1cS0xOS43OTUgMTkuODM2LTQ4LjkxMyAxOS44MzZabTI1Ni44IDBxLTI5LjA0OCAwLTQ4Ljg4NS0xOS43NDUtMTkuODM2LTE5Ljc0Ni0xOS44MzYtNDguNzk0dDE5Ljc0NS00OC44ODVxMTkuNzQ2LTE5LjgzNiA0OC43OTQtMTkuODM2dDQ4Ljg4NSAxOS43NDVxMTkuODM2IDE5Ljc0NiAxOS44MzYgNDguNzk0dC0xOS43NDUgNDguODg1cS0xOS43NDYgMTkuODM2LTQ4Ljc5NCAxOS44MzZabTI1Ni42MTggMHEtMjkuMTE4IDAtNDguOTEzLTE5Ljc0NVE2NjgtNDUwLjg2MSA2NjgtNDc5LjkwOXQxOS43OTYtNDguODg1cTE5Ljc5NS0xOS44MzYgNDguOTEzLTE5LjgzNiAyOS4xMTcgMCA0OC45NTQgMTkuNzQ1IDE5LjgzNyAxOS43NDYgMTkuODM3IDQ4Ljc5NHQtMTkuODM3IDQ4Ljg4NXEtMTkuODM3IDE5LjgzNi00OC45NTQgMTkuODM2Wk0yMjMuMjkxLTY2OHEtMjkuMTE3IDAtNDguOTU0LTE5Ljc5Ni0xOS44MzctMTkuNzk1LTE5LjgzNy00OC45MTMgMC0yOS4xMTcgMTkuODM3LTQ4Ljk1NHQ0OC45NTQtMTkuODM3cTI5LjExOCAwIDQ4LjkxMyAxOS44MzdRMjkyLTc2NS44MjYgMjkyLTczNi43MDlxMCAyOS4xMTgtMTkuNzk2IDQ4LjkxM1EyNTIuNDA5LTY2OCAyMjMuMjkxLTY2OFptMjU2LjggMHEtMjkuMDQ4IDAtNDguODg1LTE5Ljc5Ni0xOS44MzYtMTkuNzk1LTE5LjgzNi00OC45MTMgMC0yOS4xMTcgMTkuNzQ1LTQ4Ljk1NCAxOS43NDYtMTkuODM3IDQ4Ljc5NC0xOS44Mzd0NDguODg1IDE5LjgzN3ExOS44MzYgMTkuODM3IDE5LjgzNiA0OC45NTQgMCAyOS4xMTgtMTkuNzQ1IDQ4LjkxM1E1MDkuMTM5LTY2OCA0ODAuMDkxLTY2OFptMjU2LjYxOCAwcS0yOS4xMTggMC00OC45MTMtMTkuNzk2UTY2OC03MDcuNTkxIDY2OC03MzYuNzA5cTAtMjkuMTE3IDE5Ljc5Ni00OC45NTQgMTkuNzk1LTE5LjgzNyA0OC45MTMtMTkuODM3IDI5LjExNyAwIDQ4Ljk1NCAxOS44Mzd0MTkuODM3IDQ4Ljk1NHEwIDI5LjExOC0xOS44MzcgNDguOTEzUTc2NS44MjYtNjY4IDczNi43MDktNjY4WiIvPjwvc3ZnPg==';
- Editor.thinOpenImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMjQiIHZpZXdCb3g9IjAgLTk2MCA5NjAgOTYwIiB3aWR0aD0iMjQiPjxwYXRoIGQ9Ik0yNDAtODBxLTMzIDAtNTYuNS0yMy41VDE2MC0xNjB2LTY0MHEwLTMzIDIzLjUtNTYuNVQyNDAtODgwaDMyMGwyNDAgMjQwdjI0MGgtODB2LTIwMEg1MjB2LTIwMEgyNDB2NjQwaDM2MHY4MEgyNDBabTYzOCAxNUw3NjAtMTgzdjg5aC04MHYtMjI2aDIyNnY4MGgtOTBsMTE4IDExOC01NiA1N1ptLTYzOC05NXYtNjQwIDY0MFoiLz48L3N2Zz4=';
- Editor.selectImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iNDgiIHdpZHRoPSI0OCI+PHBhdGggZD0iTTkgNDJxLTEuMjUgMC0yLjEyNS0uODc1VDYgMzlWOXEwLTEuMjUuODc1LTIuMTI1VDkgNmgzMHEuNyAwIDEuMjc1LjN0LjkyNS43TDM5IDkuMlY5SDl2MzBoMzBWMjEuODVsMy0zVjM5cTAgMS4yNS0uODc1IDIuMTI1VDM5IDQyWm0xNC4wNS04LjQtMTEuMS0xMS4xIDIuMS0yLjEgOSA5IDE5LjEtMTkuMSAyLjEgMi4xWiIvPjwvc3ZnPg==';
- /**
- *
- */
- Editor.styles = [{},
- {commonStyle: {fontColor: '#393C56', strokeColor: '#E07A5F', fillColor: '#F2CC8F'},
- graph: {background: '#F4F1DE', gridColor: '#D4D0C0'}},
- {vertexStyle: {strokeColor: '#BAC8D3', fillColor: '#09555B', fontColor: '#EEEEEE'},
- edgeStyle: {strokeColor: '#0B4D6A'}},
- {vertexStyle: {strokeColor: '#FFFFFF', fillColor: '#182E3E', fontColor: '#FFFFFF'},
- edgeStyle: {strokeColor: '#23445D'},
- graph: {background: '#FCE7CD', gridColor: '#CFBDA8'}},
- {vertexStyle: {strokeColor: '#D0CEE2', fillColor: '#5D7F99'},
- edgeStyle: {strokeColor: '#736CA8'},
- commonStyle: {fontColor: '#1A1A1A'}},
- {commonStyle: {fontColor: '#46495D', strokeColor: '#788AA3', fillColor: '#B2C9AB'}},
- {commonStyle: {fontColor: '#5AA9E6', strokeColor: '#FF6392', fillColor: '#FFE45E'}},
- {commonStyle: {fontColor: '#E4FDE1', strokeColor: '#028090', fillColor: '#F45B69'},
- graph: {background: '#114B5F', gridColor: '#0B3240'}},
- {commonStyle: {fontColor: '#FEFAE0', strokeColor: '#DDA15E', fillColor: '#BC6C25'},
- graph: {background: '#283618', gridColor: '#48632C'}},
- {commonStyle: {fontColor: '#143642', strokeColor: '#0F8B8D', fillColor: '#FAE5C7'},
- edgeStyle: {strokeColor: '#A8201A'},
- graph: {background: '#DAD2D8', gridColor: '#ABA4A9'}},
- {},
- {vertexStyle: {strokeColor: '#D0CEE2', fillColor: '#FAD9D5'},
- edgeStyle: {strokeColor: '#09555B'},
- commonStyle: {fontColor: '#1A1A1A'}},
- {commonStyle: {fontColor: '#1D3557', strokeColor: '#457B9D', fillColor: '#A8DADC'},
- graph: {background: '#F1FAEE'}},
- {commonStyle: {fontColor: '#095C86', strokeColor: '#AF45ED', fillColor: '#F694C1'},
- edgeStyle: {strokeColor: '#60E696'}},
- {commonStyle: {fontColor: '#5C5C5C', strokeColor: '#006658', fillColor: '#21C0A5'}},
- {vertexStyle: {strokeColor: '#FFFFFF', fillColor: '#F08E81'},
- edgeStyle: {strokeColor: '#182E3E'},
- commonStyle: {fontColor: '#1A1A1A'},
- graph: {background: '#B0E3E6', gridColor: '#87AEB0'}},
- {vertexStyle: {strokeColor: '#909090', fillColor: '#F5AB50'},
- edgeStyle: {strokeColor: '#182E3E'},
- commonStyle: {fontColor: '#1A1A1A'},
- graph: {background: '#EEEEEE'}},
- {vertexStyle: {strokeColor: '#BAC8D3', fillColor: '#B1DDF0', fontColor: '#182E3E'},
- edgeStyle: {strokeColor: '#EEEEEE', fontColor: '#FFFFFF'},
- graph: {background: '#09555B', gridColor: '#13B4C2'}},
- {vertexStyle: {strokeColor: '#EEEEEE', fillColor: '#56517E', fontColor: '#FFFFFF'},
- edgeStyle: {strokeColor: '#182E3E'},
- graph: {background: '#FAD9D5', gridColor: '#BFA6A3'}},
- {vertexStyle: {fillColor: '#EEEEEE', fontColor: '#1A1A1A'},
- edgeStyle: {fontColor: '#FFFFFF'},
- commonStyle: {strokeColor: '#FFFFFF'},
- graph: {background: '#182E3E', gridColor: '#4D94C7'}}];
-
- /**
- *
- */
- Editor.logoImage = 'data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+CjxzdmcKICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIgogICB4bWxuczpjYz0iaHR0cDovL2NyZWF0aXZlY29tbW9ucy5vcmcvbnMjIgogICB4bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDIvMjItcmRmLXN5bnRheC1ucyMiCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciCiAgIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIKICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSIKICAgZW5hYmxlLWJhY2tncm91bmQ9Im5ldyAwIDAgMzA2LjE4NSAxMjAuMjk2IgogICB2aWV3Qm94PSIyNCAyNiA2OCA2OCIKICAgeT0iMHB4IgogICB4PSIwcHgiCiAgIHZlcnNpb249IjEuMSI+CiAgIAkgPGc+PGxpbmUKICAgICAgIHkyPSI3Mi4zOTQiCiAgICAgICB4Mj0iNDEuMDYxIgogICAgICAgeTE9IjQzLjM4NCIKICAgICAgIHgxPSI1OC4wNjkiCiAgICAgICBzdHJva2UtbWl0ZXJsaW1pdD0iMTAiCiAgICAgICBzdHJva2Utd2lkdGg9IjMuNTUyOCIKICAgICAgIHN0cm9rZT0iI0ZGRkZGRiIKICAgICAgIGZpbGw9Im5vbmUiIC8+PGxpbmUKICAgICAgIHkyPSI3Mi4zOTQiCiAgICAgICB4Mj0iNzUuMDc2IgogICAgICAgeTE9IjQzLjM4NCIKICAgICAgIHgxPSI1OC4wNjgiCiAgICAgICBzdHJva2UtbWl0ZXJsaW1pdD0iMTAiCiAgICAgICBzdHJva2Utd2lkdGg9IjMuNTAwOCIKICAgICAgIHN0cm9rZT0iI0ZGRkZGRiIKICAgICAgIGZpbGw9Im5vbmUiIC8+PGc+PHBhdGgKICAgICAgICAgZD0iTTUyLjc3Myw3Ny4wODRjMCwxLjk1NC0xLjU5OSwzLjU1My0zLjU1MywzLjU1M0gzNi45OTljLTEuOTU0LDAtMy41NTMtMS41OTktMy41NTMtMy41NTN2LTkuMzc5ICAgIGMwLTEuOTU0LDEuNTk5LTMuNTUzLDMuNTUzLTMuNTUzaDEyLjIyMmMxLjk1NCwwLDMuNTUzLDEuNTk5LDMuNTUzLDMuNTUzVjc3LjA4NHoiCiAgICAgICAgIGZpbGw9IiNGRkZGRkYiIC8+PC9nPjxnCiAgICAgICBpZD0iZzM0MTkiPjxwYXRoCiAgICAgICAgIGQ9Ik02Ny43NjIsNDguMDc0YzAsMS45NTQtMS41OTksMy41NTMtMy41NTMsMy41NTNINTEuOTg4Yy0xLjk1NCwwLTMuNTUzLTEuNTk5LTMuNTUzLTMuNTUzdi05LjM3OSAgICBjMC0xLjk1NCwxLjU5OS0zLjU1MywzLjU1My0zLjU1M0g2NC4yMWMxLjk1NCwwLDMuNTUzLDEuNTk5LDMuNTUzLDMuNTUzVjQ4LjA3NHoiCiAgICAgICAgIGZpbGw9IiNGRkZGRkYiIC8+PC9nPjxnPjxwYXRoCiAgICAgICAgIGQ9Ik04Mi43NTIsNzcuMDg0YzAsMS45NTQtMS41OTksMy41NTMtMy41NTMsMy41NTNINjYuOTc3Yy0xLjk1NCwwLTMuNTUzLTEuNTk5LTMuNTUzLTMuNTUzdi05LjM3OSAgICBjMC0xLjk1NCwxLjU5OS0zLjU1MywzLjU1My0zLjU1M2gxMi4yMjJjMS45NTQsMCwzLjU1MywxLjU5OSwzLjU1MywzLjU1M1Y3Ny4wODR6IgogICAgICAgICBmaWxsPSIjRkZGRkZGIiAvPjwvZz48L2c+PC9zdmc+';
- Editor.saveImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyNCAyNCIgZmlsbD0iYmxhY2siIHdpZHRoPSIxOHB4IiBoZWlnaHQ9IjE4cHgiPjxwYXRoIGQ9Ik0wIDBoMjR2MjRIMHoiIGZpbGw9Im5vbmUiLz48cGF0aCBkPSJNMTkgMTJ2N0g1di03SDN2N2MwIDEuMS45IDIgMiAyaDE0YzEuMSAwIDItLjkgMi0ydi03aC0yem0tNiAuNjdsMi41OS0yLjU4TDE3IDExLjVsLTUgNS01LTUgMS40MS0xLjQxTDExIDEyLjY3VjNoMnoiLz48L3N2Zz4=';
- Editor.globeImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNCIgaGVpZ2h0PSIyNCIgdmlld0JveD0iMCAwIDI0IDI0Ij48cGF0aCBkPSJNMTEuOTkgMkM2LjQ3IDIgMiA2LjQ4IDIgMTJzNC40NyAxMCA5Ljk5IDEwQzE3LjUyIDIyIDIyIDE3LjUyIDIyIDEyUzE3LjUyIDIgMTEuOTkgMnptNi45MyA2aC0yLjk1Yy0uMzItMS4yNS0uNzgtMi40NS0xLjM4LTMuNTYgMS44NC42MyAzLjM3IDEuOTEgNC4zMyAzLjU2ek0xMiA0LjA0Yy44MyAxLjIgMS40OCAyLjUzIDEuOTEgMy45NmgtMy44MmMuNDMtMS40MyAxLjA4LTIuNzYgMS45MS0zLjk2ek00LjI2IDE0QzQuMSAxMy4zNiA0IDEyLjY5IDQgMTJzLjEtMS4zNi4yNi0yaDMuMzhjLS4wOC42Ni0uMTQgMS4zMi0uMTQgMiAwIC42OC4wNiAxLjM0LjE0IDJINC4yNnptLjgyIDJoMi45NWMuMzIgMS4yNS43OCAyLjQ1IDEuMzggMy41Ni0xLjg0LS42My0zLjM3LTEuOS00LjMzLTMuNTZ6bTIuOTUtOEg1LjA4Yy45Ni0xLjY2IDIuNDktMi45MyA0LjMzLTMuNTZDOC44MSA1LjU1IDguMzUgNi43NSA4LjAzIDh6TTEyIDE5Ljk2Yy0uODMtMS4yLTEuNDgtMi41My0xLjkxLTMuOTZoMy44MmMtLjQzIDEuNDMtMS4wOCAyLjc2LTEuOTEgMy45NnpNMTQuMzQgMTRIOS42NmMtLjA5LS42Ni0uMTYtMS4zMi0uMTYtMiAwLS42OC4wNy0xLjM1LjE2LTJoNC42OGMuMDkuNjUuMTYgMS4zMi4xNiAyIDAgLjY4LS4wNyAxLjM0LS4xNiAyem0uMjUgNS41NmMuNi0xLjExIDEuMDYtMi4zMSAxLjM4LTMuNTZoMi45NWMtLjk2IDEuNjUtMi40OSAyLjkzLTQuMzMgMy41NnpNMTYuMzYgMTRjLjA4LS42Ni4xNC0xLjMyLjE0LTIgMC0uNjgtLjA2LTEuMzQtLjE0LTJoMy4zOGMuMTYuNjQuMjYgMS4zMS4yNiAycy0uMSAxLjM2LS4yNiAyaC0zLjM4eiIvPjwvc3ZnPg==';
- Editor.commentImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNCIgaGVpZ2h0PSIyNCIgdmlld0JveD0iMCAwIDI0IDI0Ij48cGF0aCBkPSJNMjEuOTkgNGMwLTEuMS0uODktMi0xLjk5LTJINGMtMS4xIDAtMiAuOS0yIDJ2MTJjMCAxLjEuOSAyIDIgMmgxNGw0IDQtLjAxLTE4ek0xOCAxNEg2di0yaDEydjJ6bTAtM0g2VjloMTJ2MnptMC0zSDZWNmgxMnYyeiIvPjxwYXRoIGQ9Ik0wIDBoMjR2MjRIMHoiIGZpbGw9Im5vbmUiLz48L3N2Zz4=';
- Editor.userImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNCIgaGVpZ2h0PSIyNCIgdmlld0JveD0iMCAwIDI0IDI0Ij48cGF0aCBkPSJNMTIgMTJjMi4yMSAwIDQtMS43OSA0LTRzLTEuNzktNC00LTQtNCAxLjc5LTQgNCAxLjc5IDQgNCA0em0wIDJjLTIuNjcgMC04IDEuMzQtOCA0djJoMTZ2LTJjMC0yLjY2LTUuMzMtNC04LTR6Ii8+PC9zdmc+';
- Editor.shareImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNCIgaGVpZ2h0PSIyNCIgdmlld0JveD0iMCAwIDI0IDI0Ij48cGF0aCBkPSJNMTggMTYuMDhjLS43NiAwLTEuNDQuMy0xLjk2Ljc3TDguOTEgMTIuN2MuMDUtLjIzLjA5LS40Ni4wOS0uN3MtLjA0LS40Ny0uMDktLjdsNy4wNS00LjExYy41NC41IDEuMjUuODEgMi4wNC44MSAxLjY2IDAgMy0xLjM0IDMtM3MtMS4zNC0zLTMtMy0zIDEuMzQtMyAzYzAgLjI0LjA0LjQ3LjA5LjdMOC4wNCA5LjgxQzcuNSA5LjMxIDYuNzkgOSA2IDljLTEuNjYgMC0zIDEuMzQtMyAzczEuMzQgMyAzIDNjLjc5IDAgMS41LS4zMSAyLjA0LS44MWw3LjEyIDQuMTZjLS4wNS4yMS0uMDguNDMtLjA4LjY1IDAgMS42MSAxLjMxIDIuOTIgMi45MiAyLjkyIDEuNjEgMCAyLjkyLTEuMzEgMi45Mi0yLjkycy0xLjMxLTIuOTItMi45Mi0yLjkyeiIvPjwvc3ZnPg==';
- Editor.syncImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNCIgaGVpZ2h0PSIyNCIgdmlld0JveD0iMCAwIDI0IDI0Ij48cGF0aCBkPSJNMTIgNFYxTDggNWw0IDRWNmMzLjMxIDAgNiAyLjY5IDYgNiAwIDEuMDEtLjI1IDEuOTctLjcgMi44bDEuNDYgMS40NkMxOS41NCAxNS4wMyAyMCAxMy41NyAyMCAxMmMwLTQuNDItMy41OC04LTgtOHptMCAxNGMtMy4zMSAwLTYtMi42OS02LTYgMC0xLjAxLjI1LTEuOTcuNy0yLjhMNS4yNCA3Ljc0QzQuNDYgOC45NyA0IDEwLjQzIDQgMTJjMCA0LjQyIDMuNTggOCA4IDh2M2w0LTQtNC00djN6Ii8+PC9zdmc+';
- Editor.cloudImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMjRweCIgdmlld0JveD0iMCAwIDI0IDI0IiB3aWR0aD0iMjRweCIgZmlsbD0iIzAwMDAwMCI+PHBhdGggZD0iTTAgMGgyNHYyNEgwVjB6IiBmaWxsPSJub25lIi8+PHBhdGggZD0iTTEyIDZjMi42MiAwIDQuODggMS44NiA1LjM5IDQuNDNsLjMgMS41IDEuNTMuMTFjMS41Ni4xIDIuNzggMS40MSAyLjc4IDIuOTYgMCAxLjY1LTEuMzUgMy0zIDNINmMtMi4yMSAwLTQtMS43OS00LTQgMC0yLjA1IDEuNTMtMy43NiAzLjU2LTMuOTdsMS4wNy0uMTEuNS0uOTVDOC4wOCA3LjE0IDkuOTQgNiAxMiA2bTAtMkM5LjExIDQgNi42IDUuNjQgNS4zNSA4LjA0IDIuMzQgOC4zNiAwIDEwLjkxIDAgMTRjMCAzLjMxIDIuNjkgNiA2IDZoMTNjMi43NiAwIDUtMi4yNCA1LTUgMC0yLjY0LTIuMDUtNC43OC00LjY1LTQuOTZDMTguNjcgNi41OSAxNS42NCA0IDEyIDR6Ii8+PC9zdmc+';
- Editor.cloudOffImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMjRweCIgdmlld0JveD0iMCAwIDI0IDI0IiB3aWR0aD0iMjRweCIgZmlsbD0iIzAwMDAwMCI+PHBhdGggZD0iTTAgMGgyNHYyNEgwVjB6IiBmaWxsPSJub25lIi8+PHBhdGggZD0iTTI0IDE1YzAtMi42NC0yLjA1LTQuNzgtNC42NS00Ljk2QzE4LjY3IDYuNTkgMTUuNjQgNCAxMiA0Yy0xLjMzIDAtMi41Ny4zNi0zLjY1Ljk3bDEuNDkgMS40OUMxMC41MSA2LjE3IDExLjIzIDYgMTIgNmMzLjA0IDAgNS41IDIuNDYgNS41IDUuNXYuNUgxOWMxLjY2IDAgMyAxLjM0IDMgMyAwIC45OS0uNDggMS44NS0xLjIxIDIuNGwxLjQxIDEuNDFjMS4wOS0uOTIgMS44LTIuMjcgMS44LTMuODF6TTQuNDEgMy44NkwzIDUuMjdsMi43NyAyLjc3aC0uNDJDMi4zNCA4LjM2IDAgMTAuOTEgMCAxNGMwIDMuMzEgMi42OSA2IDYgNmgxMS43M2wyIDIgMS40MS0xLjQxTDQuNDEgMy44NnpNNiAxOGMtMi4yMSAwLTQtMS43OS00LTRzMS43OS00IDQtNGgxLjczbDggOEg2eiIvPjwvc3ZnPg==';
- Editor.calendarImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGVuYWJsZS1iYWNrZ3JvdW5kPSJuZXcgMCAwIDI0IDI0IiBoZWlnaHQ9IjI0cHgiIHZpZXdCb3g9IjAgMCAyNCAyNCIgd2lkdGg9IjI0cHgiIGZpbGw9IiMwMDAwMDAiPjxnPjxwYXRoIGQ9Ik0wLDBoMjR2MjRIMFYweiIgZmlsbD0ibm9uZSIvPjwvZz48Zz48cGF0aCBkPSJNMjAsNEg0QzIuOSw0LDIsNC45LDIsNnYxMmMwLDEuMSwwLjksMiwyLDJoMTZjMS4xLDAsMi0wLjksMi0yVjZDMjIsNC45LDIxLjEsNCwyMCw0eiBNOCwxMUg0VjZoNFYxMXogTTE0LDExaC00VjZoNFYxMXogTTIwLDExaC00VjZoNFYxMXogTTgsMThINHYtNWg0VjE4eiBNMTQsMThoLTR2LTVoNFYxOHogTTIwLDE4aC00di01aDRWMTh6Ii8+PC9nPjwvc3ZnPg==';
- Editor.syncProblemImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNCIgaGVpZ2h0PSIyNCIgdmlld0JveD0iMCAwIDI0IDI0Ij48cGF0aCBkPSJNMyAxMmMwIDIuMjEuOTEgNC4yIDIuMzYgNS42NEwzIDIwaDZ2LTZsLTIuMjQgMi4yNEM1LjY4IDE1LjE1IDUgMTMuNjYgNSAxMmMwLTIuNjEgMS42Ny00LjgzIDQtNS42NVY0LjI2QzUuNTUgNS4xNSAzIDguMjcgMyAxMnptOCA1aDJ2LTJoLTJ2MnpNMjEgNGgtNnY2bDIuMjQtMi4yNEMxOC4zMiA4Ljg1IDE5IDEwLjM0IDE5IDEyYzAgMi42MS0xLjY3IDQuODMtNCA1LjY1djIuMDljMy40NS0uODkgNi00LjAxIDYtNy43NCAwLTIuMjEtLjkxLTQuMi0yLjM2LTUuNjRMMjEgNHptLTEwIDloMlY3aC0ydjZ6Ii8+PC9zdmc+';
- Editor.tailSpin = 'data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMTYiIGhlaWdodD0iMTYiIHZpZXdCb3g9Ii0yIC0yIDQ0IDQ0IiB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciPgogICAgPGRlZnM+CiAgICAgICAgPGxpbmVhckdyYWRpZW50IHgxPSI4LjA0MiUiIHkxPSIwJSIgeDI9IjY1LjY4MiUiIHkyPSIyMy44NjUlIiBpZD0iYSI+CiAgICAgICAgICAgIDxzdG9wIHN0b3AtY29sb3I9IiM4MDgwODAiIHN0b3Atb3BhY2l0eT0iMCIgb2Zmc2V0PSIwJSIvPgogICAgICAgICAgICA8c3RvcCBzdG9wLWNvbG9yPSIjODA4MDgwIiBzdG9wLW9wYWNpdHk9Ii42MzEiIG9mZnNldD0iNjMuMTQ2JSIvPgogICAgICAgICAgICA8c3RvcCBzdG9wLWNvbG9yPSIjODA4MDgwIiBvZmZzZXQ9IjEwMCUiLz4KICAgICAgICA8L2xpbmVhckdyYWRpZW50PgogICAgPC9kZWZzPgogICAgPGcgZmlsbD0ibm9uZSIgZmlsbC1ydWxlPSJldmVub2RkIj4KICAgICAgICA8ZyB0cmFuc2Zvcm09InRyYW5zbGF0ZSgxIDEpIj4KICAgICAgICAgICAgPHBhdGggZD0iTTM2IDE4YzAtOS45NC04LjA2LTE4LTE4LTE4IiBzdHJva2U9InVybCgjYSkiIHN0cm9rZS13aWR0aD0iNiI+CiAgICAgICAgICAgICAgICA8YW5pbWF0ZVRyYW5zZm9ybQogICAgICAgICAgICAgICAgICAgIGF0dHJpYnV0ZU5hbWU9InRyYW5zZm9ybSIKICAgICAgICAgICAgICAgICAgICB0eXBlPSJyb3RhdGUiCiAgICAgICAgICAgICAgICAgICAgZnJvbT0iMCAxOCAxOCIKICAgICAgICAgICAgICAgICAgICB0bz0iMzYwIDE4IDE4IgogICAgICAgICAgICAgICAgICAgIGR1cj0iMC45cyIKICAgICAgICAgICAgICAgICAgICByZXBlYXRDb3VudD0iaW5kZWZpbml0ZSIgLz4KICAgICAgICAgICAgPC9wYXRoPgogICAgICAgICAgICA8Y2lyY2xlIGZpbGw9IiM4MDgwODAiIGN4PSIzNiIgY3k9IjE4IiByPSIxIj4KICAgICAgICAgICAgICAgIDxhbmltYXRlVHJhbnNmb3JtCiAgICAgICAgICAgICAgICAgICAgYXR0cmlidXRlTmFtZT0idHJhbnNmb3JtIgogICAgICAgICAgICAgICAgICAgIHR5cGU9InJvdGF0ZSIKICAgICAgICAgICAgICAgICAgICBmcm9tPSIwIDE4IDE4IgogICAgICAgICAgICAgICAgICAgIHRvPSIzNjAgMTggMTgiCiAgICAgICAgICAgICAgICAgICAgZHVyPSIwLjlzIgogICAgICAgICAgICAgICAgICAgIHJlcGVhdENvdW50PSJpbmRlZmluaXRlIiAvPgogICAgICAgICAgICA8L2NpcmNsZT4KICAgICAgICA8L2c+CiAgICA8L2c+Cjwvc3ZnPgo=';
- Editor.mailImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMjRweCIgdmlld0JveD0iMCAwIDI0IDI0IiB3aWR0aD0iMjRweCIgZmlsbD0iIzAwMDAwMCI+PHBhdGggZD0iTTAgMGgyNHYyNEgwVjB6IiBmaWxsPSJub25lIi8+PHBhdGggZD0iTTIyIDZjMC0xLjEtLjktMi0yLTJINGMtMS4xIDAtMiAuOS0yIDJ2MTJjMCAxLjEuOSAyIDIgMmgxNmMxLjEgMCAyLS45IDItMlY2em0tMiAwbC04IDQuOTlMNCA2aDE2em0wIDEySDRWOGw4IDUgOC01djEweiIvPjwvc3ZnPg==';
- Editor.cameraImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMThweCIgdmlld0JveD0iMCAwIDI0IDI0IiB3aWR0aD0iMThweCIgZmlsbD0iIzAwMDAwMCI+PHBhdGggZD0iTTAgMGgyNHYyNEgwVjB6IiBmaWxsPSJub25lIi8+PHBhdGggZD0iTTE0LjEyIDRsMS44MyAySDIwdjEySDRWNmg0LjA1bDEuODMtMmg0LjI0TTE1IDJIOUw3LjE3IDRINGMtMS4xIDAtMiAuOS0yIDJ2MTJjMCAxLjEuOSAyIDIgMmgxNmMxLjEgMCAyLS45IDItMlY2YzAtMS4xLS45LTItMi0yaC0zLjE3TDE1IDJ6bS0zIDdjMS42NSAwIDMgMS4zNSAzIDNzLTEuMzUgMy0zIDMtMy0xLjM1LTMtMyAxLjM1LTMgMy0zbTAtMmMtMi43NiAwLTUgMi4yNC01IDVzMi4yNCA1IDUgNSA1LTIuMjQgNS01LTIuMjQtNS01LTV6Ii8+PC9zdmc+';
- Editor.tagsImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGVuYWJsZS1iYWNrZ3JvdW5kPSJuZXcgMCAwIDI0IDI0IiBoZWlnaHQ9IjE4cHgiIHZpZXdCb3g9IjAgMCAyNCAyNCIgd2lkdGg9IjE4cHgiIGZpbGw9IiMwMDAwMDAiPjxnPjxwYXRoIGQ9Ik0wLDBoMjR2MjRIMFYweiIgZmlsbD0ibm9uZSIvPjwvZz48Zz48Zz48cGF0aCBkPSJNMjEuNDEsMTEuNDFsLTguODMtOC44M0MxMi4yMSwyLjIxLDExLjcsMiwxMS4xNywySDRDMi45LDIsMiwyLjksMiw0djcuMTdjMCwwLjUzLDAuMjEsMS4wNCwwLjU5LDEuNDFsOC44Myw4LjgzIGMwLjc4LDAuNzgsMi4wNSwwLjc4LDIuODMsMGw3LjE3LTcuMTdDMjIuMiwxMy40NiwyMi4yLDEyLjIsMjEuNDEsMTEuNDF6IE0xMi44MywyMEw0LDExLjE3VjRoNy4xN0wyMCwxMi44M0wxMi44MywyMHoiLz48Y2lyY2xlIGN4PSI2LjUiIGN5PSI2LjUiIHI9IjEuNSIvPjwvZz48L2c+PC9zdmc+';
- Editor.darkModeImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMjQiIHdpZHRoPSIyNCI+PHBhdGggZD0iTTEyIDIxcS0zLjc1IDAtNi4zNzUtMi42MjVUMyAxMnEwLTMuNzUgMi42MjUtNi4zNzVUMTIgM3EuMzUgMCAuNjg4LjAyNS4zMzcuMDI1LjY2Mi4wNzUtMS4wMjUuNzI1LTEuNjM3IDEuODg3UTExLjEgNi4xNSAxMS4xIDcuNXEwIDIuMjUgMS41NzUgMy44MjVRMTQuMjUgMTIuOSAxNi41IDEyLjlxMS4zNzUgMCAyLjUyNS0uNjEzIDEuMTUtLjYxMiAxLjg3NS0xLjYzNy4wNS4zMjUuMDc1LjY2MlEyMSAxMS42NSAyMSAxMnEwIDMuNzUtMi42MjUgNi4zNzVUMTIgMjFabTAtMnEyLjIgMCAzLjk1LTEuMjEyIDEuNzUtMS4yMTMgMi41NS0zLjE2My0uNS4xMjUtMSAuMi0uNS4wNzUtMSAuMDc1LTMuMDc1IDAtNS4yMzgtMi4xNjJROS4xIDEwLjU3NSA5LjEgNy41cTAtLjUuMDc1LTF0LjItMXEtMS45NS44LTMuMTYyIDIuNTVRNSA5LjggNSAxMnEwIDIuOSAyLjA1IDQuOTVROS4xIDE5IDEyIDE5Wm0tLjI1LTYuNzVaIi8+PC9zdmc+';
- Editor.lightModeImage = 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGhlaWdodD0iMjQiIHdpZHRoPSIyNCI+PHBhdGggZD0iTTEyIDE1cTEuMjUgMCAyLjEyNS0uODc1VDE1IDEycTAtMS4yNS0uODc1LTIuMTI1VDEyIDlxLTEuMjUgMC0yLjEyNS44NzVUOSAxMnEwIDEuMjUuODc1IDIuMTI1VDEyIDE1Wm0wIDJxLTIuMDc1IDAtMy41MzctMS40NjNRNyAxNC4wNzUgNyAxMnQxLjQ2My0zLjUzOFE5LjkyNSA3IDEyIDd0My41MzggMS40NjJRMTcgOS45MjUgMTcgMTJxMCAyLjA3NS0xLjQ2MiAzLjUzN1ExNC4wNzUgMTcgMTIgMTdaTTIgMTNxLS40MjUgMC0uNzEyLS4yODhRMSAxMi40MjUgMSAxMnQuMjg4LS43MTNRMS41NzUgMTEgMiAxMWgycS40MjUgMCAuNzEzLjI4N1E1IDExLjU3NSA1IDEydC0uMjg3LjcxMlE0LjQyNSAxMyA0IDEzWm0xOCAwcS0uNDI1IDAtLjcxMi0uMjg4UTE5IDEyLjQyNSAxOSAxMnQuMjg4LS43MTNRMTkuNTc1IDExIDIwIDExaDJxLjQyNSAwIC43MTIuMjg3LjI4OC4yODguMjg4LjcxM3QtLjI4OC43MTJRMjIuNDI1IDEzIDIyIDEzWm0tOC04cS0uNDI1IDAtLjcxMi0uMjg4UTExIDQuNDI1IDExIDRWMnEwLS40MjUuMjg4LS43MTNRMTEuNTc1IDEgMTIgMXQuNzEzLjI4N1ExMyAxLjU3NSAxMyAydjJxMCAuNDI1LS4yODcuNzEyUTEyLjQyNSA1IDEyIDVabTAgMThxLS40MjUgMC0uNzEyLS4yODhRMTEgMjIuNDI1IDExIDIydi0ycTAtLjQyNS4yODgtLjcxMlExMS41NzUgMTkgMTIgMTl0LjcxMy4yODhRMTMgMTkuNTc1IDEzIDIwdjJxMCAuNDI1LS4yODcuNzEyUTEyLjQyNSAyMyAxMiAyM1pNNS42NSA3LjA1IDQuNTc1IDZxLS4zLS4yNzUtLjI4OC0uNy4wMTMtLjQyNS4yODgtLjcyNS4zLS4zLjcyNS0uM3QuNy4zTDcuMDUgNS42NXEuMjc1LjMuMjc1LjcgMCAuNC0uMjc1LjctLjI3NS4zLS42ODcuMjg3LS40MTMtLjAxMi0uNzEzLS4yODdaTTE4IDE5LjQyNWwtMS4wNS0xLjA3NXEtLjI3NS0uMy0uMjc1LS43MTIgMC0uNDEzLjI3NS0uNjg4LjI3NS0uMy42ODgtLjI4Ny40MTIuMDEyLjcxMi4yODdMMTkuNDI1IDE4cS4zLjI3NS4yODguNy0uMDEzLjQyNS0uMjg4LjcyNS0uMy4zLS43MjUuM3QtLjctLjNaTTE2Ljk1IDcuMDVxLS4zLS4yNzUtLjI4Ny0uNjg4LjAxMi0uNDEyLjI4Ny0uNzEyTDE4IDQuNTc1cS4yNzUtLjMuNy0uMjg4LjQyNS4wMTMuNzI1LjI4OC4zLjMuMy43MjV0LS4zLjdMMTguMzUgNy4wNXEtLjMuMjc1LS43LjI3NS0uNCAwLS43LS4yNzVaTTQuNTc1IDE5LjQyNXEtLjMtLjMtLjMtLjcyNXQuMy0uN2wxLjA3NS0xLjA1cS4zLS4yNzUuNzEzLS4yNzUuNDEyIDAgLjY4Ny4yNzUuMy4yNzUuMjg4LjY4OC0uMDEzLjQxMi0uMjg4LjcxMkw2IDE5LjQyNXEtLjI3NS4zLS43LjI4Ny0uNDI1LS4wMTItLjcyNS0uMjg3Wk0xMiAxMloiLz48L3N2Zz4=';
- Editor.spinImage = 'data:image/gif;base64,R0lGODlhDAAMAPUxAEVriVp7lmCAmmGBm2OCnGmHn3OPpneSqYKbr4OcsIScsI2kto6kt46lt5KnuZmtvpquvpuvv56ywaCzwqK1xKu7yay9yq+/zLHAzbfF0bjG0bzJ1LzK1MDN18jT28nT3M3X3tHa4dTc49Xd5Njf5dng5t3k6d/l6uDm6uru8e7x8/Dz9fT29/b4+Pj5+fj5+vr6+v///wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAkKADEAIf8LTkVUU0NBUEUyLjADAQAAACwAAAAADAAMAAAGR8CYcEgsOgYAIax4CCQuQldrCBEsiK8VS2hoFGOrlJDA+cZQwkLnqyoJFZKviSS0ICrE0ec0jDAwIiUeGyBFGhMPFBkhZo1BACH5BAkKAC4ALAAAAAAMAAwAhVB0kFR3k1V4k2CAmmWEnW6Lo3KOpXeSqH2XrIOcsISdsImhtIqhtJCmuJGnuZuwv52wwJ+ywZ+ywqm6yLHBzbLCzrXEz7fF0LnH0rrI0r7L1b/M1sXR2cfT28rV3czW3s/Z4Nfe5Nvi6ODm6uLn6+Ln7OLo7OXq7efs7+zw8u/y9PDy9PX3+Pr7+////wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAZDQJdwSCxGDAIAoVFkFBwYSyIwGE4OkCJxIdG6WkJEx8sSKj7elfBB0a5SQg1EQ0SVVMPKhDM6iUIkRR4ZFxsgJl6JQQAh+QQJCgAxACwAAAAADAAMAIVGa4lcfZdjgpxkg51nhp5ui6N3kqh5lKqFnbGHn7KIoLOQp7iRp7mSqLmTqbqarr6br7+fssGitcOitcSuvsuuv8uwwMyzw861xNC5x9K6x9K/zNbDztjE0NnG0drJ1NzQ2eDS2+LT2+LV3ePZ4Oba4ebb4ufc4+jm6+7t8PLt8PPt8fPx8/Xx9PX09vf19/j3+Pn///8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGQ8CYcEgsUhQFggFSjCQmnE1jcBhqGBXiIuAQSi7FGEIgfIzCFoCXFCZiPO0hKBMiwl7ET6eUYqlWLkUnISImKC1xbUEAIfkECQoAMgAsAAAAAAwADACFTnKPT3KPVHaTYoKcb4yjcY6leZSpf5mtgZuvh5+yiqG0i6K1jqW3kae5nrHBnrLBn7LCoLPCobTDqbrIqrvIs8LOtMPPtcPPtcTPuMbRucfSvcrUvsvVwMzWxdHaydTcytXdzNbezdff0drh2ODl2+Ln3eTp4Obq4ujs5Ont5uvu6O3w6u7w6u7x7/L09vj5+vr7+vv7////AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABkdAmXBILHIcicOCUqxELKKPxKAYgiYd4oMAEWo8RVmjIMScwhmBcJMKXwLCECmMGAhPI1QRwBiaSixCMDFhLSorLi8wYYxCQQAh+QQJCgAxACwAAAAADAAMAIVZepVggJphgZtnhp5vjKN2kah3kqmBmq+KobSLorWNpLaRp7mWq7ybr7+gs8KitcSktsWnuManucexwM2ywc63xtG6yNO9ytS+ytW/zNbDz9jH0tvL1d3N197S2+LU3OPU3ePV3eTX3+Xa4efb4ufd5Onl6u7r7vHs7/Lt8PLw8/Xy9Pby9fb09ff2+Pn3+Pn6+vr///8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGSMCYcEgseiwSR+RS7GA4JFGF8RiWNiEiJTERgkjFGAQh/KTCGoJwpApnBkITKrwoCFWnFlEhaAxXLC9CBwAGRS4wQgELYY1CQQAh+QQJCgAzACwAAAAADAAMAIVMcI5SdZFhgZtti6JwjaR4k6mAma6Cm6+KobSLorWLo7WNo7aPpredsMCescGitMOitcSmuMaqu8ixwc2zws63xdC4xtG5x9K9ytXAzdfCztjF0NnF0drK1d3M1t7P2N/P2eDT2+LX3+Xe5Onh5+vi5+vj6Ozk6e3n7O/o7O/q7vHs7/Lt8PPu8fPx8/X3+Pn6+vv7+/v8/Pz///8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGRcCZcEgsmkIbTOZTLIlGqZNnchm2SCgiJ6IRqljFmQUiXIVnoITQde4chC9Y+LEQxmTFRkFSNFAqDAMIRQoCAAEEDmeLQQAh+QQJCgAwACwAAAAADAAMAIVXeZRefplff5lhgZtph59yjqV2kaeAmq6FnbGFnrGLorWNpLaQp7mRqLmYrb2essGgs8Klt8apusitvcquv8u2xNC7yNO8ydS8ytTAzdfBzdfM1t7N197Q2eDU3OPX3+XZ4ObZ4ebc4+jf5erg5erg5uvp7fDu8fPv8vTz9fb09vf19/j3+Pn4+fn5+vr6+/v///8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGRUCYcEgspkwjEKhUVJ1QsBNp0xm2VixiSOMRvlxFGAcTJook5eEHIhQcwpWIkAFQECkNy9AQWFwyEAkPRQ4FAwQIE2llQQAh+QQJCgAvACwAAAAADAAMAIVNcY5SdZFigptph6BvjKN0kKd8lquAmq+EnbGGn7KHn7ONpLaOpbearr+csMCdscCescGhtMOnuMauvsuzws60w862xdC9ytW/y9a/zNbCztjG0drH0tvK1N3M1t7N19/U3ePb4uff5urj6Ozk6e3l6u7m6u7o7PDq7vDt8PPv8vTw8vTw8/X19vf6+vv///8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGQ8CXcEgsvlytVUplJLJIpSEDUESFTELBwSgCCQEV42kjDFiMo4uQsDB2MkLHoEHUTD7DRAHC8VAiZ0QSCgYIDxhNiUEAOw==';
- Editor.errorImage = 'data:image/gif;base64,R0lGODlhEAAQAPcAAADGAIQAAISEhP8AAP///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////yH5BAEAAAAALAAAAAAQABAAAAhoAAEIFBigYMGBCAkGGMCQ4cGECxtKHBAAYUQCEzFSHLiQgMeGHjEGEAAg4oCQJz86LCkxpEqHAkwyRClxpEyXGmGaREmTIsmOL1GO/DkzI0yOE2sKIMlRJsWhCQHENDiUaVSpS5cmDAgAOw==';
- Editor.smallPlusImage = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyJpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMC1jMDYwIDYxLjEzNDc3NywgMjAxMC8wMi8xMi0xNzozMjowMCAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNSBNYWNpbnRvc2giIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6MDdCMTdENjVCOEM4MTFFNDlCRjVBNDdCODU5NjNBNUMiIHhtcE1NOkRvY3VtZW50SUQ9InhtcC5kaWQ6MDdCMTdENjZCOEM4MTFFNDlCRjVBNDdCODU5NjNBNUMiPiA8eG1wTU06RGVyaXZlZEZyb20gc3RSZWY6aW5zdGFuY2VJRD0ieG1wLmlpZDowN0IxN0Q2M0I4QzgxMUU0OUJGNUE0N0I4NTk2M0E1QyIgc3RSZWY6ZG9jdW1lbnRJRD0ieG1wLmRpZDowN0IxN0Q2NEI4QzgxMUU0OUJGNUE0N0I4NTk2M0E1QyIvPiA8L3JkZjpEZXNjcmlwdGlvbj4gPC9yZGY6UkRGPiA8L3g6eG1wbWV0YT4gPD94cGFja2V0IGVuZD0iciI/PtjrjmgAAAAtSURBVHjaYvz//z8DMigvLwcLdHZ2MiKLMzEQCaivkLGsrOw/dU0cAr4GCDAARQsQbTFrv10AAAAASUVORK5CYII=';
- Editor.hiResImage = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAFAAAAA+CAMAAACLMWy1AAAAh1BMVEUAAABMTExERERBQUFBQUFFRUVAQEBCQkJAQEA6OjpDQ0NKSkpBQUFBQUFERERERERBQUFCQkJCQkJCQkJJSUlBQUFCQkJDQ0NDQ0NCQkJDQ0NBQUFBQUFCQkJBQUFCQkJCQkJDQ0NCQkJHR0dBQUFCQkJCQkJAQEBCQkJDQ0NAQEBERERCQkIk1hS2AAAAKnRSTlMAAjj96BL7PgQFRwfu3TYazKuVjRXl1V1DPCn1uLGjnWNVIgy9hU40eGqPkM38AAACG0lEQVRYw+2X63KbMBCFzwZblgGDceN74muatpLe//m6MHV3gHGFAv2RjM94MAbxzdnVsQbBDKwH8AH8MDAyafzjqYeyOG04XE7RS8nIRDXg6BlT+rA0nmtAPh+NQRDxIASIMG44rAMrGunBgHwy3uUldxggIStGKp2f+DQc2O4h4eQsX3O2IFB/oEbsjOKbStnjAEA+zJ0ylZTbgvoDn8xNyn6Dbj5Kd4GsNpABa6duQPfSdEj88TgMAhKuCWjAkgmFXPLnsD0pWd3OFGdrMugQII/eOMPEiGOzqPMIeWrcSoMCg71W1pXBPvCP+gS/OdXqQ3uW23+93XGWLl/OaBb805bNcBPoEIcVJsnHzcxpZH86u5KZ9gDby5dQCcnKqdbke4ItI4Tzd7IW9hZQt4EO6GG9b9sYuuK9Wwn8TIr2xKbF2+3Nhr+qxChJ/AI6pIfCu4z4Zowp4ZUNihz79vewzctnHDwTvQO/hCdFBzrUGDOPn2Y/F8YKT4oOATLvlhOznzmBSdFBJWtc58y7r+UVFOCQczy3wpN6pegDqHtsCPTGvH9JuTO0Dyg8icldYPk+RB6g8Aofj4m2EKBvtTmUPD9xDd1pPcSReV2U5iD/ik2yrngtvvqBfPzOvKiDTKTsCdoHZJ7pLLffgTwlJ5vJdtJV2/jiAYaLvLGhMAEDO5QcDg2M/jOw/8Zn+K3ZwJvHT7ZffgC/NvA3zcybTeIfE4EAAAAASUVORK5CYII=';
- Editor.loResImage = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAFAAAAA+CAMAAACLMWy1AAAAS1BMVEVAQEAAAAA1NTVBQUFDQ0NDQ0NFRUVERERBQUFBQUFBQUFAQEBBQUFBQUFCQkJCQkJCQkJBQUFCQkJDQ0NDQ0NCQkJCQkJCQkJGRkb5/XqTAAAAGXRSTlP+AAWODlASCsesX+Lc2LyWe3pwa1tCPjohjSJfoAAAAI1JREFUWMPt1MkKhTAMRuG0anvneXr/J71nUypKcdqI/N8yhLMKMZE1CahnClDQzMPB44ED3EgeCubgDWnWQMHpwTtKwTe+UHD4sJ94wbUEHHFGhILlYDeSnsQeabeCgsPBgB0MOZZ9oGA5GJFiJSfUULAfjLjARrhCwX7wh2YCDwVbwZkUBKqFFJRN+wOcwSgR2sREcgAAAABJRU5ErkJggg==';
- Editor.blankImage = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAACklEQVR4nGMAAQAABQABDQottAAAAABJRU5ErkJggg==';
- Editor.facebookImage = IMAGE_PATH + '/facebook.png';
- Editor.tweetImage = IMAGE_PATH + '/tweet.png';
- /**
- * Broken image symbol for offline SVG.
- */
- Editor.svgBrokenImage = Graph.createSvgImage(10, 10, '<rect x="0" y="0" width="10" height="10" stroke="#000" fill="transparent"/><path d="m 0 0 L 10 10 L 0 10 L 10 0" stroke="#000" fill="transparent"/>');
- /**
- * Error image for not found images
- */
- Editor.configurationKey = '.configuration';
-
- /**
- * Error image for not found images
- */
- Editor.settingsKey = '.drawio-config';
-
- /**
- * Default value for custom libraries in mxSettings.
- */
- Editor.defaultCustomLibraries = [];
-
- /**
- * Default value for custom libraries in mxSettings.
- */
- Editor.enableCustomLibraries = true;
-
- /**
- * Not yet implemented. Reading uncompressed supported.
- */
- Editor.enableUncompressedLibraries = false;
-
- /**
- * Specifies if custom properties should be enabled.
- */
- Editor.enableCustomProperties = true;
-
- /**
- * Specifies if the simple theme should be enabled. This theme can be used
- * at runtime in the kennedy theme.
- */
- Editor.enableSimpleTheme = true;
-
- /**
- * Specifies if the URL should be rewritten to contain the selected page.
- * Default is true for online app without embed.
- */
- Editor.enableHashObjects = !mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp &&
- urlParams['embed'] != '1' && window.top == window.self;
-
- /**
- * Sets the default value for including a copy of the diagram.
- * Default is true.
- */
- Editor.defaultIncludeDiagram = true;
- /**
- * Specifies if custom properties should be enabled.
- */
- Editor.enableServiceWorker = urlParams['pwa'] != '0' &&
- 'serviceWorker' in navigator && (urlParams['offline'] == '1' ||
- /.*\.diagrams\.net$/.test(window.location.hostname) ||
- /.*\.draw\.io$/.test(window.location.hostname));
- // Checks access to service worker in sandboxed context
- try
- {
- Editor.enableServiceWorker && navigator.serviceWorker;
- }
- catch (e)
- {
- Editor.enableServiceWorker = false;
- }
- /**
- * Specifies if web fonts are enabled.
- */
- Editor.enableWebFonts = urlParams['safe-style-src'] != '1' && !window.mxIsElectron;
- /**
- * Disables the shadow option in the format panel.
- */
- Editor.enableShadowOption = !mxClient.IS_SF;
- /**
- * Disables the export URL function.
- */
- Editor.enableExportUrl = true;
- /**
- * Disables fast real time collaboration while keeping slower real time collaboration enabled.
- */
- Editor.enableRealtime = true;
- /**
- * Enables cache for patches and Pusher for messages. Default is true.
- */
- Editor.enableRealtimeCache = true;
-
- /**
- * Enables P2P instead of Pusher for messages. (Ignored if enableRealtimeCache is false.)
- * Default is false.
- */
- Editor.p2pSyncNotify = false;
-
- /**
- * Specifies if XML files should be compressed. Default is true.
- */
- Editor.compressXml = true;
- /**
- * Specifies if XML files should be compressed by default. Default is false.
- */
- Editor.defaultCompressed = false;
-
- /**
- * Specifies if XML files should be compressed. Default is true.
- */
- Editor.oneDriveInlinePicker = (window.urlParams != null && window.urlParams['inlinePicker'] == '0') ? false : true;
- /**
- * Specifies global variables.
- */
- Editor.globalVars = null;
- /**
- * Default border for image export (to allow for sketch style).
- */
- Editor.defaultBorder = 5;
- /**
- * Specifies if cell metadata should be added to SVG output. Default is false.
- */
- Editor.addSvgMetadata = false;
- /**
- * Specifies animations should be enabled. Default is true.
- */
- Editor.enableAnimations = true;
- /**
- * Specifies if ChatGPT should be enabled. Default is true only
- * on app.diagrams.net (including test and preprod).
- */
- Editor.enableChatGpt = (/test\.draw\.io$/.test(window.location.hostname)) ||
- (/preprod\.diagrams\.net$/.test(window.location.hostname)) ||
- (/app\.diagrams\.net$/.test(window.location.hostname));
- /**
- * Specifies the ChatGPT API key. Default is null.
- */
- Editor.gptApiKey = (urlParams['gpt-api-key'] != null) ?
- decodeURIComponent(urlParams['gpt-api-key']) : null;
- /**
- * Specifies the ChatGPT model. Default is 'gpt-3.5-turbo'.
- */
- Editor.gptModel = (urlParams['gpt-model'] != null) ?
- decodeURIComponent(urlParams['gpt-model']) : 'gpt-3.5-turbo';
-
- /**
- * Specifies the ChatGPT endpoint URL. Default is
- * 'https://api.openai.com/v1/chat/completions'.
- */
- Editor.gptUrl = (urlParams['gpt-url'] != null) ?
- decodeURIComponent(urlParams['gpt-url']) :
- 'https://api.openai.com/v1/chat/completions';
-
- /**
- * Specifies if data URIs should be replaced with SVG sub-trees in SVG export.
- * Default is true.
- */
- Editor.replaceSvgDataUris = true;
-
- /**
- * Specifies if foreignObject alternate content should be replaced with an image
- * of the HTML text. Default is true.
- */
- Editor.foreignObjectImages = true;
-
- /**
- * Specifies the scale used to rasterize SVG images. Default is 4.
- */
- Editor.svgRasterScale = 4;
-
- /**
- * Specifies the scale used to rasterize HTML markup. Default is 4.
- */
- Editor.htmlRasterScale = 4;
-
- /**
- * Reference to the config object passed to <configure>.
- */
- Editor.config = null;
- /**
- * Reference to the version of the last config object in
- * <configure>. If this is different to the last version in
- * mxSettings.parse, then the settings are reset.
- */
- Editor.configVersion = null;
- /**
- * Common properties for all edges.
- */
- Editor.commonProperties = [
- {name: 'enumerate', dispName: 'Enumerate', type: 'bool', defVal: false, onChange: function(graph)
- {
- graph.refresh();
- }},
- {name: 'enumerateValue', dispName: 'Enumerate Value', type: 'string', defVal: '', isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'enumerate', '0') == '1';
- }},
- {name: 'comic', dispName: 'Comic', type: 'bool', defVal: false, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', '0') != '1';
- }},
- {name: 'jiggle', dispName: 'Jiggle', type: 'float', min: 0, defVal: 1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'comic', '0') == '1' ||
- mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
- }},
- {name: 'fillWeight', dispName: 'Fill Weight', type: 'int', defVal: -1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'hachureGap', dispName: 'Hachure Gap', type: 'int', defVal: -1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'hachureAngle', dispName: 'Hachure Angle', type: 'int', defVal: -41, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'curveFitting', dispName: 'Curve Fitting', type: 'float', defVal: 0.95, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
- }},
- {name: 'simplification', dispName: 'Simplification', type: 'float', defVal: 0, min: 0, max: 1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
- }},
- {name: 'disableMultiStroke', dispName: 'Disable Multi Stroke', type: 'bool', defVal: false, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
- }},
- {name: 'disableMultiStrokeFill', dispName: 'Disable Multi Stroke Fill', type: 'bool', defVal: false, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'dashOffset', dispName: 'Dash Offset', type: 'int', defVal: -1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'dashGap', dispName: 'Dash Gap', type: 'int', defVal: -1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'zigzagOffset', dispName: 'ZigZag Offset', type: 'int', defVal: -1, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1' &&
- state.vertices.length > 0;
- }},
- {name: 'sketchStyle', dispName: 'Sketch Style', type: 'enum', defVal: 'rough',
- enumList: [{val: 'rough', dispName: 'Rough'}, {val: 'comic', dispName: 'Comic'}],
- isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, 'sketch', (urlParams['rough'] == '1') ? '1' : '0') == '1';
- }},
- {name: mxConstants.STYLE_SHADOWCOLOR, dispName: 'Shadow Color', type: 'color', getDefaultValue: function()
- {
- return mxConstants.SHADOWCOLOR;
- }, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_SHADOW, '0') == '1' ||
- mxUtils.getValue(state.style, mxConstants.STYLE_TEXT_SHADOW, '0') == '1';
- }},
- {name: mxConstants.STYLE_SHADOW_OPACITY, dispName: 'Shadow Opacity', type: 'int', min: 0, max: 100, getDefaultValue: function()
- {
- return mxConstants.SHADOW_OPACITY * 100;
- }, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_SHADOW, '0') == '1' ||
- mxUtils.getValue(state.style, mxConstants.STYLE_TEXT_SHADOW, '0') == '1';
- }},
- {name: mxConstants.STYLE_SHADOW_OFFSET_X, dispName: 'Shadow Offset X', type: 'int', getDefaultValue: function()
- {
- return mxConstants.SHADOW_OFFSET_X;
- }, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_SHADOW, '0') == '1' ||
- mxUtils.getValue(state.style, mxConstants.STYLE_TEXT_SHADOW, '0') == '1';
- }},
- {name: mxConstants.STYLE_SHADOW_OFFSET_Y, dispName: 'Shadow Offset Y', type: 'int', getDefaultValue: function()
- {
- return mxConstants.SHADOW_OFFSET_Y;
- }, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_SHADOW, '0') == '1' ||
- mxUtils.getValue(state.style, mxConstants.STYLE_TEXT_SHADOW, '0') == '1';
- }},
- {name: mxConstants.STYLE_SHADOW_BLUR, dispName: 'Shadow Blur', type: 'int', min: 0, getDefaultValue: function()
- {
- return mxConstants.SHADOW_BLUR;
- }, isVisible: function(state, format)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_SHADOW, '0') == '1' ||
- mxUtils.getValue(state.style, mxConstants.STYLE_TEXT_SHADOW, '0') == '1';
- }},
- {name: 'linecap', dispName: 'Line Cap', type: 'enum', defVal: null,
- enumList: [{val: null, dispName: 'Flat'}, {val: 'round', dispName: 'Round'}, {val: 'square', dispName: 'Square'}]
- },
- {name: 'linejoin', dispName: 'Line Join', type: 'enum', defVal: null,
- enumList: [{val: null, dispName: 'Miter'}, {val: 'arcs', dispName: 'Arcs'}, {val: 'bevel', dispName: 'Bevel'},
- {val: 'miter-clip', dispName: 'Miter-Clip'}, {val: 'round', dispName: 'Round'}]
- },
- ];
-
- /**
- * Common properties for all edges.
- */
- Editor.commonEdgeProperties = [
- {type: 'separator'},
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'sourcePortConstraint', dispName: 'Source Constraint', type: 'enum', defVal: 'none',
- enumList: [{val: 'none', dispName: 'None'}, {val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
- },
- {name: 'targetPortConstraint', dispName: 'Target Constraint', type: 'enum', defVal: 'none',
- enumList: [{val: 'none', dispName: 'None'}, {val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
- },
- {name: 'jettySize', dispName: 'Jetty Size', type: 'int', min: 0, defVal: 'auto', allowAuto: true, isVisible: function(state)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_EDGE, null) == 'orthogonalEdgeStyle';
- }},
- {name: 'segment', dispName: 'Segment Size', type: 'int', min: 0, defVal: mxConstants.ENTITY_SEGMENT, isVisible: function(state)
- {
- return mxUtils.getValue(state.style, mxConstants.STYLE_EDGE, null) == 'entityRelationEdgeStyle';
- }},
- {name: 'fillOpacity', dispName: 'Fill Opacity', type: 'int', min: 0, max: 100, defVal: 100},
- {name: 'strokeOpacity', dispName: 'Stroke Opacity', type: 'int', min: 0, max: 100, defVal: 100},
- {name: 'startFill', dispName: 'Start Fill', type: 'bool', defVal: true},
- {name: 'startFillColor', dispName: 'Start Fill Color', type: 'color', defVal: null},
- {name: 'endFill', dispName: 'End Fill', type: 'bool', defVal: true},
- {name: 'endFillColor', dispName: 'End Fill Color', type: 'color', defVal: null},
- {name: 'perimeterSpacing', dispName: 'Terminal Spacing', type: 'float', defVal: 0},
- {name: 'anchorPointDirection', dispName: 'Anchor Direction', type: 'bool', defVal: true},
- {name: 'snapToPoint', dispName: 'Snap to Point', type: 'bool', defVal: false},
- {name: 'dashPattern', dispName: 'Dash Pattern', type: 'numbers', defVal: ''},
- {name: 'fixDash', dispName: 'Fixed Dash', type: 'bool', defVal: false},
- {name: 'flowAnimationDuration', dispName: 'Flow Duration', type: 'int', defVal: 500, isVisible: function(state)
- {
- return mxUtils.getValue(state.style, 'flowAnimation', null) == '1';
- }},
- {name: 'flowAnimationTimingFunction', dispName: 'Flow Timing', type: 'enum', defVal: 'linear',
- enumList: [{val: 'linear', dispName: 'Linear'}, {val: 'ease', dispName: 'Ease'}, {val: 'ease-in', dispName: 'Ease-in'},
- {val: 'ease-out', dispName: 'Ease-out'}, {val: 'ease-in-out', dispName: 'Ease-in-out'}], isVisible: function(state)
- {
- return mxUtils.getValue(state.style, 'flowAnimation', null) == '1';
- }
- },
- {name: 'flowAnimationDirection', dispName: 'Flow Direction', type: 'enum', defVal: 'normal',
- enumList: [{val: 'normal', dispName: 'Normal'}, {val: 'reverse', dispName: 'Reverse'},
- {val: 'alternate', dispName: 'Alternate'}, {val: 'alternate-reverse', dispName: 'Alternate-Reverse'}],
- isVisible: function(state)
- {
- return mxUtils.getValue(state.style, 'flowAnimation', null) == '1';
- }
- },
- {name: 'editable', dispName: 'Editable', type: 'bool', defVal: true},
- {name: 'metaEdit', dispName: 'Edit Dialog', type: 'bool', defVal: false},
- {name: 'backgroundOutline', dispName: 'Background Outline', type: 'bool', defVal: false},
- {name: 'bendable', dispName: 'Bendable', type: 'bool', defVal: true},
- {name: 'movable', dispName: 'Movable', type: 'bool', defVal: true},
- {name: 'cloneable', dispName: 'Cloneable', type: 'bool', defVal: true},
- {name: 'deletable', dispName: 'Deletable', type: 'bool', defVal: true},
- {name: 'noJump', dispName: 'No Jumps', type: 'bool', defVal: false},
- {name: 'ignoreEdge', dispName: 'Ignore Edge', type: 'bool', defVal: false},
- {name: 'orthogonalLoop', dispName: 'Loop Routing', type: 'bool', defVal: false},
- {name: 'orthogonal', dispName: 'Orthogonal', type: 'bool', defVal: false}
- ].concat(Editor.commonProperties);
- /**
- * Common properties for all vertices.
- */
- Editor.commonVertexProperties = [
- {name: 'colspan', dispName: 'Colspan', type: 'int', min: 1, defVal: 1, isVisible: function(state, format)
- {
- var graph = format.editorUi.editor.graph;
-
- return state.vertices.length == 1 && state.edges.length == 0 && graph.isTableCell(state.vertices[0]);
- }},
- {name: 'rowspan', dispName: 'Rowspan', type: 'int', min: 1, defVal: 1, isVisible: function(state, format)
- {
- var graph = format.editorUi.editor.graph;
-
- return state.vertices.length == 1 && state.edges.length == 0 && graph.isTableCell(state.vertices[0]);
- }},
- {type: 'separator'},
- {name: 'resizeLastRow', dispName: 'Resize Last Row', type: 'bool', getDefaultValue: function(state, format)
- {
- var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
- var graph = format.editorUi.editor.graph;
- var style = graph.getCellStyle(cell);
-
- return mxUtils.getValue(style, 'resizeLastRow', '0') == '1';
- }, isVisible: function(state, format)
- {
- var graph = format.editorUi.editor.graph;
-
- return state.vertices.length == 1 && state.edges.length == 0 &&
- graph.isTable(state.vertices[0]);
- }},
- {name: 'resizeLast', dispName: 'Resize Last Column', type: 'bool', getDefaultValue: function(state, format)
- {
- var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
- var graph = format.editorUi.editor.graph;
- var style = graph.getCellStyle(cell);
-
- return mxUtils.getValue(style, 'resizeLast', '0') == '1';
- }, isVisible: function(state, format)
- {
- var graph = format.editorUi.editor.graph;
-
- return state.vertices.length == 1 && state.edges.length == 0 &&
- graph.isTable(state.vertices[0]);
- }},
- {name: 'fillOpacity', dispName: 'Fill Opacity', type: 'int', min: 0, max: 100, defVal: 100},
- {name: 'strokeOpacity', dispName: 'Stroke Opacity', type: 'int', min: 0, max: 100, defVal: 100},
- {name: 'overflow', dispName: 'Text Overflow', defVal: 'visible', type: 'enum',
- enumList: [{val: 'visible', dispName: 'Visible'}, {val: 'hidden', dispName: 'Hidden'}, {val: 'block', dispName: 'Block'},
- {val: 'fill', dispName: 'Fill'}, {val: 'width', dispName: 'Width'}]
- },
- {name: 'noLabel', dispName: 'Hide Label', type: 'bool', defVal: false},
- {name: 'labelPadding', dispName: 'Label Padding', type: 'float', defVal: 0},
- {name: 'direction', dispName: 'Direction', type: 'enum', defVal: 'east',
- enumList: [{val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
- },
- {name: 'portConstraint', dispName: 'Constraint', type: 'enum', defVal: 'none',
- enumList: [{val: 'none', dispName: 'None'}, {val: 'north', dispName: 'North'}, {val: 'east', dispName: 'East'}, {val: 'south', dispName: 'South'}, {val: 'west', dispName: 'West'}]
- },
- {name: 'portConstraintRotation', dispName: 'Rotate Constraint', type: 'bool', defVal: false},
- {name: 'connectable', dispName: 'Connectable', type: 'bool', getDefaultValue: function(state, format)
- {
- var cell = (state.vertices.length > 0 && state.edges.length == 0) ? state.vertices[0] : null;
- var graph = format.editorUi.editor.graph;
-
- return graph.isCellConnectable(cell);
- }, isVisible: function(state, format)
- {
- return state.vertices.length > 0 && state.edges.length == 0;
- }},
- {name: 'allowArrows', dispName: 'Allow Arrows', type: 'bool', defVal: true},
- {name: 'snapToPoint', dispName: 'Snap to Point', type: 'bool', defVal: false},
- {name: 'perimeter', dispName: 'Perimeter', defVal: 'none', type: 'enum',
- enumList: [{val: 'none', dispName: 'None'},
- {val: 'rectanglePerimeter', dispName: 'Rectangle'}, {val: 'ellipsePerimeter', dispName: 'Ellipse'},
- {val: 'rhombusPerimeter', dispName: 'Rhombus'}, {val: 'trianglePerimeter', dispName: 'Triangle'},
- {val: 'hexagonPerimeter2', dispName: 'Hexagon'}, {val: 'lifelinePerimeter', dispName: 'Lifeline'},
- {val: 'orthogonalPerimeter', dispName: 'Orthogonal'}, {val: 'backbonePerimeter', dispName: 'Backbone'},
- {val: 'calloutPerimeter', dispName: 'Callout'}, {val: 'parallelogramPerimeter', dispName: 'Parallelogram'},
- {val: 'trapezoidPerimeter', dispName: 'Trapezoid'}, {val: 'stepPerimeter', dispName: 'Step'},
- {val: 'centerPerimeter', dispName: 'Center'}]
- },
- {name: 'fixDash', dispName: 'Fixed Dash', type: 'bool', defVal: false},
- {name: 'container', dispName: 'Container', type: 'bool', getDefaultValue: function(state, format)
- {
- var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
- var graph = format.editorUi.editor.graph;
- return cell != null && graph.isSwimlane(cell);
- }, isVisible: function(state, format)
- {
- return state.vertices.length == 1 && state.edges.length == 0;
- }},
- {name: 'dropTarget', dispName: 'Drop Target', type: 'bool', getDefaultValue: function(state, format)
- {
- var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
- var graph = format.editorUi.editor.graph;
-
- return cell != null && (graph.isSwimlane(cell) || graph.model.getChildCount(cell) > 0);
- }, isVisible: function(state, format)
- {
- return state.vertices.length == 1 && state.edges.length == 0;
- }},
- {name: 'collapsible', dispName: 'Collapsible', type: 'bool', getDefaultValue: function(state, format)
- {
- var cell = (state.vertices.length == 1 && state.edges.length == 0) ? state.vertices[0] : null;
- var graph = format.editorUi.editor.graph;
-
- return cell != null && ((graph.isContainer(cell) && state.style['collapsible'] != '0') ||
- (!graph.isContainer(cell) && state.style['collapsible'] == '1'));
- }, isVisible: function(state, format)
- {
- return state.vertices.length == 1 && state.edges.length == 0;
- }},
- {name: 'recursiveResize', dispName: 'Resize Children', type: 'bool', defVal: true, isVisible: function(state, format)
- {
- return state.vertices.length == 1 && state.edges.length == 0 &&
- !format.editorUi.editor.graph.isSwimlane(state.vertices[0]) &&
- mxUtils.getValue(state.style, 'childLayout', null) == null;
- }},
- {name: 'expand', dispName: 'Expand', type: 'bool', defVal: true},
- {name: 'part', dispName: 'Part', type: 'bool', defVal: false, isVisible: function(state, format)
- {
- var model = format.editorUi.editor.graph.model;
-
- return (state.vertices.length > 0) ? model.isVertex(model.getParent(state.vertices[0])) : false;
- }},
- {name: 'editable', dispName: 'Editable', type: 'bool', defVal: true},
- {name: 'metaEdit', dispName: 'Edit Dialog', type: 'bool', defVal: false},
- {name: 'backgroundOutline', dispName: 'Background Outline', type: 'bool', defVal: false},
- {name: 'movable', dispName: 'Movable', type: 'bool', defVal: true},
- {name: 'movableLabel', dispName: 'Movable Label', type: 'bool', defVal: false, isVisible: function(state, format)
- {
- var geo = (state.vertices.length > 0) ? format.editorUi.editor.graph.getCellGeometry(state.vertices[0]) : null;
-
- return geo != null && !geo.relative;
- }},
- {name: 'autosize', dispName: 'Autosize', type: 'bool', defVal: false},
- {name: 'autosizeGrid', dispName: 'Autosize Grid', type: 'enum', defVal: null,
- enumList: [{val: null, dispName: 'Default'}, {val: '1', dispName: 'Enabled'}, {val: '0', dispName: 'Disabled'}], isVisible: function(state, format)
- {
- return state.vertices.length > 0 && format.editorUi.editor.graph.isAutoSizeCell(state.vertices[0]);
- }
- },
- {name: 'fixedWidth', dispName: 'Fixed Width', type: 'bool', defVal: false},
- {name: 'resizable', dispName: 'Resizable', type: 'bool', defVal: true},
- {name: 'resizeWidth', dispName: 'Resize Width', type: 'bool', defVal: false},
- {name: 'resizeHeight', dispName: 'Resize Height', type: 'bool', defVal: false},
- {name: 'rotatable', dispName: 'Rotatable', type: 'bool', defVal: true},
- {name: 'cloneable', dispName: 'Cloneable', type: 'bool', defVal: true},
- {name: 'deletable', dispName: 'Deletable', type: 'bool', defVal: true},
- {name: 'treeFolding', dispName: 'Tree Folding', type: 'bool', defVal: false},
- {name: 'treeMoving', dispName: 'Tree Moving', type: 'bool', defVal: false},
- {name: 'pointerEvents', dispName: 'Pointer Events', type: 'bool', defVal: true, isVisible: function(state, format)
- {
- var fillColor = mxUtils.getValue(state.style, mxConstants.STYLE_FILLCOLOR, null);
-
- return !mxShape.forceFilledPointerEvents ||
- (format.editorUi.editor.graph.isSwimlane(state.vertices[0]) ||
- fillColor == null || fillColor == mxConstants.NONE ||
- mxUtils.getValue(state.style, mxConstants.STYLE_FILL_OPACITY, 100) == 0 ||
- mxUtils.getValue(state.style, mxConstants.STYLE_OPACITY, 100) == 0 ||
- state.style['pointerEvents'] != null);
- }},
- {name: 'moveCells', dispName: 'Move Cells on Fold', type: 'bool', defVal: false, isVisible: function(state, format)
- {
- return state.vertices.length > 0 && format.editorUi.editor.graph.isContainer(state.vertices[0]);
- }}
- ].concat(Editor.commonProperties);
- /**
- * Default value for the CSV import dialog.
- */
- Editor.defaultCsvValue = '##\n' +
- '## Example CSV import. Use ## for comments and # for configuration. Paste CSV below.\n' +
- '## The following names are reserved and should not be used (or ignored):\n' +
- '## id, tooltip, placeholder(s), link and label (see below)\n' +
- '##\n' +
- '#\n' +
- '## Node label with placeholders and HTML.\n' +
- '## Default is \'%name_of_first_column%\'.\n' +
- '#\n' +
- '# label: %name%<br><i style="color:gray;">%position%</i><br><a href="mailto:%email%">Email</a>\n' +
- '#\n' +
- '## Node style (placeholders are replaced once).\n' +
- '## Default is the current style for nodes.\n' +
- '#\n' +
- '# style: label;image=%image%;whiteSpace=wrap;html=1;rounded=1;fillColor=%fill%;strokeColor=%stroke%;\n' +
- '#\n' +
- '## Parent style for nodes with child nodes (placeholders are replaced once).\n' +
- '#\n' +
- '# parentstyle: swimlane;whiteSpace=wrap;html=1;childLayout=stackLayout;horizontal=1;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;\n' +
- '#\n' +
- '## Style to be used for objects not in the CSV. If this is - then such objects are ignored,\n' +
- '## else they are created using this as their style, eg. whiteSpace=wrap;html=1;\n' +
- '#\n' +
- '# unknownStyle: -\n' +
- '#\n' +
- '## Optional column name that contains a reference to a named style in styles.\n' +
- '## Default is the current style for nodes.\n' +
- '#\n' +
- '# stylename: -\n' +
- '#\n' +
- '## JSON for named styles of the form {"name": "style", "name": "style"} where style is a cell style with\n' +
- '## placeholders that are replaced once.\n' +
- '#\n' +
- '# styles: -\n' +
- '#\n' +
- '## JSON for variables in styles of the form {"name": "value", "name": "value"} where name is a string\n' +
- '## that will replace a placeholder in a style.\n' +
- '#\n' +
- '# vars: -\n' +
- '#\n' +
- '## Optional column name that contains a reference to a named label in labels.\n' +
- '## Default is the current label.\n' +
- '#\n' +
- '# labelname: -\n' +
- '#\n' +
- '## JSON for named labels of the form {"name": "label", "name": "label"} where label is a cell label with\n' +
- '## placeholders.\n' +
- '#\n' +
- '# labels: -\n' +
- '#\n' +
- '## Uses the given column name as the identity for cells (updates existing cells).\n' +
- '## Default is no identity (empty value or -).\n' +
- '#\n' +
- '# identity: -\n' +
- '#\n' +
- '## Uses the given column name as the parent reference for cells. Default is no parent (empty or -).\n' +
- '## The identity above is used for resolving the reference so it must be specified.\n' +
- '#\n' +
- '# parent: -\n' +
- '#\n' +
- '## Adds a prefix to the identity of cells to make sure they do not collide with existing cells (whose\n' +
- '## IDs are numbers from 0..n, sometimes with a GUID prefix in the context of realtime collaboration).\n' +
- '## Default is csvimport-.\n' +
- '#\n' +
- '# namespace: csvimport-\n' +
- '#\n' +
- '## Connections between rows ("from": source colum, "to": target column).\n' +
- '## Label, style and invert are optional. Defaults are \'\', current style and false.\n' +
- '## If placeholders are used in the style, they are replaced with data from the source.\n' +
- '## An optional placeholders can be set to target to use data from the target instead.\n' +
- '## In addition to label, an optional fromlabel and tolabel can be used to name the column\n' +
- '## that contains the text for the label in the edges source or target (invert ignored).\n' +
- '## In addition to those, an optional source and targetlabel can be used to specify a label\n' +
- '## that contains placeholders referencing the respective columns in the source or target row.\n' +
- '## The label is created in the form fromlabel + sourcelabel + label + tolabel + targetlabel.\n' +
- '## Additional labels can be added by using an optional labels array with entries of the\n' +
- '## form {"label": string, "x": number, "y": number, "dx": number, "dy": number} where\n' +
- '## x is from -1 to 1 along the edge, y is orthogonal, and dx/dy are offsets in pixels.\n' +
- '## An optional placeholders with the string value "source" or "target" can be specified\n' +
- '## to replace placeholders in the additional label with data from the source or target.\n' +
- '## An optional data object can be specified to define the metadata for the connector.\n' +
- '## The target column may contain a comma-separated list of values.\n' +
- '## Multiple connect entries are allowed.\n' +
- '#\n' +
- '# connect: {"from": "manager", "to": "name", "invert": true, "label": "manages", \\\n' +
- '# "style": "curved=1;endArrow=blockThin;endFill=1;fontSize=11;"}\n' +
- '# connect: {"from": "refs", "to": "id", "style": "curved=1;fontSize=11;"}\n' +
- '#\n' +
- '## Node x-coordinate. Possible value is a column name. Default is empty. Layouts will\n' +
- '## override this value.\n' +
- '#\n' +
- '# left: \n' +
- '#\n' +
- '## Node y-coordinate. Possible value is a column name. Default is empty. Layouts will\n' +
- '## override this value.\n' +
- '#\n' +
- '# top: \n' +
- '#\n' +
- '## Node width. Possible value is a number (in px), auto or an @ sign followed by a column\n' +
- '## name that contains the value for the width. Default is auto.\n' +
- '#\n' +
- '# width: auto\n' +
- '#\n' +
- '## Node height. Possible value is a number (in px), auto, width or an @ sign followed by a column\n' +
- '## name that contains the value for the height. Default is auto.\n' +
- '#\n' +
- '# height: auto\n' +
- '#\n' +
- '## Collapsed state for vertices. Possible values are true or false. Default is false.\n' +
- '#\n' +
- '# collapsed: false\n' +
- '#\n' +
- '## Padding for autosize. Default is 0.\n' +
- '#\n' +
- '# padding: -12\n' +
- '#\n' +
- '## Comma-separated list of ignored columns for metadata. (These can be\n' +
- '## used for connections and styles but will not be added as metadata.)\n' +
- '#\n' +
- '# ignore: id,image,fill,stroke,refs,manager\n' +
- '#\n' +
- '## Column to be renamed to link attribute (used as link).\n' +
- '#\n' +
- '# link: url\n' +
- '#\n' +
- '## Spacing between nodes. Default is 40.\n' +
- '#\n' +
- '# nodespacing: 40\n' +
- '#\n' +
- '## Spacing between levels of hierarchical layouts. Default is 100.\n' +
- '#\n' +
- '# levelspacing: 100\n' +
- '#\n' +
- '## Spacing between parallel edges. Default is 40. Use 0 to disable.\n' +
- '#\n' +
- '# edgespacing: 40\n' +
- '#\n' +
- '## Name or JSON of layout. Possible values are auto, none, verticaltree, horizontaltree,\n' +
- '## verticalflow, horizontalflow, organic, circle, orgchart or a JSON string as used in\n' +
- '## Layout, Apply. Default is auto.\n' +
- '#\n' +
- '# layout: auto\n' +
- '#\n' +
- '## ---- CSV below this line. First line are column names. ----\n' +
- 'name,position,id,location,manager,email,fill,stroke,refs,url,image\n' +
- 'Tessa Miller,CFO,emi,Office 1,,me@example.com,default,#6c8ebf,,https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-3-128.png\n' +
- 'Edward Morrison,Brand Manager,emo,Office 2,Tessa Miller,me@example.com,default,#82b366,,https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-10-3-128.png\n' +
- 'Alison Donovan,System Admin,rdo,Office 3,Tessa Miller,me@example.com,default,#82b366,"emo,tva",https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-2-128.png\n' +
- 'Evan Valet,HR Director,tva,Office 4,Tessa Miller,me@example.com,default,#82b366,,https://www.draw.io,https://cdn3.iconfinder.com/data/icons/user-avatars-1/512/users-9-2-128.png\n';
- /**
- * Compresses the given string.
- */
- Editor.createRoughCanvas = function(c)
- {
- var rc = rough.canvas(
- {
- // Provides expected function but return value is not used
- getContext: function()
- {
- return c;
- }
- });
-
- rc.draw = function(drawable)
- {
- var sets = drawable.sets || [];
- var o = drawable.options || this.getDefaultOptions();
- for (var i = 0; i < sets.length; i++)
- {
- var drawing = sets[i];
-
- switch (drawing.type)
- {
- case 'path':
- if (o.stroke != null)
- {
- this._drawToContext(c, drawing, o);
- }
- break;
- case 'fillPath':
- this._drawToContext(c, drawing, o);
- break;
- case 'fillSketch':
- this.fillSketch(c, drawing, o);
- break;
- }
- }
- };
-
- rc.fillSketch = function(ctx, drawing, o)
- {
- var strokeColor = c.state.strokeColor;
- var strokeWidth = c.state.strokeWidth;
- var strokeAlpha = c.state.strokeAlpha;
- var dashed = c.state.dashed;
-
- var fweight = o.fillWeight;
- if (fweight < 0)
- {
- fweight = o.strokeWidth / 2;
- }
- c.setStrokeAlpha(c.state.fillAlpha);
- c.setStrokeColor(o.fill || '');
- c.setStrokeWidth(fweight);
- c.setDashed(false);
-
- this._drawToContext(ctx, drawing, o);
-
- c.setDashed(dashed);
- c.setStrokeWidth(strokeWidth);
- c.setStrokeColor(strokeColor);
- c.setStrokeAlpha(strokeAlpha);
- };
-
- rc._drawToContext = function(ctx, drawing, o)
- {
- ctx.begin();
-
- for (var i = 0; i < drawing.ops.length; i++)
- {
- var item = drawing.ops[i];
- var data = item.data;
-
- switch (item.op)
- {
- case 'move':
- ctx.moveTo(data[0], data[1]);
- break;
- case 'bcurveTo':
- ctx.curveTo(data[0], data[1], data[2], data[3], data[4], data[5]);
- break;
- case 'lineTo':
- ctx.lineTo(data[0], data[1]);
- break;
- }
- };
-
- ctx.end();
-
- if (drawing.type === 'fillPath' && o.filled)
- {
- ctx.fill();
- }
- else
- {
- ctx.stroke();
- }
- };
-
- return rc;
- };
-
- /**
- * Uses RoughJs for drawing comic shapes.
- */
- (function()
- {
- /**
- * Adds handJiggle style (jiggle=n sets jiggle)
- */
- function RoughCanvas(canvas, rc, shape)
- {
- this.canvas = canvas;
- this.rc = rc;
- this.shape = shape;
-
- // Avoids "spikes" in the output
- this.canvas.setLineJoin('round');
- this.canvas.setLineCap('round');
-
- this.originalBegin = this.canvas.begin;
- this.canvas.begin = mxUtils.bind(this, RoughCanvas.prototype.begin);
-
- this.originalEnd = this.canvas.end;
- this.canvas.end = mxUtils.bind(this, RoughCanvas.prototype.end);
-
- this.originalRect = this.canvas.rect;
- this.canvas.rect = mxUtils.bind(this, RoughCanvas.prototype.rect);
-
- this.originalRoundrect = this.canvas.roundrect;
- this.canvas.roundrect = mxUtils.bind(this, RoughCanvas.prototype.roundrect);
-
- this.originalEllipse = this.canvas.ellipse;
- this.canvas.ellipse = mxUtils.bind(this, RoughCanvas.prototype.ellipse);
-
- this.originalLineTo = this.canvas.lineTo;
- this.canvas.lineTo = mxUtils.bind(this, RoughCanvas.prototype.lineTo);
-
- this.originalMoveTo = this.canvas.moveTo;
- this.canvas.moveTo = mxUtils.bind(this, RoughCanvas.prototype.moveTo);
-
- this.originalQuadTo = this.canvas.quadTo;
- this.canvas.quadTo = mxUtils.bind(this, RoughCanvas.prototype.quadTo);
-
- this.originalCurveTo = this.canvas.curveTo;
- this.canvas.curveTo = mxUtils.bind(this, RoughCanvas.prototype.curveTo);
-
- this.originalArcTo = this.canvas.arcTo;
- this.canvas.arcTo = mxUtils.bind(this, RoughCanvas.prototype.arcTo);
-
- this.originalClose = this.canvas.close;
- this.canvas.close = mxUtils.bind(this, RoughCanvas.prototype.close);
-
- this.originalFill = this.canvas.fill;
- this.canvas.fill = mxUtils.bind(this, RoughCanvas.prototype.fill);
-
- this.originalStroke = this.canvas.stroke;
- this.canvas.stroke = mxUtils.bind(this, RoughCanvas.prototype.stroke);
-
- this.originalFillAndStroke = this.canvas.fillAndStroke;
- this.canvas.fillAndStroke = mxUtils.bind(this, RoughCanvas.prototype.fillAndStroke);
-
- this.path = [];
- this.passThrough = false;
- };
-
- RoughCanvas.prototype.moveOp = 'M';
- RoughCanvas.prototype.lineOp = 'L';
- RoughCanvas.prototype.quadOp = 'Q';
- RoughCanvas.prototype.curveOp = 'C';
- RoughCanvas.prototype.closeOp = 'Z';
-
- RoughCanvas.prototype.getStyle = function(stroke, fill)
- {
- // Random seed created from cell ID
- var seed = 1;
- if (this.shape.state != null)
- {
- var str = this.shape.state.cell.id;
-
- if (str != null)
- {
- for (var i = 0; i < str.length; i++)
- {
- seed = ((seed << 5) - seed + str.charCodeAt(i)) << 0;
- }
- }
- }
- var style = {strokeWidth: this.canvas.state.strokeWidth, seed: seed, preserveVertices: true};
- var defs = this.rc.getDefaultOptions();
-
- if (stroke)
- {
- style.stroke = this.canvas.state.strokeColor === mxConstants.NONE ?
- 'transparent' : this.canvas.state.strokeColor;
- }
- else
- {
- style.stroke = mxConstants.NONE;
- }
-
- var gradient = null;
- style.filled = fill;
-
- if (fill)
- {
- style.fill = this.canvas.state.fillColor === mxConstants.NONE ?
- '' : this.canvas.state.fillColor;
- gradient = this.canvas.state.gradientColor === mxConstants.NONE ?
- null : this.canvas.state.gradientColor;
- }
- else
- {
- style.fill = '';
- }
-
- // Applies cell style
- style['bowing'] = mxUtils.getValue(this.shape.style, 'bowing', defs['bowing']);
- style['hachureAngle'] = mxUtils.getValue(this.shape.style, 'hachureAngle', defs['hachureAngle']);
- style['curveFitting'] = mxUtils.getValue(this.shape.style, 'curveFitting', defs['curveFitting']);
- style['roughness'] = mxUtils.getValue(this.shape.style, 'jiggle', defs['roughness']);
- style['simplification'] = mxUtils.getValue(this.shape.style, 'simplification', defs['simplification']);
- style['disableMultiStroke'] = mxUtils.getValue(this.shape.style, 'disableMultiStroke', defs['disableMultiStroke']);
- style['disableMultiStrokeFill'] = mxUtils.getValue(this.shape.style, 'disableMultiStrokeFill', defs['disableMultiStrokeFill']);
-
- var hachureGap = mxUtils.getValue(this.shape.style, 'hachureGap', -1);
- style['hachureGap'] = (hachureGap == 'auto') ? -1 : hachureGap;
- style['dashGap'] = mxUtils.getValue(this.shape.style, 'dashGap', hachureGap);
- style['dashOffset'] = mxUtils.getValue(this.shape.style, 'dashOffset', hachureGap);
- style['zigzagOffset'] = mxUtils.getValue(this.shape.style, 'zigzagOffset', hachureGap);
-
- var fillWeight = mxUtils.getValue(this.shape.style, 'fillWeight', -1);
- style['fillWeight'] = (fillWeight == 'auto') ? -1 : fillWeight;
-
- var fillStyle = mxUtils.getValue(this.shape.style, 'fillStyle', 'auto');
-
- // Dots fill style is disable due to performance problems
- if (fillStyle == 'dots')
- {
- fillStyle = 'auto';
- }
-
- if (fillStyle == 'auto')
- {
- // One of the following backgrounds for solid fill
- var bg = [mxUtils.hex2rgb('#ffffff')];
-
- if (this.shape.state != null)
- {
- bg.push(mxUtils.hex2rgb(this.shape.state.view.graph.shapeBackgroundColor));
- }
- if (Editor.isDarkMode())
- {
- bg.push(mxUtils.hex2rgb(Editor.darkColor));
- }
- fillStyle = (style.fill != null && (gradient != null || mxUtils.indexOf(
- bg, mxUtils.hex2rgb(style.fill)) >= 0)) ? 'solid' : defs['fillStyle'];
- }
- style['fillStyle'] = fillStyle;
-
- return style;
- };
-
- RoughCanvas.prototype.begin = function()
- {
- if (this.passThrough)
- {
- this.originalBegin.apply(this.canvas, arguments);
- }
- else
- {
- this.path = [];
- }
- };
-
- RoughCanvas.prototype.end = function()
- {
- if (this.passThrough)
- {
- this.originalEnd.apply(this.canvas, arguments);
- }
- else
- {
- // do nothing
- }
- };
-
- RoughCanvas.prototype.addOp = function()
- {
- if (this.path != null)
- {
- this.path.push(arguments[0]);
-
- if (arguments.length > 2)
- {
- var s = this.canvas.state;
-
- for (var i = 2; i < arguments.length; i += 2)
- {
- this.lastX = arguments[i - 1];
- this.lastY = arguments[i];
-
- this.path.push(this.canvas.format((this.lastX)));
- this.path.push(this.canvas.format((this.lastY)));
- }
- }
- }
- };
-
- RoughCanvas.prototype.lineTo = function(endX, endY)
- {
- if (this.passThrough)
- {
- this.originalLineTo.apply(this.canvas, arguments);
- }
- else
- {
- this.addOp(this.lineOp, endX, endY);
- this.lastX = endX;
- this.lastY = endY;
- }
- };
-
- RoughCanvas.prototype.moveTo = function(endX, endY)
- {
- if (this.passThrough)
- {
- this.originalMoveTo.apply(this.canvas, arguments);
- }
- else
- {
- this.addOp(this.moveOp, endX, endY);
- this.lastX = endX;
- this.lastY = endY;
- this.firstX = endX;
- this.firstY = endY;
- }
- };
-
- RoughCanvas.prototype.close = function()
- {
- if (this.passThrough)
- {
- this.originalClose.apply(this.canvas, arguments);
- }
- else
- {
- this.addOp(this.closeOp);
- }
- };
-
- RoughCanvas.prototype.quadTo = function(x1, y1, x2, y2)
- {
- if (this.passThrough)
- {
- this.originalQuadTo.apply(this.canvas, arguments);
- }
- else
- {
- this.addOp(this.quadOp, x1, y1, x2, y2);
- this.lastX = x2;
- this.lastY = y2;
- }
- };
-
- RoughCanvas.prototype.curveTo = function(x1, y1, x2, y2, x3, y3)
- {
- if (this.passThrough)
- {
- this.originalCurveTo.apply(this.canvas, arguments);
- }
- else
- {
- this.addOp(this.curveOp, x1, y1, x2, y2, x3, y3);
- this.lastX = x3;
- this.lastY = y3;
- }
- };
-
- RoughCanvas.prototype.arcTo = function(rx, ry, angle, largeArcFlag, sweepFlag, x, y)
- {
- if (this.passThrough)
- {
- this.originalArcTo.apply(this.canvas, arguments);
- }
- else
- {
- var curves = mxUtils.arcToCurves(this.lastX, this.lastY, rx, ry, angle, largeArcFlag, sweepFlag, x, y);
-
- if (curves != null)
- {
- for (var i = 0; i < curves.length; i += 6)
- {
- this.curveTo(curves[i], curves[i + 1], curves[i + 2],
- curves[i + 3], curves[i + 4], curves[i + 5]);
- }
- }
-
- this.lastX = x;
- this.lastY = y;
- }
- };
-
- RoughCanvas.prototype.rect = function(x, y, w, h)
- {
- if (this.passThrough)
- {
- this.originalRect.apply(this.canvas, arguments);
- }
- else
- {
- this.path = [];
- this.nextShape = this.rc.generator.rectangle(x, y, w, h, this.getStyle(true, true));
- }
- };
-
- RoughCanvas.prototype.ellipse = function(x, y, w, h)
- {
- if (this.passThrough)
- {
- this.originalEllipse.apply(this.canvas, arguments);
- }
- else
- {
- this.path = [];
- this.nextShape = this.rc.generator.ellipse(x + w / 2, y + h / 2, w, h, this.getStyle(true, true));
- }
- };
-
- RoughCanvas.prototype.roundrect = function(x, y, w, h, dx, dy)
- {
- if (this.passThrough)
- {
- this.originalRoundrect.apply(this.canvas, arguments);
- }
- else
- {
- this.begin();
- this.moveTo(x + dx, y);
- this.lineTo(x + w - dx, y);
- this.quadTo(x + w, y, x + w, y + dy);
- this.lineTo(x + w, y + h - dy);
- this.quadTo(x + w, y + h, x + w - dx, y + h);
- this.lineTo(x + dx, y + h);
- this.quadTo(x, y + h, x, y + h - dy);
- this.lineTo(x, y + dy);
- this.quadTo(x, y, x + dx, y);
- }
- };
-
- RoughCanvas.prototype.drawPath = function(style)
- {
- if (this.path.length > 0)
- {
- this.passThrough = true;
- try
- {
- this.rc.path(this.path.join(' '), style);
- }
- catch (e)
- {
- // ignore
- }
- this.passThrough = false;
- }
- else if (this.nextShape != null)
- {
- for (var key in style)
- {
- this.nextShape.options[key] = style[key];
- }
-
- if (style['stroke'] == mxConstants.NONE ||
- style['stroke'] == null)
- {
- delete this.nextShape.options['stroke'];
- }
-
- if (!style.filled)
- {
- delete this.nextShape.options['fill'];
- }
-
- this.passThrough = true;
- this.rc.draw(this.nextShape);
- this.passThrough = false;
- }
- };
-
- RoughCanvas.prototype.stroke = function()
- {
- if (this.passThrough)
- {
- this.originalStroke.apply(this.canvas, arguments);
- }
- else
- {
- this.drawPath(this.getStyle(true, false));
- }
- };
-
- RoughCanvas.prototype.fill = function()
- {
- if (this.passThrough)
- {
- this.originalFill.apply(this.canvas, arguments);
- }
- else
- {
- this.drawPath(this.getStyle(false, true));
- }
- };
-
- RoughCanvas.prototype.fillAndStroke = function()
- {
- if (this.passThrough)
- {
- this.originalFillAndStroke.apply(this.canvas, arguments);
- }
- else
- {
- this.drawPath(this.getStyle(true, true));
- }
- };
-
- RoughCanvas.prototype.destroy = function()
- {
- this.canvas.lineTo = this.originalLineTo;
- this.canvas.moveTo = this.originalMoveTo;
- this.canvas.close = this.originalClose;
- this.canvas.quadTo = this.originalQuadTo;
- this.canvas.curveTo = this.originalCurveTo;
- this.canvas.arcTo = this.originalArcTo;
- this.canvas.close = this.originalClose;
- this.canvas.fill = this.originalFill;
- this.canvas.stroke = this.originalStroke;
- this.canvas.fillAndStroke = this.originalFillAndStroke;
- this.canvas.begin = this.originalBegin;
- this.canvas.end = this.originalEnd;
- this.canvas.rect = this.originalRect;
- this.canvas.ellipse = this.originalEllipse;
- this.canvas.roundrect = this.originalRoundrect;
- };
-
- // Returns a new HandJiggle canvas
- mxShape.prototype.createRoughCanvas = function(c)
- {
- return new RoughCanvas(c, Editor.createRoughCanvas(c), this);
- };
-
- // Overrides to include sketch style
- var shapeCreateHandJiggle = mxShape.prototype.createHandJiggle;
- mxShape.prototype.createHandJiggle = function(c)
- {
- if (!this.outline && this.style != null &&
- mxUtils.getValue(this.style, 'sketch', '0') != '0')
- {
- if (mxUtils.getValue(this.style, 'sketchStyle', 'rough') == 'comic')
- {
- return this.createComicCanvas(c);
- }
- else
- {
- return this.createRoughCanvas(c);
- }
- }
- else
- {
- return shapeCreateHandJiggle.apply(this, arguments);
- }
- };
- // Avoids duplicate painting of images
- var imageShapePaintVertexShape = mxImageShape.prototype.paintVertexShape;
- mxImageShape.prototype.paintVertexShape = function(c, x, y, w, h)
- {
- if (c.handJiggle == null || !c.handJiggle.passThrough)
- {
- imageShapePaintVertexShape.apply(this, arguments);
- }
- };
- // Overrides for event handling on transparent background for sketch style
- var shapePaint = mxShape.prototype.paint;
- mxShape.prototype.paint = function(c)
- {
- var addTolerance = c.addTolerance;
- var events = true;
-
- if (this.style != null)
- {
- events = mxUtils.getValue(this.style, mxConstants.STYLE_POINTER_EVENTS, '1') == '1';
- }
-
- if (c.handJiggle != null && c.handJiggle.constructor == RoughCanvas && !this.outline)
- {
- // Save needed for possible transforms applied during paint
- c.save();
- var fill = this.fill;
- var stroke = this.stroke;
- this.fill = null;
- this.stroke = null;
-
- var configurePointerEvents = this.configurePointerEvents;
-
- // Ignores color changes during paint
- var setStrokeColor = c.setStrokeColor;
-
- c.setStrokeColor = function()
- {
- // ignore
- };
-
- var setFillColor = c.setFillColor;
-
- c.setFillColor = function()
- {
- // ignore
- };
-
- // Adds stroke tolerance for plain rendering if filled
- if (!events && fill != null)
- {
- this.configurePointerEvents = function()
- {
- // ignore
- };
- }
-
- c.handJiggle.passThrough = true;
- shapePaint.apply(this, arguments);
- c.handJiggle.passThrough = false;
- c.setFillColor = setFillColor;
- c.setStrokeColor = setStrokeColor;
- this.configurePointerEvents = configurePointerEvents;
- this.stroke = stroke;
- this.fill = fill;
- c.restore();
-
- // Bypasses stroke tolerance for sketched rendering if filled
- if (events && fill != null)
- {
- c.addTolerance = function()
- {
- // ignore
- };
- }
- }
-
- shapePaint.apply(this, arguments);
- c.addTolerance = addTolerance;
- };
- // Overrides glass effect to disable sketch style
- var shapePaintGlassEffect = mxShape.prototype.paintGlassEffect;
- mxShape.prototype.paintGlassEffect = function(c, x, y, w, h, arc)
- {
- if (c.handJiggle != null && c.handJiggle.constructor == RoughCanvas)
- {
- c.handJiggle.passThrough = true;
- shapePaintGlassEffect.apply(this, arguments);
- c.handJiggle.passThrough = false;
- }
- else
- {
- shapePaintGlassEffect.apply(this, arguments);
- }
- };
- })();
- /**
- * Compresses the given string.
- */
- Editor.fastCompress = function(data)
- {
- if (data == null || data.length == 0 || typeof(pako) === 'undefined')
- {
- return data;
- }
- else
- {
- return Graph.arrayBufferToString(pako.deflateRaw(data));
- }
- };
- /**
- * Decompresses the given string.
- */
- Editor.fastDecompress = function(data)
- {
- if (data == null || data.length == 0 || typeof(pako) === 'undefined')
- {
- return data;
- }
- else
- {
- return pako.inflateRaw(Graph.stringToArrayBuffer(atob(data)), {to: 'string'});
- }
- };
- /**
- * Helper function to extract the graph model XML node.
- */
- Editor.extractGraphModel = function(node, allowMxFile, checked)
- {
- if (node != null && typeof(pako) !== 'undefined')
- {
- var tmp = node.ownerDocument.getElementsByTagName('div');
- var divs = [];
-
- if (tmp != null && tmp.length > 0)
- {
- for (var i = 0; i < tmp.length; i++)
- {
- if (tmp[i].getAttribute('class') == 'mxgraph')
- {
- divs.push(tmp[i]);
- break;
- }
- }
- }
-
- if (divs.length > 0)
- {
- var data = divs[0].getAttribute('data-mxgraph');
- if (data != null)
- {
- var config = JSON.parse(data);
- if (config != null && config.xml != null)
- {
- var doc2 = mxUtils.parseXml(config.xml);
- node = doc2.documentElement;
- }
- }
- else
- {
- var divs2 = divs[0].getElementsByTagName('div');
-
- if (divs2.length > 0)
- {
- var data = mxUtils.getTextContent(divs2[0]);
- data = Graph.decompress(data, null, checked);
-
- if (data.length > 0)
- {
- var doc2 = mxUtils.parseXml(data);
- node = doc2.documentElement;
- }
- }
- }
- }
- }
-
- if (node != null && node.nodeName == 'svg')
- {
- var tmp = node.getAttribute('content');
-
- if (tmp != null && tmp.charAt(0) != '<' && tmp.charAt(0) != '%')
- {
- tmp = unescape((window.atob) ? atob(tmp) : Base64.decode(cont, tmp));
- }
-
- if (tmp != null && tmp.charAt(0) == '%')
- {
- tmp = decodeURIComponent(tmp);
- }
-
- if (tmp != null && tmp.length > 0)
- {
- node = mxUtils.parseXml(tmp).documentElement;
- }
- else
- {
- throw {message: mxResources.get('notADiagramFile')};
- }
- }
-
- if (node != null && !allowMxFile)
- {
- var diagramNode = null;
-
- if (node.nodeName == 'diagram')
- {
- diagramNode = node;
- }
- else if (node.nodeName == 'mxfile')
- {
- var diagrams = node.getElementsByTagName('diagram');
- if (diagrams.length > 0)
- {
- diagramNode = diagrams[Math.max(0, Math.min(diagrams.length - 1, urlParams['page'] || 0))];
- }
- }
-
- if (diagramNode != null)
- {
- node = Editor.parseDiagramNode(diagramNode, checked);
- }
- }
-
- if (node != null && node.nodeName != 'mxGraphModel' && (!allowMxFile || node.nodeName != 'mxfile'))
- {
- node = null;
- }
-
- return node;
- };
-
- /**
- * Extracts the XML from the compressed or non-compressed text chunk.
- */
- Editor.parseDiagramNode = function(diagramNode, checked, allowRecurse)
- {
- Editor.validateDiagramNode(diagramNode);
- var text = mxUtils.trim(mxUtils.getTextContent(diagramNode));
- var node = null;
-
- if (text.length > 0)
- {
- var tmp = Graph.decompress(text, null, checked);
-
- if (tmp != null && tmp.length > 0)
- {
- node = mxUtils.parseXml(tmp).documentElement;
- }
- }
- else
- {
- var temp = mxUtils.getChildNodes(diagramNode);
-
- if (temp.length > 0)
- {
- var tempNode = temp[0];
- if (allowRecurse)
- {
- tempNode = Editor.parseDiagramNode(tempNode, checked, false);
- }
- // Creates new document for unique IDs within mxGraphModel
- var doc = mxUtils.createXmlDocument();
- doc.appendChild(doc.importNode(tempNode, true));
- node = doc.documentElement;
- }
- }
-
- return node;
- };
-
- /**
- * Extracts the XML from the compressed or non-compressed text chunk.
- */
- Editor.getDiagramNodeXml = function(diagramNode)
- {
- Editor.validateDiagramNode(diagramNode);
- var text = mxUtils.getNodeValue(diagramNode);
- var xml = null;
-
- if (text.length > 0)
- {
- xml = Graph.decompress(text);
- }
- else
- {
- var temp = diagramNode.getElementsByTagName('mxGraphModel');
- if (temp != null && temp.length > 0)
- {
- xml = mxUtils.getXml(temp[0]);
- }
- }
-
- return xml;
- };
- /**
- * Initializes diagram nodes with no content.
- */
- Editor.validateDiagramNode = function(diagramNode)
- {
- if (mxUtils.trim(mxUtils.getTextContent(diagramNode)).length == 0 &&
- mxUtils.getChildNodes(diagramNode).length == 0)
- {
- var codec = new mxCodec();
- var model = new mxGraphModel();
- diagramNode.appendChild(codec.encode(model));
- }
- return diagramNode;
- };
- /**
- * Static method for parsing PDF files.
- */
- Editor.extractGraphModelFromPdf = function(base64)
- {
- var result = null;
- base64 = base64.substring(base64.indexOf(',') + 1);
- // Workaround for invalid character error in Safari
- var f = (window.atob && !mxClient.IS_SF) ? atob(base64) : Base64.decode(base64, true);
- // Extracts Subject or Embedded file attachment from PDF 1.7
- if (f.substring(0, 8) == '%PDF-1.7')
- {
- var blockStart = f.indexOf('EmbeddedFile');
-
- if (blockStart > -1)
- {
- var streamStart = f.indexOf('stream', blockStart) + 9; //the start of the stream [skipping header check]
- var fileInfo = f.substring(blockStart, streamStart);
-
- if (fileInfo.indexOf('application#2Fvnd.jgraph.mxfile') > 0)
- {
- var streamEnd = f.indexOf('endstream', streamStart - 1);
-
- return pako.inflateRaw(Graph.stringToArrayBuffer(
- f.substring(streamStart, streamEnd)), {to: 'string'});
- }
- }
- var last = f.indexOf('/ObjStm');
- while (last > 0)
- {
- var streamStart = f.indexOf('stream', last) + 9; //the start of the stream [skipping header check]
- var streamEnd = f.indexOf('endstream', streamStart - 1);
-
- function hex_to_ascii(hex)
- {
- var str = [];
-
- for (var n = 0; n < hex.length; n += 2)
- {
- var code = hex.substr(n, 2);
- // Encoded mxfile is URI encoded ASCII
- if (code != '00')
- {
- str.push(String.fromCharCode(parseInt(code, 16)));
- }
- }
- return str.join('');
- };
- var text = pako.inflateRaw(Graph.stringToArrayBuffer(
- f.substring(streamStart, streamEnd)), {to: 'string'});
- var subj = text.indexOf('/Subject <');
- // Extracts Subject from PDF 1.4
- if (subj > 0)
- {
- var temp = text.substring(subj + 14, text.indexOf('>', subj));
- if (temp != null)
- {
- result = hex_to_ascii(temp);
- }
- break;
- }
- last = f.indexOf('/ObjStm', last + 1);
- }
- }
- // Extracts subject from PDF 1.4
- if (result == null && f.substring(0, 8) == '%PDF-1.4')
- {
- var check = '/Subject (%3Cmxfile';
- var curline = '';
- var checked = 0;
- var pos = 0;
- var obj = [];
- var buf = null;
-
- while (pos < f.length)
- {
- var b = f.charCodeAt(pos);
- pos += 1;
-
- if (b != 10)
- {
- curline += String.fromCharCode(b);
- }
-
- if (b == check.charCodeAt(checked))
- {
- checked++;
- }
- else
- {
- checked = 0;
- }
-
- if (checked == check.length)
- {
- var end = f.indexOf('%3C%2Fmxfile%3E', pos) + 15; //15 is the length of encoded </mxfile>
- pos -= 9; //9 is the length of encoded <mxfile
- // Default case is XML inlined in Subject metadata
- if (end > pos)
- {
- result = f.substring(pos, end);
- break;
- }
- }
-
- // Creates table for lookup if no inline data is found
- if (b == 10)
- {
- if (curline == 'endobj')
- {
- buf = null;
- }
- else if (curline.substring(curline.length - 3, curline.length) == 'obj' ||
- curline == 'xref' || curline == 'trailer')
- {
- buf = [];
- obj[curline.split(' ')[0]] = buf;
- }
- else if (buf != null)
- {
- buf.push(curline);
- }
-
- curline = '';
- }
- }
- }
-
- // Extract XML via references
- if (result == null && obj != null)
- {
- result = Editor.extractGraphModelFromXref(obj);
- }
-
- if (result != null)
- {
- result = decodeURIComponent(result.
- replace(/\\\(/g, "(").
- replace(/\\\)/g, ")"));
- }
-
- return result;
- };
- /**
- * Static method for extracting Subject via references of the form
- *
- * << /Size 33 /Root 20 0 R /Info 1 0 R and 1 0 obj << /Subject 22 0 R
- *
- * Where Info is the metadata block and Subject is the data block.
- */
- Editor.extractGraphModelFromXref = function(obj)
- {
- var trailer = obj['trailer'];
- var result = null;
- // Gets Info object
- if (trailer != null)
- {
- var arr = /.* \/Info (\d+) (\d+) R/g.exec(trailer.join('\n'));
-
- if (arr != null && arr.length > 0)
- {
- var info = obj[arr[1]];
-
- if (info != null)
- {
- arr = /.* \/Subject (\d+) (\d+) R/g.exec(info.join('\n'));
-
- if (arr != null && arr.length > 0)
- {
- var subj = obj[arr[1]];
-
- if (subj != null)
- {
- subj = subj.join('\n');
- result = subj.substring(1, subj.length - 1);
- }
- }
- }
- }
- }
-
- return result;
- };
- /**
- * Extracts any parsers errors in the given XML.
- */
- Editor.extractParserError = function(node, defaultCause)
- {
- var cause = null;
- var errors = (node != null) ? node.getElementsByTagName('parsererror') : null;
-
- if (errors != null && errors.length > 0)
- {
- cause = defaultCause || mxResources.get('invalidChars');
- var divs = errors[0].getElementsByTagName('div');
-
- if (divs.length > 0)
- {
- cause = mxUtils.getTextContent(divs[0]);
- }
- }
-
- return (cause != null) ? mxUtils.trim(cause) : cause;
- };
-
- /**
- * Adds the given retry function to the given error.
- */
- Editor.addRetryToError = function(err, retry)
- {
- if (err != null)
- {
- var e = (err.error != null) ? err.error : err;
-
- if (e.retry == null)
- {
- e.retry = retry;
- }
- }
- };
-
- /**
- *
- * Hook for mermaid to draw.io converter.
- */
- Editor.mermaidToDrawio = function(graph, diagramtype, exta)
- {
- if (typeof mxMermaidToDrawio === 'function')
- {
- return mxMermaidToDrawio(graph, diagramtype, exta);
- }
- };
-
- /**
- * Global configuration of the Editor
- * see https://www.drawio.com/doc/faq/configure-diagram-editor
- *
- * For defaultVertexStyle, defaultEdgeStyle and defaultLibraries, this must be called before
- * mxSettings.load via global config variable window.mxLoadSettings = false.
- */
- Editor.configure = function(config)
- {
- if (config != null)
- {
- Editor.config = config;
- Editor.configVersion = config.version;
- // Enables debug output
- if (config.debug)
- {
- urlParams['test'] = '1'
- }
-
- if (config.defaultFonts != null)
- {
- Menus.prototype.defaultFonts = config.defaultFonts
- }
- if (config.presetColors != null)
- {
- ColorDialog.prototype.presetColors = config.presetColors
- }
- if (config.defaultColors != null)
- {
- ColorDialog.prototype.defaultColors = config.defaultColors
- }
- if (config.colorNames != null)
- {
- ColorDialog.prototype.colorNames = config.colorNames
- }
- if (config.defaultColorSchemes != null)
- {
- StyleFormatPanel.prototype.defaultColorSchemes = config.defaultColorSchemes
- }
- if (config.defaultEdgeLength != null)
- {
- Graph.prototype.defaultEdgeLength = config.defaultEdgeLength
- }
- if (config.selectParentLayer != null)
- {
- Graph.selectParentLayer = config.selectParentLayer
- }
- if (config.autosaveDelay != null)
- {
- DrawioFile.prototype.autosaveDelay = config.autosaveDelay
- }
-
- if (config.templateFile != null)
- {
- EditorUi.templateFile = config.templateFile;
- }
-
- if (config.styles != null)
- {
- if (Array.isArray(config.styles))
- {
- Editor.styles = config.styles;
- }
- else
- {
- EditorUi.debug('Configuration Error: Array expected for styles');
- }
- }
-
- if (config.globalVars != null)
- {
- Editor.globalVars = config.globalVars;
- }
- if (config.compressXml != null)
- {
- Editor.defaultCompressed = config.compressXml;
- Editor.compressXml = config.compressXml;
- }
-
- if (config.includeDiagram != null)
- {
- Editor.defaultIncludeDiagram = config.includeDiagram;
- }
-
- if (config.simpleLabels != null)
- {
- Editor.simpleLabels = config.simpleLabels;
- }
- if (config.oneDriveInlinePicker != null)
- {
- Editor.oneDriveInlinePicker = config.oneDriveInlinePicker;
- }
- if (config.enableCssDarkMode != null)
- {
- Editor.enableCssDarkMode = config.enableCssDarkMode;
- }
- if (config.darkColor != null)
- {
- Editor.darkColor = config.darkColor;
- }
- if (config.lightColor != null)
- {
- Editor.lightColor = config.lightColor;
- }
- if (config.settingsName != null)
- {
- Editor.configurationKey = '.' + config.settingsName + '-configuration';
- Editor.settingsKey = '.' + config.settingsName + '-config';
- mxSettings.key = Editor.settingsKey;
- }
-
- if (config.customFonts != null)
- {
- Menus.prototype.defaultFonts = config.customFonts.
- concat(Menus.prototype.defaultFonts);
- }
-
- if (config.customPresetColors != null)
- {
- ColorDialog.prototype.presetColors = config.customPresetColors.
- concat(ColorDialog.prototype.presetColors);
- }
-
- if (config.customColorSchemes != null)
- {
- StyleFormatPanel.prototype.defaultColorSchemes = config.customColorSchemes.
- concat(StyleFormatPanel.prototype.defaultColorSchemes);
- }
-
- // Custom CSS injected directly into the page
- if (config.css != null)
- {
- var s = document.createElement('style');
- s.setAttribute('type', 'text/css');
- s.appendChild(document.createTextNode(config.css));
-
- var t = document.getElementsByTagName('script')[0];
- t.parentNode.insertBefore(s, t);
- }
- if (config.expandLibraries != null)
- {
- Sidebar.prototype.expandLibraries = config.expandLibraries;
- }
-
- if (config.appendCustomLibraries != null)
- {
- Sidebar.prototype.appendCustomLibraries = config.appendCustomLibraries;
- }
-
- // Configures the custom libraries
- if (config.libraries != null)
- {
- Sidebar.prototype.customEntries = config.libraries;
- }
-
- // Defines the enabled built-in libraries.
- if (config.enabledLibraries != null)
- {
- if (Array.isArray(config.enabledLibraries))
- {
- Sidebar.prototype.enabledLibraries = config.enabledLibraries;
- }
- else
- {
- EditorUi.debug('Configuration Error: Array expected for enabledLibraries');
- }
- }
-
- // Overrides default libraries
- if (config.defaultLibraries != null)
- {
- Sidebar.prototype.defaultEntries = config.defaultLibraries;
- }
-
- // Overrides default custom libraries
- if (config.defaultCustomLibraries != null)
- {
- Editor.defaultCustomLibraries = config.defaultCustomLibraries;
- }
-
- // Disables custom libraries
- if (config.enableCustomLibraries != null)
- {
- Editor.enableCustomLibraries = config.enableCustomLibraries;
- }
-
- // Overrides default vertex style
- if (config.defaultVertexStyle != null)
- {
- Graph.prototype.defaultVertexStyle = config.defaultVertexStyle;
- }
- // Overrides default edge style
- if (config.defaultEdgeStyle != null)
- {
- Graph.prototype.defaultEdgeStyle = config.defaultEdgeStyle;
- }
- // Overrides default page visible
- if (config.defaultPageVisible != null)
- {
- Graph.prototype.defaultPageVisible = config.defaultPageVisible;
- }
- // Overrides default grid enabled
- if (config.defaultGridEnabled != null)
- {
- Graph.prototype.defaultGridEnabled = config.defaultGridEnabled;
- }
- // Overrides mouse wheel function
- if (config.zoomWheel != null)
- {
- Graph.zoomWheel = config.zoomWheel;
- }
- // Overrides zoom factor
- if (config.zoomFactor != null)
- {
- var val = parseFloat(config.zoomFactor);
-
- if (!isNaN(val) && val > 1)
- {
- Graph.prototype.zoomFactor = val;
- }
- else
- {
- EditorUi.debug('Configuration Error: Float > 1 expected for zoomFactor');
- }
- }
- // Overrides default grid size
- if (config.defaultGridSize != null)
- {
- var val = parseInt(config.defaultGridSize);
-
- if (!isNaN(val) && val > 0)
- {
- mxGraph.prototype.gridSize = val;
- }
- else
- {
- EditorUi.debug('Configuration Error: Int > 0 expected for defaultGridSize');
- }
- }
- // Overrides grid steps
- if (config.gridSteps != null)
- {
- var val = parseInt(config.gridSteps);
-
- if (!isNaN(val) && val > 0)
- {
- mxGraphView.prototype.gridSteps = val;
- }
- else
- {
- EditorUi.debug('Configuration Error: Int > 0 expected for gridSteps');
- }
- }
- if (config.pageFormat != null)
- {
- var w = parseInt(config.pageFormat.width);
- var h = parseInt(config.pageFormat.height);
- if (!isNaN(w) && w > 0 && !isNaN(h) && h > 0)
- {
- mxGraph.prototype.defaultPageFormat = new mxRectangle(0, 0, w, h);
- mxGraph.prototype.pageFormat = mxGraph.prototype.defaultPageFormat;
- }
- else
- {
- EditorUi.debug('Configuration Error: {width: int, height: int} expected for pageFormat');
- }
- }
-
- if (config.thumbWidth != null)
- {
- Sidebar.prototype.thumbWidth = config.thumbWidth;
- }
-
- if (config.thumbHeight != null)
- {
- Sidebar.prototype.thumbHeight = config.thumbHeight;
- }
-
- if (config.emptyLibraryXml != null)
- {
- EditorUi.prototype.emptyLibraryXml = config.emptyLibraryXml;
- }
- if (config.emptyDiagramXml != null)
- {
- EditorUi.prototype.emptyDiagramXml = config.emptyDiagramXml;
- }
-
- if (config.sidebarWidth != null)
- {
- EditorUi.prototype.hsplitPosition = config.sidebarWidth;
- }
- if (config.updateDefaultStyle != null)
- {
- EditorUi.prototype.updateDefaultStyle = config.updateDefaultStyle;
- }
-
- if (config.sidebarTitles != null)
- {
- Sidebar.prototype.sidebarTitles = config.sidebarTitles;
- }
-
- if (config.sidebarTitleSize != null)
- {
- var val = parseInt(config.sidebarTitleSize);
-
- if (!isNaN(val) && val > 0)
- {
- Sidebar.prototype.sidebarTitleSize = val;
- }
- else
- {
- EditorUi.debug('Configuration Error: Int > 0 expected for sidebarTitleSize');
- }
- }
-
- if (config.fontCss != null)
- {
- if (typeof config.fontCss === 'string')
- {
- Editor.configureFontCss(config.fontCss);
- }
- else
- {
- EditorUi.debug('Configuration Error: String expected for fontCss');
- }
- }
-
- if (config.autosaveDelay != null)
- {
- var val = parseInt(config.autosaveDelay);
-
- if (!isNaN(val) && val > 0)
- {
- DrawioFile.prototype.autosaveDelay = val;
- }
- else
- {
- EditorUi.debug('Configuration Error: Int > 0 expected for autosaveDelay');
- }
- }
- if(config.maxImageBytes != null)
- {
- EditorUi.prototype.maxImageBytes = config.maxImageBytes;
- }
-
- if(config.maxImageSize != null)
- {
- EditorUi.prototype.maxImageSize = config.maxImageSize;
- }
-
- if (config.shareCursorPosition != null)
- {
- EditorUi.prototype.shareCursorPosition = config.shareCursorPosition;
- }
- if (config.showRemoteCursors != null)
- {
- EditorUi.prototype.showRemoteCursors = config.showRemoteCursors;
- }
- if (config.restrictExport != null)
- {
- DrawioFile.RESTRICT_EXPORT = config.restrictExport;
- }
-
- if (config.replaceSvgDataUris != null)
- {
- Editor.replaceSvgDataUris = config.replaceSvgDataUris;
- }
- if (config.foreignObjectImages != null)
- {
- Editor.foreignObjectImages = config.foreignObjectImages;
- }
- if (config.shadowColor != null)
- {
- mxConstants.SHADOW_COLOR = config.shadowColor;
- }
- if (config.shadowOpacity != null)
- {
- mxConstants.SHADOW_OPACITY = config.shadowOpacity;
- }
- if (config.shadowOffsetX != null)
- {
- mxConstants.SHADOW_OFFSET_X = config.shadowOffsetX;
- }
- if (config.shadowOffsetY != null)
- {
- mxConstants.SHADOW_OFFSET_Y = config.shadowOffsetY;
- }
- if (config.shadowBlur != null)
- {
- mxConstants.SHADOW_BLUR = config.shadowBlur;
- }
- if (config.enableAnimations != null)
- {
- Editor.enableAnimations = config.enableAnimations;
- }
-
- if (config.enableChatGpt != null)
- {
- Editor.enableChatGpt = config.enableChatGpt;
- }
- if (config.gptApiKey != null)
- {
- Editor.gptApiKey = config.gptApiKey;
- }
- if (config.gptModel != null)
- {
- Editor.gptModel = config.gptModel;
- }
- if (config.gptUrl != null)
- {
- Editor.gptUrl = config.gptUrl;
- }
- }
- };
-
- /**
- *
- */
- Editor.isSettingsEnabled = function()
- {
- return typeof window.mxSettings !== 'undefined' && (isLocalStorage || mxClient.IS_CHROMEAPP);
- };
-
- /**
- * Adds the global fontCss configuration.
- */
- Editor.configureFontCss = function(fontCss)
- {
- if (fontCss != null)
- {
- Editor.prototype.fontCss = fontCss;
- var t = document.getElementsByTagName('script')[0];
-
- if (t != null && t.parentNode != null)
- {
- var s = document.createElement('style');
- s.setAttribute('type', 'text/css');
- s.appendChild(document.createTextNode(fontCss));
- t.parentNode.insertBefore(s, t);
-
- // Preloads fonts where supported
- var parts = fontCss.split('url(');
-
- for (var i = 1; i < parts.length; i++)
- {
- var idx = parts[i].indexOf(')');
- var url = Editor.trimCssUrl(parts[i].substring(0, idx));
-
- var l = document.createElement('link');
- l.setAttribute('rel', 'preload');
- l.setAttribute('href', url);
- l.setAttribute('as', 'font');
- l.setAttribute('crossorigin', '');
-
- t.parentNode.insertBefore(l, t);
- }
- }
- }
- };
-
- /**
- * Strips leading and trailing quotes and spaces
- */
- Editor.trimCssUrl = function(str)
- {
- return str.replace(new RegExp("^[\\s\"']+", "g"), "").replace(new RegExp("[\\s\"']+$", "g"), "");
- }
-
- /**
- * Prefix for URLs that reference Google fonts.
- */
- Editor.GOOGLE_FONTS = 'https://fonts.googleapis.com/css?family=';
-
- /**
- * Prefix for URLs that reference Google fonts with CSS2.
- */
- Editor.GOOGLE_FONTS_CSS2 = 'https://fonts.googleapis.com/css2?family=';
-
- /**
- * Alphabet for global unique IDs.
- */
- Editor.GUID_ALPHABET = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ-_';
- /**
- * Default length for global unique IDs.
- */
- Editor.GUID_LENGTH = 20;
-
- /**
- * Default length for global unique IDs.
- */
- Editor.guid = function(length)
- {
- var len = (length != null) ? length : Editor.GUID_LENGTH;
- var rtn = [];
-
- for (var i = 0; i < len; i++)
- {
- rtn.push(Editor.GUID_ALPHABET.charAt(Math.floor(Math.random() * Editor.GUID_ALPHABET.length)));
- }
- return rtn.join('');
- };
- /**
- * Interval for updating the file status.
- */
- Editor.updateStatusInterval = 10000;
- /**
- * Specifies the app name. Default is document.title.
- */
- Editor.prototype.appName = 'draw.io';
-
- /**
- * Known file types.
- */
- Editor.prototype.diagramFileTypes = [
- {description: 'diagramXmlDesc', extension: 'drawio', mimeType: 'text/xml'},
- {description: 'diagramPngDesc', extension: 'png', mimeType: 'image/png'},
- {description: 'diagramSvgDesc', extension: 'svg', mimeType: 'image/svg'},
- {description: 'diagramHtmlDesc', extension: 'html', mimeType: 'text/html'}];
-
- /**
- * Known file types.
- */
- Editor.prototype.libraryFileTypes = [{description: 'Library (.drawiolib, .xml)', extensions: ['drawiolib', 'xml']}];
-
- /**
- * Additional help text for special file extensions.
- */
- Editor.prototype.fileExtensions = [
- {ext: 'html', title: 'filetypeHtml'},
- {ext: 'png', title: 'filetypePng'},
- {ext: 'svg', title: 'filetypeSvg'}];
-
- /**
- * General timeout is 25 seconds.
- */
- Editor.prototype.timeout = 25000;
-
- /**
- * Executes the first step for connecting to Google Drive.
- */
- Editor.prototype.editButtonLink = (urlParams['edit'] != null) ? decodeURIComponent(urlParams['edit']) : null;
- /**
- * Specifies if img.crossOrigin is supported. This is true for all browsers except IE10 and earlier.
- */
- Editor.prototype.crossOriginImages = !mxClient.IS_IE;
-
- /**
- * Adds support for old stylesheets and compressed files
- */
- var editorSetGraphXml = Editor.prototype.setGraphXml;
- Editor.prototype.setGraphXml = function(node)
- {
- node = (node != null && node.nodeName != 'mxlibrary') ? this.extractGraphModel(node) : null;
- if (node != null)
- {
- // Checks for parser errors
- var cause = Editor.extractParserError(node, mxResources.get('invalidOrMissingFile'));
- if (cause)
- {
- EditorUi.debug('Editor.setGraphXml ParserError', [this],
- 'node', [node], 'cause', [cause]);
- throw new Error(mxResources.get('notADiagramFile') + ' (' + cause + ')');
- }
- else if (node.nodeName == 'mxGraphModel')
- {
- var style = node.getAttribute('style') || 'default-style2';
-
- // Decodes the style if required
- if (urlParams['embed'] != '1' && (style == null || style == ''))
- {
- var node2 = (this.graph.themes != null) ?
- this.graph.themes['default-old'] :
- mxUtils.load(STYLE_PATH + '/default-old.xml').getDocumentElement();
-
- if (node2 != null)
- {
- var dec2 = new mxCodec(node2.ownerDocument);
- dec2.decode(node2, this.graph.getStylesheet());
- }
- }
- else if (style != this.graph.currentStyle)
- {
- var node2 = (this.graph.themes != null) ?
- this.graph.themes[style] :
- mxUtils.load(STYLE_PATH + '/' + style + '.xml').getDocumentElement()
-
- if (node2 != null)
- {
- var dec2 = new mxCodec(node2.ownerDocument);
- dec2.decode(node2, this.graph.getStylesheet());
- }
- }
-
- this.graph.currentStyle = style;
- this.graph.mathEnabled = (urlParams['math'] == '1' || node.getAttribute('math') == '1');
-
- var bgImg = node.getAttribute('backgroundImage');
-
- if (bgImg != null)
- {
- this.graph.setBackgroundImage(this.graph.parseBackgroundImage(bgImg));
- }
- else
- {
- this.graph.setBackgroundImage(null);
- }
-
- this.graph.useCssTransforms = !mxClient.NO_FO &&
- this.isChromelessView() &&
- this.graph.isCssTransformsSupported();
- this.graph.updateCssTransform();
- this.graph.setShadowVisible(node.getAttribute('shadow') == '1', false);
-
- var extFonts = node.getAttribute('extFonts');
-
- if (extFonts)
- {
- try
- {
- extFonts = extFonts.split('|').map(function(ef)
- {
- var parts = ef.split('^');
- return {name: parts[0], url: parts[1]};
- });
-
- for (var i = 0; i < extFonts.length; i++)
- {
- this.graph.addExtFont(extFonts[i].name, extFonts[i].url);
- }
- }
- catch(e)
- {
- console.log('ExtFonts format error: ' + e.message);
- }
- }
- else if (this.graph.extFonts != null && this.graph.extFonts.length > 0)
- {
- this.graph.extFonts = [];
- }
- }
-
- // Calls updateGraphComponents
- editorSetGraphXml.apply(this, arguments);
- }
- else
- {
- throw {
- message: mxResources.get('notADiagramFile') || 'Invalid data',
- toString: function() { return this.message; }
- };
- }
- };
- /**
- * Adds persistent style to file
- */
- var editorGetGraphXml = Editor.prototype.getGraphXml;
- Editor.prototype.getGraphXml = function(ignoreSelection, resolveReferences)
- {
- ignoreSelection = (ignoreSelection != null) ? ignoreSelection : true;
- var node = editorGetGraphXml.apply(this, arguments);
-
- // Adds the current style
- if (this.graph.currentStyle != null && this.graph.currentStyle != 'default-style2')
- {
- node.setAttribute('style', this.graph.currentStyle);
- }
- var bgImg = this.graph.getBackgroundImageObject(
- this.graph.backgroundImage,
- resolveReferences);
-
- // Adds the background image
- if (bgImg != null)
- {
- node.setAttribute('backgroundImage', JSON.stringify(bgImg));
- }
-
- node.setAttribute('math', (this.graph.mathEnabled) ? '1' : '0');
- node.setAttribute('shadow', (this.graph.shadowVisible) ? '1' : '0');
-
- if (this.graph.extFonts != null && this.graph.extFonts.length > 0)
- {
- var strExtFonts = this.graph.extFonts.map(function(ef)
- {
- return ef.name + '^' + ef.url;
- });
-
- node.setAttribute('extFonts', strExtFonts.join('|'));
- }
-
- return node;
- };
-
- /**
- * Helper function to extract the graph model XML node.
- */
- Editor.prototype.isDataSvg = function(svg)
- {
- try
- {
- var svgRoot = mxUtils.parseXml(svg).documentElement;
- var tmp = svgRoot.getAttribute('content');
-
- if (tmp != null)
- {
- if (tmp != null && tmp.charAt(0) != '<' && tmp.charAt(0) != '%')
- {
- tmp = unescape((window.atob) ? atob(tmp) : Base64.decode(cont, tmp));
- }
-
- if (tmp != null && tmp.charAt(0) == '%')
- {
- tmp = decodeURIComponent(tmp);
- }
-
- if (tmp != null && tmp.length > 0)
- {
- var node = mxUtils.parseXml(tmp).documentElement;
-
- return node.nodeName == 'mxfile' || node.nodeName == 'mxGraphModel';
- }
- }
- }
- catch (e)
- {
- // ignore
- }
-
- return false;
- };
-
- /**
- * Helper function to extract the graph model XML node.
- */
- Editor.prototype.extractGraphModel = function(node, allowMxFile, checked)
- {
- return Editor.extractGraphModel.apply(this, arguments);
- };
-
- /**
- * Overrides reset graph.
- */
- var editorResetGraph = Editor.prototype.resetGraph;
- Editor.prototype.resetGraph = function()
- {
- this.graph.mathEnabled = (urlParams['math'] == '1');
- this.graph.view.x0 = null;
- this.graph.view.y0 = null;
-
- this.graph.useCssTransforms = !mxClient.NO_FO &&
- this.isChromelessView() &&
- this.graph.isCssTransformsSupported();
- this.graph.updateCssTransform();
-
- editorResetGraph.apply(this, arguments);
- };
- /**
- * Math support.
- */
- var editorUpdateGraphComponents = Editor.prototype.updateGraphComponents;
- Editor.prototype.updateGraphComponents = function()
- {
- editorUpdateGraphComponents.apply(this, arguments);
-
- this.graph.useCssTransforms = !mxClient.NO_FO &&
- this.isChromelessView() &&
- this.graph.isCssTransformsSupported();
- this.graph.updateCssTransform();
- };
-
- /**
- * Initializes math typesetting and loads respective code.
- */
- Editor.initMath = function(src, config)
- {
- if (typeof window.MathJax === 'undefined' && !mxClient.IS_IE && !mxClient.IS_IE11)
- {
- src = (src != null) ? src : DRAW_MATH_URL + '/startup.js';
- Editor.mathJaxQueue = [];
- // Blocks concurrent rendering while
- // async rendering is in progress
- var rendering = null;
- function mathJaxDone()
- {
- rendering = null;
- if (Editor.mathJaxQueue.length > 0)
- {
- Editor.doMathJaxRender(Editor.mathJaxQueue.shift());
- }
- else
- {
- Editor.onMathJaxDone();
- }
- };
-
- Editor.doMathJaxRender = function(container)
- {
- try
- {
- if (rendering == null)
- {
- MathJax.typesetClear([container]);
- MathJax.typeset([container]);
- mathJaxDone();
- }
- else if (rendering != container)
- {
- Editor.mathJaxQueue.push(container);
- }
- }
- catch (e)
- {
- MathJax.typesetClear([container]);
- if (e.retry != null)
- {
- rendering = container;
- e.retry.then(function()
- {
- MathJax.typesetPromise([container]).then(mathJaxDone)['catch'](function(e)
- {
- console.log('Error in MathJax.typesetPromise: ' + e.toString());
- mathJaxDone();
- });
- })['catch'](function(e)
- {
- console.log('Error in MathJax.retry: ' + e.toString());
- mathJaxDone();
- });;
- }
- else if (window.console != null)
- {
- console.log('Error in MathJax.typeset: ' + e.toString());
- }
- }
- };
-
- window.MathJax = (config != null) ? config :
- {
- options:
- {
- skipHtmlTags: {'[+]': ['text']},
- ignoreHtmlClass: 'geDisableMathJax'
- },
- loader:
- {
- load: [(urlParams['math-output'] == 'html') ?
- 'output/chtml' : 'output/svg', 'input/tex',
- 'input/asciimath', 'ui/safe']
- },
- startup:
- {
- pageReady: function()
- {
- for (var i = 0; i < Editor.mathJaxQueue.length; i++)
- {
- Editor.doMathJaxRender(Editor.mathJaxQueue[i]);
- }
- }
- }
- };
- // Adds global enqueue method for async rendering
- Editor.MathJaxRender = function(container)
- {
- if (typeof MathJax !== 'undefined' && typeof MathJax.typeset === 'function')
- {
- Editor.doMathJaxRender(container);
- }
- else
- {
- Editor.mathJaxQueue.push(container);
- }
- };
-
- // Adds global MathJax render callback
- Editor.onMathJaxDone = function()
- {
- // Hook for listeners
- };
- // Updates math typesetting after changes
- var editorInit = Editor.prototype.init;
-
- Editor.prototype.init = function()
- {
- editorInit.apply(this, arguments);
- var renderMath = mxUtils.bind(this, function(sender, evt)
- {
- if (this.graph.container != null &&
- this.graph.mathEnabled)
- {
- Editor.MathJaxRender(this.graph.container);
- }
- });
-
- this.graph.model.addListener(mxEvent.CHANGE, renderMath);
- this.graph.addListener(mxEvent.REFRESH, renderMath);
- };
-
- var tags = document.getElementsByTagName('script');
-
- if (tags != null && tags.length > 0)
- {
- var s = document.createElement('script');
- s.setAttribute('type', 'text/javascript');
- s.setAttribute('src', src);
- tags[0].parentNode.appendChild(s);
- }
- }
- };
- /**
- * Parses line of CSV values according to RFC 4180.
- */
- Editor.prototype.csvToArray = function(text)
- {
- if (text.length > 0)
- {
- var p = '', row = [''], i = 0, s = !0, l;
- for (l of text)
- {
- if ('"' === l)
- {
- if (s && l === p)
- {
- row[i] += l;
- }
- s = !s;
- }
- else if (',' === l && s)
- {
- l = row[++i] = '';
- }
- else
- {
- row[i] += l;
- }
- p = l;
- }
-
- return row;
- }
- else
- {
- return [];
- }
- };
- /**
- * Returns an URL that is proxied if the given URL is blocked. Uses
- * direct URL if no CSP is used as proxy blocks unknown text content.
- */
- Editor.prototype.getProxiedUrl = function(url)
- {
- if ((/test\.draw\.io$/.test(window.location.hostname) ||
- /app\.diagrams\.net$/.test(window.location.hostname)) &&
- !this.isCorsEnabledForUrl(url))
- {
- var isVisioFilename = EditorUi.isVisioFilename(url);
- var binary = /\.png$/i.test(url) || /\.pdf$/i.test(url);
- var base64 = binary || isVisioFilename;
- var nocache = 't=' + new Date().getTime();
- url = PROXY_URL + '?url=' + encodeURIComponent(url) +
- '&' + nocache + ((base64) ? '&base64=1' : '');
- }
- return url;
- };
- /**
- * Returns true if the given URL is known to have CORS headers and is
- * allowed by CSP.
- */
- Editor.prototype.isCorsEnabledForUrl = function(url)
- {
- // Disables proxy for desktop and chrome app as it is served locally
- if (mxClient.IS_CHROMEAPP || EditorUi.isElectronApp)
- {
- return true;
- }
-
- // Does not use proxy for same domain
- if (url.substring(0, window.location.origin.length) == window.location.origin)
- {
- return true;
- }
- // Blocked by CSP in production but allowed for hosted deployment
- if (urlParams['cors'] != null && this.corsRegExp == null)
- {
- this.corsRegExp = new RegExp(decodeURIComponent(urlParams['cors']));
- }
-
- // No access-control-allow-origin for some Iconfinder images, add this when fixed:
- // /^https?:\/\/[^\/]*\.iconfinder.com\//.test(url) ||
- return (this.corsRegExp != null && this.corsRegExp.test(url)) ||
- url.substring(0, 34) === 'https://raw.githubusercontent.com/' ||
- url.substring(0, 29) === 'https://fonts.googleapis.com/' ||
- url.substring(0, 26) === 'https://fonts.gstatic.com/';
- };
- /**
- * Converts all images in the SVG output to data URIs for immediate rendering
- */
- Editor.prototype.createImageUrlConverter = function()
- {
- var converter = new mxUrlConverter();
- converter.updateBaseUrl();
- // Extends convert to avoid CORS using an image proxy server where needed
- var convert = converter.convert;
- var self = this;
-
- converter.convert = function(src)
- {
- if (src != null && navigator.onLine)
- {
- var remote = src.substring(0, 7) == 'http://' || src.substring(0, 8) == 'https://';
-
- if (remote && src.substring(0, converter.baseUrl.length) != converter.baseUrl &&
- (!self.crossOriginImages || !self.isCorsEnabledForUrl(src)))
- {
- src = PROXY_URL + '?url=' + encodeURIComponent(src);
- }
- else if (src.substring(0, 19) != 'chrome-extension://')
- {
- src = convert.apply(this, arguments);
- }
- }
-
- return src;
- };
-
- return converter;
- };
- /**
- *
- */
- Editor.createSvgDataUri = function(svg)
- {
- return 'data:image/svg+xml;base64,' + btoa(unescape(encodeURIComponent(svg)));
- };
- /**
- *
- */
- Editor.prototype.convertImageToDataUri = function(url, callback, error, convertScale, forceConvert)
- {
- try
- {
- var acceptResponse = true;
-
- var timeoutThread = window.setTimeout(mxUtils.bind(this, function()
- {
- acceptResponse = false;
- callback(url);
- }), this.timeout);
- // Fallback to raster image if SVG cannot be loaded
- var svgError = mxUtils.bind(this, function()
- {
- if (convertScale != null)
- {
- this.convertImageToDataUri(url, callback, error, convertScale, true);
- }
- else
- {
- callback(url);
- }
- });
-
- if (/(\.svg)$/i.test(url) && !forceConvert)
- {
- mxUtils.get(url, mxUtils.bind(this, function(req)
- {
- window.clearTimeout(timeoutThread);
- if (acceptResponse)
- {
- if (req.getStatus() < 200 || req.getStatus() > 299)
- {
- svgError();
- }
- else
- {
- callback(Editor.createSvgDataUri(req.getText()));
- }
-
- }
- }),
- mxUtils.bind(this, function()
- {
- window.clearTimeout(timeoutThread);
- if (acceptResponse)
- {
- svgError();
- }
- }));
- }
- else
- {
- var img = new Image();
-
- if (this.crossOriginImages)
- {
- img.crossOrigin = 'anonymous';
- }
-
- img.onload = function()
- {
- window.clearTimeout(timeoutThread);
-
- if (acceptResponse)
- {
- try
- {
- convertScale = (convertScale != null &&
- forceConvert) ? convertScale : 1;
- var canvas = document.createElement('canvas');
- var ctx = canvas.getContext('2d');
- ctx.scale(convertScale, convertScale);
- canvas.height = img.height * convertScale;
- canvas.width = img.width * convertScale;
- ctx.drawImage(img, 0, 0);
-
- callback(canvas.toDataURL());
- }
- catch (e)
- {
- callback(url);
- }
- }
- };
-
- img.onerror = function()
- {
- window.clearTimeout(timeoutThread);
-
- if (acceptResponse)
- {
- if (error != null)
- {
- error();
- }
- else
- {
- callback(url);
- }
- }
- };
-
- img.src = url;
- }
- }
- catch (e)
- {
- if (error != null)
- {
- error();
- }
- else
- {
- callback(url);
- }
- }
- };
-
- /**
- * Converts all images in the SVG output to data URIs for immediate rendering
- */
- Editor.prototype.convertImages = function(svgRoot, callback, imageCache, converter)
- {
- // Converts images to data URLs for immediate painting
- if (converter == null)
- {
- converter = this.createImageUrlConverter();
- }
-
- // Queues image conversion and executes in order
- var pending = [];
- function next()
- {
- if (pending.length > 0)
- {
- pending.shift()();
- }
- else
- {
- callback(svgRoot);
- }
- };
- var cache = imageCache || new Object();
-
- var convertImages = mxUtils.bind(this, function(tagName, srcAttr)
- {
- var images = svgRoot.getElementsByTagName(tagName);
-
- for (var i = 0; i < images.length; i++)
- {
- (mxUtils.bind(this, function(img)
- {
- pending.push(mxUtils.bind(this, function()
- {
- try
- {
- if (img != null)
- {
- var src = converter.convert(img.getAttribute(srcAttr));
- // Data URIs are pass-through
- if (src != null && src.substring(0, 5) != 'data:')
- {
- var tmp = cache[src];
- if (tmp == null)
- {
- this.convertImageToDataUri(src, function(uri)
- {
- if (uri != null)
- {
- cache[src] = uri;
- img.setAttribute(srcAttr, uri);
- }
-
- next();
- }, null, Editor.svgRasterScale);
- }
- else
- {
- img.setAttribute(srcAttr, tmp);
- next();
- }
- }
- else
- {
- if (src != null)
- {
- img.setAttribute(srcAttr, src);
- }
- next();
- }
- }
- }
- catch (e)
- {
- next();
- }
- }));
- }))(images[i]);
- }
- });
-
- // Converts all known image tags in output
- // LATER: Add support for images in CSS
- convertImages('image', 'xlink:href');
- convertImages('img', 'src');
- next();
- };
-
- /**
- * Base64 encodes the given string. This method seems to be more
- * robust for encoding PNG from binary AJAX responses.
- */
- Editor.base64Encode = function(str)
- {
- var CHARS = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
- var out = "", i = 0, len = str.length, c1, c2, c3;
-
- while (i < len)
- {
- c1 = str.charCodeAt(i++) & 0xff;
-
- if (i == len)
- {
- out += CHARS.charAt(c1 >> 2);
- out += CHARS.charAt((c1 & 0x3) << 4);
- out += "==";
- break;
- }
-
- c2 = str.charCodeAt(i++);
-
- if (i == len)
- {
- out += CHARS.charAt(c1 >> 2);
- out += CHARS.charAt(((c1 & 0x3)<< 4) | ((c2 & 0xF0) >> 4));
- out += CHARS.charAt((c2 & 0xF) << 2);
- out += "=";
- break;
- }
-
- c3 = str.charCodeAt(i++);
- out += CHARS.charAt(c1 >> 2);
- out += CHARS.charAt(((c1 & 0x3) << 4) | ((c2 & 0xF0) >> 4));
- out += CHARS.charAt(((c2 & 0xF) << 2) | ((c3 & 0xC0) >> 6));
- out += CHARS.charAt(c3 & 0x3F);
- }
-
- return out;
- };
- /**
- * Checks if the client is authorized and calls the next step.
- */
- Editor.prototype.loadUrl = function(url, success, error, forceBinary, retry, dataUriPrefix, noBinary, headers)
- {
- try
- {
- var binary = !noBinary && (forceBinary || /(\.png)($|\?)/i.test(url) ||
- /(\.jpe?g)($|\?)/i.test(url) || /(\.gif)($|\?)/i.test(url) ||
- /(\.pdf)($|\?)/i.test(url));
- retry = (retry != null) ? retry : true;
-
- var fn = mxUtils.bind(this, function()
- {
- mxUtils.get(url, mxUtils.bind(this, function(req)
- {
- if (req.getStatus() >= 200 && req.getStatus() <= 299)
- {
- if (success != null)
- {
- var data = req.getText();
-
- // Returns PNG as base64 encoded data URI
- if (binary)
- {
- // NOTE: This requires BinaryToArray VB script in the page
- if ((document.documentMode == 9 || document.documentMode == 10) &&
- typeof window.mxUtilsBinaryToArray !== 'undefined')
- {
- var bin = mxUtilsBinaryToArray(req.request.responseBody).toArray();
- var tmp = new Array(bin.length);
-
- for (var i = 0; i < bin.length; i++)
- {
- tmp[i] = String.fromCharCode(bin[i]);
- }
-
- data = tmp.join('');
- }
-
- // LATER: Could be JPG but modern browsers
- // ignore the mime type in the data URI
- dataUriPrefix = (dataUriPrefix != null) ? dataUriPrefix : 'data:image/png;base64,';
- data = dataUriPrefix + Editor.base64Encode(data);
- }
-
- success(data);
- }
- }
- else if (error != null)
- {
- if (req.getStatus() == 0)
- {
- // Handles CORS errors
- error({message: mxResources.get('accessDenied')}, req);
- }
- else if (req.getStatus() == 404)
- {
- error({message: mxResources.get('fileNotFound'),
- code: req.getStatus()}, req);
- }
- else
- {
- error({message: this.getErrorMessage(req)}, req);
- }
- }
- }), function(req)
- {
- if (error != null)
- {
- error({message: mxResources.get('error') + ' ' + req.getStatus()});
- }
- }, binary, this.timeout, function()
- {
- if (retry && error != null)
- {
- error({code: App.ERROR_TIMEOUT, retry: fn});
- }
- }, headers);
- });
-
- fn();
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- };
- /**
- * Adds the listener for automatically saving the diagram for local changes.
- */
- Editor.prototype.getErrorMessage = function(req)
- {
- var msg = mxResources.get('error') + ' ' + req.getStatus();
- try
- {
- var data = req.getText();
- var obj = JSON.parse(data);
- if (obj != null && obj.error != null &&
- obj.error.message != null)
- {
- msg = obj.error.message + ' (' + msg + ')';
- }
- }
- catch (e)
- {
- // ignore
- }
- return msg;
- };
- /**
- * Makes all relative font URLs absolute in the given font CSS.
- */
- Editor.prototype.absoluteCssFonts = function(fontCss)
- {
- var result = null;
-
- if (fontCss != null)
- {
- var parts = fontCss.split('url(');
-
- if (parts.length > 0)
- {
- result = [parts[0]];
-
- // Gets path for URL
- var path = window.location.pathname;
- var idx = (path != null) ? path.lastIndexOf('/') : -1;
-
- if (idx >= 0)
- {
- path = path.substring(0, idx + 1);
- }
-
- // Gets base tag from head
- var temp = document.getElementsByTagName('base');
- var base = null;
-
- if (temp != null && temp.length > 0)
- {
- base = temp[0].getAttribute('href');
- }
-
- for (var i = 1; i < parts.length; i++)
- {
- var idx = parts[i].indexOf(')');
-
- if (idx > 0)
- {
- var url = Editor.trimCssUrl(parts[i].substring(0, idx));
-
- if (this.graph.isRelativeUrl(url))
- {
- url = (base != null) ? base + url : (window.location.protocol + '//' + window.location.hostname +
- ((url.charAt(0) == '/') ? '' : path) + url);
- }
-
- result.push('url("' + url + '"' + parts[i].substring(idx));
- }
- else
- {
- result.push(parts[i]);
- }
- }
- }
- else
- {
- result = [fontCss]
- }
- }
-
- return (result != null) ? result.join('') : null;
- };
-
- /**
- * Returns the URL and mime type to be used for the given font.
- */
- Editor.prototype.mapFontUrl = function(mime, url, fn)
- {
- if ((/^https?:\/\//.test(url)) && !this.isCorsEnabledForUrl(url))
- {
- url = PROXY_URL + '?url=' + encodeURIComponent(url);
- }
- fn(mime, url);
- };
- /**
- * For the fonts in CSS to be applied when rendering images on canvas, the actual
- * font data must be made available via a data URI encoding of the file.
- */
- Editor.prototype.embedCssFonts = function(fontCss, then)
- {
- var parts = fontCss.split('url(');
- var waiting = 0;
-
- if (this.cachedFonts == null)
- {
- this.cachedFonts = {};
- }
- var finish = mxUtils.bind(this, function()
- {
- if (waiting == 0)
- {
- // Constructs string
- var result = [parts[0]];
-
- for (var j = 1; j < parts.length; j++)
- {
- var idx = parts[j].indexOf(')');
- result.push('url("');
- result.push(this.cachedFonts[Editor.trimCssUrl(parts[j].substring(0, idx))]);
- result.push('"' + parts[j].substring(idx));
- }
-
- then(result.join(''));
- }
- });
-
- if (parts.length > 0)
- {
- for (var i = 1; i < parts.length; i++)
- {
- var idx = parts[i].indexOf(')');
- var format = null;
-
- // Checks if there is a format directive
- var fmtIdx = parts[i].indexOf('format(', idx);
-
- if (fmtIdx > 0)
- {
- format = Editor.trimCssUrl(parts[i].substring(fmtIdx + 7, parts[i].indexOf(')', fmtIdx)));
- }
- (mxUtils.bind(this, function(url)
- {
- if (this.cachedFonts[url] == null)
- {
- // Mark font as being fetched and fetch it
- this.cachedFonts[url] = url;
- waiting++;
-
- var mime = 'application/x-font-ttf';
-
- // See https://stackoverflow.com/questions/2871655/proper-mime-type-for-fonts
- if (format == 'svg' || /(\.svg)($|\?)/i.test(url))
- {
- mime = 'image/svg+xml';
- }
- else if (format == 'otf' || format == 'embedded-opentype' || /(\.otf)($|\?)/i.test(url))
- {
- mime = 'application/x-font-opentype';
- }
- else if (format == 'woff' || /(\.woff)($|\?)/i.test(url))
- {
- mime = 'application/font-woff';
- }
- else if (format == 'woff2' || /(\.woff2)($|\?)/i.test(url))
- {
- mime = 'application/font-woff2';
- }
- else if (format == 'eot' || /(\.eot)($|\?)/i.test(url))
- {
- mime = 'application/vnd.ms-fontobject';
- }
- else if (format == 'sfnt' || /(\.sfnt)($|\?)/i.test(url))
- {
- mime = 'application/font-sfnt';
- }
- this.mapFontUrl(mime, url, mxUtils.bind(this, function(realMime, realUrl)
- {
- // LATER: Remove cache-control header
- this.loadUrl(realUrl, mxUtils.bind(this, function(uri)
- {
- this.cachedFonts[url] = uri;
- waiting--;
- finish();
- }), mxUtils.bind(this, function(err)
- {
- // LATER: handle error
- waiting--;
- finish();
- }), true, null, 'data:' + realMime + ';charset=utf-8;base64,');
- }));
- }
- }))(Editor.trimCssUrl(parts[i].substring(0, idx)), format);
- }
-
- //In case all fonts are cached
- finish();
- }
- else
- {
- //No font urls found
- then(fontCss);
- }
- };
-
- /**
- * For the fontCSS to be applied when rendering images on canvas, the actual
- * font data must be made available via a data URI encoding of the file.
- */
- Editor.prototype.loadFonts = function(then)
- {
- if (this.fontCss != null && this.resolvedFontCss == null)
- {
- this.embedCssFonts(this.fontCss, mxUtils.bind(this, function(resolvedFontCss)
- {
- this.resolvedFontCss = resolvedFontCss;
- if (then != null)
- {
- then();
- }
- }));
- }
- else if (then != null)
- {
- then();
- }
- };
- /**
- * Returns a CSS mapping for the given CSS URL.
- */
- Editor.prototype.createGoogleFontCache = function()
- {
- var cache = {};
- for (var key in Graph.fontMapping)
- {
- if (Graph.isCssFontUrl(key))
- {
- cache[key] = Graph.fontMapping[key];
- }
- }
- return cache;
- };
-
- /**
- * Embeds external fonts
- */
- Editor.prototype.embedExtFonts = function(callback)
- {
- var extFonts = this.graph.getCustomFonts();
-
- if (extFonts.length > 0)
- {
- var content = [];
- var waiting = 0;
-
- if (this.cachedGoogleFonts == null)
- {
- this.cachedGoogleFonts = this.createGoogleFontCache();
- }
-
- var googleCssDone = mxUtils.bind(this, function()
- {
- if (waiting == 0)
- {
- this.embedCssFonts(content.join(''), callback);
- }
- });
-
- for (var i = 0; i < extFonts.length; i++)
- {
- (mxUtils.bind(this, function(fontName, fontUrl)
- {
- if (Graph.isCssFontUrl(fontUrl))
- {
- if (this.cachedGoogleFonts[fontUrl] == null)
- {
- waiting++;
-
- this.loadUrl(Graph.rewriteGoogleFontUrl(fontUrl), mxUtils.bind(this, function(css)
- {
- this.cachedGoogleFonts[fontUrl] = css;
- content.push(css + '\n');
- waiting--;
- googleCssDone();
- }), mxUtils.bind(this, function(err)
- {
- waiting--;
- content.push('@import url(' + fontUrl + ');\n');
- googleCssDone();
- }));
- }
- else
- {
- content.push(this.cachedGoogleFonts[fontUrl] + '\n');
- }
- }
- else
- {
- content.push('@font-face {' +
- 'font-family: "' + fontName + '";' +
- 'src: url("' + fontUrl + '")}\n');
- }
- }))(extFonts[i].name, extFonts[i].url);
- }
-
- googleCssDone();
- }
- else
- {
- callback();
- }
- };
-
- /**
- * Copies MathJax CSS into the SVG output.
- */
- Editor.prototype.addMathCss = function(svgRoot)
- {
- var defs = svgRoot.getElementsByTagName('defs');
-
- if (defs != null && defs.length > 0)
- {
- var styles = document.getElementsByTagName('style');
-
- for (var i = 0; i < styles.length; i++)
- {
- // Ignores style elements with no MathJax CSS
- var content = mxUtils.getTextContent(styles[i]);
- if (content.indexOf('mxPageSelector') < 0 &&
- content.indexOf('MathJax') > 0)
- {
- defs[0].appendChild(styles[i].cloneNode(true));
- }
- }
- }
- };
- /**
- * Adds the global fontCss configuration.
- */
- Editor.prototype.addFontCss = function(svgRoot, fontCss)
- {
- fontCss = (fontCss != null) ? fontCss : this.absoluteCssFonts(this.fontCss);
- // Creates defs element if not available
- if (fontCss != null)
- {
- var defsElt = mxUtils.getSvgDefs(svgRoot);
- // Moves imports to separate style element
- var lines = fontCss.split('\n');
- var imports = [];
- var other = [];
- for (var i = 0; i < lines.length; i++)
- {
- if (lines[i].substring(0, 7) == '@import')
- {
- imports.push(lines[i]);
- }
- else
- {
- other.push(lines[i]);
- }
- }
-
- var svgDoc = svgRoot.ownerDocument;
- var style = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'style') :
- svgDoc.createElement('style');
- style.setAttribute('type', 'text/css');
- if (imports.length > 0)
- {
- mxUtils.setTextContent(style, imports.join('\n'));
- defsElt.appendChild(style);
- }
- if (other.length > 0)
- {
- style = style.cloneNode(false);
- mxUtils.setTextContent(style, other.join('\n'));
- defsElt.appendChild(style);
- }
- }
- };
-
- /**
- * Disables client-side image export if math is enabled.
- */
- Editor.prototype.isExportToCanvas = function()
- {
- return mxClient.IS_CHROMEAPP || this.useCanvasForExport;
- };
- /**
- * Returns the maximum possible scale for the given canvas dimension and scale.
- * This will return the given scale or the maximum scale that can be used to
- * generate a valid image in the current browser.
- *
- * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/canvas
- */
- Editor.prototype.getMaxCanvasScale = function(w, h, scale)
- {
- var max = (mxClient.IS_FF) ? 8192 : 16384;
- return Math.min(scale, Math.min(max / w, max / h));
- };
-
- /**
- *
- */
- Editor.prototype.exportToCanvas = function(callback, width, imageCache, background, error, limitHeight,
- ignoreSelection, scale, transparentBackground, addShadow, converter, graph, border, noCrop, grid,
- theme, exportType, cells)
- {
- try
- {
- limitHeight = (limitHeight != null) ? limitHeight : true;
- ignoreSelection = (ignoreSelection != null) ? ignoreSelection : true;
- graph = (graph != null) ? graph : this.graph;
- border = (border != null) ? border : 0;
-
- var bg = (transparentBackground) ? null : graph.background;
-
- if (bg == mxConstants.NONE)
- {
- bg = null;
- }
-
- if (bg == null)
- {
- bg = background;
- }
-
- // Handles special case where background is null but transparent is false
- if (bg == null && transparentBackground == false)
- {
- bg = (theme == 'dark') ? Editor.darkColor : '#ffffff';
- }
- this.convertImages(graph.getSvg(null, null, border, noCrop, null, ignoreSelection,
- null, null, null, addShadow, null, theme, exportType, cells),
- mxUtils.bind(this, function(svgRoot)
- {
- try
- {
- var img = new Image();
-
- img.onload = mxUtils.bind(this, function()
- {
- try
- {
- var canvas = document.createElement('canvas');
- var w = parseInt(svgRoot.getAttribute('width'));
- var h = parseInt(svgRoot.getAttribute('height'));
- scale = (scale != null) ? scale : 1;
-
- if (width != null)
- {
- scale = (!limitHeight) ? width / w : Math.min(1, Math.min((width * 3) / (h * 4), width / w));
- }
-
- scale = this.getMaxCanvasScale(w, h, scale);
- w = Math.ceil(scale * w);
- h = Math.ceil(scale * h);
-
- canvas.setAttribute('width', w);
- canvas.setAttribute('height', h);
- var ctx = canvas.getContext('2d');
-
- if (bg != null)
- {
- ctx.beginPath();
- ctx.rect(0, 0, w, h);
- ctx.fillStyle = bg;
- ctx.fill();
- }
-
- if (scale != 1)
- {
- ctx.scale(scale, scale);
- }
-
- function drawImage()
- {
- // Workaround for broken data URI images in Safari on first export
- if (mxClient.IS_SF)
- {
- window.setTimeout(function()
- {
- ctx.drawImage(img, 0, 0);
- callback(canvas, svgRoot);
- }, 0);
- }
- else
- {
- ctx.drawImage(img, 0, 0);
- callback(canvas, svgRoot);
- }
- };
-
- if (grid)
- {
- var view = graph.view;
- var curViewScale = view.scale;
- view.scale = 1; //Reset the scale temporary to generate unscaled grid image which is then scaled
- var gridImage = btoa(unescape(encodeURIComponent(view.createSvgGrid(view.gridColor))));
- view.scale = curViewScale;
- gridImage = 'data:image/svg+xml;base64,' + gridImage;
- var phase = graph.gridSize * view.gridSteps * scale;
-
- var b = graph.getGraphBounds();
- var tx = view.translate.x * curViewScale;
- var ty = view.translate.y * curViewScale;
- var x0 = tx + (b.x - tx) / curViewScale - border;
- var y0 = ty + (b.y - ty) / curViewScale - border;
-
- var background = new Image();
-
- background.onload = function()
- {
- try
- {
- var x = -Math.round(phase - mxUtils.mod((tx - x0) * scale, phase));
- var y = -Math.round(phase - mxUtils.mod((ty - y0) * scale, phase));
-
- for (var i = x; i < w; i += phase)
- {
- for (var j = y; j < h; j += phase)
- {
- ctx.drawImage(background, i / scale, j / scale);
- }
- }
-
- drawImage();
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- };
-
- background.onerror = function(e)
- {
- if (error != null)
- {
- error(e);
- }
- };
-
- background.src = gridImage;
- }
- else
- {
- drawImage();
- }
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- });
-
- img.onerror = function(e)
- {
- //console.log('img', e, img.src);
-
- if (error != null)
- {
- error(e);
- }
- };
- if (addShadow)
- {
- this.graph.addSvgShadow(svgRoot);
- }
-
- if (this.graph.mathEnabled)
- {
- this.addMathCss(svgRoot);
- }
-
- var done = mxUtils.bind(this, function()
- {
- try
- {
- if (this.resolvedFontCss != null)
- {
- this.addFontCss(svgRoot, this.resolvedFontCss);
- }
-
- img.src = Editor.createSvgDataUri(mxUtils.getXml(svgRoot));
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- });
-
- this.embedExtFonts(mxUtils.bind(this, function(extFontsEmbeddedCss)
- {
- try
- {
- if (extFontsEmbeddedCss != null)
- {
- this.addFontCss(svgRoot, extFontsEmbeddedCss);
- }
-
- this.loadFonts(done);
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- }));
- }
- catch (e)
- {
- //console.log('src', e, img.src);
-
- if (error != null)
- {
- error(e);
- }
- }
- }), imageCache, converter);
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- };
-
- Editor.crcTable = [];
-
- for (var n = 0; n < 256; n++)
- {
- var c = n;
-
- for (var k = 0; k < 8; k++)
- {
- if ((c & 1) == 1)
- {
- c = 0xedb88320 ^ (c >>> 1);
- }
- else
- {
- c >>>= 1;
- }
- Editor.crcTable[n] = c;
- }
- }
-
- Editor.updateCRC = function(crc, data, off, len)
- {
- var c = crc;
-
- for (var n = 0; n < len; n++)
- {
- c = Editor.crcTable[(c ^ data.charCodeAt(off + n)) & 0xff] ^ (c >>> 8);
- }
-
- return c;
- };
- Editor.crc32 = function(str)
- {
- var crc = 0 ^ (-1);
- for (var i = 0; i < str.length; i++ )
- {
- crc = (crc >>> 8) ^ Editor.crcTable[(crc ^ str.charCodeAt(i)) & 0xFF];
- }
- return (crc ^ (-1)) >>> 0;
- };
- /**
- * Adds the given text to the compressed or non-compressed text chunk.
- */
- Editor.writeGraphModelToPng = function(data, type, key, value, error)
- {
- var base64 = data.substring(data.indexOf(',') + 1);
- var f = (window.atob) ? atob(base64) : Base64.decode(base64, true);
- var pos = 0;
-
- function fread(d, count)
- {
- var start = pos;
- pos += count;
-
- return d.substring(start, pos);
- };
-
- // Reads unsigned long 32 bit big endian
- function _freadint(d)
- {
- var bytes = fread(d, 4);
-
- return bytes.charCodeAt(3) + (bytes.charCodeAt(2) << 8) +
- (bytes.charCodeAt(1) << 16) + (bytes.charCodeAt(0) << 24);
- };
-
- function writeInt(num)
- {
- return String.fromCharCode((num >> 24) & 0x000000ff, (num >> 16) & 0x000000ff,
- (num >> 8) & 0x000000ff, num & 0x000000ff);
- };
-
- // Checks signature
- if (fread(f,8) != String.fromCharCode(137) + 'PNG' + String.fromCharCode(13, 10, 26, 10))
- {
- if (error != null)
- {
- error();
- }
-
- return;
- }
-
- // Reads header chunk
- fread(f,4);
-
- if (fread(f,4) != 'IHDR')
- {
- if (error != null)
- {
- error();
- }
-
- return;
- }
-
- fread(f, 17);
- var result = f.substring(0, pos);
-
- do
- {
- var n = _freadint(f);
- var chunk = fread(f,4);
-
- if (chunk == 'IDAT')
- {
- result = f.substring(0, pos - 8);
-
- if (type == 'pHYs' && key == 'dpi')
- {
- var dpm = Math.round(value / 0.0254); //One inch is equal to exactly 0.0254 meters.
- var chunkData = writeInt(dpm) + writeInt(dpm) + String.fromCharCode(1);
- }
- else
- {
- var chunkData = key + String.fromCharCode(0) +
- ((type == 'zTXt') ? String.fromCharCode(0) : '') +
- value;
- }
-
- var crc = 0xffffffff;
- crc = Editor.updateCRC(crc, type, 0, 4);
- crc = Editor.updateCRC(crc, chunkData, 0, chunkData.length);
-
- result += writeInt(chunkData.length) + type + chunkData + writeInt(crc ^ 0xffffffff);
- result += f.substring(pos - 8, f.length);
-
- break;
- }
-
- result += f.substring(pos - 8, pos - 4 + n);
- fread(f,n);
- fread(f,4);
- }
- while (n);
-
- return 'data:image/png;base64,' + ((window.btoa) ? btoa(result) : Base64.encode(result, true));
- };
- /**
- * Adds persistence for recent colors
- */
- if (window.ColorDialog)
- {
- FilenameDialog.filenameHelpLink = 'https://www.drawio.com/doc/faq/save-file-formats';
-
- var colorDialogAddRecentColor = ColorDialog.addRecentColor;
-
- ColorDialog.addRecentColor = function(color, max)
- {
- colorDialogAddRecentColor.apply(this, arguments);
-
- mxSettings.setRecentColors(ColorDialog.recentColors);
- mxSettings.save();
- };
-
- var colorDialogResetRecentColors = ColorDialog.resetRecentColors;
-
- ColorDialog.resetRecentColors = function()
- {
- colorDialogResetRecentColors.apply(this, arguments);
-
- mxSettings.setRecentColors(ColorDialog.recentColors);
- mxSettings.save();
- };
- }
-
- // Overrides ID for pages
- if (typeof window.EditDataDialog !== 'undefined')
- {
- EditDataDialog.getDisplayIdForCell = function(ui, cell)
- {
- var id = null;
-
- if (ui.editor.graph.getModel().getParent(cell) != null)
- {
- id = cell.getId();
- }
- else if (ui.currentPage != null)
- {
- id = ui.currentPage.getId();
- }
-
- return id;
- };
- }
- var AddCustomPropertyDialog = function(editorUi, callback)
- {
- var row, td;
-
- var table = document.createElement('table');
- var tbody = document.createElement('tbody');
- table.setAttribute('cellpadding', (mxClient.IS_SF) ? '0' : '2');
-
- row = document.createElement('tr');
-
- td = document.createElement('td');
- td.style.fontSize = '10pt';
- td.style.width = '100px';
- mxUtils.write(td, mxResources.get('name', null, 'Name') + ':');
-
- row.appendChild(td);
-
- var nameInput = document.createElement('input');
- nameInput.style.width = '180px';
- td = document.createElement('td');
- td.appendChild(nameInput);
- row.appendChild(td);
-
- tbody.appendChild(row);
-
- row = document.createElement('tr');
-
- td = document.createElement('td');
- td.style.fontSize = '10pt';
- mxUtils.write(td, mxResources.get('type', null, 'Type') + ':');
-
- row.appendChild(td);
-
- var typeSelect = document.createElement('select');
- typeSelect.style.width = '180px';
- var boolOption = document.createElement('option');
- boolOption.setAttribute('value', 'bool');
- mxUtils.write(boolOption, mxResources.get('bool', null, 'Boolean'));
- typeSelect.appendChild(boolOption);
-
- var clrOption = document.createElement('option');
- clrOption.setAttribute('value', 'color');
- mxUtils.write(clrOption, mxResources.get('color', null, 'Color'));
- typeSelect.appendChild(clrOption);
-
- var enumOption = document.createElement('option');
- enumOption.setAttribute('value', 'enum');
- mxUtils.write(enumOption, mxResources.get('enum', null, 'Enumeration'));
- typeSelect.appendChild(enumOption);
- var floatOption = document.createElement('option');
- floatOption.setAttribute('value', 'float');
- mxUtils.write(floatOption, mxResources.get('float', null, 'Float'));
- typeSelect.appendChild(floatOption);
- var intOption = document.createElement('option');
- intOption.setAttribute('value', 'int');
- mxUtils.write(intOption, mxResources.get('int', null, 'Int'));
- typeSelect.appendChild(intOption);
-
- var strOption = document.createElement('option');
- strOption.setAttribute('value', 'string');
- mxUtils.write(strOption, mxResources.get('string', null, 'String'));
- typeSelect.appendChild(strOption);
- td = document.createElement('td');
- td.appendChild(typeSelect);
- row.appendChild(td);
-
- tbody.appendChild(row);
-
- row = document.createElement('tr');
- td = document.createElement('td');
- td.style.fontSize = '10pt';
- mxUtils.write(td, mxResources.get('dispName', null, 'Display Name') + ':');
-
- row.appendChild(td);
-
- var dispNameInput = document.createElement('input');
- dispNameInput.style.width = '180px';
- td = document.createElement('td');
- td.appendChild(dispNameInput);
- row.appendChild(td);
- tbody.appendChild(row);
- var listRow = document.createElement('tr');
- td = document.createElement('td');
- td.style.fontSize = '10pt';
- mxUtils.write(td, mxResources.get('enumList', null, 'Enum List') + ' (csv):');
-
- listRow.appendChild(td);
-
- var enumListInput = document.createElement('input');
- enumListInput.style.width = '180px';
- td = document.createElement('td');
- td.appendChild(enumListInput);
- listRow.appendChild(td);
- listRow.style.display = 'none';
- tbody.appendChild(listRow);
-
- table.appendChild(tbody);
-
- function typeChanged()
- {
- if (typeSelect.value === 'enum')
- {
- listRow.style.display = '';
- this.container.parentNode.style.height = "150px";
-
- }
- else
- {
- listRow.style.display = 'none';
- this.container.parentNode.style.height = "130px";
- }
- };
-
- mxEvent.addListener(typeSelect, 'change', mxUtils.bind(this, typeChanged));
- row = document.createElement('tr');
- td = document.createElement('td');
- td.setAttribute('align', 'right');
- td.style.paddingTop = '22px';
- td.colSpan = 2;
-
- var addBtn = mxUtils.button(mxResources.get('add', null, 'Add'), mxUtils.bind(this, function()
- {
- var name = nameInput.value;
- if (name == "")
- {
- nameInput.style.border = "1px solid red";
- return;
- }
-
- var type = typeSelect.value;
- var dispName = dispNameInput.value;
- if (dispName == "")
- {
- dispNameInput.style.border = "1px solid red";
- return;
- }
- var enumList = enumListInput.value;
-
- if (enumList == "" && type == "enum")
- {
- enumListInput.style.border = "1px solid red";
- return;
-
- }
-
- if (enumList != null)
- {
- enumList = enumList.split(',');
-
- for (var i = 0; i < enumList.length; i++)
- {
- enumList[i] = enumList[i].trim();
- }
- }
-
- if (callback)
- {
- callback(editorUi, name, type, dispName, enumList);
- editorUi.hideDialog();
- }
- }));
- addBtn.className = 'geBtn gePrimaryBtn';
-
- var cancelBtn = mxUtils.button(mxResources.get('cancel'), function()
- {
- editorUi.hideDialog();
- });
- cancelBtn.className = 'geBtn';
-
- if (editorUi.editor.cancelFirst)
- {
- td.appendChild(cancelBtn);
- td.appendChild(addBtn);
- }
- else
- {
- td.appendChild(addBtn);
- td.appendChild(cancelBtn);
- }
- row.appendChild(td);
- tbody.appendChild(row);
- table.appendChild(tbody);
- this.container = table;
- };
-
- if (window.StyleFormatPanel != null)
- {
- var formatInit = Format.prototype.init;
-
- Format.prototype.init = function()
- {
- formatInit.apply(this, arguments);
- this.editorUi.editor.addListener('fileLoaded', this.update);
- };
- var formatRefresh = Format.prototype.refresh;
-
- Format.prototype.refresh = function()
- {
- if (this.editorUi.getCurrentFile() != null || urlParams['embed'] == '1' ||
- this.editorUi.editor.chromeless)
- {
- formatRefresh.apply(this, arguments);
- }
- else
- {
- this.clear();
- }
- };
- /**
- * Option is not visible in default theme.
- */
- DiagramFormatPanel.prototype.isMathOptionVisible = function(div)
- {
- return (Editor.currentTheme == 'simple' ||
- Editor.currentTheme == 'sketch' ||
- Editor.currentTheme == 'min');
- };
-
- /**
- * Adds autosave and math typesetting options.
- */
- var diagramFormatPanelAddOptions = DiagramFormatPanel.prototype.addOptions;
- DiagramFormatPanel.prototype.addOptions = function(div)
- {
- div = diagramFormatPanelAddOptions.apply(this, arguments);
-
- var ui = this.editorUi;
- var editor = ui.editor;
- var graph = editor.graph;
- // Adds autosave option
- if (graph.isEnabled())
- {
- var file = ui.getCurrentFile();
- if (file != null && file.isAutosaveOptional())
- {
- div.appendChild(this.createOption(mxResources.get('autosave'), function()
- {
- return ui.editor.autosave;
- }, function(checked)
- {
- ui.editor.setAutosave(checked);
-
- if (ui.editor.autosave && file.isModified())
- {
- file.fileChanged();
- }
- },
- {
- install: function(apply)
- {
- this.listener = function()
- {
- apply(ui.editor.autosave);
- };
-
- ui.editor.addListener('autosaveChanged', this.listener);
- },
- destroy: function()
- {
- ui.editor.removeListener(this.listener);
- }
- }));
- }
- }
-
- // Adds math option
- if (this.isMathOptionVisible() && graph.isEnabled())
- {
- if (typeof(MathJax) !== 'undefined')
- {
- // Math
- var option = this.createOption(mxResources.get('mathematicalTypesetting'), function()
- {
- return graph.mathEnabled;
- }, function(checked)
- {
- ui.actions.get('mathematicalTypesetting').funct();
- },
- {
- install: function(apply)
- {
- this.listener = function()
- {
- apply(graph.mathEnabled);
- };
-
- ui.addListener('mathEnabledChanged', this.listener);
- },
- destroy: function()
- {
- ui.removeListener(this.listener);
- }
- });
-
- div.appendChild(option);
-
- var help = ui.menus.createHelpLink('https://www.drawio.com/doc/faq/math-typesetting');
- help.style.position = 'relative';
- help.style.marginLeft = '6px';
- option.appendChild(help);
- }
- }
-
- return div;
- };
- mxCellRenderer.prototype.defaultVertexShape.prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'absoluteArcSize', dispName: 'Abs. Arc Size', type: 'bool', defVal: false}
- ];
- mxCellRenderer.defaultShapes['link'].prototype.customProperties = [
- {name: 'width', dispName: 'Width', type: 'float', min:0, defVal: 4}
- ];
- mxCellRenderer.defaultShapes['flexArrow'].prototype.customProperties = [
- {name: 'width', dispName: 'Width', type: 'float', min:0, defVal: 10},
- {name: 'startWidth', dispName: 'Start Width', type: 'float', min:0, defVal: 20},
- {name: 'endWidth', dispName: 'End Width', type: 'float', min:0, defVal: 20}
- ];
- mxCellRenderer.defaultShapes['process'].prototype.customProperties = [
- {name: 'size', dispName: 'Indent', type: 'float', min: 0, max: 0.5, defVal: 0.1}
- ];
- mxCellRenderer.defaultShapes['rhombus'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, max: 50, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'double', dispName: 'Double', type: 'bool', defVal: false}
- ];
-
- mxCellRenderer.defaultShapes['partialRectangle'].prototype.customProperties = [
- {name: 'top', dispName: 'Top Line', type: 'bool', defVal: true},
- {name: 'bottom', dispName: 'Bottom Line', type: 'bool', defVal: true},
- {name: 'left', dispName: 'Left Line', type: 'bool', defVal: true},
- {name: 'right', dispName: 'Right Line', type: 'bool', defVal: true}
- ];
-
- mxCellRenderer.defaultShapes['parallelogram'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'size', dispName: 'Slope Angle', type: 'float', min:0, max: 1, defVal: 0.2}
- ];
-
- mxCellRenderer.defaultShapes['hexagon'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'size', dispName: 'Slope Angle', type: 'float', min:0, max: 1, defVal: 0.25}
- ];
-
- mxCellRenderer.defaultShapes['triangle'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE}
- ];
-
- mxCellRenderer.defaultShapes['document'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', defVal: 0.3, min:0, max:1}
- ];
-
- mxCellRenderer.defaultShapes['internalStorage'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'dx', dispName: 'Left Line', type: 'float', min:0, defVal: 20},
- {name: 'dy', dispName: 'Top Line', type: 'float', min:0, defVal: 20}
- ];
-
- mxCellRenderer.defaultShapes['cube'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min:0, defVal:20 },
- {name: 'darkOpacity', dispName: 'Dark Opacity', type: 'float', min:-1, max:1, defVal:0 },
- {name: 'darkOpacity2', dispName: 'Dark Opacity 2', type: 'float', min:-1, max:1, defVal:0 }
- ];
-
- mxCellRenderer.defaultShapes['step'].prototype.customProperties = [
- {name: 'size', dispName: 'Notch Size', type: 'float', min:0, defVal:20},
- {name: 'fixedSize', dispName: 'Fixed Size', type: 'bool', defVal:true}
- ];
-
- mxCellRenderer.defaultShapes['trapezoid'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'size', dispName: 'Slope Angle', type: 'float', min:0, max: 1, defVal: 0.2}
- ];
-
- mxCellRenderer.defaultShapes['tape'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min:0, max:1, defVal:0.4 }
- ];
-
- mxCellRenderer.defaultShapes['note'].prototype.customProperties = [
- {name: 'size', dispName: 'Fold Size', type: 'float', min:0, defVal: 30},
- {name: 'darkOpacity', dispName: 'Dark Opacity', type: 'float', min:-1, max:1, defVal:0 },
- ];
-
- mxCellRenderer.defaultShapes['card'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'size', dispName: 'Cutoff Size', type: 'float', min:0, defVal: 30}
- ];
-
- mxCellRenderer.defaultShapes['callout'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: mxConstants.LINE_ARCSIZE},
- {name: 'base', dispName: 'Callout Width', type: 'float', min:0, defVal: 20},
- {name: 'size', dispName: 'Callout Length', type: 'float', min:0, defVal: 30},
- {name: 'position', dispName: 'Callout Position', type: 'float', min:0, max:1, defVal: 0.5},
- {name: 'position2', dispName: 'Callout Tip Position', type: 'float', min:0, max:1, defVal: 0.5}
- ];
-
- mxCellRenderer.defaultShapes['folder'].prototype.customProperties = [
- {name: 'tabWidth', dispName: 'Tab Width', type: 'float'},
- {name: 'tabHeight', dispName: 'Tab Height', type: 'float'},
- {name: 'tabPosition', dispName: 'Tap Position', type: 'enum',
- enumList: [{val: 'left', dispName: 'Left'}, {val: 'right', dispName: 'Right'}]
- }
- ];
-
- mxCellRenderer.defaultShapes['swimlane'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: 15},
- {name: 'absoluteArcSize', dispName: 'Abs. Arc Size', type: 'bool', defVal: false},
- {name: 'startSize', dispName: 'Header Size', type: 'float'},
- {name: 'swimlaneHead', dispName: 'Head Border', type: 'bool', defVal: true},
- {name: 'swimlaneBody', dispName: 'Body Border', type: 'bool', defVal: true},
- {name: 'horizontal', dispName: 'Horizontal', type: 'bool', defVal: true},
- {name: 'separatorColor', dispName: 'Separator Color', type: 'color', defVal: null},
- ];
-
- mxCellRenderer.defaultShapes['table'].prototype.customProperties = [
- {name: 'rowLines', dispName: 'Row Lines', type: 'bool', defVal: true},
- {name: 'columnLines', dispName: 'Column Lines', type: 'bool', defVal: true},
- {name: 'fixedRows', dispName: 'Fixed Rows', type: 'bool', defVal: false},
- {name: 'resizeLast', dispName: 'Resize Last Column', type: 'bool', defVal: false},
- {name: 'resizeLastRow', dispName: 'Resize Last Row', type: 'bool', defVal: false}].
- concat(mxCellRenderer.defaultShapes['swimlane'].prototype.customProperties).
- concat(mxCellRenderer.defaultShapes['partialRectangle'].prototype.customProperties);
- mxCellRenderer.defaultShapes['tableRow'].prototype.customProperties =
- mxCellRenderer.defaultShapes['swimlane'].prototype.customProperties.
- concat(mxCellRenderer.defaultShapes['partialRectangle'].prototype.customProperties);
-
- mxCellRenderer.defaultShapes['doubleEllipse'].prototype.customProperties = [
- {name: 'margin', dispName: 'Indent', type: 'float', min:0, defVal:4}
- ];
-
- mxCellRenderer.defaultShapes['ext'].prototype.customProperties = [
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: 15},
- {name: 'double', dispName: 'Double', type: 'bool', defVal: false},
- {name: 'margin', dispName: 'Indent', type: 'float', min: 0, defVal:0}
- ];
-
- mxCellRenderer.defaultShapes['curlyBracket'].prototype.customProperties = [
- {name: 'rounded', dispName: 'Rounded', type: 'bool', defVal: true},
- {name: 'size', dispName: 'Size', type: 'float', min:0, max: 1, defVal: 0.5}
- ];
-
- mxCellRenderer.defaultShapes['image'].prototype.customProperties = [
- {name: 'imageAspect', dispName: 'Fixed Image Aspect', type: 'bool', defVal:true}
- ];
-
- mxCellRenderer.defaultShapes['label'].prototype.customProperties = [
- {name: 'imageAspect', dispName: 'Fixed Image Aspect', type: 'bool', defVal:true},
- {name: 'imageAlign', dispName: 'Image Align', type: 'enum',
- enumList: [{val: 'left', dispName: 'Left'},
- {val: 'center', dispName: 'Center'},
- {val: 'right', dispName: 'Right'}], defVal: 'left'},
- {name: 'imageVerticalAlign', dispName: 'Image Vertical Align', type: 'enum',
- enumList: [{val: 'top', dispName: 'Top'},
- {val: 'middle', dispName: 'Middle'},
- {val: 'bottom', dispName: 'Bottom'}], defVal: 'middle'},
- {name: 'imageWidth', dispName: 'Image Width', type: 'float', min:0, defVal: 24},
- {name: 'imageHeight', dispName: 'Image Height', type: 'float', min:0, defVal: 24},
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: 12},
- {name: 'absoluteArcSize', dispName: 'Abs. Arc Size', type: 'bool', defVal: false}
- ];
-
- mxCellRenderer.defaultShapes['dataStorage'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min:0, max:1, defVal:0.1 }
- ];
-
- mxCellRenderer.defaultShapes['manualInput'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min:0, defVal:30 },
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: 20}
- ];
-
- mxCellRenderer.defaultShapes['loopLimit'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min:0, defVal:20 },
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: 20}
- ];
-
- mxCellRenderer.defaultShapes['offPageConnector'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min:0, defVal:38 },
- {name: 'arcSize', dispName: 'Arc Size', type: 'float', min:0, defVal: 20}
- ];
-
- mxCellRenderer.defaultShapes['display'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min: 0, max: 1, defVal: 0.25 }
- ];
-
- mxCellRenderer.defaultShapes['singleArrow'].prototype.customProperties = [
- {name: 'arrowWidth', dispName: 'Arrow Width', type: 'float', min: 0, max: 1, defVal: 0.3 },
- {name: 'arrowSize', dispName: 'Arrowhead Length', type: 'float', min: 0, max: 1, defVal: 0.2 }
- ];
-
- mxCellRenderer.defaultShapes['doubleArrow'].prototype.customProperties = [
- {name: 'arrowWidth', dispName: 'Arrow Width', type: 'float', min: 0, max: 1, defVal: 0.3 },
- {name: 'arrowSize', dispName: 'Arrowhead Length', type: 'float', min: 0, max: 1, defVal: 0.2 }
- ];
-
- mxCellRenderer.defaultShapes['cross'].prototype.customProperties = [
- {name: 'size', dispName: 'Size', type: 'float', min: 0, max: 1, defVal: 0.2 }
- ];
-
- mxCellRenderer.defaultShapes['corner'].prototype.customProperties = [
- {name: 'dx', dispName: 'Width1', type: 'float', min: 0, defVal: 20 },
- {name: 'dy', dispName: 'Width2', type: 'float', min: 0, defVal: 20 }
- ];
-
- mxCellRenderer.defaultShapes['tee'].prototype.customProperties = [
- {name: 'dx', dispName: 'Width1', type: 'float', min: 0, defVal: 20 },
- {name: 'dy', dispName: 'Width2', type: 'float', min: 0, defVal: 20 }
- ];
-
- mxCellRenderer.defaultShapes['umlLifeline'].prototype.customProperties = [
- {name: 'participant', dispName:'Participant', type:'enum', defVal:'none', enumList:[
- {val:'none', dispName: 'Default'},
- {val:'umlActor', dispName: 'Actor'},
- {val:'umlBoundary', dispName: 'Boundary'},
- {val:'umlEntity', dispName: 'Entity'},
- {val:'umlControl', dispName: 'Control'},
- ]},
- {name: 'size', dispName:'Height', type:'float', defVal:40, min:0}
- ];
-
- mxCellRenderer.defaultShapes['umlFrame'].prototype.customProperties = [
- {name: 'width', dispName:'Title Width', type:'float', defVal:60, min:0},
- {name: 'height', dispName:'Title Height', type:'float', defVal:30, min:0}
- ];
-
- /**
- * Configures global color schemes.
- */
- StyleFormatPanel.prototype.defaultColorSchemes = [[{fill: '', stroke: ''}, {fill: '#f5f5f5', stroke: '#666666', font: '#333333'},
- {fill: '#dae8fc', stroke: '#6c8ebf'}, {fill: '#d5e8d4', stroke: '#82b366'},
- {fill: '#ffe6cc', stroke: '#d79b00'}, {fill: '#fff2cc', stroke: '#d6b656'},
- {fill: '#f8cecc', stroke: '#b85450'}, {fill: '#e1d5e7', stroke: '#9673a6'}],
- [{fill: '', stroke: ''}, {fill: '#60a917', stroke: '#2D7600', font: '#ffffff'},
- {fill: '#008a00', stroke: '#005700', font: '#ffffff'}, {fill: '#1ba1e2', stroke: '#006EAF', font: '#ffffff'},
- {fill: '#0050ef', stroke: '#001DBC', font: '#ffffff'}, {fill: '#6a00ff', stroke: '#3700CC', font: '#ffffff'},
- //{fill: '#aa00ff', stroke: '#7700CC', font: '#ffffff'},
- {fill: '#d80073', stroke: '#A50040', font: '#ffffff'}, {fill: '#a20025', stroke: '#6F0000', font: '#ffffff'}],
- [{fill: '#e51400', stroke: '#B20000', font: '#ffffff'}, {fill: '#fa6800', stroke: '#C73500', font: '#000000'},
- {fill: '#f0a30a', stroke: '#BD7000', font: '#000000'}, {fill: '#e3c800', stroke: '#B09500', font: '#000000'},
- {fill: '#6d8764', stroke: '#3A5431', font: '#ffffff'}, {fill: '#647687', stroke: '#314354', font: '#ffffff'},
- {fill: '#76608a', stroke: '#432D57', font: '#ffffff'}, {fill: '#a0522d', stroke: '#6D1F00', font: '#ffffff'}],
- [{fill: '', stroke: ''}, {fill: mxConstants.NONE, stroke: ''},
- {fill: '#fad7ac', stroke: '#b46504'}, {fill: '#fad9d5', stroke: '#ae4132'},
- {fill: '#b0e3e6', stroke: '#0e8088'}, {fill: '#b1ddf0', stroke: '#10739e'},
- {fill: '#d0cee2', stroke: '#56517e'}, {fill: '#bac8d3', stroke: '#23445d'}],
- [{fill: '', stroke: ''},
- {fill: '#f5f5f5', stroke: '#666666', gradient: '#b3b3b3'},
- {fill: '#dae8fc', stroke: '#6c8ebf', gradient: '#7ea6e0'},
- {fill: '#d5e8d4', stroke: '#82b366', gradient: '#97d077'},
- {fill: '#ffcd28', stroke: '#d79b00', gradient: '#ffa500'},
- {fill: '#fff2cc', stroke: '#d6b656', gradient: '#ffd966'},
- {fill: '#f8cecc', stroke: '#b85450', gradient: '#ea6b66'},
- {fill: '#e6d0de', stroke: '#996185', gradient: '#d5739d'}],
- [{fill: '', stroke: ''}, {fill: '#eeeeee', stroke: '#36393d'},
- {fill: '#f9f7ed', stroke: '#36393d'}, {fill: '#ffcc99', stroke: '#36393d'},
- {fill: '#cce5ff', stroke: '#36393d'}, {fill: '#ffff88', stroke: '#36393d'},
- {fill: '#cdeb8b', stroke: '#36393d'}, {fill: '#ffcccc', stroke: '#36393d'}]];
-
- /**
- * Configures custom color schemes.
- */
- StyleFormatPanel.prototype.customColorSchemes = null;
- StyleFormatPanel.prototype.findCommonProperties = function(cell, properties, addAll)
- {
- if (properties == null) return;
-
- var handleCustomProp = function(custProperties)
- {
- if (custProperties != null)
- {
- if (addAll)
- {
- for (var i = 0; i < custProperties.length; i++)
- {
- properties[custProperties[i].name] = custProperties[i];
- }
- }
- else
- {
- for (var key in properties)
- {
- var found = false;
-
- for (var i = 0; i < custProperties.length; i++)
- {
- if (custProperties[i].name == key && custProperties[i].type == properties[key].type)
- {
- found = true;
- break;
- }
- }
-
- if (!found)
- {
- delete properties[key];
- }
- }
- }
- }
- };
-
- var view = this.editorUi.editor.graph.view;
- var state = view.getState(cell);
-
- if (state != null && state.shape != null)
- {
- //Add common properties to all shapes
- if (!state.shape.commonCustomPropAdded)
- {
- state.shape.commonCustomPropAdded = true;
- state.shape.customProperties = state.shape.customProperties || [];
-
- if (state.cell.vertex)
- {
- Array.prototype.push.apply(state.shape.customProperties, Editor.commonVertexProperties);
- }
- else
- {
- Array.prototype.push.apply(state.shape.customProperties, Editor.commonEdgeProperties);
- }
- }
-
- handleCustomProp(state.shape.customProperties);
- }
-
- //This currently is not needed but let's keep it in case we needed in the future
- var userCustomProp = cell.getAttribute('customProperties');
-
- if (userCustomProp != null)
- {
- try
- {
- handleCustomProp(JSON.parse(userCustomProp));
- }
- catch(e){}
- }
- };
-
- /**
- * Adds predefiend styles.
- */
- var styleFormatPanelInit = StyleFormatPanel.prototype.init;
-
- StyleFormatPanel.prototype.init = function()
- {
- var sstate = this.editorUi.getSelectionState();
- if (this.defaultColorSchemes != null && this.defaultColorSchemes.length > 0 &&
- sstate.style.shape != 'image' && !sstate.containsLabel &&
- sstate.cells.length > 0)
- {
- this.container.appendChild(this.addStyles(this.createPanel()));
- }
-
- styleFormatPanelInit.apply(this, arguments);
- if (Editor.enableCustomProperties)
- {
- var properties = {};
- var vertices = sstate.vertices;
- var edges = sstate.edges;
-
- for (var i = 0; i < vertices.length; i++)
- {
- this.findCommonProperties(vertices[i], properties, i == 0);
- }
-
- for (var i = 0; i < edges.length; i++)
- {
- this.findCommonProperties(edges[i], properties, vertices.length == 0 && i == 0);
- }
- if (Object.getOwnPropertyNames != null && Object.getOwnPropertyNames(properties).length > 0)
- {
- this.container.appendChild(this.addProperties(this.createPanel(), properties, sstate));
- }
- }
- };
- /**
- * Overridden to add copy and paste style.
- */
- var styleFormatPanelAddStyleOps = StyleFormatPanel.prototype.addStyleOps;
-
- StyleFormatPanel.prototype.addStyleOps = function(div)
- {
- var ss = this.editorUi.getSelectionState();
- if (ss.cells.length == 1)
- {
- this.addActions(div, ['copyStyle', 'pasteStyle']);
- }
- else if (ss.cells.length >= 1)
- {
- this.addActions(div, ['pasteStyle', 'pasteData']);
- }
- styleFormatPanelAddStyleOps.apply(this, arguments);
-
- return div;
- };
- /**
- * Initial collapsed state of the properties panel.
- */
- EditorUi.prototype.propertiesCollapsed = true;
- /**
- * Create Properties Panel
- */
- BaseFormatPanel.prototype.addProperties = function(div, properties, state, hideId)
- {
- var that = this;
- var graph = this.editorUi.editor.graph;
- var secondLevel = [];
-
- function insertAfter(newElem, curElem)
- {
- curElem.parentNode.insertBefore(newElem, curElem.nextSibling);
- };
-
- function applyStyleVal(pName, newVal, prop, delIndex, input)
- {
- if (prop.valueChanged != null)
- {
- prop.valueChanged(newVal, input);
- }
- else
- {
- graph.getModel().beginUpdate();
- try
- {
- var changedProps = [];
- var changedVals = [];
- if (prop.index != null)
- {
- var allVals = [];
- var curVal = prop.parentRow.nextSibling;
-
- while(curVal && curVal.getAttribute('data-pName') == pName)
- {
- allVals.push(curVal.getAttribute('data-pValue'));
- curVal = curVal.nextSibling;
- }
-
- if (prop.index < allVals.length)
- {
- if (delIndex != null)
- {
- allVals.splice(delIndex, 1);
- }
- else
- {
- allVals[prop.index] = newVal;
- }
- }
- else
- {
- allVals.push(newVal);
- }
-
- if (prop.size != null && allVals.length > prop.size) //trim the array to the specifies size
- {
- allVals = allVals.slice(0, prop.size);
- }
-
- newVal = allVals.join(',');
-
- if (prop.countProperty != null)
- {
- graph.setCellStyles(prop.countProperty, allVals.length, graph.getSelectionCells());
-
- changedProps.push(prop.countProperty);
- changedVals.push(allVals.length);
- }
- }
- graph.setCellStyles(pName, newVal, graph.getSelectionCells());
- changedProps.push(pName);
- changedVals.push(newVal);
-
- if (prop.dependentProps != null)
- {
- for (var i = 0; i < prop.dependentProps.length; i++)
- {
- var defVal = prop.dependentPropsDefVal[i];
- var vals = prop.dependentPropsVals[i];
-
- if (vals.length > newVal)
- {
- vals = vals.slice(0, newVal);
- }
- else
- {
- for (var j = vals.length; j < newVal; j++)
- {
- vals.push(defVal);
- }
- }
-
- vals = vals.join(',');
- graph.setCellStyles(prop.dependentProps[i], vals, graph.getSelectionCells());
- changedProps.push(prop.dependentProps[i]);
- changedVals.push(vals);
- }
- }
-
- if (typeof(prop.onChange) == 'function')
- {
- prop.onChange(graph, newVal);
- }
-
- that.editorUi.fireEvent(new mxEventObject('styleChanged', 'keys', changedProps,
- 'values', changedVals, 'cells', graph.getSelectionCells()));
- }
- finally
- {
- graph.getModel().endUpdate();
- }
- }
- }
-
- function setElementPos(td, elem, adjustHeight)
- {
- var divPos = mxUtils.getOffset(div, true);
- var pos = mxUtils.getOffset(td, true);
- elem.style.position = 'absolute';
- elem.style.left = (pos.x - divPos.x) + 'px';
- elem.style.top = (pos.y - divPos.y) + 'px';
- elem.style.width = td.offsetWidth + 'px';
- elem.style.height = (td.offsetHeight - (adjustHeight? 4 : 0)) + 'px';
- elem.style.zIndex = 5;
- };
-
- function createColorBtn(pName, pValue, prop)
- {
- var clrDiv = document.createElement('div');
- clrDiv.style.width = '32px';
- clrDiv.style.height = '4px';
- clrDiv.style.margin = '2px';
- clrDiv.style.border = '1px solid black';
- clrDiv.style.background = !pValue || pValue == 'none'?
- 'url(\'' + Dialog.prototype.noColorImage + '\')' : pValue;
- btn = mxUtils.button('', mxUtils.bind(that, function(evt)
- {
- this.editorUi.pickColor(pValue, function(color)
- {
- clrDiv.style.background = color == 'none'?
- 'url(\'' + Dialog.prototype.noColorImage + '\')' : color;
- applyStyleVal(pName, color, prop);
- });
-
- mxEvent.consume(evt);
- }));
-
- btn.style.height = '12px';
- btn.style.width = '40px';
- btn.className = 'geColorBtn';
-
- btn.appendChild(clrDiv);
- return btn;
- };
-
- function createDynArrList(pName, pValue, subType, defVal, countProperty, myRow, flipBkg)
- {
- if (pValue != null)
- {
- var vals = pValue.split(',');
- secondLevel.push({name: pName, values: vals, type: subType,
- defVal: defVal, countProperty: countProperty,
- parentRow: myRow, isDeletable: true, flipBkg: flipBkg});
- }
-
- btn = mxUtils.button('+', mxUtils.bind(that, function(evt)
- {
- var beforeElem = myRow;
- var index = 0;
-
- while (beforeElem.nextSibling != null)
- {
- var cur = beforeElem.nextSibling;
- var elemPName = cur.getAttribute('data-pName');
- if (elemPName == pName)
- {
- beforeElem = beforeElem.nextSibling;
- index++;
- }
- else
- {
- break;
- }
- }
-
- var newProp = {type: subType, parentRow: myRow, index: index,
- isDeletable: true, defVal: defVal, countProperty: countProperty};
- var arrItem = createPropertyRow(pName, '', newProp, index % 2 == 0, flipBkg);
- applyStyleVal(pName, defVal, newProp);
- insertAfter(arrItem, beforeElem);
-
- mxEvent.consume(evt);
- }));
-
- btn.style.height = '16px';
- btn.style.width = '25px';
- btn.className = 'geColorBtn';
-
- return btn;
- };
-
- function createStaticArrList(pName, pValue, subType, defVal, size, myRow, flipBkg)
- {
- if (size > 0)
- {
- var vals = new Array(size);
-
- var curVals = pValue != null? pValue.split(',') : [];
-
- for (var i = 0; i < size; i++)
- {
- vals[i] = curVals[i] != null? curVals[i] : (defVal != null? defVal : '');
- }
-
- secondLevel.push({name: pName, values: vals, type: subType, defVal: defVal, parentRow: myRow, flipBkg: flipBkg, size: size});
- }
-
- return document.createElement('div'); //empty cell
- };
-
- function createCheckbox(pName, pValue, prop)
- {
- var input = document.createElement('input');
- input.type = 'checkbox';
- input.checked = pValue == '1';
-
- mxEvent.addListener(input, 'change', function()
- {
- applyStyleVal(pName, input.checked? '1' : '0', prop);
- });
- return input;
- };
-
- function createPropertyRow(pName, pValue, prop, isOdd, flipBkg)
- {
- var pDiplayName = prop.dispName;
- var pType = prop.type;
- var row = document.createElement('tr');
- row.className = 'gePropRow' + (flipBkg? 'Dark' : '') + (isOdd? 'Alt' : '') + ' gePropNonHeaderRow';
- row.setAttribute('data-pName', pName);
- row.setAttribute('data-pValue', pValue);
- var rightAlig = false;
-
- if (prop.index != null)
- {
- row.setAttribute('data-index', prop.index);
- pDiplayName = (pDiplayName != null? pDiplayName : '') + '[' + prop.index + ']';
- rightAlig = true;
- }
-
- var td = document.createElement('td');
- td.className = 'gePropRowCell';
- var label = mxResources.get(pDiplayName, null, pDiplayName);
- mxUtils.write(td, label);
- td.setAttribute('title', label);
-
- if (rightAlig)
- {
- td.style.textAlign = 'right';
- }
-
- row.appendChild(td);
- td = document.createElement('td');
- td.className = 'gePropRowCell';
- td.setAttribute('title', (pValue != null) ? pValue : mxResources.get('none'));
-
- if (pType == 'color')
- {
- td.appendChild(createColorBtn(pName, pValue, prop));
- }
- else if (pType == 'bool' || pType == 'boolean')
- {
- td.appendChild(createCheckbox(pName, pValue, prop));
- }
- else if (pType == 'enum')
- {
- var pEnumList = prop.enumList;
-
- for (var i = 0; i < pEnumList.length; i++)
- {
- var op = pEnumList[i];
-
- if (op.val == pValue)
- {
- mxUtils.write(td, mxResources.get(op.dispName, null, op.dispName));
- break;
- }
- }
-
- mxEvent.addListener(td, 'click', mxUtils.bind(that, function()
- {
- var select = document.createElement('select');
- var nullValue = 'null';
- var nullOption = null;
- setElementPos(td, select);
- for (var i = 0; i < pEnumList.length; i++)
- {
- var op = pEnumList[i];
- var opElem = document.createElement('option');
- opElem.value = mxUtils.htmlEntities(op.val);
- mxUtils.write(opElem, mxResources.get(op.dispName, null, op.dispName));
- select.appendChild(opElem);
- if (op.val == null)
- {
- opElem.value = nullValue;
- nullOption = opElem;
- }
- }
-
- select.value = (pValue == null && nullOption != null) ? nullValue : pValue;
-
- div.appendChild(select);
- mxEvent.addListener(select, 'change', function()
- {
- var newVal = mxUtils.htmlEntities(select.value);
- if (select[select.selectedIndex] == nullOption ||
- newVal.value == nullValue)
- {
- newVal = null;
- }
- applyStyleVal(pName, newVal, prop);
- //set value triggers a redraw of the panel which removes the select and updates the row
- });
-
- select.focus();
- //FF calls blur on focus! so set the event after focusing (not with selects but to be safe)
- mxEvent.addListener(select, 'blur', function()
- {
- div.removeChild(select);
- });
- }));
- }
- else if (pType == 'dynamicArr')
- {
- td.appendChild(createDynArrList(pName, pValue, prop.subType, prop.subDefVal, prop.countProperty, row, flipBkg));
- }
- else if (pType == 'staticArr')
- {
- td.appendChild(createStaticArrList(pName, pValue, prop.subType, prop.subDefVal, prop.size, row, flipBkg));
- }
- else if (pType == 'readOnly')
- {
- var inp = document.createElement('input');
- inp.setAttribute('readonly', '');
- inp.value = pValue;
- inp.style.borderWidth = '0px';
- if (pName == 'id')
- {
- inp.style.width = '190px';
- inp.style.position = 'relative';
- inp.style.right = '6px';
- inp.style.float = 'right';
- inp.style.background = 'none';
- inp.style.textAlign = 'right';
- row.firstChild.setAttribute('colspan', '2');
- inp.setAttribute('title', pValue);
- row.firstChild.appendChild(inp);
- }
- else
- {
- inp.style.width = '96px';
- td.appendChild(inp);
- }
- }
- else
- {
- td.innerHTML = mxUtils.htmlEntities(decodeURIComponent(pValue));
-
- mxEvent.addListener(td, 'click', mxUtils.bind(that, function()
- {
- var input = document.createElement('input');
- setElementPos(td, input, true);
- input.value = decodeURIComponent(pValue);
- input.className = 'gePropEditor';
-
- if ((pType == 'int' || pType == 'float') && !prop.allowAuto)
- {
- input.type = 'number';
- input.step = pType == 'int'? '1' : 'any';
-
- if (prop.min != null)
- {
- input.min = parseFloat(prop.min);
- }
-
- if (prop.max != null)
- {
- input.max = parseFloat(prop.max);
- }
- }
-
- div.appendChild(input);
- function setInputVal()
- {
- var inputVal = input.value;
- inputVal = inputVal.length == 0 && pType != 'string' &&
- pType != 'numbers'? 0 : inputVal;
-
- if (prop.allowAuto)
- {
- if (inputVal.trim != null && inputVal.trim().
- toLowerCase() == 'auto')
- {
- inputVal = 'auto';
- pType = 'string';
- }
- else
- {
- inputVal = parseFloat(inputVal);
- inputVal = isNaN(inputVal)? 0 : inputVal;
- }
- }
-
- if (prop.min != null && inputVal < prop.min)
- {
- inputVal = prop.min;
- }
- else if (prop.max != null && inputVal > prop.max)
- {
- inputVal = prop.max;
- }
- var newVal = null;
- try
- {
- newVal = (pType == 'numbers') ? inputVal.match(/\d+/g).map(Number).join(' ') :
- encodeURIComponent((pType == 'int'? parseInt(inputVal) : inputVal) + '');
- }
- catch(e)
- {
- // ignores parsing errors
- }
-
- applyStyleVal(pName, newVal, prop, null, input);
- }
-
- mxEvent.addListener(input, 'keypress', function(e)
- {
- if (e.keyCode == 13)
- {
- setInputVal();
- //set value triggers a redraw of the panel which removes the input
- }
- });
-
- input.focus();
-
- //FF calls blur on focus! so set the event after focusing
- mxEvent.addListener(input, 'blur', function()
- {
- setInputVal();
- });
- }));
- }
- if (prop.isDeletable)
- {
- var delBtn = mxUtils.button('-', mxUtils.bind(that, function(evt)
- {
- //delete the node by refreshing the properties
- applyStyleVal(pName, '', prop, prop.index);
-
- mxEvent.consume(evt);
- }));
-
- delBtn.style.height = '16px';
- delBtn.style.width = '25px';
- delBtn.style.float = 'right';
- delBtn.className = 'geColorBtn';
- td.appendChild(delBtn);
- }
-
- row.appendChild(td);
- return row;
- };
-
- div.style.position = 'relative';
- div.style.padding = '0';
- var grid = document.createElement('table');
- grid.className = 'geProperties';
- grid.style.whiteSpace = 'nowrap';
- grid.style.width = '100%';
- //create header row
- var hrow = document.createElement('tr');
- hrow.className = 'gePropHeader';
- var th = document.createElement('th');
- th.className = 'gePropHeaderCell';
- var collapseImg = document.createElement('img');
- collapseImg.src = Sidebar.prototype.expandedImage;
- collapseImg.style.verticalAlign = 'middle';
- th.appendChild(collapseImg);
- mxUtils.write(th, mxResources.get('property'));
- hrow.style.cursor = 'pointer';
-
- var onFold = function()
- {
- var rows = grid.querySelectorAll('.gePropNonHeaderRow');
- var display;
-
- if (!that.editorUi.propertiesCollapsed)
- {
- collapseImg.src = Sidebar.prototype.expandedImage;
- display = '';
- }
- else
- {
- collapseImg.src = Sidebar.prototype.collapsedImage;
- display = 'none';
-
- for (var e = div.childNodes.length - 1; e >= 0 ; e--)
- {
- //Blur can be executed concurrently with this method and the element is removed before removing it here
- try
- {
- var child = div.childNodes[e];
- var nodeName = child.nodeName.toUpperCase();
-
- if (nodeName == 'INPUT' || nodeName == 'SELECT')
- {
- div.removeChild(child);
- }
- }
- catch(ex){}
- }
- }
-
- for (var r = 0; r < rows.length; r++)
- {
- rows[r].style.display = display;
- }
- };
- mxEvent.addListener(hrow, 'click', function()
- {
- that.editorUi.propertiesCollapsed = !that.editorUi.propertiesCollapsed;
- onFold();
- });
- hrow.appendChild(th);
- th = document.createElement('th');
- th.className = 'gePropHeaderCell';
- th.innerHTML = mxResources.get('value');
- hrow.appendChild(th);
- grid.appendChild(hrow);
-
- var isOdd = false;
- var flipBkg = false;
-
- var cellId = null;
-
- if (state.vertices.length == 1 && state.edges.length == 0)
- {
- cellId = state.vertices[0].id;
- }
- else if (state.vertices.length == 0 && state.edges.length == 1)
- {
- cellId = state.edges[0].id;
- }
-
- //Add it to top (always)
- if (cellId != null && !hideId)
- {
- grid.appendChild(createPropertyRow('id', mxUtils.htmlEntities(cellId),
- {dispName: 'id', type: 'readOnly'}, true, false));
- }
-
- for (var key in properties)
- {
- var prop = properties[key];
-
- if (typeof(prop.isVisible) == 'function')
- {
- if (!prop.isVisible(state, this)) continue;
- }
-
- var pValue = (prop.getValue != null) ? prop.getValue() : (state.style[key] != null? mxUtils.htmlEntities(state.style[key] + '') :
- ((prop.getDefaultValue != null) ? prop.getDefaultValue(state, this) : prop.defVal)); //or undefined if defVal is undefined
- if (prop.type == 'separator')
- {
- flipBkg = !flipBkg;
- continue;
- }
- else if (prop.type == 'staticArr') //if dynamic values are needed, a more elegant technique is needed to replace such values
- {
- prop.size = parseInt(state.style[prop.sizeProperty] || properties[prop.sizeProperty].defVal) || 0;
- }
- else if (prop.dependentProps != null)
- {
- var dependentProps = prop.dependentProps;
- var dependentPropsVals = [];
- var dependentPropsDefVal = [];
-
- for (var i = 0; i < dependentProps.length; i++)
- {
- var propVal = state.style[dependentProps[i]];
- dependentPropsDefVal.push(properties[dependentProps[i]].subDefVal);
- dependentPropsVals.push(propVal != null? propVal.split(',') : []);
- }
-
- prop.dependentPropsDefVal = dependentPropsDefVal;
- prop.dependentPropsVals = dependentPropsVals;
- }
-
- grid.appendChild(createPropertyRow(key, pValue, prop, isOdd, flipBkg));
-
- isOdd = !isOdd;
- }
-
- for (var i = 0; i < secondLevel.length; i++)
- {
- var prop = secondLevel[i];
- var insertElem = prop.parentRow;
-
- for (var j = 0; j < prop.values.length; j++)
- {
- //mxUtils.clone failed because of the HTM element, so manual cloning is used
- var iProp = {type: prop.type, parentRow: prop.parentRow, isDeletable: prop.isDeletable, index: j,
- defVal: prop.defVal, countProperty: prop.countProperty, size: prop.size};
- var arrItem = createPropertyRow(prop.name, prop.values[j], iProp, j % 2 == 0, prop.flipBkg);
- insertAfter(arrItem, insertElem);
- insertElem = arrItem;
- }
- }
-
- div.appendChild(grid);
- onFold();
-
- return div;
- };
-
- /**
- * Creates the buttons for the predefined styles.
- */
- StyleFormatPanel.prototype.addStyles = function(div)
- {
- if (this.defaultColorSchemes != null)
- {
- var ui = this.editorUi;
- var graph = ui.editor.graph;
- var picker = document.createElement('div');
- picker.style.whiteSpace = 'nowrap';
- picker.style.paddingLeft = '24px';
- picker.style.paddingRight = '20px';
- div.style.paddingLeft = '16px';
- div.style.paddingBottom = '6px';
- div.style.position = 'relative';
- div.appendChild(picker);
- var stylenames = ['plain-gray', 'plain-blue', 'plain-green', 'plain-turquoise',
- 'plain-orange', 'plain-yellow', 'plain-red', 'plain-pink', 'plain-purple', 'gray',
- 'blue', 'green', 'turquoise', 'orange', 'yellow', 'red', 'pink', 'purple'];
-
- // Maximum palettes to switch the switcher
- var maxEntries = 10;
-
- // Selector
- var switcher = document.createElement('div');
- switcher.style.whiteSpace = 'nowrap';
- switcher.style.position = 'relative';
- switcher.style.textAlign = 'center';
- switcher.style.width = '210px';
-
- var dots = [];
-
- for (var i = 0; i < this.defaultColorSchemes.length; i++)
- {
- var dot = document.createElement('div');
- dot.style.display = 'inline-block';
- dot.style.width = '6px';
- dot.style.height = '6px';
- dot.style.marginLeft = '4px';
- dot.style.marginRight = '3px';
- dot.style.borderRadius = '3px';
- dot.style.cursor = 'pointer';
- dot.style.background = 'transparent';
- dot.style.border = '1px solid #b5b6b7';
-
- (mxUtils.bind(this, function(index)
- {
- mxEvent.addListener(dot, 'click', mxUtils.bind(this, function()
- {
- setScheme(index);
- }));
- }))(i);
-
- dots.push(dot);
- switcher.appendChild(dot);
- }
-
- var setScheme = mxUtils.bind(this, function(index)
- {
- if (dots[index] != null)
- {
- if (this.format.currentScheme != null && dots[this.format.currentScheme] != null)
- {
- dots[this.format.currentScheme].style.background = 'transparent';
- }
-
- this.format.currentScheme = index;
- updateScheme(this.defaultColorSchemes[this.format.currentScheme]);
- dots[this.format.currentScheme].style.background = '#84d7ff';
- }
- });
-
- var updateScheme = mxUtils.bind(this, function(colorsets)
- {
- var addButton = mxUtils.bind(this, function(colorset)
- {
- var btn = mxUtils.button('', mxUtils.bind(this, function(evt)
- {
- graph.getModel().beginUpdate();
- try
- {
- var cells = ui.getSelectionState().cells;
-
- for (var i = 0; i < cells.length; i++)
- {
- var style = graph.getModel().getStyle(cells[i]);
-
- for (var j = 0; j < stylenames.length; j++)
- {
- style = mxUtils.removeStylename(style, stylenames[j]);
- }
- var defaults = (graph.getModel().isVertex(cells[i])) ? graph.defaultVertexStyle : graph.defaultEdgeStyle;
-
- if (colorset != null)
- {
- if (!mxEvent.isShiftDown(evt))
- {
- if (colorset['fill'] == '')
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_FILLCOLOR, null);
- }
- else
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_FILLCOLOR, colorset['fill'] ||
- mxUtils.getValue(defaults, mxConstants.STYLE_FILLCOLOR, null));
- }
- style = mxUtils.setStyle(style, mxConstants.STYLE_GRADIENTCOLOR, colorset['gradient'] ||
- mxUtils.getValue(defaults, mxConstants.STYLE_GRADIENTCOLOR, null));
-
- if (!mxEvent.isControlDown(evt) && (!mxClient.IS_MAC || !mxEvent.isMetaDown(evt)) &&
- graph.getModel().isVertex(cells[i]))
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_FONTCOLOR, colorset['font'] ||
- mxUtils.getValue(defaults, mxConstants.STYLE_FONTCOLOR, null));
- }
- }
-
- if (!mxEvent.isAltDown(evt))
- {
- if (colorset['stroke'] == '')
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_STROKECOLOR, null);
- }
- else
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_STROKECOLOR, colorset['stroke'] ||
- mxUtils.getValue(defaults, mxConstants.STYLE_STROKECOLOR, null));
- }
- }
- }
- else
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_FILLCOLOR,
- mxUtils.getValue(defaults, mxConstants.STYLE_FILLCOLOR, '#ffffff'));
- style = mxUtils.setStyle(style, mxConstants.STYLE_STROKECOLOR,
- mxUtils.getValue(defaults, mxConstants.STYLE_STROKECOLOR, '#000000'));
- style = mxUtils.setStyle(style, mxConstants.STYLE_GRADIENTCOLOR,
- mxUtils.getValue(defaults, mxConstants.STYLE_GRADIENTCOLOR, null));
-
- if (graph.getModel().isVertex(cells[i]))
- {
- style = mxUtils.setStyle(style, mxConstants.STYLE_FONTCOLOR,
- mxUtils.getValue(defaults, mxConstants.STYLE_FONTCOLOR, null));
- }
- }
- graph.getModel().setStyle(cells[i], style);
- }
- }
- finally
- {
- graph.getModel().endUpdate();
- }
- }));
-
- btn.className = 'geStyleButton';
- btn.style.width = '36px';
- btn.style.height = (this.defaultColorSchemes.length <= maxEntries) ? '24px' : '30px';
- btn.style.margin = '0px 6px 6px 0px';
-
- if (colorset != null)
- {
- var b = (Editor.isDarkMode()) ? '2px solid' : '1px solid';
-
- if (colorset['border'] != null)
- {
- b = colorset['border'];
- }
-
- if (colorset['gradient'] != null)
- {
- if (mxClient.IS_IE && (document.documentMode < 10))
- {
- btn.style.filter = 'progid:DXImageTransform.Microsoft.Gradient('+
- 'StartColorStr=\'' + colorset['fill'] +
- '\', EndColorStr=\'' + colorset['gradient'] + '\', GradientType=0)';
- }
- else
- {
- btn.style.backgroundImage = 'linear-gradient(' + colorset['fill'] + ' 0px,' +
- colorset['gradient'] + ' 100%)';
- }
- }
- else if (colorset['fill'] == mxConstants.NONE)
- {
- btn.style.background = 'url(\'' + Dialog.prototype.noColorImage + '\')';
- }
- else if (colorset['fill'] == '')
- {
- btn.style.backgroundColor = mxUtils.getValue(graph.defaultVertexStyle,
- mxConstants.STYLE_FILLCOLOR, (Editor.isDarkMode()) ? Editor.darkColor : '#ffffff');
- }
- else
- {
- btn.style.backgroundColor = colorset['fill'] || mxUtils.getValue(graph.defaultVertexStyle,
- mxConstants.STYLE_FILLCOLOR, (Editor.isDarkMode()) ? Editor.darkColor : '#ffffff');
- }
-
- if (colorset['stroke'] == mxConstants.NONE)
- {
- btn.style.border = b + ' transparent';
- }
- else if (colorset['stroke'] == '')
- {
- btn.style.border = b + ' ' + mxUtils.getValue(graph.defaultVertexStyle,
- mxConstants.STYLE_STROKECOLOR, (!Editor.isDarkMode()) ? Editor.darkColor : '#ffffff');
- }
- else
- {
- btn.style.border = b + ' ' + (colorset['stroke'] || mxUtils.getValue(graph.defaultVertexStyle,
- mxConstants.STYLE_STROKECOLOR, (!Editor.isDarkMode()) ? Editor.darkColor : '#ffffff'));
- }
- if (colorset['title'] != null)
- {
- btn.setAttribute('title', colorset['title']);
- }
- }
- else
- {
- var bg = mxUtils.getValue(graph.defaultVertexStyle, mxConstants.STYLE_FILLCOLOR, '#ffffff');
- var bd = mxUtils.getValue(graph.defaultVertexStyle, mxConstants.STYLE_STROKECOLOR, '#000000');
-
- btn.style.backgroundColor = bg;
- btn.style.border = '1px solid ' + bd;
- }
- btn.style.borderRadius = '0';
-
- picker.appendChild(btn);
- });
-
- picker.innerText = '';
-
- if (colorsets != null)
- {
- for (var i = 0; i < colorsets.length; i++)
- {
- if (i > 0 && mxUtils.mod(i, 4) == 0)
- {
- mxUtils.br(picker);
- }
-
- addButton(colorsets[i]);
- }
- }
- });
- if (this.format.currentScheme == null)
- {
- setScheme(Math.min(dots.length - 1, Editor.isDarkMode()
- ? 1 : (urlParams['sketch'] == '1' ? 5 : 0)));
- }
- else
- {
- setScheme(this.format.currentScheme);
- }
-
- var bottom = (this.defaultColorSchemes.length <= maxEntries) ? 28 : 8;
- var left = document.createElement('div');
- left.style.cssText = 'position:absolute;left:10px;top:8px;bottom:' + bottom + 'px;width:20px;margin:4px;opacity:0.5;' +
- 'background-repeat:no-repeat;background-position:center center;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAwAAAAQBAMAAADQT4M0AAAAIVBMVEUAAAB2dnZ4eHh3d3d1dXVxcXF2dnZ2dnZ2dnZxcXF2dnYmb3w1AAAACnRSTlMAfCTkhhvb7cQSPH2JPgAAADRJREFUCNdjwACMAmBKaiGYs2oJmLPKAZ3DabU8AMRTXpUKopislqFyVzCAuUZgikkBZjoAcMYLnp53P/UAAAAASUVORK5CYII=);';
-
- mxEvent.addListener(left, 'click', mxUtils.bind(this, function()
- {
- setScheme(mxUtils.mod(this.format.currentScheme - 1, this.defaultColorSchemes.length));
- }));
-
- var right = document.createElement('div');
- right.style.cssText = 'position:absolute;left:202px;top:8px;bottom:' + bottom + 'px;width:20px;margin:4px;opacity:0.5;' +
- 'background-repeat:no-repeat;background-position:center center;background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAwAAAAQBAMAAADQT4M0AAAAIVBMVEUAAAB2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnYBuwCcAAAACnRSTlMAfCTkhhvb7cQSPH2JPgAAADZJREFUCNdjQAOMAmBKaiGY8loF5rKswsZlrVo8AUiFrTICcbIWK8A5DF1gDoMymMPApIAwHwCS0Qx/U7qCBQAAAABJRU5ErkJggg==);';
- if (this.defaultColorSchemes.length > 1)
- {
- div.appendChild(left);
- div.appendChild(right);
- }
-
- mxEvent.addListener(right, 'click', mxUtils.bind(this, function()
- {
- setScheme(mxUtils.mod(this.format.currentScheme + 1, this.defaultColorSchemes.length));
- }));
-
- // Hover state
- function addHoverState(elt)
- {
- mxEvent.addListener(elt, 'mouseenter', function()
- {
- elt.style.opacity = '1';
- });
- mxEvent.addListener(elt, 'mouseleave', function()
- {
- elt.style.opacity = '0.5';
- });
- };
-
- addHoverState(left);
- addHoverState(right);
-
- updateScheme(this.defaultColorSchemes[this.format.currentScheme]);
-
- if (this.defaultColorSchemes.length <= maxEntries)
- {
- div.appendChild(switcher);
- }
- }
-
- return div;
- };
- }
-
- /**
- * Maps fonts to font-face CSS.
- */
- Graph.fontMapping = {'https://fonts.googleapis.com/css?family=Architects+Daughter':
- '@font-face { font-family: "Architects Daughter"; ' +
- 'src: url(' + STYLE_PATH + '/fonts/ArchitectsDaughter-Regular.ttf) format("truetype"); }'};
- /**
- * Lookup table for mapping from font URL and name to elements in the DOM.
- */
- Graph.customFontElements = {};
- /**
- * Returns true if the given font URL references a Google font.
- */
- Graph.isGoogleFontUrl = function(url)
- {
- return url.substring(0, Editor.GOOGLE_FONTS.length) == Editor.GOOGLE_FONTS ||
- url.substring(0, Editor.GOOGLE_FONTS_CSS2.length) == Editor.GOOGLE_FONTS_CSS2;
- };
- /**
- * Returns true if the given font URL is a CSS file.
- */
- Graph.isCssFontUrl = function(url)
- {
- return Graph.isGoogleFontUrl(url);
- };
- /**
- * Uses CSS2 for Google fonts to support bold font style eg.
- * https://fonts.googleapis.com/css?family=IBM+Plex+Sans is rewritten as
- * https://fonts.googleapis.com/css2?family=IBM+Plex+Sans:wght@400;500
- */
- Graph.rewriteGoogleFontUrl = function(url)
- {
- if (url != null && url.substring(0, Editor.GOOGLE_FONTS.length) == Editor.GOOGLE_FONTS)
- {
- url = Editor.GOOGLE_FONTS_CSS2 + url.substring(Editor.GOOGLE_FONTS.length) + ':wght@400;500';
- }
- return url;
- };
- /**
- * Creates the DOM node for the custom font.
- */
- Graph.createFontElement = function(name, url)
- {
- var elt = null;
- var style = Graph.fontMapping[url];
- if (style == null && Graph.isCssFontUrl(url))
- {
- elt = document.createElement('link');
- elt.setAttribute('rel', 'stylesheet');
- elt.setAttribute('type', 'text/css');
- elt.setAttribute('charset', 'UTF-8');
- elt.setAttribute('href', Graph.rewriteGoogleFontUrl(url));
- }
- else
- {
- if (style == null)
- {
- style = '@font-face {\n' +
- 'font-family: "' + name + '";\n' +
- 'src: url("' + url + '");\n}'
- }
- elt = document.createElement('style');
- mxUtils.write(elt, style);
- }
-
- return elt;
- };
-
- /**
- * Adds an entry to the recent custom fonts list.
- */
- Graph.addRecentCustomFont = function(key, entry)
- {
- // Hook for registering recent custom fonts in the UI
- };
- /**
- * Adds a font to the document.
- */
- Graph.addFont = function(name, url, callback)
- {
- if (name != null && name.length > 0 && url != null && url.length > 0)
- {
- var key = name.toLowerCase();
-
- // Blocks UI fonts from being overwritten
- if (key != 'helvetica' && name != 'arial' && key != 'sans-serif')
- {
- var entry = Graph.customFontElements[key];
-
- // Replaces element if URL has changed
- if (entry != null && entry.url != url)
- {
- entry.elt.parentNode.removeChild(entry.elt);
- entry = null;
- }
-
- if (entry == null)
- {
- var realUrl = url;
-
- // Fixes possible mixed content by using proxy
- if (url.substring(0, 5) == 'http:')
- {
- realUrl = PROXY_URL + '?url=' + encodeURIComponent(url);
- }
- entry = {name: name, url: url, elt: Graph.createFontElement(name, realUrl)};
- Graph.customFontElements[key] = entry;
- var head = document.getElementsByTagName('head')[0];
-
- if (callback != null)
- {
- if (entry.elt.nodeName.toLowerCase() == 'link')
- {
- entry.elt.onload = callback;
- entry.elt.onerror = callback;
- }
- else
- {
- callback();
- }
- }
-
- if (head != null)
- {
- head.appendChild(entry.elt);
- }
- }
- else if (callback != null)
- {
- callback();
- }
- }
- else if (callback != null)
- {
- callback();
- }
- }
- else if (callback != null)
- {
- callback();
- }
-
- return name;
- };
- /**
- * Returns the URL for the given font name if it exists in the document.
- * Otherwise it returns the given URL.
- */
- Graph.getFontUrl = function(name, url)
- {
- var font = Graph.customFontElements[name.toLowerCase()];
-
- if (font != null)
- {
- url = font.url;
- }
-
- return url;
- };
-
- /**
- * Processes the fonts in the given element and its descendants.
- */
- Graph.processFontAttributes = function(elt)
- {
- var elts = elt.getElementsByTagName('*');
-
- for (var i = 0; i < elts.length; i++)
- {
- var url = elts[i].getAttribute('data-font-src');
-
- if (url != null)
- {
- var name = (elts[i].nodeName == 'FONT') ?
- elts[i].getAttribute('face') :
- elts[i].style.fontFamily;
-
- if (name != null)
- {
- Graph.addFont(name, url);
- }
- }
- }
- };
-
- /**
- * Processes the font in the given cell style.
- */
- Graph.processFontStyle = function(style)
- {
- if (style != null)
- {
- var url = mxUtils.getValue(style, 'fontSource', null);
-
- if (url != null)
- {
- var name = mxUtils.getValue(style, mxConstants.STYLE_FONTFAMILY, null);
-
- if (name != null)
- {
- try
- {
- Graph.addFont(name, decodeURIComponent(url));
- }
- catch (e)
- {
- // ignore
- }
- }
- }
- }
-
- return style;
- };
- /**
- * Changes the default stylename so that it matches the old named style
- * if one was specified in the XML.
- */
- Graph.prototype.defaultThemeName = 'default-style2';
-
- /**
- * Contains the last XML that was pasted.
- */
- Graph.prototype.lastPasteXml = null;
-
- /**
- * Contains the number of times the last XML was pasted.
- */
- Graph.prototype.pasteCounter = 0;
-
- /**
- * Graph Overrides
- */
- Graph.prototype.defaultScrollbars = urlParams['sb'] != '0';
- /**
- * Specifies if the page should be visible for new files. Default is true.
- */
- Graph.prototype.defaultPageVisible = urlParams['pv'] != '0';
- /**
- * Specifies if the page should be visible for new files. Default is true.
- */
- Graph.prototype.shadowId = 'dropShadow';
-
- /**
- * Properties for the SVG shadow effect.
- */
- Graph.prototype.svgShadowColor = '#3D4574';
-
- /**
- * Properties for the SVG shadow effect.
- */
- Graph.prototype.svgShadowOpacity = '0.4';
- /**
- * Properties for the SVG shadow effect.
- */
- Graph.prototype.svgShadowBlur = '1.7';
-
- /**
- * Properties for the SVG shadow effect.
- */
- Graph.prototype.svgShadowSize = '3';
- /**
- * Enables move of bends/segments without selecting.
- */
- Graph.prototype.hiddenTags = [];
-
- /**
- * Enables move of bends/segments without selecting.
- */
- Graph.prototype.defaultMathEnabled = false;
-
- /**
- * Adds rack child layout style.
- */
- var graphInit = Graph.prototype.init;
-
- Graph.prototype.init = function()
- {
- graphInit.apply(this, arguments);
- // Array of hidden tags used in isCellVisible override
- this.hiddenTags = [];
- //TODO initialize Freehand in the correct location!
- if (window.mxFreehand)
- {
- this.freehand = new mxFreehand(this);
- }
-
- // Override insert location for current mouse point
- var mouseEvent = null;
-
- function setMouseEvent(evt)
- {
- mouseEvent = evt;
- };
-
- mxEvent.addListener(this.container, 'mouseenter', setMouseEvent);
- mxEvent.addListener(this.container, 'mousemove', setMouseEvent);
- mxEvent.addListener(this.container, 'mouseleave', function(evt)
- {
- mouseEvent = null;
- });
-
- // Extends getInsertPoint to use the current mouse location
- this.isMouseInsertPoint = function()
- {
- return mouseEvent != null;
- };
-
- var getInsertPoint = this.getInsertPoint;
-
- this.getInsertPoint = function()
- {
- if (mouseEvent != null)
- {
- return this.getPointForEvent(mouseEvent);
- }
-
- return getInsertPoint.apply(this, arguments);
- };
-
- var layoutManagerGetLayout = this.layoutManager.getLayout;
-
- this.layoutManager.getLayout = function(cell)
- {
- // Workaround for possible invalid style after change and before view validation
- var style = this.graph.getCellStyle(cell);
-
- // mxRackContainer may be undefined as it is dynamically loaded at render time
- if (style != null)
- {
- if (style['childLayout'] == 'rack')
- {
- var rackLayout = new mxStackLayout(this.graph, false);
- var unitSize = 20;
-
- if (style['rackUnitSize'] != null)
- {
- rackLayout.gridSize = parseFloat(style['rackUnitSize']);
- }
- else
- {
- rackLayout.gridSize = (typeof mxRackContainer !== 'undefined') ? mxRackContainer.unitSize : unitSize;
- }
-
- rackLayout.marginLeft = style['marginLeft'] || 0;
- rackLayout.marginRight = style['marginRight'] || 0;
- rackLayout.marginTop = style['marginTop'] || 0;
- rackLayout.marginBottom = style['marginBottom'] || 0;
- rackLayout.allowGaps = style['allowGaps'] || 0;
- rackLayout.horizontal = mxUtils.getValue(style, 'horizontalRack', '0') == '1';
- rackLayout.resizeParent = false;
- rackLayout.fill = true;
-
- return rackLayout;
- }
- }
-
- return layoutManagerGetLayout.apply(this, arguments);
- }
-
- this.updateGlobalUrlVariables();
- };
- /**
- * Adds support for custom fonts in cell styles.
- */
- var graphPostProcessCellStyle = Graph.prototype.postProcessCellStyle;
- Graph.prototype.postProcessCellStyle = function(cell, style)
- {
- return Graph.processFontStyle(graphPostProcessCellStyle.apply(this, arguments));
- };
- /**
- * Handles custom fonts in labels.
- */
- var mxSvgCanvas2DUpdateTextNodes = mxSvgCanvas2D.prototype.updateTextNodes;
- mxSvgCanvas2D.prototype.updateTextNodes = function(x, y, w, h, align, valign, wrap, overflow, clip, rotation, dir, g)
- {
- mxSvgCanvas2DUpdateTextNodes.apply(this, arguments);
- Graph.processFontAttributes(g);
- };
-
- /**
- * Handles custom fonts in labels.
- */
- var mxTextRedraw = mxText.prototype.redraw;
- mxText.prototype.redraw = function()
- {
- mxTextRedraw.apply(this, arguments);
-
- // Handles label rendered without foreign object
- if (this.node != null && this.node.nodeName == 'DIV')
- {
- Graph.processFontAttributes(this.node);
- }
- };
- /**
- * Creates the tags dialog.
- */
- Graph.prototype.createTagsDialog = function(isEnabled, invert, addFn, helpButton)
- {
- var graph = this;
- var allTags = graph.hiddenTags.slice();
-
- var div = document.createElement('div');
- div.style.userSelect = 'none';
- div.style.overflow = 'hidden';
- div.style.padding = '10px';
- div.style.height = '100%';
-
- var tagCloud = document.createElement('div');
- tagCloud.style.boxSizing = 'border-box';
- tagCloud.style.borderRadius = '4px';
- tagCloud.style.userSelect = 'none';
- tagCloud.style.overflow = 'auto';
- tagCloud.style.position = 'absolute';
- tagCloud.style.left = '10px';
- tagCloud.style.right = '10px';
- tagCloud.style.top = '10px';
- tagCloud.style.border = (graph.isEnabled()) ? '1px solid #808080' : 'none';
- tagCloud.style.bottom = (graph.isEnabled()) ? '48px' : '10px';
-
- div.appendChild(tagCloud);
- function removeInvisibleSelectionCells()
- {
- var cells = graph.getSelectionCells();
- var visible = [];
- for (var i = 0; i < cells.length; i++)
- {
- if (graph.isCellVisible(cells[i]))
- {
- visible.push(cells[i]);
- }
- }
- graph.setSelectionCells(visible);
- };
- function setAllVisible(visible)
- {
- graph.setHiddenTags(visible ? [] : allTags.slice());
- removeInvisibleSelectionCells();
- graph.refresh();
- };
- var resetBtn = mxUtils.button(mxResources.get('reset'), function(evt)
- {
- graph.setHiddenTags([]);
- if (!mxEvent.isShiftDown(evt))
- {
- allTags = graph.hiddenTags.slice();
- }
- removeInvisibleSelectionCells();
- graph.refresh();
- });
-
- resetBtn.setAttribute('title', mxResources.get('reset'));
- resetBtn.className = 'geBtn';
- resetBtn.style.margin = '0 4px 0 0';
- var addBtn = mxUtils.button(mxResources.get('add'), function()
- {
- if (addFn != null)
- {
- // Takes all tags and callback to update all tags
- addFn(allTags, function(newAllTags)
- {
- allTags = newAllTags;
- refreshUi();
- });
- }
- });
-
- addBtn.setAttribute('title', mxResources.get('add'));
- addBtn.className = 'geBtn';
- addBtn.style.margin = '0';
-
- graph.addListener(mxEvent.ROOT, function()
- {
- allTags = graph.hiddenTags.slice();
- });
-
- function refreshTags(tags, selected)
- {
- tagCloud.innerText = '';
-
- if (tags.length > 0)
- {
- var table = document.createElement('table');
- table.setAttribute('cellpadding', '2');
- table.style.boxSizing = 'border-box';
- table.style.tableLayout = 'fixed';
- table.style.width = '100%';
-
- var tbody = document.createElement('tbody');
- if (tags != null && tags.length > 0)
- {
- for (var i = 0; i < tags.length; i++)
- {
- (function(tag)
- {
- function setTagVisible()
- {
- var temp = allTags.slice();
- var index = mxUtils.indexOf(temp, tag);
- temp.splice(index, 1);
- graph.setHiddenTags(temp);
- removeInvisibleSelectionCells();
- graph.refresh();
- };
- function selectCells()
- {
- var cells = graph.getCellsForTags(
- [tag], null, null, true);
- if (graph.isEnabled())
- {
- graph.setSelectionCells(cells);
- }
- else
- {
- graph.highlightCells(cells, null, null, 70);
- }
- };
- var visible = mxUtils.indexOf(graph.hiddenTags, tag) < 0;
- var row = document.createElement('tr');
- var td = document.createElement('td');
- td.style.align = 'center';
- td.style.width = '16px';
- var img = document.createElement('img');
- img.setAttribute('src', visible ? Editor.visibleImage : Editor.hiddenImage);
- img.setAttribute('title', mxResources.get(visible ? 'hideIt' : 'show', [tag]));
- mxUtils.setOpacity(img, visible ? 75 : 25);
- img.className = 'geAdaptiveAsset';
- img.style.verticalAlign = 'middle';
- img.style.cursor = 'pointer';
- img.style.width = '16px';
-
- if (invert || Editor.isDarkMode())
- {
- img.style.filter = 'invert(100%)';
- }
-
- td.appendChild(img);
- mxEvent.addListener(img, 'click', function(evt)
- {
- if (mxEvent.isShiftDown(evt))
- {
- setAllVisible(mxUtils.indexOf(graph.hiddenTags, tag) >= 0);
- }
- else
- {
- graph.toggleHiddenTag(tag);
- removeInvisibleSelectionCells();
- graph.refresh();
- }
- mxEvent.consume(evt);
- });
-
- row.appendChild(td);
- td = document.createElement('td');
- td.style.align = 'center';
- td.style.width = '16px';
- var img = document.createElement('img');
- img.setAttribute('src', Editor.selectImage);
- img.setAttribute('title', mxResources.get('select'));
- mxUtils.setOpacity(img, visible ? 75 : 25);
- img.className = 'geAdaptiveAsset';
- img.style.verticalAlign = 'middle';
- img.style.cursor = 'pointer';
- img.style.width = '16px';
-
- if (invert || Editor.isDarkMode())
- {
- img.style.filter = 'invert(100%)';
- }
- mxEvent.addListener(img, 'click', function(evt)
- {
- setAllVisible(true);
- selectCells();
- mxEvent.consume(evt);
- });
-
- td.appendChild(img);
- row.appendChild(td);
- td = document.createElement('td');
- td.style.overflow = 'hidden';
- td.style.whiteSpace = 'nowrap';
- td.style.textOverflow = 'ellipsis';
- td.style.verticalAlign = 'middle';
- td.style.cursor = 'pointer';
- td.setAttribute('title', tag);
-
- a = document.createElement('a');
- mxUtils.write(a, tag);
- a.style.textOverflow = 'ellipsis';
- a.style.position = 'relative';
- mxUtils.setOpacity(a, visible ? 100 : 40);
- td.appendChild(a);
-
- mxEvent.addListener(td, 'click', (function(evt)
- {
- if (mxEvent.isShiftDown(evt))
- {
- setAllVisible(true);
- selectCells();
- }
- else
- {
- if (visible && graph.hiddenTags.length > 0)
- {
- setAllVisible(true);
- }
- else
- {
- setTagVisible();
- }
- }
- mxEvent.consume(evt);
- }));
- row.appendChild(td);
- if (graph.isEnabled())
- {
- td = document.createElement('td');
- td.style.verticalAlign = 'middle';
- td.style.textAlign = 'center';
- td.style.width = '18px';
-
- if (selected == null)
- {
- td.style.align = 'center';
- td.style.width = '16px';
-
- var img = document.createElement('img');
- img.setAttribute('src', Editor.trashImage);
- img.setAttribute('title', mxResources.get('removeIt', [tag]));
- mxUtils.setOpacity(img, visible ? 75 : 25);
- img.className = 'geAdaptiveAsset';
- img.style.verticalAlign = 'middle';
- img.style.cursor = 'pointer';
- img.style.width = '16px';
- if (invert || Editor.isDarkMode())
- {
- img.style.filter = 'invert(100%)';
- }
- mxEvent.addListener(img, 'click', function(evt)
- {
- var idx = mxUtils.indexOf(allTags, tag);
- if (idx >= 0)
- {
- allTags.splice(idx, 1);
- }
- graph.removeTagsForCells(
- graph.model.getDescendants(
- graph.model.getRoot()), [tag]);
- graph.refresh();
- mxEvent.consume(evt);
- });
- td.appendChild(img);
- }
- else
- {
- var cb2 = document.createElement('input');
- cb2.setAttribute('type', 'checkbox');
- cb2.style.margin = '0px';
-
- cb2.defaultChecked = (selected != null &&
- mxUtils.indexOf(selected, tag) >= 0);
- cb2.checked = cb2.defaultChecked;
- cb2.style.background = 'transparent';
- cb2.setAttribute('title', mxResources.get(
- cb2.defaultChecked ?
- 'removeIt' : 'add', [tag]));
-
- mxEvent.addListener(cb2, 'change', function(evt)
- {
- if (cb2.checked)
- {
- graph.addTagsForCells(graph.getSelectionCells(), [tag]);
- }
- else
- {
- graph.removeTagsForCells(graph.getSelectionCells(), [tag]);
- }
-
- mxEvent.consume(evt);
- });
-
- td.appendChild(cb2);
- }
- row.appendChild(td);
- }
-
- tbody.appendChild(row);
- })(tags[i]);
- }
- }
-
- table.appendChild(tbody);
- tagCloud.appendChild(table);
- }
- };
-
- var refreshUi = mxUtils.bind(this, function(sender, evt)
- {
- if (isEnabled())
- {
- var tags = graph.getAllTags();
-
- for (var i = 0; i < tags.length; i++)
- {
- if (mxUtils.indexOf(allTags, tags[i]) < 0)
- {
- allTags.push(tags[i]);
- }
- }
-
- allTags.sort();
-
- if (graph.isSelectionEmpty())
- {
- refreshTags(allTags);
- }
- else
- {
- refreshTags(allTags, graph.getCommonTagsForCells(
- graph.getSelectionCells()));
- }
- }
- });
-
- graph.selectionModel.addListener(mxEvent.CHANGE, refreshUi);
- graph.model.addListener(mxEvent.CHANGE, refreshUi);
- graph.addListener(mxEvent.REFRESH, refreshUi);
-
- var footer = document.createElement('div');
- footer.style.display = 'flex';
- footer.style.alignItems = 'center';
- footer.style.boxSizing = 'border-box';
- footer.style.whiteSpace = 'nowrap';
- footer.style.position = 'absolute';
- footer.style.overflow = 'hidden';
- footer.style.bottom = '6px';
- footer.style.height = '42px';
- footer.style.right = '10px';
- footer.style.left = '10px';
- if (graph.isEnabled())
- {
- footer.appendChild(resetBtn);
- footer.appendChild(addBtn);
- div.appendChild(footer);
- }
- if (helpButton != null)
- {
- footer.appendChild(helpButton);
- }
- return {div: div, refresh: refreshUi};
- };
-
- /**
- * Returns all custom fonts (old and new).
- */
- Graph.prototype.getCustomFonts = function()
- {
- var fonts = this.extFonts;
-
- if (fonts != null)
- {
- fonts = fonts.slice();
- }
- else
- {
- fonts = [];
- }
-
- for (var key in Graph.customFontElements)
- {
- var font = Graph.customFontElements[key];
- fonts.push({name: font.name, url: font.url});
- }
-
- return fonts;
- };
-
- /**
- * Assigns the given custom font to the selected text.
- */
- Graph.prototype.setFont = function(name, url)
- {
- // Adds the font element to the document
- Graph.addFont(name, url);
- // Marks the element with a random font name so
- // that it can be found in the code below
- var temp = Editor.guid();
- document.execCommand('fontname', false, temp);
- // Finds the new or updated element and changes or
- // removes is data-font-src attribute as required
- var fonts = this.cellEditor.textarea.getElementsByTagName('font');
- var elt = null;
- for (var i = 0; i < fonts.length; i++)
- {
- if (fonts[i].getAttribute('face') == temp)
- {
- elt = fonts[i];
- break;
- }
- }
- if (elt != null)
- {
- fonts[i].setAttribute('face', name);
- if (url != null)
- {
- fonts[i].setAttribute('data-font-src', url);
- }
- else
- {
- fonts[i].removeAttribute('data-font-src');
- }
- }
- else
- {
- // Fallback to use the real font name if a new
- // or updated element can not be found above
- document.execCommand('fontname', false, name);
- }
- };
-
- /**
- * Disables fast zoom with shadow in lightbox for Safari
- * to work around blank output on retina screen.
- */
- var graphIsFastZoomEnabled = Graph.prototype.isFastZoomEnabled;
-
- Graph.prototype.isFastZoomEnabled = function()
- {
- return graphIsFastZoomEnabled.apply(this, arguments) && (!this.shadowVisible || !mxClient.IS_SF);
- };
-
- /**
- * Updates the global variables from the vars URL parameter.
- */
- Graph.prototype.updateGlobalUrlVariables = function()
- {
- this.globalVars = Editor.globalVars;
-
- if (urlParams['vars'] != null)
- {
- try
- {
- this.globalVars = (this.globalVars != null) ? mxUtils.clone(this.globalVars) : {};
- var vars = JSON.parse(decodeURIComponent(urlParams['vars']));
-
- if (vars != null)
- {
- for (var key in vars)
- {
- this.globalVars[key] = vars[key];
- }
- }
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.log('Error in vars URL parameter: ' + e);
- }
- }
- }
- };
- /**
- * Returns all global variables used for export. This function never returns null.
- * This can be overridden by plugins to return global variables for export.
- */
- Graph.prototype.getExportVariables = function()
- {
- return (this.globalVars != null) ? mxUtils.clone(this.globalVars) : {};
- };
-
- /**
- * Adds support for vars URL parameter.
- */
- var graphGetGlobalVariable = Graph.prototype.getGlobalVariable;
-
- Graph.prototype.getGlobalVariable = function(name)
- {
- var val = graphGetGlobalVariable.apply(this, arguments);
-
- if (val == null && this.globalVars != null)
- {
- val = this.globalVars[name];
- }
-
- return val;
- };
- /**
- * Cached default stylesheet for image export in dark mode.
- */
- Graph.prototype.getDefaultStylesheet = function()
- {
- if (this.defaultStylesheet == null)
- {
- var node = this.themes['default-style2'];
- var dec = new mxCodec(node.ownerDocument);
- this.defaultStylesheet = dec.decode(node);
- }
-
- return this.defaultStylesheet;
- };
-
- /**
- * Overiddes function to use url parameter
- */
- Graph.prototype.isViewer = function()
- {
- return urlParams['viewer'];
- };
- /**
- * Returns true if the given string contains an mxfile.
- */
- Graph.prototype.adaptBackgroundPage = function(image, theme)
- {
- if (image != null && image.src != null && Graph.isPageLink(image.originalSrc))
- {
- try
- {
- var svg = Graph.getSvgFromDataUri(image.src);
- if (svg != null)
- {
- var doc = new DOMParser().parseFromString(svg, 'text/xml');
- // Removes dark theme CSS
- var styles = doc.getElementsByTagName('style');
- var css = mxUtils.htmlEntities(Graph.createSvgDarkModeCss(), false);
- for (var i = 0; i < styles.length; i++)
- {
- if (styles[i].innerHTML == css)
- {
- styles[i].parentNode.removeChild(styles[i]);
-
- break;
- }
- }
- // Adds new dark theme CSS
- if (theme == 'dark' || theme == 'auto')
- {
- var style = Graph.createSvgDarkModeStyle(doc, theme);
- doc.getElementsByTagName('defs')[0].appendChild(style);
- }
- image = new mxImage(Editor.createSvgDataUri(mxUtils.getXml(
- doc.documentElement)), image.width, image.height,
- image.x, image.y)
- }
- }
- catch (e)
- {
- // ignore
- }
- }
- return image;
- };
- /**
- * Temporarily overrides stylesheet during image export in dark mode.
- */
- var graphGetSvg = Graph.prototype.getSvg;
-
- Graph.prototype.getSvg = function(background, scale, border, nocrop, crisp,
- ignoreSelection, showText, imgExport, linkTarget, hasShadow,
- incExtFonts, theme, exportType, cells, noCssClass, disableLinks)
- {
- var temp = null;
- var tempFg = null;
- var tempBg = null;
- if (!Editor.enableCssDarkMode && this.themes != null && theme != null &&
- ((theme == 'dark') != (this.defaultThemeName == 'darkTheme')))
- {
- temp = this.stylesheet;
- tempFg = this.shapeForegroundColor;
- tempBg = this.shapeBackgroundColor;
- this.shapeForegroundColor = (this.defaultThemeName == 'darkTheme') ?
- '#000000' : Editor.lightColor;
- this.shapeBackgroundColor = (this.defaultThemeName == 'darkTheme') ?
- '#ffffff' : Editor.darkColor;
- this.stylesheet = this.getDefaultStylesheet();
- this.refresh();
- }
- var bgImg = null;
- // Adapts background page to given theme
- if (Editor.enableCssDarkMode && this.backgroundImage != null)
- {
- bgImg = this.backgroundImage;
- this.backgroundImage = this.adaptBackgroundPage(bgImg, theme);
- }
- var result = graphGetSvg.apply(this, arguments);
- if (Editor.enableCssDarkMode && (theme == 'dark' || theme == 'auto'))
- {
- var cssClass = (noCssClass) ? null : 'ge-export-svg-' + theme;
-
- if (cssClass != null)
- {
- result.setAttribute('class', cssClass);
- }
- var style = Graph.createSvgDarkModeStyle(result.ownerDocument, theme, cssClass, background);
- result.getElementsByTagName('defs')[0].appendChild(style);
- }
-
- var extFonts = this.getCustomFonts();
-
- // Adds external fonts
- if (incExtFonts && extFonts.length > 0)
- {
- var svgDoc = result.ownerDocument;
- var style = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'style') : svgDoc.createElement('style');
- svgDoc.setAttributeNS != null? style.setAttributeNS('type', 'text/css') : style.setAttribute('type', 'text/css');
-
- var prefix = '';
- var postfix = '';
-
- for (var i = 0; i < extFonts.length; i++)
- {
- var fontName = extFonts[i].name, fontUrl = extFonts[i].url;
-
- if (Graph.isCssFontUrl(fontUrl))
- {
- prefix += '@import url(' + Graph.rewriteGoogleFontUrl(fontUrl) + ');\n';
- }
- else
- {
- postfix += '@font-face {\n' +
- 'font-family: "' + fontName + '";\n' +
- 'src: url("' + fontUrl + '");\n}\n';
- }
- }
-
- style.appendChild(svgDoc.createTextNode(prefix + postfix));
- result.getElementsByTagName('defs')[0].appendChild(style);
- }
- // Converts background pages to subtrees
- if (Editor.replaceSvgDataUris && bgImg != null &&
- bgImg.originalSrc != null)
- {
- EditorUi.embedSvgImages(result);
- }
- // SVG element must be added to DOM for MathJax to work
- if (this.mathEnabled)
- {
- document.body.appendChild(result);
- Editor.MathJaxRender(result);
- result.parentNode.removeChild(result);
- // Copies MathJax CSS to output
- var style = result.ownerDocument.getElementById('MJX-SVG-styles');
- if (style != null)
- {
- result.getElementsByTagName('defs')[0].appendChild(style.cloneNode(true));
- }
- }
- if (bgImg != null)
- {
- this.backgroundImage = bgImg;
- }
-
- if (temp != null)
- {
- this.shapeBackgroundColor = tempBg;
- this.shapeForegroundColor = tempFg;
- this.stylesheet = temp;
- this.refresh();
- }
-
- return result;
- };
-
- /**
- * Overridden to destroy the shape number.
- */
- var cellRendererDestroy = mxCellRenderer.prototype.destroy;
- mxCellRenderer.prototype.destroy = function(state)
- {
- cellRendererDestroy.apply(this, arguments);
-
- if (state.secondLabel != null)
- {
- state.secondLabel.destroy();
- state.secondLabel = null;
- }
- };
-
- /**
- * Includes the shape number in the return value.
- */
- mxCellRenderer.prototype.getShapesForState = function(state)
- {
- return [state.shape, state.text, state.secondLabel, state.control];
- };
- /**
- * Resets the global shape counter.
- */
- var graphViewResetValidationState = mxGraphView.prototype.resetValidationState;
-
- mxGraphView.prototype.resetValidationState = function()
- {
- graphViewResetValidationState.apply(this, arguments);
- this.enumerationState = 0;
- };
-
- /**
- * Adds shape number update the validation step.
- */
- var graphViewStateValidated = mxGraphView.prototype.stateValidated;
-
- mxGraphView.prototype.stateValidated = function(state)
- {
- if (state.shape != null)
- {
- this.redrawEnumerationState(state);
- }
- return graphViewStateValidated.apply(this, arguments);
- };
- /**
- * Returns the markup to be used for the enumeration shape.
- */
- mxGraphView.prototype.createEnumerationValue = function(state)
- {
- var value = decodeURIComponent(mxUtils.getValue(state.style, 'enumerateValue', ''));
- if (value == '')
- {
- value = ++this.enumerationState;
- }
- return '<div style="padding:2px;border:1px solid gray;background:yellow;border-radius:2px;">' +
- mxUtils.htmlEntities(value) + '</div>';
- };
- /**
- * Adds drawing and update of the shape number.
- */
- mxGraphView.prototype.redrawEnumerationState = function(state)
- {
- var enumerate = mxUtils.getValue(state.style, 'enumerate', 0) == '1';
- if (enumerate && state.secondLabel == null)
- {
- state.secondLabel = new mxText('', new mxRectangle(),
- mxConstants.ALIGN_LEFT, mxConstants.ALIGN_BOTTOM);
- state.secondLabel.size = 12;
- state.secondLabel.state = state;
- state.secondLabel.dialect = mxConstants.DIALECT_STRICTHTML;
- this.graph.cellRenderer.initializeLabel(state, state.secondLabel);
- }
- else if (!enumerate && state.secondLabel != null)
- {
- state.secondLabel.destroy();
- state.secondLabel = null;
- }
- var shape = state.secondLabel;
- if (shape != null)
- {
- var s = state.view.scale;
- var value = this.createEnumerationValue(state);
- var bounds = this.graph.model.isVertex(state.cell) ?
- new mxRectangle(state.x + state.width - 4 * s, state.y + 4 * s, 0, 0) :
- mxRectangle.fromPoint(state.view.getPoint(state));
- if (!shape.bounds.equals(bounds) || shape.value != value || shape.scale != s)
- {
- shape.bounds = bounds;
- shape.value = value;
- shape.scale = s;
- shape.redraw();
- }
- }
- };
- /**
- * Updates the SVG for the background image if it references another page.
- */
- var graphRefresh = Graph.prototype.refresh;
- Graph.prototype.refresh = function()
- {
- this.refreshBackgroundImage();
- graphRefresh.apply(this, arguments);
- };
-
- /**
- * Updates the SVG for the background image if it references another page.
- */
- Graph.prototype.refreshBackgroundImage = function()
- {
- if (this.backgroundImage != null && this.backgroundImage.originalSrc != null)
- {
- this.setBackgroundImage(this.backgroundImage);
- this.view.validateBackgroundImage();
- }
- };
-
- /**
- * Sets default style (used in editor.get/setGraphXml below)
- */
- var graphLoadStylesheet = Graph.prototype.loadStylesheet;
-
- Graph.prototype.loadStylesheet = function()
- {
- graphLoadStylesheet.apply(this, arguments);
- this.currentStyle = 'default-style2';
- };
- /**
- * Adds support for data:action/json,{"actions":[actions]} where actions is
- * a comma-separated list of JSON objects.
- *
- * An example action is:
- *
- * data:action/json,{"actions":[{"toggle": {"cells": ["3", "4"]}}]}
- *
- * This toggles the visible state of the cells with ID 3 and 4.
- */
- Graph.prototype.handleCustomLink = function(href, cell)
- {
- if (href.substring(0, 17) == 'data:action/json,')
- {
- var link = JSON.parse(href.substring(17));
- if (link.actions != null)
- {
- this.executeCustomActions(link.actions, null, cell);
- }
- }
- };
-
- /**
- * Runs the given actions and invokes done when all actions have been executed.
- * When adding new actions that reference cell IDs support for updating
- * those cell IDs must be handled in Graph.updateCustomLinkActions
- */
- Graph.prototype.executeCustomActions = function(actions, done, cell)
- {
- if (!this.executingCustomActions)
- {
- this.executingCustomActions = true;
- var updatingModel = false;
- var waitCounter = 0;
- var index = 0;
- var beginUpdate = mxUtils.bind(this, function()
- {
- if (!updatingModel)
- {
- updatingModel = true;
- this.model.beginUpdate();
- }
- });
- var endUpdate = mxUtils.bind(this, function()
- {
- if (updatingModel)
- {
- updatingModel = false;
- this.model.endUpdate();
- }
- });
- var waitAndExecute = mxUtils.bind(this, function()
- {
- if (waitCounter > 0)
- {
- waitCounter--;
- }
- if (waitCounter == 0)
- {
- executeNextAction()
- }
- });
- var executeNextAction = mxUtils.bind(this, function()
- {
- if (index < actions.length)
- {
- var stop = this.stoppingCustomActions;
- var action = actions[index++];
- var animations = [];
- // Executes open actions before starting transaction
- if (action.open != null)
- {
- endUpdate();
- if (this.isCustomLink(action.open))
- {
- if (!this.customLinkClicked(action.open, cell))
- {
- return;
- }
- }
- else
- {
- this.openLink(action.open);
- }
- }
- if (action.wait != null && !stop)
- {
- this.pendingExecuteNextAction = mxUtils.bind(this, function()
- {
- this.pendingExecuteNextAction = null;
- this.pendingWaitThread = null;
- waitAndExecute();
- });
- waitCounter++;
- this.pendingWaitThread = window.setTimeout(this.pendingExecuteNextAction,
- (action.wait != '') ? parseInt(action.wait) : 1000);
- endUpdate();
- }
- if (action.opacity != null && action.opacity.value != null)
- {
- Graph.setOpacityForNodes(this.getNodesForCells(
- this.getCellsForAction(action.opacity, true)),
- action.opacity.value);
- }
- if (action.fadeIn != null)
- {
- waitCounter++;
- Graph.fadeNodes(this.getNodesForCells(
- this.getCellsForAction(action.fadeIn, true)),
- 0, 1, waitAndExecute, (stop) ?
- 0 : action.fadeIn.delay);
- }
- if (action.fadeOut != null)
- {
- waitCounter++;
- Graph.fadeNodes(this.getNodesForCells(
- this.getCellsForAction(action.fadeOut, true)),
- 1, 0, waitAndExecute, (stop) ?
- 0 : action.fadeOut.delay);
- }
- if (action.wipeIn != null)
- {
- animations = animations.concat(this.createWipeAnimations(
- this.getCellsForAction(action.wipeIn, true), true));
- }
- if (action.wipeOut != null)
- {
- animations = animations.concat(this.createWipeAnimations(
- this.getCellsForAction(action.wipeOut, true), false));
- }
- // Executes all actions that change cell states
- if (action.toggle != null)
- {
- beginUpdate();
- this.toggleCells(this.getCellsForAction(action.toggle, true));
- }
- if (action.show != null)
- {
- beginUpdate();
- var temp = this.getCellsForAction(action.show, true);
- Graph.setOpacityForNodes(this.getNodesForCells(temp), 1);
- this.setCellsVisible(temp, true);
- }
- if (action.hide != null)
- {
- beginUpdate();
- var temp = this.getCellsForAction(action.hide, true);
- Graph.setOpacityForNodes(this.getNodesForCells(temp), 0);
- this.setCellsVisible(temp, false);
- }
-
- if (action.toggleStyle != null && action.toggleStyle.key != null)
- {
- beginUpdate();
- this.toggleCellStyles(action.toggleStyle.key, (action.toggleStyle.defaultValue != null) ?
- action.toggleStyle.defaultValue : '0', this.getCellsForAction(action.toggleStyle, true));
- }
- if (action.style != null && action.style.key != null)
- {
- beginUpdate();
- this.setCellStyles(action.style.key, action.style.value,
- this.getCellsForAction(action.style, true));
- }
- // Executes stateless actions on cells
- var cells = [];
-
- if (action.select != null && this.isEnabled())
- {
- cells = this.getCellsForAction(action.select);
- this.setSelectionCells(cells);
- }
- if (action.highlight != null)
- {
- cells = this.getCellsForAction(action.highlight);
- this.highlightCells(cells, action.highlight.color,
- action.highlight.duration,
- action.highlight.opacity);
- }
- if (action.scroll != null)
- {
- cells = this.getCellsForAction(action.scroll);
- }
-
- if (action.viewbox != null)
- {
- this.fitWindow(action.viewbox, action.viewbox.border);
- }
-
- if (cells.length > 0)
- {
- this.scrollCellToVisible(cells[0]);
- }
-
- if (cell != null && action.explore != null)
- {
- Graph.exploreFromCell(this, cell, action.explore);
- }
- if (action.tags != null)
- {
- if (action.tags.toggle != null)
- {
- var tags = action.tags.toggle;
- if (tags.length == 0)
- {
- tags = this.getAllTags();
- }
- for (var i = 0; i < tags.length; i++)
- {
- this.toggleHiddenTag(tags[i]);
- }
- }
-
- var hidden = null;
- if (action.tags.hidden != null)
- {
- if (hidden == null)
- {
- hidden = [];
- }
-
- hidden = hidden.concat(action.tags.hidden);
- }
- if (action.tags.visible != null)
- {
- if (hidden == null)
- {
- hidden = [];
- }
- var all = this.getAllTags();
- for (var i = 0; i < all.length; i++)
- {
- if (mxUtils.indexOf(action.tags.visible, all[i]) < 0 &&
- mxUtils.indexOf(hidden, all[i]) < 0)
- {
- hidden.push(all[i]);
- }
- }
- }
- if (hidden != null)
- {
- this.setHiddenTags(hidden);
- }
- this.refresh();
- }
- if (animations.length > 0)
- {
- waitCounter++;
- this.executeAnimations(animations, waitAndExecute,
- (stop) ? 1 : action.steps,
- (stop) ? 0 : action.delay);
- }
- if (waitCounter == 0)
- {
- executeNextAction();
- }
- else
- {
- endUpdate();
- }
- }
- else
- {
- this.executingCustomActions = false;
- this.stoppingCustomActions = false;
- endUpdate();
- if (done != null)
- {
- done();
- }
- }
- });
- executeNextAction();
- }
- else
- {
- this.stoppingCustomActions = true;
- if (this.pendingWaitThread != null)
- {
- window.clearTimeout(this.pendingWaitThread);
- }
- if (this.pendingExecuteNextAction != null)
- {
- this.pendingExecuteNextAction();
- }
- this.fireEvent(new mxEventObject('stopExecutingCustomActions'));
- }
- };
- /**
- * Updates cell IDs in custom links on the given cell and its label.
- */
- Graph.prototype.doUpdateCustomLinksForCell = function(mapping, cell)
- {
- var href = this.getLinkForCell(cell);
-
- if (href != null && href.substring(0, 17) == 'data:action/json,')
- {
- this.setLinkForCell(cell, this.updateCustomLink(mapping, href));
- }
-
- if (this.isHtmlLabel(cell))
- {
- var temp = document.createElement('div');
- temp.innerHTML = Graph.sanitizeHtml(this.getLabel(cell));
- var links = temp.getElementsByTagName('a');
- var changed = false;
-
- for (var i = 0; i < links.length; i++)
- {
- href = links[i].getAttribute('href');
-
- if (href != null && href.substring(0, 17) == 'data:action/json,')
- {
- links[i].setAttribute('href', this.updateCustomLink(mapping, href));
- changed = true;
- }
- }
-
- if (changed)
- {
- this.labelChanged(cell, temp.innerHTML);
- }
- }
- };
-
- /**
- * Updates cell IDs in the given custom link and returns the updated link.
- */
- Graph.prototype.updateCustomLink = function(mapping, href)
- {
- if (href.substring(0, 17) == 'data:action/json,')
- {
- try
- {
- var link = JSON.parse(href.substring(17));
- if (link.actions != null)
- {
- this.updateCustomLinkActions(mapping, link.actions);
- href = 'data:action/json,' + JSON.stringify(link);
- }
- }
- catch (e)
- {
- // Ignore
- }
- }
-
- return href;
- };
- /**
- * Updates cell IDs in the given custom link actions.
- */
- Graph.prototype.updateCustomLinkActions = function(mapping, actions)
- {
- for (var i = 0; i < actions.length; i++)
- {
- var action = actions[i];
- for (var name in action)
- {
- this.updateCustomLinkAction(mapping, action[name], 'cells');
- this.updateCustomLinkAction(mapping, action[name], 'excludeCells');
- }
- }
- };
-
- /**
- * Updates cell IDs in the given custom link action.
- */
- Graph.prototype.updateCustomLinkAction = function(mapping, action, name)
- {
- if (action != null && action[name] != null)
- {
- var result = [];
-
- for (var i = 0; i < action[name].length; i++)
- {
- if (action[name][i] == '*')
- {
- result.push(action[name][i]);
- }
- else
- {
- var temp = mapping[action[name][i]];
-
- if (temp != null)
- {
- if (temp != '')
- {
- result.push(temp);
- }
- }
- else
- {
- result.push(action[name][i]);
- }
- }
- }
-
- action[name] = result;
- }
- };
-
- /**
- * Handles each action in the action array of a custom link. This code
- * handles toggle actions for cell IDs.
- */
- Graph.prototype.getCellsForAction = function(action, layers)
- {
- var result = this.getCellsById(action.cells).concat(
- this.getCellsForTags(action.tags, null, layers));
- // Removes excluded cells
- if (action.excludeCells != null)
- {
- var temp = [];
- for (var i = 0; i < result.length; i++)
- {
- if (action.excludeCells.indexOf(result[i].id) < 0)
- {
- temp.push(result[i]);
- }
- }
- result = temp;
- }
- return result;
- };
-
- /**
- * Returns the cells in the model (or given array) that have all of the
- * given tags in their tags property.
- */
- Graph.prototype.getCellsById = function(ids)
- {
- var result = [];
-
- if (ids != null)
- {
- for (var i = 0; i < ids.length; i++)
- {
- if (ids[i] == '*')
- {
- var parent = this.model.getRoot();
-
- result = result.concat(this.model.filterDescendants(function(cell)
- {
- return cell != parent;
- }, parent));
- }
- else
- {
- var cell = this.model.getCell(ids[i]);
-
- if (cell != null)
- {
- result.push(cell);
- }
- }
- }
- }
-
- return result;
- };
- /**
- * Adds support for custom fonts in cell styles.
- */
- var graphIsCellVisible = Graph.prototype.isCellVisible;
- Graph.prototype.isCellVisible = function(cell)
- {
- return graphIsCellVisible.apply(this, arguments) &&
- !this.isAllTagsHidden(this.getTagsForCell(cell));
- };
-
- /**
- * Returns the tags for the given cell as a string.
- */
- Graph.prototype.setHiddenTags = function(tags)
- {
- this.hiddenTags = tags;
- this.fireEvent(new mxEventObject('hiddenTagsChanged'));
- };
- /**
- * Returns the tags for the given cell as a string.
- */
- Graph.prototype.toggleHiddenTag = function(tag)
- {
- var idx = mxUtils.indexOf(this.hiddenTags, tag);
-
- if (idx < 0)
- {
- this.hiddenTags.push(tag);
- }
- else if (idx >= 0)
- {
- this.hiddenTags.splice(idx, 1);
- }
-
- this.fireEvent(new mxEventObject('hiddenTagsChanged'));
- };
-
- /**
- * Returns the cells in the model (or given array) that have all of the
- * given tags in their tags property.
- */
- Graph.prototype.isAllTagsHidden = function(tags)
- {
- if (tags == null || tags.length == 0 ||
- this.hiddenTags.length == 0)
- {
- return false;
- }
- else
- {
- var tmp = tags.split(' ');
- if (tmp.length > this.hiddenTags.length)
- {
- return false;
- }
- else
- {
- for (var i = 0; i < tmp.length; i++)
- {
- if (mxUtils.indexOf(this.hiddenTags, tmp[i]) < 0)
- {
- return false;
- }
- }
-
- return true;
- }
- }
- };
- /**
- * Returns the cells in the model (or given array) that have all of the
- * given tags in their tags property.
- */
- Graph.prototype.getCellsForTags = function(tagList, cells, includeLayers, checkVisible)
- {
- var result = [];
-
- if (tagList != null)
- {
- cells = (cells != null) ? cells : this.model.getDescendants(this.model.getRoot());
-
- var tagCount = 0;
- var lookup = {};
-
- for (var i = 0; i < tagList.length; i++)
- {
- if (tagList[i].length > 0)
- {
- lookup[tagList[i]] = true;
- tagCount++;
- }
- }
-
- for (var i = 0; i < cells.length; i++)
- {
- if ((includeLayers && this.model.getParent(cells[i]) == this.model.root) ||
- this.model.isVertex(cells[i]) || this.model.isEdge(cells[i]))
- {
- var tags = this.getTagsForCell(cells[i]);
- var match = false;
-
- if (tags.length > 0)
- {
- var tmp = tags.split(' ');
-
- if (tmp.length >= tagList.length)
- {
- var matchCount = 0;
-
- for (var j = 0; j < tmp.length && (matchCount < tagCount); j++)
- {
- if (lookup[tmp[j]] != null)
- {
- matchCount++;
- }
- }
-
- match = matchCount == tagCount;
- }
- }
-
- if (match && ((checkVisible != true) || this.isCellVisible(cells[i])))
- {
- result.push(cells[i]);
- }
- }
- }
- }
-
- return result;
- };
- /**
- * Returns all tags in the diagram.
- */
- Graph.prototype.getAllTags = function()
- {
- return this.getTagsForCells(
- this.model.getDescendants(
- this.model.getRoot()));
-
- };
- /**
- * Returns the common tags for the given cells as a array.
- */
- Graph.prototype.getCommonTagsForCells = function(cells)
- {
- var commonTokens = null;
- var validTags = [];
-
- for (var i = 0; i < cells.length; i++)
- {
- var tags = this.getTagsForCell(cells[i]);
- validTags = [];
-
- if (tags.length > 0)
- {
- var tokens = tags.split(' ');
- var temp = {};
-
- for (var j = 0; j < tokens.length; j++)
- {
- if (commonTokens == null || commonTokens[tokens[j]] != null)
- {
- temp[tokens[j]] = true;
- validTags.push(tokens[j]);
- }
- }
-
- commonTokens = temp;
- }
- else
- {
- return [];
- }
- }
-
- return validTags;
- };
-
- /**
- * Returns all tags for the given cells as an array.
- */
- Graph.prototype.getTagsForCells = function(cells)
- {
- var tokens = [];
- var temp = {};
-
- for (var i = 0; i < cells.length; i++)
- {
- var tags = this.getTagsForCell(cells[i]);
- if (tags.length > 0)
- {
- var t = tags.split(' ');
-
- for (var j = 0; j < t.length; j++)
- {
- if (temp[t[j]] == null)
- {
- temp[t[j]] = true;
- tokens.push(t[j]);
- }
- }
- }
- }
-
- return tokens;
- };
- /**
- * Returns the tags for the given cell as a string.
- */
- Graph.prototype.getTagsForCell = function(cell)
- {
- return this.getAttributeForCell(cell, 'tags', '');
- };
- /**
- * Adds the given array of tags to the given array cells.
- */
- Graph.prototype.addTagsForCells = function(cells, tagList)
- {
- if (cells.length > 0 && tagList.length > 0)
- {
- this.model.beginUpdate();
-
- try
- {
- for (var i = 0; i < cells.length; i++)
- {
- var temp = this.getTagsForCell(cells[i]);
- var tags = temp.split(' ');
- var changed = false;
-
- for (var j = 0; j < tagList.length; j++)
- {
- var tag = mxUtils.trim(tagList[j]);
- if (tag != '' && mxUtils.indexOf(tags, tag) < 0)
- {
- temp = (temp.length > 0) ? temp + ' ' + tag : tag;
- changed = true;
- }
- }
-
- if (changed)
- {
- this.setAttributeForCell(cells[i], 'tags', temp);
- }
- }
- }
- finally
- {
- this.model.endUpdate();
- }
- }
- };
- /**
- * Removes the given array of tags from the given array cells.
- */
- Graph.prototype.removeTagsForCells = function(cells, tagList)
- {
- if (cells.length > 0 && tagList.length > 0)
- {
- this.model.beginUpdate();
-
- try
- {
- for (var i = 0; i < cells.length; i++)
- {
- var tags = this.getTagsForCell(cells[i]);
-
- if (tags.length > 0)
- {
- var tokens = tags.split(' ');
- var changed = false;
-
- for (var j = 0; j < tagList.length; j++)
- {
- var idx = mxUtils.indexOf(tokens, tagList[j]);
-
- if (idx >= 0)
- {
- tokens.splice(idx, 1);
- changed = true;
- }
- }
- if (changed)
- {
- this.setAttributeForCell(cells[i], 'tags', tokens.join(' '));
- }
- }
- }
- }
- finally
- {
- this.model.endUpdate();
- }
- }
- };
-
- /**
- * Shows or hides the given cells.
- */
- Graph.prototype.toggleCells = function(cells)
- {
- this.model.beginUpdate();
- try
- {
- for (var i = 0; i < cells.length; i++)
- {
- this.model.setVisible(cells[i], !this.model.isVisible(cells[i]))
- }
- }
- finally
- {
- this.model.endUpdate();
- }
- };
-
- /**
- * Shows or hides the given cells.
- */
- Graph.prototype.setCellsVisible = function(cells, visible)
- {
- this.model.beginUpdate();
- try
- {
- for (var i = 0; i < cells.length; i++)
- {
- this.model.setVisible(cells[i], visible);
- }
- }
- finally
- {
- this.model.endUpdate();
- }
- };
-
- /**
- * Highlights the given cell.
- */
- Graph.prototype.highlightCells = function(cells, color, duration, opacity)
- {
- for (var i = 0; i < cells.length; i++)
- {
- this.highlightCell(cells[i], color, duration, opacity);
- }
- };
-
- /**
- * Highlights the given cell.
- */
- Graph.prototype.highlightCell = function(cell, color, duration, opacity, strokeWidth)
- {
- color = (color != null) ? color : mxConstants.DEFAULT_VALID_COLOR;
- duration = (duration != null) ? duration : 1000;
- var state = this.view.getState(cell);
- var hl = null;
-
- if (state != null)
- {
- strokeWidth = (strokeWidth != null) ? strokeWidth : 4;
- var sw = Math.max(strokeWidth + 1, mxUtils.getValue(state.style,
- mxConstants.STYLE_STROKEWIDTH, 1) + strokeWidth);
- hl = new mxCellHighlight(this, color, sw, false);
-
- if (opacity != null)
- {
- hl.opacity = opacity;
- }
-
- hl.highlight(state);
-
- // Fades out the highlight after a duration
- window.setTimeout(function()
- {
- if (hl.shape != null)
- {
- mxUtils.setPrefixedStyle(hl.shape.node.style,
- 'transition', 'all 1200ms ease-in-out');
- hl.shape.node.style.opacity = 0;
- }
-
- // Destroys the highlight after the fade
- window.setTimeout(function()
- {
- hl.destroy();
- }, 1200);
- }, duration);
- }
- return hl;
- };
- /**
- * Adds a shadow filter to the given svg root.
- */
- Graph.prototype.addSvgShadow = function(svgRoot, group, createOnly, extend)
- {
- createOnly = (createOnly != null) ? createOnly : false;
- extend = (extend != null) ? extend : true;
-
- var svgDoc = svgRoot.ownerDocument;
-
- var filter = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'filter') : svgDoc.createElement('filter');
- filter.setAttribute('id', this.shadowId);
- var blur = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'feGaussianBlur') : svgDoc.createElement('feGaussianBlur');
- blur.setAttribute('in', 'SourceAlpha');
- blur.setAttribute('stdDeviation', this.svgShadowBlur);
- blur.setAttribute('result', 'blur');
- filter.appendChild(blur);
-
- var offset = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'feOffset') : svgDoc.createElement('feOffset');
- offset.setAttribute('in', 'blur');
- offset.setAttribute('dx', this.svgShadowSize);
- offset.setAttribute('dy', this.svgShadowSize);
- offset.setAttribute('result', 'offsetBlur');
- filter.appendChild(offset);
-
- var flood = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'feFlood') : svgDoc.createElement('feFlood');
- flood.setAttribute('flood-color', this.svgShadowColor);
- flood.setAttribute('flood-opacity', this.svgShadowOpacity);
- flood.setAttribute('result', 'offsetColor');
- filter.appendChild(flood);
-
- var composite = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'feComposite') : svgDoc.createElement('feComposite');
- composite.setAttribute('in', 'offsetColor');
- composite.setAttribute('in2', 'offsetBlur');
- composite.setAttribute('operator', 'in');
- composite.setAttribute('result', 'offsetBlur');
- filter.appendChild(composite);
- var feBlend = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'feBlend') : svgDoc.createElement('feBlend');
- feBlend.setAttribute('in', 'SourceGraphic');
- feBlend.setAttribute('in2', 'offsetBlur');
- filter.appendChild(feBlend);
-
- // Creates defs element if not available
- var defs = svgRoot.getElementsByTagName('defs');
- var defsElt = null;
-
- if (defs.length == 0)
- {
- defsElt = (svgDoc.createElementNS != null) ?
- svgDoc.createElementNS(mxConstants.NS_SVG, 'defs') : svgDoc.createElement('defs');
-
- if (svgRoot.firstChild != null)
- {
- svgRoot.insertBefore(defsElt, svgRoot.firstChild);
- }
- else
- {
- svgRoot.appendChild(defsElt);
- }
- }
- else
- {
- defsElt = defs[0];
- }
-
- defsElt.appendChild(filter);
-
- if (!createOnly)
- {
- group = (group != null) ? group : svgRoot.getElementsByTagName('g')[0];
-
- if (group != null)
- {
- group.setAttribute('filter', 'url(#' + this.shadowId + ')');
-
- if (!isNaN(parseInt(svgRoot.getAttribute('width'))) && extend)
- {
- svgRoot.setAttribute('width', parseInt(svgRoot.getAttribute('width')) + 6);
- svgRoot.setAttribute('height', parseInt(svgRoot.getAttribute('height')) + 6);
-
- // Updates viewbox if one exists
- var vb = svgRoot.getAttribute('viewBox');
-
- if (vb != null && vb.length > 0)
- {
- var tokens = vb.split(' ');
-
- if (tokens.length > 3)
- {
- w = parseFloat(tokens[2]) + 6;
- h = parseFloat(tokens[3]) + 6;
-
- svgRoot.setAttribute('viewBox', tokens[0] + ' ' + tokens[1] + ' ' + w + ' ' + h);
- }
- }
- }
- }
- }
-
- return filter;
- };
-
- /**
- * Loads the stylesheet for this graph.
- */
- Graph.prototype.setShadowVisible = function(value, fireEvent)
- {
- if (mxClient.IS_SVG && !mxClient.IS_SF)
- {
- fireEvent = (fireEvent != null) ? fireEvent : true;
- this.shadowVisible = value;
-
- if (this.shadowVisible)
- {
- this.view.getDrawPane().setAttribute('filter', 'url(#' + this.shadowId + ')');
- }
- else
- {
- this.view.getDrawPane().removeAttribute('filter');
- }
-
- if (fireEvent)
- {
- this.fireEvent(new mxEventObject('shadowVisibleChanged'));
- }
- }
- };
-
- /**
- * Selects first unlocked layer if one exists
- */
- Graph.prototype.checkDefaultParent = function()
- {
- if (this.defaultParent != null &&
- !this.model.contains(this.defaultParent))
- {
- this.setDefaultParent(null);
- this.selectUnlockedLayer();
- }
- };
-
- /**
- * Selects first unlocked layer if one exists
- */
- Graph.prototype.selectUnlockedLayer = function()
- {
- if (this.defaultParent == null)
- {
- var childCount = this.model.getChildCount(this.model.root);
- var cell = null;
- var index = 0;
-
- do
- {
- cell = this.model.getChildAt(this.model.root, index);
- } while (index++ < childCount && mxUtils.getValue(this.getCellStyle(cell), 'locked', '0') == '1')
-
- if (cell != null)
- {
- this.setDefaultParent(cell);
- }
- }
- };
- /**
- * Specifies special libraries that are loaded via dynamic JS. Add cases
- * where the filename cannot be worked out from the package name. The
- * standard scheme for this mapping is stencils/packagename.xml. If there
- * are multiple XML files, any JS files or any anomalies in the filename or
- * directory that contains the file, then an entry must be added here and
- * in EmbedServlet2 for the loading of the shapes to work.
- */
- // Required to avoid 404 for mockup.xml since naming of mxgraph.mockup.anchor does not contain
- // buttons even though it is defined in the mxMockupButtons.js file. This could only be fixed
- // with aliases for existing shapes or aliases for basenames, but this is essentially the same.
- mxStencilRegistry.libraries['mockup'] = [SHAPES_PATH + '/mockup/mxMockupButtons.js'];
-
- mxStencilRegistry.libraries['arrows2'] = [SHAPES_PATH + '/mxArrows.js'];
- mxStencilRegistry.libraries['atlassian'] = [STENCIL_PATH + '/atlassian.xml', SHAPES_PATH + '/mxAtlassian.js'];
- mxStencilRegistry.libraries['bpmn'] = [SHAPES_PATH + '/mxBasic.js', STENCIL_PATH + '/bpmn.xml', SHAPES_PATH + '/bpmn/mxBpmnShape2.js'];
- mxStencilRegistry.libraries['bpmn2'] = [SHAPES_PATH + '/mxBasic.js', STENCIL_PATH + '/bpmn.xml', SHAPES_PATH + '/bpmn/mxBpmnShape2.js'];
- mxStencilRegistry.libraries['c4'] = [SHAPES_PATH + '/mxC4.js'];
- mxStencilRegistry.libraries['cisco19'] = [SHAPES_PATH + '/mxCisco19.js', STENCIL_PATH + '/cisco19.xml'];
- mxStencilRegistry.libraries['cisco_safe'] = [SHAPES_PATH + '/mxCiscoSafe.js', STENCIL_PATH + '/cisco_safe/architecture.xml', STENCIL_PATH + '/cisco_safe/business_icons.xml', STENCIL_PATH + '/cisco_safe/capability.xml', STENCIL_PATH + '/cisco_safe/design.xml', STENCIL_PATH + '/cisco_safe/iot_things_icons.xml', STENCIL_PATH + '/cisco_safe/people_places_things_icons.xml', STENCIL_PATH + '/cisco_safe/security_icons.xml', STENCIL_PATH + '/cisco_safe/technology_icons.xml', STENCIL_PATH + '/cisco_safe/threat.xml'];
- mxStencilRegistry.libraries['dfd'] = [SHAPES_PATH + '/mxDFD.js'];
- mxStencilRegistry.libraries['er'] = [SHAPES_PATH + '/er/mxER.js'];
- mxStencilRegistry.libraries['kubernetes'] = [SHAPES_PATH + '/mxKubernetes.js', STENCIL_PATH + '/kubernetes.xml', STENCIL_PATH + '/kubernetes2.xml'];
- mxStencilRegistry.libraries['flowchart'] = [SHAPES_PATH + '/mxFlowchart.js', STENCIL_PATH + '/flowchart.xml'];
- mxStencilRegistry.libraries['ios'] = [SHAPES_PATH + '/mockup/mxMockupiOS.js'];
- mxStencilRegistry.libraries['rackGeneral'] = [SHAPES_PATH + '/rack/mxRack.js', STENCIL_PATH + '/rack/general.xml'];
- mxStencilRegistry.libraries['rackF5'] = [STENCIL_PATH + '/rack/f5.xml'];
- mxStencilRegistry.libraries['lean_mapping'] = [SHAPES_PATH + '/mxLeanMap.js', STENCIL_PATH + '/lean_mapping.xml'];
- mxStencilRegistry.libraries['basic'] = [SHAPES_PATH + '/mxBasic.js', STENCIL_PATH + '/basic.xml'];
- mxStencilRegistry.libraries['ios7icons'] = [STENCIL_PATH + '/ios7/icons.xml'];
- mxStencilRegistry.libraries['ios7ui'] = [SHAPES_PATH + '/ios7/mxIOS7Ui.js', STENCIL_PATH + '/ios7/misc.xml'];
- mxStencilRegistry.libraries['android'] = [SHAPES_PATH + '/mxAndroid.js', STENCIL_PATH + '/android/android.xml'];
- mxStencilRegistry.libraries['electrical/abstract'] = [SHAPES_PATH + '/mxElectrical.js', STENCIL_PATH + '/electrical/abstract.xml'];
- mxStencilRegistry.libraries['electrical/logic_gates'] = [SHAPES_PATH + '/mxElectrical.js', STENCIL_PATH + '/electrical/logic_gates.xml'];
- mxStencilRegistry.libraries['electrical/miscellaneous'] = [SHAPES_PATH + '/mxElectrical.js', STENCIL_PATH + '/electrical/miscellaneous.xml'];
- mxStencilRegistry.libraries['electrical/signal_sources'] = [SHAPES_PATH + '/mxElectrical.js', STENCIL_PATH + '/electrical/signal_sources.xml'];
- mxStencilRegistry.libraries['electrical/electro-mechanical'] = [SHAPES_PATH + '/mxElectrical.js', STENCIL_PATH + '/electrical/electro-mechanical.xml'];
- mxStencilRegistry.libraries['electrical/transmission'] = [SHAPES_PATH + '/mxElectrical.js', STENCIL_PATH + '/electrical/transmission.xml'];
- mxStencilRegistry.libraries['infographic'] = [SHAPES_PATH + '/mxInfographic.js'];
- mxStencilRegistry.libraries['mockup/buttons'] = [SHAPES_PATH + '/mockup/mxMockupButtons.js'];
- mxStencilRegistry.libraries['mockup/containers'] = [SHAPES_PATH + '/mockup/mxMockupContainers.js'];
- mxStencilRegistry.libraries['mockup/forms'] = [SHAPES_PATH + '/mockup/mxMockupForms.js'];
- mxStencilRegistry.libraries['mockup/graphics'] = [SHAPES_PATH + '/mockup/mxMockupGraphics.js', STENCIL_PATH + '/mockup/misc.xml'];
- mxStencilRegistry.libraries['mockup/markup'] = [SHAPES_PATH + '/mockup/mxMockupMarkup.js'];
- mxStencilRegistry.libraries['mockup/misc'] = [SHAPES_PATH + '/mockup/mxMockupMisc.js', STENCIL_PATH + '/mockup/misc.xml'];
- mxStencilRegistry.libraries['mockup/navigation'] = [SHAPES_PATH + '/mockup/mxMockupNavigation.js', STENCIL_PATH + '/mockup/misc.xml'];
- mxStencilRegistry.libraries['mockup/text'] = [SHAPES_PATH + '/mockup/mxMockupText.js'];
- mxStencilRegistry.libraries['floorplan'] = [SHAPES_PATH + '/mxFloorplan.js', STENCIL_PATH + '/floorplan.xml'];
- mxStencilRegistry.libraries['bootstrap'] = [SHAPES_PATH + '/mxBootstrap.js', SHAPES_PATH + '/mxBasic.js', STENCIL_PATH + '/bootstrap.xml'];
- mxStencilRegistry.libraries['gmdl'] = [SHAPES_PATH + '/mxGmdl.js', STENCIL_PATH + '/gmdl.xml'];
- mxStencilRegistry.libraries['gcp2'] = [SHAPES_PATH + '/mxGCP2.js', STENCIL_PATH + '/gcp2.xml'];
- mxStencilRegistry.libraries['ibm'] = [SHAPES_PATH + '/mxIBM.js', STENCIL_PATH + '/ibm.xml'];
- mxStencilRegistry.libraries['cabinets'] = [SHAPES_PATH + '/mxCabinets.js', STENCIL_PATH + '/cabinets.xml'];
- mxStencilRegistry.libraries['archimate'] = [SHAPES_PATH + '/mxArchiMate.js'];
- mxStencilRegistry.libraries['archimate3'] = [SHAPES_PATH + '/mxArchiMate3.js'];
- mxStencilRegistry.libraries['sysml'] = [SHAPES_PATH + '/mxSysML.js'];
- mxStencilRegistry.libraries['eip'] = [SHAPES_PATH + '/mxEip.js', STENCIL_PATH + '/eip.xml'];
- mxStencilRegistry.libraries['networks'] = [SHAPES_PATH + '/mxNetworks.js', STENCIL_PATH + '/networks.xml'];
- mxStencilRegistry.libraries['aws3d'] = [SHAPES_PATH + '/mxAWS3D.js', STENCIL_PATH + '/aws3d.xml'];
- mxStencilRegistry.libraries['aws4'] = [SHAPES_PATH + '/mxAWS4.js', STENCIL_PATH + '/aws4.xml'];
- mxStencilRegistry.libraries['aws4b'] = [SHAPES_PATH + '/mxAWS4.js', STENCIL_PATH + '/aws4.xml'];
- mxStencilRegistry.libraries['uml25'] = [SHAPES_PATH + '/mxUML25.js'];
- mxStencilRegistry.libraries['veeam'] = [STENCIL_PATH + '/veeam/2d.xml', STENCIL_PATH + '/veeam/3d.xml', STENCIL_PATH + '/veeam/veeam.xml'];
- mxStencilRegistry.libraries['veeam2'] = [STENCIL_PATH + '/veeam/2d.xml', STENCIL_PATH + '/veeam/3d.xml', STENCIL_PATH + '/veeam/veeam2.xml'];
- mxStencilRegistry.libraries['pid2inst'] = [SHAPES_PATH + '/pid2/mxPidInstruments.js'];
- mxStencilRegistry.libraries['pid2misc'] = [SHAPES_PATH + '/pid2/mxPidMisc.js', STENCIL_PATH + '/pid/misc.xml'];
- mxStencilRegistry.libraries['pid2valves'] = [SHAPES_PATH + '/pid2/mxPidValves.js'];
- mxStencilRegistry.libraries['pidFlowSensors'] = [STENCIL_PATH + '/pid/flow_sensors.xml'];
- mxStencilRegistry.libraries['salesforce'] = [STENCIL_PATH + '/salesforce.xml'];
- mxStencilRegistry.libraries['sap'] = [SHAPES_PATH + '/mxSAP.js', STENCIL_PATH + '/sap.xml'];
- mxStencilRegistry.libraries['emoji'] = [SHAPES_PATH + '/emoji/mxEmoji.js'];
- // Triggers dynamic loading for markers
- mxMarker.getPackageForType = function(type)
- {
- var name = null;
-
- if (type != null && type.length > 0)
- {
- if (type.substring(0, 2) == 'ER')
- {
- name = 'mxgraph.er';
- }
- else if (type.substring(0, 5) == 'sysML')
- {
- name = 'mxgraph.sysml';
- }
- }
-
- return name;
- };
-
- var mxMarkerCreateMarker = mxMarker.createMarker;
-
- mxMarker.createMarker = function(canvas, shape, type, pe, unitX, unitY, size, source, sw, filled)
- {
- if (type != null)
- {
- var f = mxMarker.markers[type];
-
- if (f == null)
- {
- var name = this.getPackageForType(type);
-
- if (name != null)
- {
- mxStencilRegistry.getStencil(name);
- }
- }
- }
-
- return mxMarkerCreateMarker.apply(this, arguments);
- };
- /**
- * Adds style to mark stencils as lines.
- */
- var mxStencilDrawShape = mxStencil.prototype.drawShape;
-
- mxStencil.prototype.drawShape = function(canvas, shape, x, y, w, h)
- {
- if (mxUtils.getValue(shape.style, 'lineShape', null) == '1')
- {
- canvas.setFillColor(mxUtils.getValue(shape.style,
- mxConstants.STYLE_STROKECOLOR, this.stroke));
- }
- return mxStencilDrawShape.apply(this, arguments);
- };
- /**
- * Constructs a new print dialog.
- */
- PrintDialog.prototype.create = function(editorUi, titleText, fn)
- {
- var graph = editorUi.editor.graph;
- var div = document.createElement('div');
-
- var title = document.createElement('h3');
- title.style.width = '100%';
- title.style.textAlign = 'center';
- title.style.marginTop = '0px';
- mxUtils.write(title, titleText || mxResources.get('print'));
- div.appendChild(title);
- var currentPage = 1;
- var pageCount = 1;
- // Pages
- var pagesSection = document.createElement('div');
- pagesSection.style.whiteSpace = 'nowrap';
-
- var allPagesRadio = document.createElement('input');
- allPagesRadio.style.marginRight = '8px';
- allPagesRadio.style.marginBottom = '8px';
- allPagesRadio.setAttribute('type', 'radio');
- allPagesRadio.setAttribute('name', 'pages-printdialog');
-
- pagesSection.appendChild(allPagesRadio);
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('allPages'));
- mxEvent.addListener(span, 'click', function()
- {
- allPagesRadio.checked = true;
- });
- pagesSection.appendChild(span);
- mxUtils.br(pagesSection);
- // Page range
- var pagesRadio = allPagesRadio.cloneNode(true);
- pagesSection.appendChild(pagesRadio);
-
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('pages') + ':');
- pagesSection.appendChild(span);
- mxEvent.addListener(span, 'click', function()
- {
- pagesRadio.checked = true;
- });
-
- var pagesFromInput = document.createElement('input');
- pagesFromInput.style.margin = '0 4px';
- pagesFromInput.setAttribute('value', '1');
- pagesFromInput.setAttribute('type', 'number');
- pagesFromInput.setAttribute('min', '1');
- pagesFromInput.style.width = '40px';
- pagesSection.appendChild(pagesFromInput);
-
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('to'));
- pagesSection.appendChild(span);
-
- var pagesToInput = pagesFromInput.cloneNode(true);
- pagesSection.appendChild(pagesToInput);
- mxEvent.addListener(pagesFromInput, 'focus', function()
- {
- pagesRadio.checked = true;
- });
-
- mxEvent.addListener(pagesToInput, 'focus', function()
- {
- pagesRadio.checked = true;
- });
-
- function validatePageRange()
- {
- pagesToInput.value = Math.max(1, Math.min(pageCount,
- Math.max(parseInt(pagesToInput.value), parseInt(pagesFromInput.value))));
- pagesFromInput.value = Math.max(1, Math.min(pageCount,
- Math.min(parseInt(pagesToInput.value), parseInt(pagesFromInput.value))));
- };
-
- mxEvent.addListener(pagesFromInput, 'change', validatePageRange);
- mxEvent.addListener(pagesToInput, 'change', validatePageRange);
-
- if (editorUi.pages != null)
- {
- pageCount = editorUi.pages.length;
- if (editorUi.currentPage != null)
- {
- for (var i = 0; i < pageCount; i++)
- {
- if (editorUi.currentPage == editorUi.pages[i])
- {
- currentPage = i + 1;
- break;
- }
- }
- }
- }
-
- if (editorUi.lastPrintPagesFromInput != null &&
- editorUi.lastPrintPagesToInput != null)
- {
- pagesFromInput.value = Math.min(pageCount,
- Math.max(1, editorUi.lastPrintPagesFromInput));
- pagesToInput.value = Math.min(pageCount,
- Math.max(1, editorUi.lastPrintPagesToInput));
- }
- else
- {
- pagesFromInput.value = currentPage;
- pagesToInput.value = currentPage;
- }
-
- pagesFromInput.setAttribute('max', pageCount);
- pagesToInput.setAttribute('max', pageCount);
- var currPage = mxUtils.button(mxResources.get('currentPage'), function(evt)
- {
- pagesFromInput.value = currentPage;
- pagesToInput.value = currentPage;
- pagesRadio.checked = true;
- });
- currPage.setAttribute('title', mxResources.get('currentPage'));
- currPage.style.marginLeft = '4px';
- currPage.style.maxWidth = '100px';
- currPage.style.overflow = 'hidden';
- currPage.style.textOverflow = 'ellipsis';
- currPage.style.whiteSpace = 'nowrap';
- pagesSection.appendChild(currPage);
-
- if (pageCount > 1)
- {
- div.appendChild(pagesSection);
- }
-
- // Selection only
- var selectionSection = document.createElement('div');
- selectionSection.style.borderBottom = '1px solid lightGray';
- selectionSection.style.paddingBottom = '12px';
- selectionSection.style.marginBottom = '12px';
- selectionSection.style.whiteSpace = 'nowrap';
- var selectionOnlyRadio = document.createElement('input');
- selectionOnlyRadio.setAttribute('name', 'pages-printdialog');
- selectionOnlyRadio.setAttribute('type', (pageCount == 1) ? 'checkbox' : 'radio');
- selectionOnlyRadio.style.marginRight = '8px';
-
- if (graph.isSelectionEmpty())
- {
- selectionOnlyRadio.setAttribute('disabled', 'disabled');
- }
- if (graph.isEnabled())
- {
- selectionSection.appendChild(selectionOnlyRadio);
-
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('selectionOnly'));
- selectionSection.appendChild(span);
- }
- if (graph.isEnabled() || pageCount > 1)
- {
- div.appendChild(selectionSection);
- }
- if (!editorUi.isPagesEnabled() || editorUi.lastPrintPagesRadioChecked)
- {
- pagesRadio.checked = true;
- }
- else if (!graph.isSelectionEmpty() && editorUi.lastPrintSelectionOnlyChecked)
- {
- selectionOnlyRadio.checked = true;
- }
- else
- {
- allPagesRadio.checked = true;
- }
- if (!graph.isSelectionEmpty())
- {
- mxEvent.addListener(span, 'click', function()
- {
- selectionOnlyRadio.checked = !selectionOnlyRadio.checked;
- });
- }
-
- // Page view
- var pageViewSection = document.createElement('div');
- pageViewSection.style.whiteSpace = 'nowrap';
- var pageViewRadio = document.createElement('input');
- pageViewRadio.style.marginBottom = '8px';
- pageViewRadio.style.marginRight = '8px';
- pageViewRadio.setAttribute('type', 'radio');
- pageViewRadio.setAttribute('name', 'printSize');
- pageViewSection.appendChild(pageViewRadio);
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('pageView'));
- pageViewSection.appendChild(span);
- mxEvent.addListener(pageViewSection, 'click', function()
- {
- pageViewRadio.checked = true;
- });
- div.appendChild(pageViewSection);
-
- // Crop
- var cropSection = document.createElement('div');
- cropSection.style.whiteSpace = 'nowrap';
- var cropRadio = document.createElement('input');
- cropRadio.style.marginBottom = '8px';
- cropRadio.style.marginRight = '8px';
- cropRadio.setAttribute('type', 'radio');
- cropRadio.setAttribute('name', 'printSize');
- cropSection.appendChild(cropRadio);
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('crop'));
- cropSection.appendChild(span);
- mxEvent.addListener(cropSection, 'click', function()
- {
- cropRadio.checked = true;
- });
- div.appendChild(cropSection);
- // Fit to ...
- var fitSection = document.createElement('div');
- fitSection.style.whiteSpace = 'nowrap';
- var fitRadio = document.createElement('input');
- fitRadio.style.marginBottom = '8px';
- fitRadio.style.marginRight = '8px';
- fitRadio.setAttribute('type', 'radio');
- fitRadio.setAttribute('name', 'printSize');
-
- var spanFitRadio = document.createElement('div');
- spanFitRadio.style.display = 'inline-block';
- spanFitRadio.style.verticalAlign = 'top';
- spanFitRadio.style.paddingTop = '2px';
- spanFitRadio.appendChild(fitRadio);
- fitSection.appendChild(spanFitRadio);
-
- var table = document.createElement('table');
- table.style.display = 'inline-block';
- table.style.borderSpacing = '0';
- var tbody = document.createElement('tbody');
-
- var row1 = document.createElement('tr');
- var row2 = row1.cloneNode(true);
-
- var td1 = document.createElement('td');
- var td2 = td1.cloneNode(true);
- var td3 = td1.cloneNode(true);
-
- var td4 = td1.cloneNode(true);
- var td5 = td1.cloneNode(true);
- var td6 = td1.cloneNode(true);
-
- td1.style.textAlign = 'right';
- td4.style.textAlign = 'right';
- mxUtils.write(td1, mxResources.get('fitTo'));
- mxEvent.addListener(fitSection, 'click', function(e)
- {
- if (mxEvent.getSource(e) != fitRadio)
- {
- fitRadio.checked = !fitRadio.checked;
- }
- });
-
- var sheetsAcrossInput = document.createElement('input');
- sheetsAcrossInput.style.margin = '0 2px';
- sheetsAcrossInput.style.boxSizing = 'border-box';
- sheetsAcrossInput.setAttribute('value', editorUi.lastPrintSheetsAcross || '1');
- sheetsAcrossInput.setAttribute('min', '1');
- sheetsAcrossInput.setAttribute('type', 'number');
- sheetsAcrossInput.style.width = '40px';
- td2.appendChild(sheetsAcrossInput);
-
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('fitToSheetsAcross'));
- td3.appendChild(span);
- mxUtils.write(td4, mxResources.get('fitToBy'));
-
- var sheetsDownInput = sheetsAcrossInput.cloneNode(true);
- sheetsDownInput.setAttribute('value', editorUi.lastPrintSheetsDown || '1');
- td5.appendChild(sheetsDownInput);
-
- mxEvent.addListener(sheetsAcrossInput, 'click', function(e)
- {
- fitRadio.checked = true;
- mxEvent.consume(e);
- });
- mxEvent.addListener(sheetsDownInput, 'click', function(e)
- {
- fitRadio.checked = true;
- mxEvent.consume(e);
- });
- if (editorUi.lastPrintCropRadioChecked)
- {
- cropRadio.checked = true;
- }
- else if (editorUi.lastPrintFitRadioChecked)
- {
- fitRadio.checked = true;
- }
- else
- {
- pageViewRadio.checked = true;
- }
- var span = document.createElement('span');
- mxUtils.write(span, mxResources.get('fitToSheetsDown'));
- td6.appendChild(span);
-
- row1.appendChild(td1);
- row1.appendChild(td2);
- row1.appendChild(td3);
-
- row2.appendChild(td4);
- row2.appendChild(td5);
- row2.appendChild(td6);
-
- tbody.appendChild(row1);
- tbody.appendChild(row2);
- table.appendChild(tbody);
- fitSection.appendChild(table);
-
- div.appendChild(fitSection);
- // Border and zoom
- var optionsSection = document.createElement('div');
- optionsSection.style.borderTop = '1px solid lightGray';
- optionsSection.style.whiteSpace = 'nowrap';
- optionsSection.style.paddingTop = '12px';
- optionsSection.style.marginTop = '12px';
- optionsSection.style.paddingLeft = '8px';
-
- mxUtils.write(optionsSection, mxResources.get('borderWidth') + ':');
- var borderInput = document.createElement('input');
- borderInput.setAttribute('type', 'number');
- borderInput.setAttribute('min', '0');
- borderInput.style.width = '40px';
- borderInput.style.marginLeft = '4px';
- borderInput.value = (editorUi.lastPrintBorder != null) ?
- editorUi.lastPrintBorder : mxPrintPreview.prototype.pageMargin;
- optionsSection.appendChild(borderInput);
- var span = document.createElement('span');
- span.style.marginLeft = '8px';
- mxUtils.write(span, mxResources.get('zoom') + ':');
- optionsSection.appendChild(span);
-
- var zoomInput = document.createElement('input');
- zoomInput.style.width = '60px';
- zoomInput.style.marginLeft = '4px';
- zoomInput.value = (editorUi.lastPrintZoom != null) ?
- editorUi.lastPrintZoom : '100%';
- optionsSection.appendChild(zoomInput);
- mxUtils.br(optionsSection);
- // Grid
- var gridInput = document.createElement('input');
- gridInput.setAttribute('type', 'checkbox');
- gridInput.style.marginTop = '12px';
- gridInput.checked = (editorUi.lastPrintGrid != null) ?
- editorUi.lastPrintGrid : false;
- optionsSection.appendChild(gridInput);
- var span = document.createElement('span');
- span.style.marginLeft = '4px';
- span.style.marginRight = '8px';
- mxUtils.write(span, mxResources.get('grid'));
- optionsSection.appendChild(span);
- mxEvent.addListener(span, 'click', function(e)
- {
- gridInput.checked = true;
- mxEvent.consume(e);
- });
- // Transparent background
- var transparentInput = document.createElement('input');
- transparentInput.setAttribute('type', 'checkbox');
- transparentInput.style.marginTop = '10px';
- transparentInput.checked = (editorUi.lastPrintTransparent != null) ?
- editorUi.lastPrintTransparent : false;
- // Export
- if (fn != null)
- {
- optionsSection.appendChild(transparentInput);
- var span = document.createElement('span');
- span.style.marginLeft = '4px';
- mxUtils.write(span, mxResources.get('transparentBackground'));
- optionsSection.appendChild(span);
- mxEvent.addListener(span, 'click', function(e)
- {
- transparentInput.checked = true;
- mxEvent.consume(e);
- });
- }
- // Include diagram
- var includeInput = document.createElement('input');
- includeInput.setAttribute('type', 'checkbox');
- includeInput.style.marginTop = '10px';
- includeInput.checked = (editorUi.lastPrintInclude != null) ?
- editorUi.lastPrintInclude : Editor.defaultIncludeDiagram;
- if (fn != null && !mxClient.IS_CHROMEAPP &&
- editorUi.getServiceName() == 'draw.io')
- {
- mxUtils.br(optionsSection);
- optionsSection.appendChild(includeInput);
- var span = document.createElement('span');
- span.style.marginLeft = '4px';
- mxUtils.write(span, mxResources.get('includeCopyOfMyDiagram'));
- optionsSection.appendChild(span);
- mxEvent.addListener(span, 'click', function(e)
- {
- includeInput.checked = true;
- mxEvent.consume(e);
- });
- }
- div.appendChild(optionsSection);
- // Buttons
- var buttons = document.createElement('div');
- buttons.style.marginTop = '40px';
- buttons.style.textAlign = 'right';
- buttons.style.whiteSpace = 'nowrap';
-
- // Preview or print
- function preview(print)
- {
- editorUi.lastPrintPagesRadioChecked = pagesRadio.checked;
- editorUi.lastPrintSelectionOnlyChecked = selectionOnlyRadio.checked;
- editorUi.lastPrintCropRadioChecked = cropRadio.checked;
- editorUi.lastPrintFitRadioChecked = fitRadio.checked;
- if (pagesRadio.checked && pagesFromInput.value < pagesToInput.value)
- {
- editorUi.lastPrintPagesFromInput = pagesFromInput.value;
- editorUi.lastPrintPagesToInput = pagesToInput.value;
- }
- else
- {
- editorUi.lastPrintPagesFromInput = null;
- editorUi.lastPrintPagesToInput = null;
- }
- zoomInput.value = Math.max(1, Math.min(1600, parseInt(zoomInput.value))) + '%';
- editorUi.lastPrintZoom = zoomInput.value;
- editorUi.lastPrintBorder = borderInput.value;
- editorUi.lastPrintGrid = gridInput.checked;
- editorUi.lastPrintTransparent = transparentInput.checked;
- editorUi.lastPrintInclude = includeInput.checked;
- editorUi.lastPrintSheetsAcross = sheetsAcrossInput.value;
- editorUi.lastPrintSheetsDown = sheetsDownInput.value;
- var args = {};
- args.border = parseInt(borderInput.value);
- args.scale = parseInt(zoomInput.value) / 100;
- args.grid = gridInput.checked;
- args.transparent = transparentInput.checked;
- args.includeCopy = includeInput.checked;
- args.sheetsAcross = parseInt(sheetsAcrossInput.value);
- args.sheetsDown = parseInt(sheetsDownInput.value);
- args.pagesFrom = parseInt(pagesFromInput.value);
- args.pagesTo = parseInt(pagesToInput.value);
- args.allPages = allPagesRadio.checked;
- args.selection = selectionOnlyRadio.checked;
- args.pageView = pageViewRadio.checked;
- args.crop = cropRadio.checked;
- args.fit = fitRadio.checked;
- return (fn != null) ? fn(!print, args) : editorUi.print(!print, args);
- };
- if (!editorUi.isOffline() && fn == null)
- {
- buttons.appendChild(editorUi.createHelpIcon('https://www.drawio.com/doc/faq/print-diagram'));
- }
-
- var cancelBtn = mxUtils.button(mxResources.get('cancel'), function()
- {
- editorUi.hideDialog();
- });
-
- cancelBtn.className = 'geBtn';
-
- if (editorUi.editor.cancelFirst)
- {
- buttons.appendChild(cancelBtn);
- }
-
- if (PrintDialog.previewEnabled && fn == null)
- {
- var previewBtn = mxUtils.button(mxResources.get('preview'), function()
- {
- try
- {
- editorUi.hideDialog();
- preview(false);
- }
- catch (e)
- {
- editorUi.handleError(e);
- }
- });
- previewBtn.className = 'geBtn';
- buttons.appendChild(previewBtn);
- }
- var printBtn = mxUtils.button(mxResources.get((fn != null) ? 'export' :
- (!PrintDialog.previewEnabled) ? 'ok' : 'print'), function()
- {
- try
- {
- editorUi.hideDialog();
- preview(true);
- }
- catch (e)
- {
- editorUi.handleError(e);
- }
- });
- printBtn.className = 'geBtn gePrimaryBtn';
- buttons.appendChild(printBtn);
-
- if (!editorUi.editor.cancelFirst)
- {
- buttons.appendChild(cancelBtn);
- }
- div.appendChild(buttons);
- this.container = div;
- };
-
- // Execute fit page on page setup changes
- var changePageSetupExecute = ChangePageSetup.prototype.execute;
-
- ChangePageSetup.prototype.execute = function()
- {
- if (this.page == null)
- {
- this.page = this.ui.currentPage;
- }
- // Workaround for redo existing change with different current page
- if (this.page != this.ui.currentPage)
- {
- if (this.page.viewState != null)
- {
- if (!this.ignoreColor)
- {
- this.page.viewState.background = this.color;
- }
-
- if (!this.ignoreImage)
- {
- var img = this.image;
- if (img != null && img.src != null && Graph.isPageLink(img.src))
- {
- img = {originalSrc: img.src};
- }
- this.page.viewState.backgroundImage = img;
- }
- if (this.format != null)
- {
- this.page.viewState.pageFormat = this.format;
- }
-
- if (this.mathEnabled != null)
- {
- this.page.viewState.mathEnabled = this.mathEnabled;
- }
-
- if (this.shadowVisible != null)
- {
- this.page.viewState.shadowVisible = this.shadowVisible;
- }
- }
- }
- else
- {
- changePageSetupExecute.apply(this, arguments);
-
- if (this.mathEnabled != null && this.mathEnabled != this.ui.isMathEnabled())
- {
- this.ui.setMathEnabled(this.mathEnabled);
- this.mathEnabled = !this.mathEnabled;
- }
- if (this.shadowVisible != null && this.shadowVisible != this.ui.editor.graph.shadowVisible)
- {
- this.ui.editor.graph.setShadowVisible(this.shadowVisible);
- this.shadowVisible = !this.shadowVisible;
- }
- }
- };
-
- /**
- * Capability check for canvas export
- */
- Editor.prototype.useCanvasForExport = false;
-
- try
- {
- var canvas = document.createElement('canvas');
- var img = new Image();
-
- // LATER: Capability check should not be async
- img.onload = function()
- {
- try
- {
- var ctx = canvas.getContext('2d');
- ctx.drawImage(img, 0, 0);
- // Works in Chrome, Firefox, Edge, Safari and Opera
- var result = canvas.toDataURL('image/png');
- Editor.prototype.useCanvasForExport = result != null && result.length > 6;
- }
- catch (e)
- {
- // ignore
- }
- };
- // Checks if SVG with foreignObject can be exported
- var svg = '<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="1px" height="1px" version="1.1"><foreignObject pointer-events="all" width="1" height="1"><div xmlns="http://www.w3.org/1999/xhtml"></div></foreignObject></svg>';
- img.src = 'data:image/svg+xml;base64,' + btoa(unescape(encodeURIComponent(svg)));
- }
- catch (e)
- {
- // ignore
- }
- Editor.prototype.useCanvasForExport = false;
-
- })();
- // Extends codec for ChangePageSetup
- (function()
- {
- var codec = new mxObjectCodec(new ChangePageSetup(), ['ui', 'previousColor', 'previousImage', 'previousFormat']);
-
- codec.beforeDecode = function(dec, node, obj)
- {
- obj.ui = dec.ui;
-
- return node;
- };
- codec.afterDecode = function(dec, node, obj)
- {
- obj.previousColor = obj.color;
- obj.previousImage = obj.image;
- obj.previousFormat = obj.format;
- if (obj.foldingEnabled != null)
- {
- obj.foldingEnabled = !obj.foldingEnabled;
- }
-
- if (obj.mathEnabled != null)
- {
- obj.mathEnabled = !obj.mathEnabled;
- }
-
- if (obj.shadowVisible != null)
- {
- obj.shadowVisible = !obj.shadowVisible;
- }
-
- return obj;
- };
- mxCodecRegistry.register(codec);
- })();
- // Extends codec for ChangeGridColor
- (function()
- {
- var codec = new mxObjectCodec(new ChangeGridColor(), ['ui']);
-
- codec.beforeDecode = function(dec, node, obj)
- {
- obj.ui = dec.ui;
-
- return node;
- };
- mxCodecRegistry.register(codec);
- })();
|