123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894 |
- /**
- * Copyright (c) 2006-2020, JGraph Ltd
- * Copyright (c) 2006-2020, draw.io AG
- */
- /**
- * Constructs a new point for the optional x and y coordinates. If no
- * coordinates are given, then the default values for <x> and <y> are used.
- * @constructor
- * @class Implements a basic 2D point. Known subclassers = {@link mxRectangle}.
- * @param {number} x X-coordinate of the point.
- * @param {number} y Y-coordinate of the point.
- */
- App = function(editor, container, lightbox)
- {
- EditorUi.call(this, editor, container, (lightbox != null) ? lightbox :
- (urlParams['lightbox'] == '1' || (uiTheme == 'min' &&
- urlParams['chrome'] != '0')));
-
- // Logs unloading of window with modifications for Google Drive file
- if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp)
- {
- window.onunload = mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
-
- if (file != null && file.isModified())
- {
- var evt = {category: 'DISCARD-FILE-' + file.getHash(),
- action: ((file.savingFile) ? 'saving' : '') +
- ((file.savingFile && file.savingFileTime != null) ? '_' +
- Math.round((Date.now() - file.savingFileTime.getTime()) / 1000) : '') +
- ((file.saveLevel != null) ? ('-sl_' + file.saveLevel) : '') +
- '-age_' + ((file.ageStart != null) ? Math.round((Date.now() - file.ageStart.getTime()) / 1000) : 'x') +
- ((this.editor.autosave) ? '' : '-nosave') +
- ((file.isAutosave()) ? '' : '-noauto') +
- '-open_' + ((file.opened != null) ? Math.round((Date.now() - file.opened.getTime()) / 1000) : 'x') +
- '-save_' + ((file.lastSaved != null) ? Math.round((Date.now() - file.lastSaved.getTime()) / 1000) : 'x') +
- '-change_' + ((file.lastChanged != null) ? Math.round((Date.now() - file.lastChanged.getTime()) / 1000) : 'x') +
- '-alive_' + Math.round((Date.now() - App.startTime.getTime()) / 1000),
- label: (file.sync != null) ? ('client_' + file.sync.clientId) : 'nosync'};
-
- if (file.constructor == DriveFile && file.desc != null && this.drive != null)
- {
- evt.label += ((this.drive.user != null) ? ('-user_' + this.drive.user.id) : '-nouser') + '-rev_' +
- file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + '-size_' + file.getSize() +
- '-mime_' + file.desc.mimeType;
- }
- EditorUi.logEvent(evt);
- }
- });
- }
- // Logs changes to autosave
- this.editor.addListener('autosaveChanged', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- EditorUi.logEvent({category: ((this.editor.autosave) ? 'ON' : 'OFF') +
- '-AUTOSAVE-FILE-' + file.getHash(), action: 'changed',
- label: 'autosave_' + ((this.editor.autosave) ? 'on' : 'off')});
- }
- }));
-
- // Pre-fetches images
- if (mxClient.IS_SVG)
- {
- mxGraph.prototype.warningImage.src = '';
- }
- else
- {
- var img = new Image();
- img.src = mxGraph.prototype.warningImage.src;
- }
-
- // Global helper method to deal with popup blockers
- window.openWindow = mxUtils.bind(this, function(url, pre, fallback)
- {
- var wnd = null;
-
- try
- {
- wnd = window.open(url);
- }
- catch (e)
- {
- // ignore
- }
-
- if (wnd == null || wnd === undefined)
- {
- this.showDialog(new PopupDialog(this, url, pre, fallback).container, 320, 140, true, true);
- }
- else if (pre != null)
- {
- pre();
- }
- });
- // Initial state for toolbar items is disabled
- this.updateDocumentTitle();
- this.updateUi();
- // Global helper method to display error messages
- window.showOpenAlert = mxUtils.bind(this, function(message)
- {
- // Cancel must be called before showing error message
- if (window.openFile != null)
- {
- window.openFile.cancel(true);
- }
-
- this.handleError(message);
- });
- // Handles opening files via drag and drop
- if (!this.editor.chromeless || this.editor.editable)
- {
- this.addFileDropHandler([document]);
- }
-
- // Process the queue for waiting plugins
- if (App.DrawPlugins != null)
- {
- for (var i = 0; i < App.DrawPlugins.length; i++)
- {
- try
- {
- App.DrawPlugins[i](this);
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.log('Plugin Error:', e, App.DrawPlugins[i]);
- }
- }
- finally
- {
- App.embedModePluginsCount--;
- this.initializeEmbedMode();
- }
- }
-
- // Installs global callback for plugins
- window.Draw.loadPlugin = mxUtils.bind(this, function(callback)
- {
- try
- {
- callback(this);
- }
- finally
- {
- App.embedModePluginsCount--;
- this.initializeEmbedMode();
- }
- });
-
- //Set a timeout in case a plugin doesn't load quickly or doesn't load at all
- setTimeout(mxUtils.bind(this, function()
- {
- //Force finish loading if its not yet called
- if (App.embedModePluginsCount > 0)
- {
- App.embedModePluginsCount = 0;
- this.initializeEmbedMode();
- }
- }), 5000); //5 sec timeout
- }
- this.load();
- };
- /**
- * Timeout error
- */
- App.ERROR_TIMEOUT = 'timeout';
- /**
- * Busy error
- */
- App.ERROR_BUSY = 'busy';
- /**
- * Unknown error
- */
- App.ERROR_UNKNOWN = 'unknown';
- /**
- * Google drive mode
- */
- App.MODE_GOOGLE = 'google';
- /**
- * Dropbox mode
- */
- App.MODE_DROPBOX = 'dropbox';
- /**
- * OneDrive Mode
- */
- App.MODE_ONEDRIVE = 'onedrive';
- /**
- * Github Mode
- */
- App.MODE_GITHUB = 'github';
- /**
- * Gitlab mode
- */
- App.MODE_GITLAB = 'gitlab';
- /**
- * Device Mode
- */
- App.MODE_DEVICE = 'device';
- /**
- * Browser Mode
- */
- App.MODE_BROWSER = 'browser';
- /**
- * Trello App Mode
- */
- App.MODE_TRELLO = 'trello';
- /**
- * Embed App Mode
- */
- App.MODE_EMBED = 'embed';
- /**
- * Sets the delay for autosave in milliseconds. Default is 2000.
- */
- App.DROPBOX_APPKEY = 'libwls2fa9szdji';
- /**
- * Sets URL to load the Dropbox SDK from
- */
- App.DROPBOX_URL = window.DRAWIO_BASE_URL + '/js/dropbox/Dropbox-sdk.min.js';
- /**
- * Sets URL to load the Dropbox dropins JS from.
- */
- App.DROPINS_URL = 'https://www.dropbox.com/static/api/2/dropins.js';
- /**
- * OneDrive Client JS (file/folder picker). This is a slightly modified version to allow using accessTokens
- * But it doesn't work for IE11, so we fallback to the original one
- */
- App.ONEDRIVE_URL = mxClient.IS_IE11? 'https://js.live.net/v7.2/OneDrive.js' : window.DRAWIO_BASE_URL + '/js/onedrive/OneDrive.js';
- /**
- * Trello URL
- */
- App.TRELLO_URL = 'https://api.trello.com/1/client.js';
- /**
- * Trello JQuery dependency
- */
- App.TRELLO_JQUERY_URL = 'https://code.jquery.com/jquery-1.7.1.min.js';
- /**
- * Specifies the key for the pusher project.
- */
- App.PUSHER_KEY = '1e756b07a690c5bdb054';
- /**
- * Specifies the key for the pusher project.
- */
- App.PUSHER_CLUSTER = 'eu';
- /**
- * Specifies the URL for the pusher API.
- */
- App.PUSHER_URL = 'https://js.pusher.com/4.3/pusher.min.js';
- /**
- * Google APIs to load. The realtime API is needed to notify collaborators of conversion
- * of the realtime files, but after Dec 11 it's read-only and hence no longer needed.
- */
- App.GOOGLE_APIS = 'drive-share';
- /**
- * Function: authorize
- *
- * Authorizes the client, gets the userId and calls <open>.
- */
- App.startTime = new Date();
- /**
- * Defines plugin IDs for loading via p URL parameter. Update the table at
- * https://desk.draw.io/solution/articles/16000042546
- */
- App.pluginRegistry = {'4xAKTrabTpTzahoLthkwPNUn': '/plugins/explore.js',
- 'ex': '/plugins/explore.js', 'p1': '/plugins/p1.js',
- 'ac': '/plugins/connect.js', 'acj': '/plugins/connectJira.js',
- 'ac148': '/plugins/cConf-1-4-8.js', 'ac148cmnt': '/plugins/cConf-comments.js',
- 'voice': '/plugins/voice.js',
- 'tips': '/plugins/tooltips.js', 'svgdata': '/plugins/svgdata.js',
- 'electron': 'plugins/electron.js',
- 'number': '/plugins/number.js', 'sql': '/plugins/sql.js',
- 'props': '/plugins/props.js', 'text': '/plugins/text.js',
- 'anim': '/plugins/animation.js', 'update': '/plugins/update.js',
- 'trees': '/plugins/trees/trees.js', 'import': '/plugins/import.js',
- 'replay': '/plugins/replay.js', 'anon': '/plugins/anonymize.js',
- 'tr': '/plugins/trello.js', 'f5': '/plugins/rackF5.js',
- 'tickets': '/plugins/tickets.js', 'flow': '/plugins/flow.js',
- 'webcola': '/plugins/webcola/webcola.js', 'rnd': '/plugins/random.js',
- 'page': '/plugins/page.js', 'gd': '/plugins/googledrive.js',
- 'tags': '/plugins/tags.js'};
- App.publicPlugin = [
- 'ex',
- 'voice',
- 'tips',
- 'svgdata',
- 'number',
- 'sql',
- 'props',
- 'text',
- 'anim',
- 'update',
- 'trees',
- // 'import',
- 'replay',
- 'anon',
- 'tickets',
- 'flow',
- 'webcola',
- // 'rnd', 'page', 'gd',
- 'tags'
- ];
- /**
- * Function: authorize
- *
- * Authorizes the client, gets the userId and calls <open>.
- */
- App.getStoredMode = function()
- {
- var mode = null;
-
- if (mode == null && isLocalStorage)
- {
- mode = localStorage.getItem('.mode');
- }
-
- if (mode == null && typeof(Storage) != 'undefined')
- {
- var cookies = document.cookie.split(";");
-
- for (var i = 0; i < cookies.length; i++)
- {
- // Removes spaces around cookie
- var cookie = mxUtils.trim(cookies[i]);
-
- if (cookie.substring(0, 5) == 'MODE=')
- {
- mode = cookie.substring(5);
- break;
- }
- }
-
- if (mode != null && isLocalStorage)
- {
- // Moves to local storage
- var expiry = new Date();
- expiry.setYear(expiry.getFullYear() - 1);
- document.cookie = 'MODE=; expires=' + expiry.toUTCString();
- localStorage.setItem('.mode', mode);
- }
- }
-
- return mode;
- };
- /**
- * Static Application initializer executed at load-time.
- */
- (function()
- {
- if (!mxClient.IS_CHROMEAPP)
- {
- if (urlParams['offline'] != '1')
- {
- // Switches to dropbox mode for db.draw.io
- if (window.location.hostname == 'db.draw.io' && urlParams['mode'] == null)
- {
- urlParams['mode'] = 'dropbox';
- }
-
- App.mode = urlParams['mode'];
- }
-
- if (App.mode == null)
- {
- // Stored mode overrides preferred mode
- App.mode = App.getStoredMode();
- }
-
- /**
- * Lazy loading backends.
- */
- if (window.mxscript != null)
- {
- // Loads gapi for all browsers but IE8 and below if not disabled or if enabled and in embed mode
- if (urlParams['embed'] != '1')
- {
- if (typeof window.DriveClient === 'function')
- {
- if (urlParams['gapi'] != '0' && isSvgBrowser &&
- (document.documentMode == null || document.documentMode >= 10))
- {
- // Immediately loads client
- if (App.mode == App.MODE_GOOGLE || (urlParams['state'] != null &&
- window.location.hash == '') || (window.location.hash != null &&
- window.location.hash.substring(0, 2) == '#G'))
- {
- mxscript('https://apis.google.com/js/api.js');
- }
- // Keeps lazy loading for fallback to authenticated Google file if not public in loadFile
- else if (urlParams['chrome'] == '0' && (window.location.hash == null ||
- window.location.hash.substring(0, 45) !== '#Uhttps%3A%2F%2Fdrive.google.com%2Fuc%3Fid%3D'))
- {
- // Disables loading of client
- window.DriveClient = null;
- }
- }
- else
- {
- // Disables loading of client
- window.DriveClient = null;
- }
- }
-
- // Loads dropbox for all browsers but IE8 and below (no CORS) if not disabled or if enabled and in embed mode
- // KNOWN: Picker does not work in IE11 (https://dropbox.zendesk.com/requests/1650781)
- if (typeof window.DropboxClient === 'function')
- {
- if (urlParams['db'] != '0' && isSvgBrowser &&
- (document.documentMode == null || document.documentMode > 9))
- {
- // Immediately loads client
- if (App.mode == App.MODE_DROPBOX || (window.location.hash != null &&
- window.location.hash.substring(0, 2) == '#D'))
- {
- mxscript(App.DROPBOX_URL, function()
- {
- // Must load this after the dropbox SDK since they use the same namespace
- mxscript(App.DROPINS_URL, null, 'dropboxjs', App.DROPBOX_APPKEY);
- });
- }
- else if (urlParams['chrome'] == '0')
- {
- window.DropboxClient = null;
- }
- }
- else
- {
- // Disables loading of client
- window.DropboxClient = null;
- }
- }
-
- // Loads OneDrive for all browsers but IE6/IOS if not disabled or if enabled and in embed mode
- if (typeof window.OneDriveClient === 'function')
- {
- if (urlParams['od'] != '0' && (navigator.userAgent == null ||
- navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10))
- {
- // Immediately loads client
- if (App.mode == App.MODE_ONEDRIVE || (window.location.hash != null &&
- window.location.hash.substring(0, 2) == '#W'))
- {
- mxscript(App.ONEDRIVE_URL);
- }
- else if (urlParams['chrome'] == '0')
- {
- window.OneDriveClient = null;
- }
- }
- else
- {
- // Disables loading of client
- window.OneDriveClient = null;
- }
- }
-
- // Loads Trello for all browsers but < IE10 if not disabled or if enabled and in embed mode
- if (typeof window.TrelloClient === 'function')
- {
- if (urlParams['tr'] != '0' && isSvgBrowser &&
- (document.documentMode == null || document.documentMode >= 10))
- {
- // Immediately loads client
- if (App.mode == App.MODE_TRELLO || (window.location.hash != null &&
- window.location.hash.substring(0, 2) == '#T'))
- {
- mxscript(App.TRELLO_JQUERY_URL, function()
- {
- mxscript(App.TRELLO_URL);
- });
- }
- else if (urlParams['chrome'] == '0')
- {
- window.TrelloClient = null;
- }
- }
- else
- {
- // Disables loading of client
- window.TrelloClient = null;
- }
- }
- }
-
- // Loads JSON for older browsers
- if (typeof(JSON) == 'undefined')
- {
- mxscript('js/json/json2.min.js');
- }
- }
- }
- })();
- /**
- * Program flow starts here.
- *
- * Optional callback is called with the app instance.
- */
- App.main = function(callback, createUi)
- {
- // Logs uncaught errors
- window.onerror = function(message, url, linenumber, colno, err)
- {
- EditorUi.logError('Global: ' + ((message != null) ? message : ''),
- url, linenumber, colno, err, null, true);
- };
-
- // Removes info text in embed mode
- if (urlParams['embed'] == '1' || urlParams['lightbox'] == '1')
- {
- var geInfo = document.getElementById('geInfo');
-
- if (geInfo != null)
- {
- geInfo.parentNode.removeChild(geInfo);
- }
- }
-
- // Redirects to the latest AWS icons
- if (document.referrer != null && urlParams['libs'] == 'aws3' &&
- document.referrer.substring(0, 42) == 'https://aws.amazon.com/architecture/icons/')
- {
- urlParams['libs'] = 'aws4';
- }
-
- if (window.mxscript != null)
- {
- // Checks for script content changes to avoid CSP errors in production
- if (urlParams['dev'] == '1' && CryptoJS != null)
- {
- var scripts = document.getElementsByTagName('script');
-
- // Checks bootstrap script
- if (scripts != null && scripts.length > 0)
- {
- var content = mxUtils.getTextContent(scripts[0]);
-
- if (CryptoJS.MD5(content).toString() != '6954d575e382bca7fc2090bf81ad77cf')
- {
- console.log('Change bootstrap script MD5 in the previous line:', CryptoJS.MD5(content).toString());
- alert('[Dev] Bootstrap script change requires update of CSP');
- }
- }
-
- // Checks main script
- if (scripts != null && scripts.length > 1)
- {
- var content = mxUtils.getTextContent(scripts[1]);
-
- if (CryptoJS.MD5(content).toString() != 'd41d8cd98f00b204e9800998ecf8427e')
- {
- console.log('Change main script MD5 in the previous line:', CryptoJS.MD5(content).toString());
- alert('[Dev] Main script change requires update of CSP');
- }
- }
- }
- // Runs as progressive web app if service workers are supported
- try
- {
- if ('serviceWorker' in navigator && (/.*\.diagrams\.net$/.test(window.location.hostname) ||
- /.*\.draw\.io$/.test(window.location.hostname) || urlParams['offline'] == '1'))
- {
- // Removes PWA cache on www.draw.io to force use of new domain via redirect
- if (urlParams['offline'] == '0' || /www\.draw\.io$/.test(window.location.hostname) ||
- (urlParams['offline'] != '1' && urlParams['dev'] == '1'))
- {
- navigator.serviceWorker.getRegistrations().then(function(registrations)
- {
- for(var i = 0; i < registrations.length; i++)
- {
- registrations[i].unregister();
- }
- });
- }
- else
- {
- mxscript('js/shapes.min.js', function()
- {
- mxscript('js/stencils.min.js', function()
- {
- mxscript('js/extensions.min.js');
- });
- });
-
- mxStencilRegistry.allowEval = false;
-
- // Use the window load event to keep the page load performant
- window.addEventListener('load', function()
- {
- navigator.serviceWorker.register('/service-worker.js');
- });
- }
- }
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.error(e);
- }
- }
-
- // Loads Pusher API
- if (('ArrayBuffer' in window) && !mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp &&
- DrawioFile.SYNC == 'auto' && (urlParams['embed'] != '1' || urlParams['embedRT'] == '1') && urlParams['local'] != '1' &&
- (urlParams['chrome'] != '0' || urlParams['rt'] == '1') &&
- urlParams['stealth'] != '1' && urlParams['offline'] != '1')
- {
- // TODO: Check if async loading is fast enough
- mxscript(App.PUSHER_URL);
- }
-
- // Loads plugins
- if (urlParams['plugins'] != '0' && urlParams['offline'] != '1')
- {
- // mxSettings is not yet initialized in configure mode, redirect parameter
- // to p URL parameter in caller for plugins in embed mode
- var plugins = (mxSettings.settings != null) ? mxSettings.getPlugins() : null;
-
- // Configured plugins in embed mode with configure=1 URL should be loaded so we
- // look ahead here and parse the config to fetch the list of custom plugins
- if (mxSettings.settings == null && isLocalStorage && typeof(JSON) !== 'undefined')
- {
- try
- {
- var temp = JSON.parse(localStorage.getItem(mxSettings.key));
-
- if (temp != null)
- {
- plugins = temp.plugins;
- }
- }
- catch (e)
- {
- // ignore
- }
- }
- var temp = urlParams['p'];
- App.initPluginCallback();
- if (temp != null)
- {
- // Mapping from key to URL in App.plugins
- App.loadPlugins(temp.split(';'));
- }
-
- if (plugins != null && plugins.length > 0 && urlParams['plugins'] != '0')
- {
- // Loading plugins inside the asynchronous block below stops the page from loading so a
- // hardcoded message for the warning dialog is used since the resources are loadd below
- var warning = 'The page has requested to load the following plugin(s):\n \n {1}\n \n Would you like to load these plugin(s) now?\n \n NOTE : Only allow plugins to run if you fully understand the security implications of doing so.\n';
- var tmp = window.location.protocol + '//' + window.location.host;
- var local = true;
-
- for (var i = 0; i < plugins.length && local; i++)
- {
- if (plugins[i].charAt(0) != '/' && plugins[i].substring(0, tmp.length) != tmp)
- {
- local = false;
- }
- }
-
- if (local || mxUtils.confirm(mxResources.replacePlaceholders(warning, [plugins.join('\n')]).replace(/\\n/g, '\n')))
- {
- for (var i = 0; i < plugins.length; i++)
- {
- try
- {
- if (App.pluginsLoaded[plugins[i]] == null)
- {
- App.pluginsLoaded[plugins[i]] = true;
- App.embedModePluginsCount++;
-
- if (plugins[i].charAt(0) == '/')
- {
- plugins[i] = PLUGINS_BASE_PATH + plugins[i];
- }
-
- mxscript(plugins[i]);
- }
- }
- catch (e)
- {
- // ignore
- }
- }
- }
- }
- }
-
- // Loads gapi for all browsers but IE8 and below if not disabled or if enabled and in embed mode
- // Special case: Cannot load in asynchronous code below
- if (typeof window.DriveClient === 'function' &&
- (typeof gapi === 'undefined' && (((urlParams['embed'] != '1' && urlParams['gapi'] != '0') ||
- (urlParams['embed'] == '1' && urlParams['gapi'] == '1')) && isSvgBrowser &&
- isLocalStorage && (document.documentMode == null || document.documentMode >= 10))))
- {
- mxscript('https://apis.google.com/js/api.js?onload=DrawGapiClientCallback', null, null, null, mxClient.IS_SVG);
- }
- // Disables client
- else if (typeof window.gapi === 'undefined')
- {
- window.DriveClient = null;
- }
- }
-
- /**
- * Asynchronous MathJax extension.
- */
- if (urlParams['math'] != '0')
- {
- Editor.initMath();
- }
- function doLoad(bundle)
- {
- // Prefetches asynchronous requests so that below code runs synchronous
- // Loading the correct bundle (one file) via the fallback system in mxResources. The stylesheet
- // is compiled into JS in the build process and is only needed for local development.
- mxUtils.getAll((urlParams['dev'] != '1') ? [bundle] : [bundle,
- STYLE_PATH + '/default.xml', STYLE_PATH + '/dark-default.xml'], function(xhr)
- {
- // Adds bundle text to resources
- mxResources.parse(xhr[0].getText());
-
- // Configuration mode
- if (isLocalStorage && localStorage != null && window.location.hash != null &&
- window.location.hash.substring(0, 9) == '#_CONFIG_')
- {
- try
- {
- var trustedPlugins = {};
-
- for (var key in App.pluginRegistry)
- {
- trustedPlugins[App.pluginRegistry[key]] = true;
- }
-
- // Only allows trusted plugins
- function checkPlugins(plugins)
- {
- if (plugins != null)
- {
- for (var i = 0; i < plugins.length; i++)
- {
- if (!trustedPlugins[plugins[i]])
- {
- throw new Error(mxResources.get('invalidInput') + ' "' + plugins[i]) + '"';
- }
- }
- }
-
- return true;
- };
-
- var value = JSON.parse(Graph.decompress(window.location.hash.substring(9)));
- if (value != null && checkPlugins(value.plugins))
- {
- EditorUi.debug('Setting configuration', JSON.stringify(value));
-
- if (confirm(mxResources.get('configLinkWarn')) &&
- confirm(mxResources.get('configLinkConfirm')))
- {
- localStorage.setItem('.configuration', JSON.stringify(value));
- window.location.hash = '';
- window.location.reload();
- }
- }
- window.location.hash = '';
- }
- catch (e)
- {
- window.location.hash = '';
- alert(e);
- }
- }
-
- // Prepares themes with mapping from old default-style to old XML file
- if (xhr.length > 2)
- {
- Graph.prototype.defaultThemes['default-style2'] = xhr[1].getDocumentElement();
- Graph.prototype.defaultThemes['darkTheme'] = xhr[2].getDocumentElement();
- }
-
- // Main
- var ui = (createUi != null) ? createUi() : new App(new Editor(
- urlParams['chrome'] == '0' || uiTheme == 'min',
- null, null, null, urlParams['chrome'] != '0'));
-
- if (window.mxscript != null)
- {
- // Loads dropbox for all browsers but IE8 and below (no CORS) if not disabled or if enabled and in embed mode
- // KNOWN: Picker does not work in IE11 (https://dropbox.zendesk.com/requests/1650781)
- if (typeof window.DropboxClient === 'function' &&
- (window.Dropbox == null && window.DrawDropboxClientCallback != null &&
- (((urlParams['embed'] != '1' && urlParams['db'] != '0') ||
- (urlParams['embed'] == '1' && urlParams['db'] == '1')) &&
- isSvgBrowser && (document.documentMode == null || document.documentMode > 9))))
- {
- mxscript(App.DROPBOX_URL, function()
- {
- // Must load this after the dropbox SDK since they use the same namespace
- mxscript(App.DROPINS_URL, function()
- {
- DrawDropboxClientCallback();
- }, 'dropboxjs', App.DROPBOX_APPKEY);
- });
- }
- // Disables client
- else if (typeof window.Dropbox === 'undefined' || typeof window.Dropbox.choose === 'undefined')
- {
- window.DropboxClient = null;
- }
-
- // Loads OneDrive for all browsers but IE6/IOS if not disabled or if enabled and in embed mode
- if (typeof window.OneDriveClient === 'function' &&
- (typeof OneDrive === 'undefined' && window.DrawOneDriveClientCallback != null &&
- (((urlParams['embed'] != '1' && urlParams['od'] != '0') || (urlParams['embed'] == '1' &&
- urlParams['od'] == '1')) && (navigator.userAgent == null ||
- navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10))))
- {
- mxscript(App.ONEDRIVE_URL, window.DrawOneDriveClientCallback);
- }
- // Disables client
- else if (typeof window.OneDrive === 'undefined')
- {
- window.OneDriveClient = null;
- }
-
- // Loads Trello for all browsers but < IE10 if not disabled or if enabled and in embed mode
- if (typeof window.TrelloClient === 'function' &&
- (typeof window.Trello === 'undefined' && window.DrawTrelloClientCallback != null &&
- (((urlParams['embed'] != '1' && urlParams['tr'] != '0') || (urlParams['embed'] == '1' &&
- urlParams['tr'] == '1')) && (navigator.userAgent == null ||
- navigator.userAgent.indexOf('MSIE') < 0 || document.documentMode >= 10))))
- {
- mxscript(App.TRELLO_JQUERY_URL, function()
- {
- // Must load this after the dropbox SDK since they use the same namespace
- mxscript(App.TRELLO_URL, function()
- {
- DrawTrelloClientCallback();
- });
- });
- }
- // Disables client
- else if (typeof window.Trello === 'undefined')
- {
- window.TrelloClient = null;
- }
-
- }
-
- if (callback != null)
- {
- callback(ui);
- }
-
- /**
- * For developers only
- */
- if (urlParams['chrome'] != '0' && urlParams['test'] == '1')
- {
- EditorUi.debug('App.start', [ui, (new Date().getTime() - t0.getTime()) + 'ms']);
-
- if (urlParams['export'] != null)
- {
- EditorUi.debug('Export:', EXPORT_URL);
- }
- }
- }, function(xhr)
- {
- var st = document.getElementById('geStatus');
-
- if (st != null)
- {
- st.innerHTML = 'Error loading page. <a>Please try refreshing.</a>';
-
- // Tries reload with default resources in case any language resources were not available
- st.getElementsByTagName('a')[0].onclick = function()
- {
- mxLanguage = 'en';
- doLoad(mxResources.getDefaultBundle(RESOURCE_BASE, mxLanguage) ||
- mxResources.getSpecialBundle(RESOURCE_BASE, mxLanguage));
- };
- }
- });
- };
- function doMain()
- {
- // Optional override for autosaveDelay and defaultEdgeLength
- try
- {
- if (mxSettings.settings != null)
- {
- if (mxSettings.settings.autosaveDelay != null)
- {
- var val = parseInt(mxSettings.settings.autosaveDelay);
-
- if (!isNaN(val) && val > 0)
- {
- DrawioFile.prototype.autosaveDelay = val;
- EditorUi.debug('Setting autosaveDelay', val);
- }
- else
- {
- EditorUi.debug('Invalid autosaveDelay', val);
- }
- }
-
- if (mxSettings.settings.defaultEdgeLength != null)
- {
- var val = parseInt(mxSettings.settings.defaultEdgeLength);
-
- if (!isNaN(val) && val > 0)
- {
- Graph.prototype.defaultEdgeLength = val;
- EditorUi.debug('Using defaultEdgeLength', val);
- }
- else
- {
- EditorUi.debug('Invalid defaultEdgeLength', val);
- }
- }
- }
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.error(e);
- }
- }
-
- // Adds required resources (disables loading of fallback properties, this can only
- // be used if we know that all keys are defined in the language specific file)
- mxResources.loadDefaultBundle = false;
- doLoad(mxResources.getDefaultBundle(RESOURCE_BASE, mxLanguage) ||
- mxResources.getSpecialBundle(RESOURCE_BASE, mxLanguage));
- };
- // Sends load event if configuration is requested and waits for configure message
- if (urlParams['configure'] == '1')
- {
- var op = window.opener || window.parent;
-
- var configHandler = function(evt)
- {
- if (evt.source == op)
- {
- try
- {
- var data = JSON.parse(evt.data);
-
- if (data != null && data.action == 'configure')
- {
- mxEvent.removeListener(window, 'message', configHandler);
- Editor.configure(data.config, true);
- mxSettings.load();
- doMain();
- }
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.log('Error in configure message: ' + e, evt.data);
- }
- }
- }
- };
-
- // Receives XML message from opener and puts it into the graph
- mxEvent.addListener(window, 'message', configHandler);
- op.postMessage(JSON.stringify({event: 'configure'}), '*');
- }
- else
- {
- if (Editor.config == null)
- {
- // Loads configuration from global scope or local storage
- if (window.DRAWIO_CONFIG != null)
- {
- try
- {
- EditorUi.debug('Using global configuration', window.DRAWIO_CONFIG);
- Editor.configure(window.DRAWIO_CONFIG);
- mxSettings.load();
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.error(e);
- }
- }
- }
-
- // Loads configuration from local storage
- if (isLocalStorage && localStorage != null && urlParams['embed'] != '1')
- {
- var configData = localStorage.getItem('.configuration');
-
- if (configData != null)
- {
- try
- {
- configData = JSON.parse(configData);
-
- if (configData != null)
- {
- EditorUi.debug('Using local configuration', configData);
- Editor.configure(configData);
- mxSettings.load();
- }
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.error(e);
- }
- }
- }
- }
- }
-
- doMain();
- }
- };
- //Extends EditorUi
- mxUtils.extend(App, EditorUi);
- /**
- * Executes the first step for connecting to Google Drive.
- */
- App.prototype.defaultUserPicture = 'https://lh3.googleusercontent.com/-HIzvXUy6QUY/AAAAAAAAAAI/AAAAAAAAAAA/giuR7PQyjEk/photo.jpg?sz=64';
- /**
- *
- */
- App.prototype.shareImage = '';
- /**
- *
- */
- App.prototype.chevronUpImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/chevron-up.png' : '';
- /**
- *
- */
- App.prototype.chevronDownImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/chevron-down.png' : '';
- /**
- *
- */
- App.prototype.formatShowImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/format-show.png' : '';
- /**
- *
- */
- App.prototype.formatHideImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/format-hide.png' : '';
- /**
- *
- */
- App.prototype.fullscreenImage = (!mxClient.IS_SVG) ? IMAGE_PATH + '/fullscreen.png' : '';
- /**
- * Interval to show dialog for unsaved data if autosave is on.
- * Default is 300000 (5 minutes).
- */
- App.prototype.warnInterval = 300000;
- /**
- *
- */
- App.prototype.compactMode = false;
- /**
- *
- */
- App.prototype.fullscreenMode = false;
-
- /**
- * Overriden UI settings depending on mode.
- */
- if (urlParams['embed'] != '1')
- {
- App.prototype.menubarHeight = 64;
- }
- else
- {
- App.prototype.footerHeight = 0;
- }
- /**
- * Queue for loading plugins and wait for UI instance
- */
- App.initPluginCallback = function()
- {
- if (App.DrawPlugins == null)
- {
- // Workaround for need to load plugins now but wait for UI instance
- App.DrawPlugins = [];
-
- // Global entry point for plugins is Draw.loadPlugin. This is the only
- // long-term supported solution for access to the EditorUi instance.
- window.Draw = new Object();
- window.Draw.loadPlugin = function(callback)
- {
- App.DrawPlugins.push(callback);
- };
- }
- };
- /**
- *
- */
- App.pluginsLoaded = {};
- App.embedModePluginsCount = 0;
- /**
- * Queue for loading plugins and wait for UI instance
- */
- App.loadPlugins = function(plugins, useInclude)
- {
- EditorUi.debug('Loading plugins', plugins);
- for (var i = 0; i < plugins.length; i++)
- {
- if (plugins[i] != null && plugins[i].length > 0)
- {
- try
- {
- var url = PLUGINS_BASE_PATH + App.pluginRegistry[plugins[i]];
-
- if (url != null)
- {
- if (App.pluginsLoaded[url] == null)
- {
- App.pluginsLoaded[url] = true;
- App.embedModePluginsCount++;
-
- if (typeof window.drawDevUrl === 'undefined')
- {
- if (useInclude)
- {
- mxinclude(url);
- }
- else
- {
- mxscript(url);
- }
- }
- else
- {
- if (useInclude)
- {
- mxinclude(url);
- }
- else
- {
- mxscript(drawDevUrl + url);
- }
- }
- }
- }
- else if (window.console != null)
- {
- console.log('Unknown plugin:', plugins[i]);
- }
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.log('Error loading plugin:', plugins[i], e);
- }
- }
- }
- }
- };
- /**
- * Delay embed mode initialization until all plugins are loaded
- */
- App.prototype.initializeEmbedMode = function()
- {
- if (urlParams['embed'] == '1')
- {
- if (window.location.hostname == 'app.diagrams.net')
- {
- this.showBanner('EmbedDeprecationFooter', 'app.diagrams.net will stop working for embed mode. Please use embed.diagrams.net.');
- }
-
- if (App.embedModePluginsCount > 0 || this.initEmbedDone)
- {
- return; //Wait for plugins to load, or this is a duplicate call due to timeout
- }
- else
- {
- this.initEmbedDone = true;
- }
-
- EditorUi.prototype.initializeEmbedMode.apply(this, arguments);
- }
- };
- /**
- * TODO: Define viewer protocol and implement new viewer style toolbar
- */
- App.prototype.initializeViewerMode = function()
- {
- var parent = window.opener || window.parent;
- if (parent != null)
- {
- this.editor.graph.addListener(mxEvent.SIZE, mxUtils.bind(this, function()
- {
- parent.postMessage(JSON.stringify(this.createLoadMessage('size')), '*');
- }));
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.init = function()
- {
- EditorUi.prototype.init.apply(this, arguments);
- /**
- * Specifies the default filename.
- */
- this.defaultLibraryName = mxResources.get('untitledLibrary');
- /**
- * Holds the listener for description changes.
- */
- this.descriptorChangedListener = mxUtils.bind(this, this.descriptorChanged);
- /**
- * Creates github client.
- */
- this.gitHub = (!mxClient.IS_IE || document.documentMode == 10 ||
- mxClient.IS_IE11 || mxClient.IS_EDGE) &&
- (urlParams['gh'] != '0' && (urlParams['embed'] != '1' ||
- urlParams['gh'] == '1')) ? new GitHubClient(this) : null;
-
- if (this.gitHub != null)
- {
- this.gitHub.addListener('userChanged', mxUtils.bind(this, function()
- {
- this.updateUserElement();
- this.restoreLibraries();
- }))
- }
-
- /**
- * Creates gitlab client.
- */
- this.gitLab = (!mxClient.IS_IE || document.documentMode == 10 ||
- mxClient.IS_IE11 || mxClient.IS_EDGE) &&
- (urlParams['gl'] != '0' && (urlParams['embed'] != '1' ||
- urlParams['gl'] == '1')) ? new GitLabClient(this) : null;
- if (this.gitLab != null)
- {
- this.gitLab.addListener('userChanged', mxUtils.bind(this, function()
- {
- this.updateUserElement();
- this.restoreLibraries();
- }));
- }
- /**
- * Lazy-loading for individual backends
- */
- if (urlParams['embed'] != '1' || urlParams['od'] == '1')
- {
- /**
- * Creates onedrive client if all required libraries are available.
- */
- var initOneDriveClient = mxUtils.bind(this, function()
- {
- if (typeof OneDrive !== 'undefined')
- {
- /**
- * Holds the x-coordinate of the point.
- */
- this.oneDrive = new OneDriveClient(this);
-
- this.oneDrive.addListener('userChanged', mxUtils.bind(this, function()
- {
- this.updateUserElement();
- this.restoreLibraries();
- }));
-
- // Notifies listeners of new client
- this.fireEvent(new mxEventObject('clientLoaded', 'client', this.oneDrive));
- }
- else if (window.DrawOneDriveClientCallback == null)
- {
- window.DrawOneDriveClientCallback = initOneDriveClient;
- }
- });
- initOneDriveClient();
- }
- /**
- * Lazy-loading for Trello
- */
- if (urlParams['embed'] != '1' || urlParams['tr'] == '1')
- {
- /**
- * Creates Trello client if all required libraries are available.
- */
- var initTrelloClient = mxUtils.bind(this, function()
- {
- if (typeof window.Trello !== 'undefined')
- {
- try
- {
- this.trello = new TrelloClient(this);
-
- //TODO we have no user info from Trello so we don't set a user
- this.trello.addListener('userChanged', mxUtils.bind(this, function()
- {
- this.updateUserElement();
- this.restoreLibraries();
- }));
-
- // Notifies listeners of new client
- this.fireEvent(new mxEventObject('clientLoaded', 'client', this.trello));
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.error(e);
- }
- }
- }
- else if (window.DrawTrelloClientCallback == null)
- {
- window.DrawTrelloClientCallback = initTrelloClient;
- }
- });
- initTrelloClient();
- }
- /**
- * Creates drive client with all required libraries are available.
- */
- if (urlParams['embed'] != '1' || urlParams['gapi'] == '1')
- {
- var initDriveClient = mxUtils.bind(this, function()
- {
- /**
- * Creates google drive client if all required libraries are available.
- */
- if (typeof gapi !== 'undefined')
- {
- var doInit = mxUtils.bind(this, function()
- {
- this.drive = new DriveClient(this);
-
- this.drive.addListener('userChanged', mxUtils.bind(this, function()
- {
- this.updateUserElement();
- this.restoreLibraries();
- this.checkLicense();
- }))
-
- // Notifies listeners of new client
- this.fireEvent(new mxEventObject('clientLoaded', 'client', this.drive));
- });
-
- if (window.DrawGapiClientCallback != null)
- {
- gapi.load(((urlParams['picker'] != '0') ? 'picker,': '') + App.GOOGLE_APIS, doInit);
-
- /**
- * Clears any callbacks.
- */
- window.DrawGapiClientCallback = null;
- }
- else
- {
- doInit();
- }
- }
- else if (window.DrawGapiClientCallback == null)
- {
- window.DrawGapiClientCallback = initDriveClient;
- }
- });
-
- initDriveClient();
- }
- if (urlParams['embed'] != '1' || urlParams['db'] == '1')
- {
- /**
- * Creates dropbox client if all required libraries are available.
- */
- var initDropboxClient = mxUtils.bind(this, function()
- {
- if (typeof Dropbox === 'function' && typeof Dropbox.choose !== 'undefined')
- {
- /**
- * Clears dropbox client callback.
- */
- window.DrawDropboxClientCallback = null;
-
- /**
- * Holds the x-coordinate of the point.
- */
- try
- {
- this.dropbox = new DropboxClient(this);
-
- this.dropbox.addListener('userChanged', mxUtils.bind(this, function()
- {
- this.updateUserElement();
- this.restoreLibraries();
- }));
-
- // Notifies listeners of new client
- this.fireEvent(new mxEventObject('clientLoaded', 'client', this.dropbox));
- }
- catch (e)
- {
- if (window.console != null)
- {
- console.error(e);
- }
- }
- }
- else if (window.DrawDropboxClientCallback == null)
- {
- window.DrawDropboxClientCallback = initDropboxClient;
- }
- });
- initDropboxClient();
- }
- if (urlParams['embed'] != '1')
- {
- /**
- * Holds the background element.
- */
- this.bg = this.createBackground();
- document.body.appendChild(this.bg);
- this.diagramContainer.style.visibility = 'hidden';
- this.formatContainer.style.visibility = 'hidden';
- this.hsplit.style.display = 'none';
- this.sidebarContainer.style.display = 'none';
- this.sidebarFooterContainer.style.display = 'none';
- // Sets the initial mode
- if (urlParams['local'] == '1')
- {
- this.setMode(App.MODE_DEVICE);
- }
- else
- {
- this.mode = App.mode;
- }
-
- // Add to Home Screen dialog for mobile devices
- if ('serviceWorker' in navigator && !this.editor.isChromelessView() &&
- (mxClient.IS_ANDROID || mxClient.IS_IOS))
- {
- window.addEventListener('beforeinstallprompt', mxUtils.bind(this, function(e)
- {
- this.showBanner('AddToHomeScreenFooter', mxResources.get('installApp'), function()
- {
- e.prompt();
- });
- }));
- }
-
- if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && !this.isOffline() &&
- !mxClient.IS_ANDROID && !mxClient.IS_IOS && urlParams['open'] == null &&
- (!this.editor.chromeless || this.editor.editable))
- {
- this.editor.addListener('fileLoaded', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- var mode = (file != null) ? file.getMode() : null;
-
- if (mode == App.MODE_DEVICE || mode == App.MODE_BROWSER)
- {
- this.showDownloadDesktopBanner();
- }
- }));
- }
-
- if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && urlParams['embed'] != '1' && DrawioFile.SYNC == 'auto' &&
- urlParams['local'] != '1' && urlParams['stealth'] != '1' && !this.isOffline() &&
- (!this.editor.chromeless || this.editor.editable))
- {
- // Checks if the cache is alive
- var acceptResponse = true;
-
- var timeoutThread = window.setTimeout(mxUtils.bind(this, function()
- {
- acceptResponse = false;
-
- // Switches to manual sync if cache cannot be reached
- DrawioFile.SYNC = 'manual';
-
- var file = this.getCurrentFile();
-
- if (file != null && file.sync != null)
- {
- file.sync.destroy();
- file.sync = null;
-
- var status = mxUtils.htmlEntities(mxResources.get('timeout'));
- this.editor.setStatus('<div title="'+ status +
- '" class="geStatusAlert" style="overflow:hidden;">' + status +
- '</div>');
- }
-
- EditorUi.logEvent({category: 'TIMEOUT-CACHE-CHECK', action: 'timeout', label: 408});
- }), Editor.cacheTimeout);
-
- var t0 = new Date().getTime();
-
- mxUtils.get(EditorUi.cacheUrl + '?alive', mxUtils.bind(this, function(req)
- {
- window.clearTimeout(timeoutThread);
- }));
- }
- }
- else if (this.menubar != null)
- {
- this.menubar.container.style.paddingTop = '0px';
- }
- this.updateHeader();
- if (this.menubar != null)
- {
- this.buttonContainer = document.createElement('div');
- this.buttonContainer.style.display = 'inline-block';
- this.buttonContainer.style.paddingRight = '48px';
- this.buttonContainer.style.position = 'absolute';
- this.buttonContainer.style.right = '0px';
-
- this.menubar.container.appendChild(this.buttonContainer);
- }
- if (uiTheme == 'atlas' && this.menubar != null)
- {
- if (this.toggleElement != null)
- {
- this.toggleElement.click();
- this.toggleElement.style.display = 'none';
- }
-
- this.icon = document.createElement('img');
- this.icon.setAttribute('src', IMAGE_PATH + '/logo-flat-small.png');
- this.icon.setAttribute('title', mxResources.get('draw.io'));
- this.icon.style.padding = '6px';
- this.icon.style.cursor = 'pointer';
-
- mxEvent.addListener(this.icon, 'click', mxUtils.bind(this, function(evt)
- {
- this.appIconClicked(evt);
- }));
-
- if (mxClient.IS_QUIRKS)
- {
- this.icon.style.marginTop = '12px';
- }
-
- this.menubar.container.insertBefore(this.icon, this.menubar.container.firstChild);
- }
-
- if (this.editor.graph.isViewer())
- {
- this.initializeViewerMode();
- }
- };
- /**
- * Schedules a sanity check.
- */
- App.prototype.scheduleSanityCheck = function()
- {
- if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp &&
- this.sanityCheckThread == null)
- {
- this.sanityCheckThread = window.setTimeout(mxUtils.bind(this, function()
- {
- this.sanityCheckThread = null;
- this.sanityCheck();
- }), this.warnInterval);
- }
- };
- /**
- * Stops sanity checks.
- */
- App.prototype.stopSanityCheck = function()
- {
- if (this.sanityCheckThread != null)
- {
- window.clearTimeout(this.sanityCheckThread);
- this.sanityCheckThread = null;
- }
- };
- /**
- * Shows a warning after some time with unsaved changes and autosave.
- */
- App.prototype.sanityCheck = function()
- {
- var file = this.getCurrentFile();
- if (file != null && file.isModified() && file.isAutosave() && file.isOverdue())
- {
- var evt = {category: 'WARN-FILE-' + file.getHash(),
- action: ((file.savingFile) ? 'saving' : '') +
- ((file.savingFile && file.savingFileTime != null) ? '_' +
- Math.round((Date.now() - file.savingFileTime.getTime()) / 1000) : '') +
- ((file.saveLevel != null) ? ('-sl_' + file.saveLevel) : '') +
- '-age_' + ((file.ageStart != null) ? Math.round((Date.now() - file.ageStart.getTime()) / 1000) : 'x') +
- ((this.editor.autosave) ? '' : '-nosave') +
- ((file.isAutosave()) ? '' : '-noauto') +
- '-open_' + ((file.opened != null) ? Math.round((Date.now() - file.opened.getTime()) / 1000) : 'x') +
- '-save_' + ((file.lastSaved != null) ? Math.round((Date.now() - file.lastSaved.getTime()) / 1000) : 'x') +
- '-change_' + ((file.lastChanged != null) ? Math.round((Date.now() - file.lastChanged.getTime()) / 1000) : 'x')+
- '-alive_' + Math.round((Date.now() - App.startTime.getTime()) / 1000),
- label: (file.sync != null) ? ('client_' + file.sync.clientId) : 'nosync'};
-
- if (file.constructor == DriveFile && file.desc != null && this.drive != null)
- {
- evt.label += ((this.drive.user != null) ? ('-user_' + this.drive.user.id) : '-nouser') + '-rev_' +
- file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + '-size_' + file.getSize() +
- '-mime_' + file.desc.mimeType;
- }
-
- EditorUi.logEvent(evt);
- var msg = mxResources.get('ensureDataSaved');
-
- if (file.lastSaved != null)
- {
- var str = this.timeSince(file.lastSaved);
-
- if (str == null)
- {
- str = mxResources.get('lessThanAMinute');
- }
- msg = mxResources.get('lastSaved', [str]);
- }
-
- // Resets possible stale state
- this.spinner.stop();
- this.showError(mxResources.get('unsavedChanges'), msg, mxResources.get('ignore'),
- mxUtils.bind(this, function()
- {
- this.hideDialog();
- }), null, mxResources.get('save'), mxUtils.bind(this, function()
- {
- this.stopSanityCheck();
- this.actions.get((this.mode == null || !file.isEditable()) ?
- 'saveAs' : 'save').funct();
- }), null, null, 360, 120, null, mxUtils.bind(this, function()
- {
- this.scheduleSanityCheck();
- }));
- }
- };
- /**
- * Returns true if the current domain is for the new drive app.
- */
- App.prototype.isDriveDomain = function()
- {
- return urlParams['drive'] != '0' &&
- (window.location.hostname == 'test.draw.io' ||
- window.location.hostname == 'www.draw.io' ||
- window.location.hostname == 'drive.draw.io' ||
- window.location.hostname == 'app.diagrams.net' ||
- window.location.hostname == 'jgraph.github.io');
- };
- /**
- * Returns the pusher instance for notifications. Creates the instance of none exists.
- */
- App.prototype.getPusher = function()
- {
- if (this.pusher == null && typeof window.Pusher === 'function')
- {
- this.pusher = new Pusher(App.PUSHER_KEY,
- {
- cluster: App.PUSHER_CLUSTER,
- encrypted: true
- });
- }
-
- return this.pusher;
- };
- /**
- * Shows a footer to download the desktop version once per session.
- */
- App.prototype.showNameChangeBanner = function()
- {
- this.showBanner('DiagramsFooter', 'draw.io is now diagrams.net', mxUtils.bind(this, function()
- {
- this.openLink('https://www.diagrams.net/blog/move-diagrams-net');
- }));
- };
- /**
- * Shows a footer to download the desktop version once per session.
- */
- App.prototype.showDownloadDesktopBanner = function()
- {
- var link = 'https://get.diagrams.net/';
-
- if (this.showBanner('DesktopFooter', mxResources.get('downloadDesktop'), mxUtils.bind(this, function()
- {
- this.openLink(link);
- })))
- {
- // Downloads installer for macOS and Windows
- mxUtils.get('https://api.github.com/repos/jgraph/drawio-desktop/releases/latest', mxUtils.bind(this, function(req)
- {
- try
- {
- var rel = JSON.parse(req.getText());
-
- if (rel != null)
- {
- if (rel.tag_name != null && rel.name != null && rel.html_url != null)
- {
- if (mxClient.IS_MAC)
- {
- link = 'https://github.com/jgraph/drawio-desktop/releases/download/' +
- rel.tag_name + '/draw.io-' + rel.name + '.dmg';
- }
- else if (mxClient.IS_WIN)
- {
- link = 'https://github.com/jgraph/drawio-desktop/releases/download/' +
- rel.tag_name + '/draw.io-' + rel.name + '-windows-installer.exe';
- }
- }
- }
- }
- catch (e)
- {
- // ignore
- }
- }));
- }
- };
- /**
- *
- */
- App.prototype.checkLicense = function()
- {
- var driveUser = this.drive.getUser();
- var email = ((urlParams['dev'] == '1') ? urlParams['lic'] : null) ||
- ((driveUser != null) ? driveUser.email : null);
-
- if (!this.isOffline() && !this.editor.chromeless && email != null)
- {
- // Anonymises the local part of the email address
- var at = email.lastIndexOf('@');
- var domain = email;
-
- if (at >= 0)
- {
- domain = email.substring(at + 1);
- email = Editor.crc32(email.substring(0, at)) + '@' + domain;
- }
- // Timestamp is workaround for cached response in certain environments
- mxUtils.post('/license', 'domain=' + encodeURIComponent(domain) + '&email=' + encodeURIComponent(email) +
- '&lc=' + encodeURIComponent(driveUser.locale) + '&ts=' + new Date().getTime(),
- mxUtils.bind(this, function(req)
- {
- try
- {
- if (req.getStatus() >= 200 && req.getStatus() <= 299)
- {
- var value = req.getText();
-
- if (value.length > 0)
- {
- var lic = JSON.parse(value);
-
- if (lic != null)
- {
- this.handleLicense(lic, domain);
- }
- }
- }
- }
- catch (e)
- {
- // ignore
- }
- }));
- }
- };
- /**
- * Returns true if the current domain is for the new drive app.
- */
- App.prototype.handleLicense = function(lic, domain)
- {
- if (lic != null && lic.plugins != null)
- {
- App.loadPlugins(lic.plugins.split(';'), true);
- }
- };
- /**
- *
- */
- App.prototype.getEditBlankXml = function()
- {
- var file = this.getCurrentFile();
-
- if (file != null && this.editor.isChromelessView() && this.editor.graph.isLightboxView())
- {
- return file.getData();
- }
- else
- {
- return this.getFileData(true);
- }
- };
- /**
- * Updates action states depending on the selection.
- */
- App.prototype.updateActionStates = function()
- {
- EditorUi.prototype.updateActionStates.apply(this, arguments);
- this.actions.get('revisionHistory').setEnabled(this.isRevisionHistoryEnabled());
- };
- /**
- * Adds the specified entry to the recent file list in local storage
- */
- App.prototype.addRecent = function(entry)
- {
- if (isLocalStorage && localStorage != null)
- {
- var recent = this.getRecent();
-
- if (recent == null)
- {
- recent = [];
- }
- else
- {
- for (var i = 0; i < recent.length; i++)
- {
- if (recent[i].id == entry.id)
- {
- recent.splice(i, 1);
- }
- }
- }
-
- if (recent != null)
- {
- recent.unshift(entry);
- recent = recent.slice(0, 10);
- localStorage.setItem('.recent', JSON.stringify(recent));
- }
- }
- };
- /**
- * Returns the recent file list from local storage
- */
- App.prototype.getRecent = function()
- {
- if (isLocalStorage && localStorage != null)
- {
- try
- {
- var recent = localStorage.getItem('.recent');
-
- if (recent != null)
- {
- return JSON.parse(recent);
- }
- }
- catch (e)
- {
- // ignore
- }
-
- return null;
- }
- };
- /**
- * Clears the recent file list in local storage
- */
- App.prototype.resetRecent = function(entry)
- {
- if (isLocalStorage && localStorage != null)
- {
- try
- {
- localStorage.removeItem('.recent');
- }
- catch (e)
- {
- // ignore
- }
- }
- };
- /**
- * Sets the onbeforeunload for the application
- */
- App.prototype.onBeforeUnload = function()
- {
- if (urlParams['embed'] == '1' && this.editor.modified)
- {
- return mxResources.get('allChangesLost');
- }
- else
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- // KNOWN: Message is ignored by most browsers
- if (file.constructor == LocalFile && file.getHash() == '' && !file.isModified() &&
- urlParams['nowarn'] != '1' && !this.isDiagramEmpty() && urlParams['url'] == null &&
- !this.editor.isChromelessView() && file.fileHandle == null)
- {
- return mxResources.get('ensureDataSaved');
- }
- else if (file.isModified())
- {
- return mxResources.get('allChangesLost');
- }
- else
- {
- file.close(true);
- }
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.updateDocumentTitle = function()
- {
- if (!this.editor.graph.isLightboxView())
- {
- var title = this.editor.appName;
- var file = this.getCurrentFile();
-
- if (this.isOfflineApp())
- {
- title += ' app';
- }
-
- if (file != null)
- {
- var filename = (file.getTitle() != null) ? file.getTitle() : this.defaultFilename;
- title = filename + ' - ' + title;
- }
-
- if (document.title != title)
- {
- document.title = title;
- var graph = this.editor.graph;
- graph.invalidateDescendantsWithPlaceholders(graph.model.getRoot());
- graph.view.validate();
- }
- }
- };
- /**
- * Returns a thumbnail of the current file.
- */
- App.prototype.getThumbnail = function(width, fn)
- {
- var result = false;
-
- try
- {
- var acceptResponse = true;
-
- var timeoutThread = window.setTimeout(mxUtils.bind(this, function()
- {
- acceptResponse = false;
- fn(null);
- }), this.timeout);
-
- var success = mxUtils.bind(this, function(canvas)
- {
- window.clearTimeout(timeoutThread);
-
- if (acceptResponse)
- {
- fn(canvas);
- }
- });
-
- if (this.thumbImageCache == null)
- {
- this.thumbImageCache = new Object();
- }
-
- var graph = this.editor.graph;
-
- // Exports PNG for first page while other page is visible by creating a graph
- // LATER: Add caching for the graph or SVG while not on first page
- // To avoid refresh during save dark theme uses separate graph instance
- var darkTheme = graph.themes != null && graph.defaultThemeName == 'darkTheme';
-
- if (darkTheme || (this.pages != null && this.currentPage != this.pages[0]))
- {
- var graphGetGlobalVariable = graph.getGlobalVariable;
- graph = this.createTemporaryGraph((darkTheme) ? graph.getDefaultStylesheet() : graph.getStylesheet());
- var page = this.pages[0];
-
- // Avoids override of stylesheet in getSvg for dark mode
- if (darkTheme)
- {
- graph.defaultThemeName = 'default';
- }
-
- graph.getGlobalVariable = function(name)
- {
- if (name == 'page')
- {
- return page.getName();
- }
- else if (name == 'pagenumber')
- {
- return 1;
- }
-
- return graphGetGlobalVariable.apply(this, arguments);
- };
-
- graph.getGlobalVariable = graphGetGlobalVariable;
- document.body.appendChild(graph.container);
- graph.model.setRoot(page.root);
- }
-
- // Uses client-side canvas export
- if (mxClient.IS_CHROMEAPP || this.useCanvasForExport)
- {
- this.editor.exportToCanvas(mxUtils.bind(this, function(canvas)
- {
- try
- {
- // Removes temporary graph from DOM
- if (graph != this.editor.graph && graph.container.parentNode != null)
- {
- graph.container.parentNode.removeChild(graph.container);
- }
- }
- catch (e)
- {
- canvas = null;
- }
-
- success(canvas);
- }), width, this.thumbImageCache, '#ffffff', function()
- {
- // Continues with null in error case
- success();
- }, null, null, null, null, null, null, graph);
-
- result = true;
- }
- else if (this.canvasSupported && this.getCurrentFile() != null)
- {
- var canvas = document.createElement('canvas');
- var bounds = graph.getGraphBounds();
- var scale = width / bounds.width;
-
- // Limits scale to 1 or 2 * width / height
- scale = Math.min(1, Math.min((width * 3) / (bounds.height * 4), scale));
-
- var x0 = Math.floor(bounds.x);
- var y0 = Math.floor(bounds.y);
-
- canvas.setAttribute('width', Math.ceil(scale * (bounds.width + 4)));
- canvas.setAttribute('height', Math.ceil(scale * (bounds.height + 4)));
-
- var ctx = canvas.getContext('2d');
-
- // Configures the canvas
- ctx.scale(scale, scale);
- ctx.translate(-x0, -y0);
-
- // Paint white background instead of transparent
- var bg = graph.background;
-
- if (bg == null || bg == '' || bg == mxConstants.NONE)
- {
- bg = '#ffffff';
- }
-
- // Paints background
- ctx.save();
- ctx.fillStyle = bg;
- ctx.fillRect(x0, y0, Math.ceil(bounds.width + 4), Math.ceil(bounds.height + 4));
- ctx.restore();
-
- var htmlCanvas = new mxJsCanvas(canvas);
-
- // NOTE: htmlCanvas passed into async canvas is only used for image
- // and canvas caching (canvas caching not used in this case as we do
- // not render text). To reuse that cache via the thumbImageCache we
- // pass that into the async canvas and override the image cache in
- // the newly created html canvas with that of the thumbImageCache.
- // LATER: Is clear thumbImageCache needed if file changes?
- var asynCanvas = new mxAsyncCanvas(this.thumbImageCache);
- htmlCanvas.images = this.thumbImageCache.images;
-
- // Render graph
- var imgExport = new mxImageExport();
-
- imgExport.drawShape = function(state, canvas)
- {
- if (state.shape instanceof mxShape && state.shape.checkBounds())
- {
- canvas.save();
- canvas.translate(0.5, 0.5);
- state.shape.paint(canvas);
- canvas.translate(-0.5, -0.5);
- canvas.restore();
- }
- };
-
- imgExport.drawText = function(state, canvas)
- {
- // No text output for thumbnails
- };
-
- imgExport.drawState(graph.getView().getState(graph.model.root), asynCanvas);
-
- asynCanvas.finish(mxUtils.bind(this, function()
- {
- try
- {
- imgExport.drawState(graph.getView().getState(graph.model.root), htmlCanvas);
-
- // Removes temporary graph from DOM
- if (graph != this.editor.graph && graph.container.parentNode != null)
- {
- graph.container.parentNode.removeChild(graph.container);
- }
- }
- catch (e)
- {
- canvas = null;
- }
- success(canvas);
- }));
-
- result = true;
- }
- }
- catch (e)
- {
- result = false;
-
- // Removes temporary graph from DOM
- if (graph != null && graph != this.editor.graph && graph.container.parentNode != null)
- {
- graph.container.parentNode.removeChild(graph.container);
- }
- }
-
- return result;
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.createBackground = function()
- {
- var bg = this.createDiv('background');
- bg.style.position = 'absolute';
- bg.style.background = 'white';
- bg.style.left = '0px';
- bg.style.top = '0px';
- bg.style.bottom = '0px';
- bg.style.right = '0px';
-
- mxUtils.setOpacity(bg, 100);
-
- if (mxClient.IS_QUIRKS)
- {
- new mxDivResizer(bg);
- }
- return bg;
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- (function()
- {
- var editorUiSetMode = EditorUi.prototype.setMode;
-
- App.prototype.setMode = function(mode, remember)
- {
- editorUiSetMode.apply(this, arguments);
-
- // Note: UseLocalStorage affects the file dialogs
- // and should not be modified if mode is undefined
- if (this.mode != null)
- {
- Editor.useLocalStorage = this.mode == App.MODE_BROWSER;
- }
- if (this.appIcon != null)
- {
- var file = this.getCurrentFile();
- mode = (file != null) ? file.getMode() : mode;
-
- if (mode == App.MODE_GOOGLE)
- {
- this.appIcon.setAttribute('title', mxResources.get('openIt', [mxResources.get('googleDrive')]));
- this.appIcon.style.cursor = 'pointer';
- }
- else if (mode == App.MODE_DROPBOX)
- {
- this.appIcon.setAttribute('title', mxResources.get('openIt', [mxResources.get('dropbox')]));
- this.appIcon.style.cursor = 'pointer';
- }
- else if (mode == App.MODE_ONEDRIVE)
- {
- this.appIcon.setAttribute('title', mxResources.get('openIt', [mxResources.get('oneDrive')]));
- this.appIcon.style.cursor = 'pointer';
- }
- else
- {
- this.appIcon.removeAttribute('title');
- this.appIcon.style.cursor = (mode == App.MODE_DEVICE) ? 'pointer' : 'default';
- }
- }
-
- if (remember)
- {
- try
- {
- if (isLocalStorage)
- {
- localStorage.setItem('.mode', mode);
- }
- else if (typeof(Storage) != 'undefined')
- {
- var expiry = new Date();
- expiry.setYear(expiry.getFullYear() + 1);
- document.cookie = 'MODE=' + mode + '; expires=' + expiry.toUTCString();
- }
- }
- catch (e)
- {
- // ignore possible access denied
- }
- }
- };
- })();
- /**
- * Function: authorize
- *
- * Authorizes the client, gets the userId and calls <open>.
- */
- App.prototype.appIconClicked = function(evt)
- {
- if (mxEvent.isAltDown(evt))
- {
- this.showSplash(true);
- }
- else
- {
- var file = this.getCurrentFile();
- var mode = (file != null) ? file.getMode() : null;
-
- if (mode == App.MODE_GOOGLE)
- {
- if (file != null && file.desc != null && file.desc.parents != null &&
- file.desc.parents.length > 0 && !mxEvent.isShiftDown(evt))
- {
- // Opens containing folder
- this.openLink('https://drive.google.com/drive/folders/' + file.desc.parents[0].id);
- }
- else if (file != null && file.getId() != null)
- {
- this.openLink('https://drive.google.com/open?id=' + file.getId());
- }
- else
- {
- this.openLink('https://drive.google.com/?authuser=0');
- }
- }
- else if (mode == App.MODE_ONEDRIVE)
- {
- if (file != null && file.meta != null && file.meta.webUrl != null)
- {
- var url = file.meta.webUrl;
- var name = encodeURIComponent(file.meta.name);
-
- if (url.substring(url.length - name.length, url.length) == name)
- {
- url = url.substring(0, url.length - name.length);
- }
-
- this.openLink(url);
- }
- else
- {
- this.openLink('https://onedrive.live.com/');
- }
- }
- else if (mode == App.MODE_DROPBOX)
- {
- if (file != null && file.stat != null && file.stat.path_display != null)
- {
- var url = 'https://www.dropbox.com/home/Apps/drawio' + file.stat.path_display;
-
- if (!mxEvent.isShiftDown(evt))
- {
- url = url.substring(0, url.length - file.stat.name.length);
- }
-
- this.openLink(url);
- }
- else
- {
- this.openLink('https://www.dropbox.com/');
- }
- }
- else if (mode == App.MODE_TRELLO)
- {
- this.openLink('https://trello.com/');
- }
- else if (mode == App.MODE_GITHUB)
- {
- if (file != null && file.constructor == GitHubFile)
- {
- this.openLink(file.meta.html_url);
- }
- else
- {
- this.openLink('https://github.com/');
- }
- }
- else if (mode == App.MODE_GITLAB)
- {
- if (file != null && file.constructor == GitLabFile)
- {
- this.openLink(file.meta.html_url);
- }
- else
- {
- this.openLink(DRAWIO_GITLAB_URL);
- }
- }
- else if (mode == App.MODE_DEVICE)
- {
- this.openLink('https://get.draw.io/');
- }
- }
-
- mxEvent.consume(evt);
- };
- /**
- * Function: authorize
- *
- * Authorizes the client, gets the userId and calls <open>.
- */
- App.prototype.clearMode = function()
- {
- if (isLocalStorage)
- {
- localStorage.removeItem('.mode');
- }
- else if (typeof(Storage) != 'undefined')
- {
- var expiry = new Date();
- expiry.setYear(expiry.getFullYear() - 1);
- document.cookie = 'MODE=; expires=' + expiry.toUTCString();
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.getDiagramId = function()
- {
- var id = window.location.hash;
-
- // Strips the hash sign
- if (id != null && id.length > 0)
- {
- id = id.substring(1);
- }
-
- // Workaround for Trello client appending data after hash
- if (id != null && id.length > 1 && id.charAt(0) == 'T')
- {
- var idx = id.indexOf('#');
-
- if (idx > 0)
- {
- id = id.substring(0, idx);
- }
- }
-
- return id;
- };
- /**
- * Opens any file specified in the URL parameters.
- */
- App.prototype.open = function()
- {
- // Cross-domain window access is not allowed in FF, so if we
- // were opened from another domain then this will fail.
- try
- {
- // If the create URL param is used in embed mode then
- // we try to open the XML from window.opener[value].
- // Use this for embedding via tab to bypass the timing
- // issues when passing messages without onload event.
- if (window.opener != null)
- {
- var value = urlParams['create'];
-
- if (value != null)
- {
- value = decodeURIComponent(value);
- }
-
- if (value != null && value.length > 0 && value.substring(0, 7) != 'http://' &&
- value.substring(0, 8) != 'https://')
- {
- var doc = mxUtils.parseXml(window.opener[value]);
- this.editor.setGraphXml(doc.documentElement);
- }
- else if (window.opener.openFile != null)
- {
- window.opener.openFile.setConsumer(mxUtils.bind(this, function(xml, filename, temp)
- {
- this.spinner.stop();
-
- if (filename == null)
- {
- var title = urlParams['title'];
- temp = true;
-
- if (title != null)
- {
- filename = decodeURIComponent(title);
- }
- else
- {
- filename = this.defaultFilename;
- }
- }
-
- // Replaces PNG with XML extension
- var dot = (!this.useCanvasForExport) ? filename.substring(filename.length - 4) == '.png' : -1;
-
- if (dot > 0)
- {
- filename = filename.substring(0, filename.length - 4) + '.drawio';
- }
-
- this.fileLoaded((mxClient.IS_IOS) ?
- new StorageFile(this, xml, filename) :
- new LocalFile(this, xml, filename, temp));
- }));
- }
- }
- }
- catch(e)
- {
- // ignore
- }
- };
- App.prototype.loadGapi = function(then)
- {
- if (typeof gapi !== 'undefined')
- {
- gapi.load(((urlParams['picker'] != '0') ? 'picker,': '') + App.GOOGLE_APIS, then);
- }
- };
- /**
- * Main function. Program starts here.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.load = function()
- {
- // Checks if we're running in embedded mode
- if (urlParams['embed'] != '1')
- {
- if (this.spinner.spin(document.body, mxResources.get('starting')))
- {
- try
- {
- this.stateArg = (urlParams['state'] != null && this.drive != null) ? JSON.parse(decodeURIComponent(urlParams['state'])) : null;
- }
- catch (e)
- {
- // ignores invalid state args
- }
-
- this.editor.graph.setEnabled(this.getCurrentFile() != null);
-
- // Passes the userId from the state parameter to the client
- if ((window.location.hash == null || window.location.hash.length == 0) &&
- this.drive != null && this.stateArg != null && this.stateArg.userId != null)
- {
- this.drive.setUserId(this.stateArg.userId);
- }
- // Legacy support for fileId parameter which is moved to the hash tag
- if (urlParams['fileId'] != null)
- {
- window.location.hash = 'G' + urlParams['fileId'];
- window.location.search = this.getSearch(['fileId']);
- }
- else
- {
- // Asynchronous or disabled loading of client
- if (this.drive == null)
- {
- if (this.mode == App.MODE_GOOGLE)
- {
- this.mode = null;
- }
-
- this.start();
- }
- else
- {
- this.loadGapi(mxUtils.bind(this, function()
- {
- this.start();
- }));
- }
- }
- }
- }
- else
- {
- this.restoreLibraries();
-
- if (urlParams['gapi'] == '1')
- {
- this.loadGapi(function() {});
- }
- }
- };
- /**
- * Adds the listener for automatically saving the diagram for local changes.
- */
- App.prototype.showRefreshDialog = function(title, message)
- {
- if (!this.showingRefreshDialog)
- {
- this.showingRefreshDialog = true;
- this.showError(title || mxResources.get('externalChanges'),
- message || mxResources.get('redirectToNewApp'),
- mxResources.get('refresh'), mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- file.setModified(false);
- }
-
- this.spinner.spin(document.body, mxResources.get('connecting'));
- this.editor.graph.setEnabled(false);
- window.location.reload();
- }), null, null, null, null, null, 340, 180);
-
- // Adds important notice to dialog
- if (this.dialog != null && this.dialog.container != null)
- {
- var alert = this.createRealtimeNotice();
- alert.style.left = '0';
- alert.style.right = '0';
- alert.style.borderRadius = '0';
- alert.style.borderLeftStyle = 'none';
- alert.style.borderRightStyle = 'none';
- alert.style.marginBottom = '26px';
- alert.style.padding = '8px 0 8px 0';
- this.dialog.container.appendChild(alert);
- }
- }
- };
- /**
- * Called in start after the spinner stops.
- */
- App.prototype.showAlert = function(message)
- {
- if (message != null && message.length > 0)
- {
- var div = document.createElement('div');
- div.className = 'geAlert';
- div.style.zIndex = 2e9;
- div.style.left = '50%';
- div.style.top = '-100%';
- mxUtils.setPrefixedStyle(div.style, 'transform', 'translate(-50%,0%)');
- mxUtils.setPrefixedStyle(div.style, 'transition', 'all 1s ease');
-
- div.innerHTML = message;
-
- var close = document.createElement('a');
- close.className = 'geAlertLink';
- close.style.textAlign = 'right';
- close.style.marginTop = '20px';
- close.style.display = 'block';
- close.setAttribute('title', mxResources.get('close'));
- close.innerHTML = mxResources.get('close');
- div.appendChild(close);
-
- mxEvent.addListener(close, 'click', function(evt)
- {
- if (div.parentNode != null)
- {
- div.parentNode.removeChild(div);
- mxEvent.consume(evt);
- }
- });
-
- document.body.appendChild(div);
-
- // Delayed to get smoother animation after DOM rendering
- window.setTimeout(function()
- {
- div.style.top = '30px';
- }, 10);
-
- // Fades out the alert after 15 secs
- window.setTimeout(function()
- {
- mxUtils.setPrefixedStyle(div.style, 'transition', 'all 2s ease');
- div.style.opacity = '0';
-
- window.setTimeout(function()
- {
- if (div.parentNode != null)
- {
- div.parentNode.removeChild(div);
- }
- }, 2000);
- }, 15000);
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.start = function()
- {
- if (this.bg != null && this.bg.parentNode != null)
- {
- this.bg.parentNode.removeChild(this.bg);
- }
-
- this.restoreLibraries();
- this.spinner.stop();
- try
- {
- // Handles all errors
- var ui = this;
-
- window.onerror = function(message, url, linenumber, colno, err)
- {
- EditorUi.logError('Uncaught: ' + ((message != null) ? message : ''),
- url, linenumber, colno, err, null, true);
- ui.handleError({message: message}, mxResources.get('unknownError'),
- null, null, null, null, true);
- };
-
- // Listens to changes of the hash if not in embed or client mode
- if (urlParams['client'] != '1' && urlParams['embed'] != '1')
- {
- // Installs listener to claim current draft if there is one
- try
- {
- if (isLocalStorage)
- {
- window.addEventListener('storage', mxUtils.bind(this, function(evt)
- {
- var file = this.getCurrentFile();
- EditorUi.debug('storage event', evt, file);
-
- if (file != null && evt.key == '.draft-alive-check' && evt.newValue != null && file.draftId != null)
- {
- this.draftAliveCheck = evt.newValue;
- file.saveDraft();
- }
- }));
- }
- if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && !this.isOfflineApp() &&
- urlParams['open'] == null && /www\.draw\.io$/.test(window.location.hostname) &&
- (!this.editor.chromeless || this.editor.editable))
- {
- this.showNameChangeBanner();
- }
- }
- catch (e)
- {
- // ignore
- }
-
- // KNOWN: Does not work in quirks mode
- mxEvent.addListener(window, 'hashchange', mxUtils.bind(this, function(evt)
- {
- try
- {
- this.hideDialog();
- var id = this.getDiagramId();
- var file = this.getCurrentFile();
- if (file == null || file.getHash() != id)
- {
- this.loadFile(id, true);
- }
- }
- catch (e)
- {
- // Workaround for possible scrollWidth of null in Dialog ctor
- if (document.body != null)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- window.location.hash = (file != null) ? file.getHash() : '';
- }));
- }
- }
- }));
- }
-
- // Redirects old url URL parameter to new #U format
- if ((window.location.hash == null || window.location.hash.length <= 1) && urlParams['url'] != null)
- {
- this.loadFile('U' + urlParams['url'], true);
- }
- else if (this.getCurrentFile() == null)
- {
- var done = mxUtils.bind(this, function()
- {
- // Starts in client mode and waits for data
- if (urlParams['client'] == '1' && (window.location.hash == null ||
- window.location.hash.length == 0 || window.location.hash.substring(0, 2) == '#P'))
- {
- var doLoadFile = mxUtils.bind(this, function(xml)
- {
- // Extracts graph model from PNG
- if (xml.substring(0, 22) == 'data:image/png;base64,')
- {
- xml = this.extractGraphModelFromPng(xml);
- }
-
- var title = urlParams['title'];
-
- if (title != null)
- {
- title = decodeURIComponent(title);
- }
- else
- {
- title = this.defaultFilename;
- }
-
- var file = new LocalFile(this, xml, title, true);
-
- if (window.location.hash != null && window.location.hash.substring(0, 2) == '#P')
- {
- file.getHash = function()
- {
- return window.location.hash.substring(1);
- };
- }
-
- this.fileLoaded(file);
- this.getCurrentFile().setModified(!this.editor.chromeless);
- });
- var parent = window.opener || window.parent;
-
- if (parent != window)
- {
- var value = urlParams['create'];
-
- if (value != null)
- {
- doLoadFile(parent[decodeURIComponent(value)]);
- }
- else
- {
- value = urlParams['data'];
-
- if (value != null)
- {
- doLoadFile(decodeURIComponent(value));
- }
- else
- {
- this.installMessageHandler(mxUtils.bind(this, function(xml, evt)
- {
- // Ignores messages from other windows
- if (evt.source == parent)
- {
- doLoadFile(xml);
- }
- }));
- }
- }
- }
- }
- // Checks if no earlier loading errors are showing
- else if (this.dialog == null)
- {
- if (urlParams['demo'] == '1')
- {
- var prev = Editor.useLocalStorage;
- this.createFile(this.defaultFilename, null, null, null, null, null, null, true);
- Editor.useLocalStorage = prev;
- }
- else
- {
- var waiting = false;
-
- // Checks if we're waiting for some asynchronous file to be loaded
- // Cross-domain window access is not allowed in FF, so if we
- // were opened from another domain then this will fail.
- try
- {
- waiting = window.opener != null && window.opener.openFile != null;
- }
- catch(e)
- {
- // ignore
- }
-
- if (waiting)
- {
- // Spinner is stopped in App.open
- this.spinner.spin(document.body, mxResources.get('loading'))
- }
- else
- {
- var id = this.getDiagramId();
-
-
- if (EditorUi.enableDrafts && (urlParams['mode'] == null || EditorUi.isElectronApp) &&
- this.getServiceName() == 'draw.io' && (id == null || id.length == 0) &&
- !this.editor.isChromelessView())
- {
- this.checkDrafts();
- }
- else if (id != null && id.length > 0)
- {
- this.loadFile(id, null, null, mxUtils.bind(this, function()
- {
- var temp = decodeURIComponent(urlParams['viewbox'] || '');
-
- if (temp != '')
- {
- try
- {
- var bounds = JSON.parse(temp);
- this.editor.graph.fitWindow(bounds, bounds.border);
- }
- catch (e)
- {
- // Ignore invalid viewport
- console.error(e);
- }
- }
- }));
- }
- else if (urlParams['splash'] != '0')
- {
- this.loadFile();
- }
- else
- {
- this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true);
- }
- }
- }
- }
- });
-
- var value = decodeURIComponent(urlParams['create'] || '');
-
- if ((window.location.hash == null || window.location.hash.length <= 1) &&
- value != null && value.length > 0 && this.spinner.spin(document.body, mxResources.get('loading')))
- {
- var reconnect = mxUtils.bind(this, function()
- {
- // Removes URL parameter and reloads the page
- if (this.spinner.spin(document.body, mxResources.get('reconnecting')))
- {
- window.location.search = this.getSearch(['create', 'title']);
- };
- });
-
- var showCreateDialog = mxUtils.bind(this, function(xml)
- {
- this.spinner.stop();
-
- // Resets mode for dialog - local file is only for preview
- if (urlParams['splash'] != '0')
- {
- this.fileLoaded(new LocalFile(this, xml, null));
-
- this.editor.graph.setEnabled(false);
- this.mode = urlParams['mode'];
- var title = urlParams['title'];
-
- if (title != null)
- {
- title = decodeURIComponent(title);
- }
- else
- {
- title = this.defaultFilename;
- }
-
- var serviceCount = this.getServiceCount(true);
-
- if (isLocalStorage)
- {
- serviceCount++;
- }
-
- var rowLimit = (serviceCount <= 4) ? 2 : (serviceCount > 6 ? 4 : 3);
-
- var dlg = new CreateDialog(this, title, mxUtils.bind(this, function(filename, mode)
- {
- if (mode == null)
- {
- this.hideDialog();
- var prev = Editor.useLocalStorage;
- this.createFile((filename.length > 0) ? filename : this.defaultFilename,
- this.getFileData(), null, null, null, true, null, true);
- Editor.useLocalStorage = prev;
- }
- else
- {
- this.pickFolder(mode, mxUtils.bind(this, function(folderId)
- {
- this.createFile(filename, this.getFileData(true),
- null, mode, null, true, folderId);
- }));
- }
- }), null, null, null, null, urlParams['browser'] == '1',
- null, null, true, rowLimit, null, null, null,
- this.editor.fileExtensions);
- this.showDialog(dlg.container, 400, (serviceCount > rowLimit) ? 390 : 270,
- true, false, mxUtils.bind(this, function(cancel)
- {
- if (cancel && this.getCurrentFile() == null)
- {
- this.showSplash();
- }
- }));
- dlg.init();
- }
- });
-
- value = decodeURIComponent(value);
-
- if (value.substring(0, 7) != 'http://' && value.substring(0, 8) != 'https://')
- {
- // Cross-domain window access is not allowed in FF, so if we
- // were opened from another domain then this will fail.
- try
- {
- if (window.opener != null && window.opener[value] != null)
- {
- showCreateDialog(window.opener[value]);
- }
- else
- {
- this.handleError(null, mxResources.get('errorLoadingFile'));
- }
- }
- catch (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }
- }
- else
- {
- this.loadTemplate(value, function(text)
- {
- showCreateDialog(text);
- }, mxUtils.bind(this, function()
- {
- this.handleError(null, mxResources.get('errorLoadingFile'), reconnect);
- }));
- }
- }
- else
- {
- // Passes the fileId from the state parameter to the hash tag and reloads
- // the page without the state parameter
- if ((window.location.hash == null || window.location.hash.length <= 1) &&
- urlParams['state'] != null && this.stateArg != null && this.stateArg.action == 'open')
- {
- if (this.stateArg.ids != null)
- {
- if (window.history && window.history.replaceState)
- {
- // Removes state URL parameter without reloading the page
- window.history.replaceState(null, null, window.location.pathname +
- this.getSearch(['state']));
- }
-
- window.location.hash = 'G' + this.stateArg.ids[0];
- }
- }
- else if ((window.location.hash == null || window.location.hash.length <= 1) &&
- this.drive != null && this.stateArg != null && this.stateArg.action == 'create')
- {
- if (window.history && window.history.replaceState)
- {
- // Removes state URL parameter without reloading the page
- window.history.replaceState(null, null, window.location.pathname +
- this.getSearch(['state']));
- }
-
- this.setMode(App.MODE_GOOGLE);
- this.actions.get('new').funct();
- }
- else
- {
- // Removes open URL parameter. Hash is also updated in Init to load client.
- if (urlParams['open'] != null && window.history && window.history.replaceState)
- {
- window.history.replaceState(null, null, window.location.pathname +
- this.getSearch(['open']));
- window.location.hash = urlParams['open'];
- }
-
- done();
- }
- }
- }
- }
- catch (e)
- {
- this.handleError(e);
- }
- };
- /**
- * Checks for orphaned drafts.
- */
- App.prototype.loadDraft = function(xml, success)
- {
- this.createFile(this.defaultFilename, xml, null, null, mxUtils.bind(this, function()
- {
- window.setTimeout(mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- file.fileChanged();
-
- if (success != null)
- {
- success();
- }
- }
- }), 0);
- }), null, null, true);
- };
- /**
- * Checks for orphaned drafts.
- */
- App.prototype.checkDrafts = function()
- {
- try
- {
- // Triggers storage event for other windows to mark active drafts
- var guid = Editor.guid();
- localStorage.setItem('.draft-alive-check', guid);
-
- window.setTimeout(mxUtils.bind(this, function()
- {
- localStorage.removeItem('.draft-alive-check');
- this.getDatabaseItems(mxUtils.bind(this, function(items)
- {
- // Collects orphaned drafts
- var drafts = [];
-
- for (var i = 0; i < items.length; i++)
- {
- try
- {
- var key = items[i].key;
-
- if (key != null && key.substring(0, 7) == '.draft_')
- {
- var obj = JSON.parse(items[i].data);
-
- if (obj != null && obj.type == 'draft' && obj.aliveCheck != guid)
- {
- obj.key = key;
- drafts.push(obj);
- }
- }
- }
- catch (e)
- {
- // ignore
- }
- }
-
- if (drafts.length == 1)
- {
- this.loadDraft(drafts[0].data, mxUtils.bind(this, function()
- {
- this.removeDatabaseItem(drafts[0].key);
- }));
- }
- else if (drafts.length > 1)
- {
- var ts = new Date(drafts[0].modified);
-
- var dlg = new DraftDialog(this, (drafts.length > 1) ? mxResources.get('selectDraft') :
- mxResources.get('draftFound', [ts.toLocaleDateString() + ' ' + ts.toLocaleTimeString()]),
- (drafts.length > 1) ? null : drafts[0].data, mxUtils.bind(this, function(index)
- {
- this.hideDialog();
- index = (index != '') ? index : 0;
-
- this.loadDraft(drafts[index].data, mxUtils.bind(this, function()
- {
- this.removeDatabaseItem(drafts[index].key);
- }));
- }), mxUtils.bind(this, function(index, success)
- {
- index = (index != '') ? index : 0;
-
- // Discard draft
- this.confirm(mxResources.get('areYouSure'), null, mxUtils.bind(this, function()
- {
- this.removeDatabaseItem(drafts[index].key);
-
- if (success != null)
- {
- success();
- }
- }), mxResources.get('no'), mxResources.get('yes'));
- }), null, null, null, (drafts.length > 1) ? drafts : null);
- this.showDialog(dlg.container, 640, 480, true, false, mxUtils.bind(this, function(cancel)
- {
- if (urlParams['splash'] != '0')
- {
- this.loadFile();
- }
- else
- {
- this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true);
- }
- }));
- dlg.init();
- }
- else if (urlParams['splash'] != '0')
- {
- this.loadFile();
- }
- else
- {
- this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true);
- }
- }), mxUtils.bind(this, function()
- {
- if (urlParams['splash'] != '0')
- {
- this.loadFile();
- }
- else
- {
- this.createFile(this.defaultFilename, this.getFileData(), null, null, null, null, null, true);
- }
- }));
- }), 0);
- }
- catch (e)
- {
- // ignore
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.showSplash = function(force)
- {
- var serviceCount = this.getServiceCount(true);
-
- var showSecondDialog = mxUtils.bind(this, function()
- {
- var dlg = new SplashDialog(this);
-
- this.showDialog(dlg.container, 340, (mxClient.IS_CHROMEAPP || EditorUi.isElectronApp) ? 200 : 230, true, true,
- mxUtils.bind(this, function(cancel)
- {
- // Creates a blank diagram if the dialog is closed
- if (cancel && !mxClient.IS_CHROMEAPP)
- {
- var prev = Editor.useLocalStorage;
- this.createFile(this.defaultFilename + (EditorUi.isElectronApp? '.drawio' : ''), null, null, null, null, null, null,
- urlParams['local'] != '1');
- Editor.useLocalStorage = prev;
- }
- }), true);
-
- if (!mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp && !this.isOfflineApp() &&
- !mxClient.IS_ANDROID && !mxClient.IS_IOS &&
- this.mode == App.MODE_DEVICE)
- {
- this.showDownloadDesktopBanner();
- }
- });
-
- if (this.editor.isChromelessView())
- {
- this.handleError({message: mxResources.get('noFileSelected')},
- mxResources.get('errorLoadingFile'), mxUtils.bind(this, function()
- {
- this.showSplash();
- }));
- }
- else if (!mxClient.IS_CHROMEAPP && (this.mode == null || force))
- {
- var rowLimit = (serviceCount == 4) ? 2 : 3;
-
- var dlg = new StorageDialog(this, mxUtils.bind(this, function()
- {
- this.hideDialog();
- showSecondDialog();
- }), rowLimit);
-
- this.showDialog(dlg.container, (rowLimit < 3) ? 200 : 300,
- ((serviceCount > 3) ? 320 : 210), true, false);
- }
- else if (urlParams['create'] == null)
- {
- showSecondDialog();
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.addLanguageMenu = function(elt, addLabel)
- {
- var img = null;
- var langMenu = this.menus.get('language');
-
- if (langMenu != null)
- {
- img = document.createElement('div');
- img.setAttribute('title', mxResources.get('language'));
- img.className = 'geIcon geSprite geSprite-globe';
- img.style.position = 'absolute';
- img.style.cursor = 'pointer';
- img.style.bottom = '20px';
- img.style.right = '20px';
-
- if (addLabel)
- {
- img.style.direction = 'rtl';
- img.style.textAlign = 'right';
- img.style.right = '24px';
- var label = document.createElement('span');
- label.style.display = 'inline-block';
- label.style.fontSize = '12px';
- label.style.margin = '5px 24px 0 0';
- label.style.color = 'gray';
- label.style.userSelect = 'none';
-
- mxUtils.write(label, mxResources.get('language'));
- img.appendChild(label);
- }
-
- mxEvent.addListener(img, 'click', mxUtils.bind(this, function(evt)
- {
- this.editor.graph.popupMenuHandler.hideMenu();
- var menu = new mxPopupMenu(this.menus.get('language').funct);
- menu.div.className += ' geMenubarMenu';
- menu.smartSeparators = true;
- menu.showDisabled = true;
- menu.autoExpand = true;
-
- // Disables autoexpand and destroys menu when hidden
- menu.hideMenu = mxUtils.bind(this, function()
- {
- mxPopupMenu.prototype.hideMenu.apply(menu, arguments);
- menu.destroy();
- });
-
- var offset = mxUtils.getOffset(img);
- menu.popup(offset.x, offset.y + img.offsetHeight, null, evt);
-
- // Allows hiding by clicking on document
- this.setCurrentMenu(menu);
- }));
-
- elt.appendChild(img);
- }
-
- return img;
- };
- /**
- * Loads the given file handle as a local file.
- */
- App.prototype.loadFileSystemEntry = function(fileHandle, success, error)
- {
- error = (error != null) ? error : mxUtils.bind(this, function(e)
- {
- this.handleError(e);
- });
-
- fileHandle.getFile().then(mxUtils.bind(this, function(file)
- {
- var reader = new FileReader();
-
- reader.onload = mxUtils.bind(this, function(e)
- {
- try
- {
- if (success != null)
- {
- var data = e.target.result;
-
- if (file.type.substring(0, 6) == 'image/')
- {
- data = this.extractGraphModelFromPng(data);
- }
- success(new LocalFile(this, data, file.name, null, fileHandle, file));
- }
- else
- {
- this.openFileHandle(e.target.result, file.name, file, false, fileHandle);
- }
- }
- catch(e)
- {
- error(e);
- }
- });
-
- reader.onerror = error;
-
- if ((file.type.substring(0, 5) === 'image' ||
- file.type === 'application/pdf') &&
- file.type.substring(0, 9) !== 'image/svg')
- {
- reader.readAsDataURL(file);
- }
- else
- {
- reader.readAsText(file);
- }
- }), error);
- };
- /**
- * Loads the given file handle as a local file.
- */
- App.prototype.createFileSystemOptions = function(name)
- {
- var ext = [];
- var temp = null;
-
- if (name != null)
- {
- var idx = name.lastIndexOf('.');
-
- if (idx > 0)
- {
- temp = name.substring(idx + 1);
- }
- }
-
- for (var i = 0; i < this.editor.diagramFileTypes.length; i++)
- {
- var obj = {description: mxResources.get(this.editor.diagramFileTypes[i].description) +
- ((mxClient.IS_MAC) ? ' (.' + this.editor.diagramFileTypes[i].extension + ')' : ''),
- extensions: [this.editor.diagramFileTypes[i].extension]};
-
- if (this.editor.diagramFileTypes[i].extension == temp)
- {
- ext.splice(0, 0, obj);
- }
- else
- {
- ext.push(obj);
- }
- }
-
- // TODO: Specify default filename
- return {type: 'save-file', accepts: ext, fileName: name};
- };
- /**
- * Loads the given file handle as a local file.
- */
- App.prototype.chooseFileSystemEntries = function(success, error, opts)
- {
- error = (error != null) ? error : mxUtils.bind(this, function(e)
- {
- if (e.name != 'AbortError')
- {
- this.handleError(e);
- }
- });
-
- opts = (opts != null) ? opts : this.createFileSystemOptions();
-
- // LATER: Specify default name via options
- window.chooseFileSystemEntries(opts).then(mxUtils.bind(this, function(fileHandle)
- {
- fileHandle.getFile().then(mxUtils.bind(this, function(desc)
- {
- success(fileHandle, desc);
- }), error);
- }), error);
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.pickFile = function(mode)
- {
- try
- {
- mode = (mode != null) ? mode : this.mode;
-
- if (mode == App.MODE_GOOGLE)
- {
- if (this.drive != null && typeof(google) != 'undefined' && typeof(google.picker) != 'undefined')
- {
- this.drive.pickFile();
- }
- else
- {
- this.openLink('https://drive.google.com');
- }
- }
- else
- {
- var peer = this.getPeerForMode(mode);
-
- if (peer != null)
- {
- peer.pickFile();
- }
- else if (mode == App.MODE_DEVICE && 'chooseFileSystemEntries' in window)
- {
- window.chooseFileSystemEntries().then(mxUtils.bind(this, function(fileHandle)
- {
- if (this.spinner.spin(document.body, mxResources.get('loading')))
- {
- this.loadFileSystemEntry(fileHandle);
- }
- }), mxUtils.bind(this, function(e)
- {
- if (e.name != 'AbortError')
- {
- this.handleError(e);
- }
- }));
- }
- else if (mode == App.MODE_DEVICE && Graph.fileSupport)
- {
- if (this.openFileInputElt == null)
- {
- var input = document.createElement('input');
- input.setAttribute('type', 'file');
-
- mxEvent.addListener(input, 'change', mxUtils.bind(this, function()
- {
- if (input.files != null)
- {
- this.openFiles(input.files);
-
- // Resets input to force change event for
- // same file (type reset required for IE)
- input.type = '';
- input.type = 'file';
- input.value = '';
- }
- }));
-
- input.style.display = 'none';
- document.body.appendChild(input);
- this.openFileInputElt = input;
- }
-
- this.openFileInputElt.click();
- }
- else
- {
- this.hideDialog();
- window.openNew = this.getCurrentFile() != null && !this.isDiagramEmpty();
- window.baseUrl = this.getUrl();
- window.openKey = 'open';
-
- window.listBrowserFiles = mxUtils.bind(this, function(success, error)
- {
- StorageFile.listFiles(this, 'F', success, error);
- });
-
- window.openBrowserFile = mxUtils.bind(this, function(title, success, error)
- {
- StorageFile.getFileContent(this, title, success, error);
- });
-
- window.deleteBrowserFile = mxUtils.bind(this, function(title, success, error)
- {
- StorageFile.deleteFile(this, title, success, error);
- });
-
- var prevValue = Editor.useLocalStorage;
- Editor.useLocalStorage = (mode == App.MODE_BROWSER);
- this.openFile();
-
- // Installs local handler for opened files in same window
- window.openFile.setConsumer(mxUtils.bind(this, function(xml, filename)
- {
- // Replaces PNG with XML extension
- var dot = !this.useCanvasForExport && filename.substring(filename.length - 4) == '.png';
-
- if (dot)
- {
- filename = filename.substring(0, filename.length - 4) + '.drawio';
- }
-
- this.fileLoaded((mode == App.MODE_BROWSER) ?
- new StorageFile(this, xml, filename) :
- new LocalFile(this, xml, filename));
- }));
-
- // Extends dialog close to show splash screen
- var dlg = this.dialog;
- var dlgClose = dlg.close;
-
- this.dialog.close = mxUtils.bind(this, function(cancel)
- {
- Editor.useLocalStorage = prevValue;
- dlgClose.apply(dlg, arguments);
-
- if (this.getCurrentFile() == null)
- {
- this.showSplash();
- }
- });
- }
- }
- }
- catch (e)
- {
- this.handleError(e);
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.pickLibrary = function(mode)
- {
- mode = (mode != null) ? mode : this.mode;
-
- if (mode == App.MODE_GOOGLE || mode == App.MODE_DROPBOX || mode == App.MODE_ONEDRIVE ||
- mode == App.MODE_GITHUB || mode == App.MODE_GITLAB || mode == App.MODE_TRELLO)
- {
- var peer = (mode == App.MODE_GOOGLE) ? this.drive :
- ((mode == App.MODE_ONEDRIVE) ? this.oneDrive :
- ((mode == App.MODE_GITHUB) ? this.gitHub :
- ((mode == App.MODE_GITLAB) ? this.gitLab :
- ((mode == App.MODE_TRELLO) ? this.trello :
- this.dropbox))));
-
- if (peer != null)
- {
- peer.pickLibrary(mxUtils.bind(this, function(id, optionalFile)
- {
- if (optionalFile != null)
- {
- try
- {
- this.loadLibrary(optionalFile);
- }
- catch (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }
- }
- else
- {
- if (this.spinner.spin(document.body, mxResources.get('loading')))
- {
- peer.getLibrary(id, mxUtils.bind(this, function(file)
- {
- this.spinner.stop();
-
- try
- {
- this.loadLibrary(file);
- }
- catch (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }
- }), mxUtils.bind(this, function(resp)
- {
- this.handleError(resp, (resp != null) ? mxResources.get('errorLoadingFile') : null);
- }));
- }
- }
- }));
- }
- }
- else if (mode == App.MODE_DEVICE && Graph.fileSupport)
- {
- if (this.libFileInputElt == null)
- {
- var input = document.createElement('input');
- input.setAttribute('type', 'file');
-
- mxEvent.addListener(input, 'change', mxUtils.bind(this, function()
- {
- if (input.files != null)
- {
- for (var i = 0; i < input.files.length; i++)
- {
- (mxUtils.bind(this, function(file)
- {
- var reader = new FileReader();
-
- reader.onload = mxUtils.bind(this, function(e)
- {
- try
- {
- this.loadLibrary(new LocalLibrary(this, e.target.result, file.name));
- }
- catch (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }
- });
-
- reader.readAsText(file);
- }))(input.files[i]);
- }
-
- // Resets input to force change event for same file (type reset required for IE)
- input.type = '';
- input.type = 'file';
- input.value = '';
- }
- }));
-
- input.style.display = 'none';
- document.body.appendChild(input);
- this.libFileInputElt = input;
- }
-
- this.libFileInputElt.click();
- }
- else
- {
- window.openNew = false;
- window.openKey = 'open';
-
- window.listBrowserFiles = mxUtils.bind(this, function(success, error)
- {
- StorageFile.listFiles(this, 'L', success, error);
- });
-
- window.openBrowserFile = mxUtils.bind(this, function(title, success, error)
- {
- StorageFile.getFileContent(this, title, success, error);
- });
-
- window.deleteBrowserFile = mxUtils.bind(this, function(title, success, error)
- {
- StorageFile.deleteFile(this, title, success, error);
- });
-
- var prevValue = Editor.useLocalStorage;
- Editor.useLocalStorage = mode == App.MODE_BROWSER;
-
- // Closes dialog after open
- window.openFile = new OpenFile(mxUtils.bind(this, function(cancel)
- {
- this.hideDialog(cancel);
- }));
-
- window.openFile.setConsumer(mxUtils.bind(this, function(xml, filename)
- {
- try
- {
- this.loadLibrary((mode == App.MODE_BROWSER) ? new StorageLibrary(this, xml, filename) :
- new LocalLibrary(this, xml, filename));
- }
- catch (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }
- }));
- // Removes openFile if dialog is closed
- this.showDialog(new OpenDialog(this).container, (Editor.useLocalStorage) ? 640 : 360,
- (Editor.useLocalStorage) ? 480 : 220, true, true, function()
- {
- Editor.useLocalStorage = prevValue;
- window.openFile = null;
- });
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.saveLibrary = function(name, images, file, mode, noSpin, noReload, fn)
- {
- try
- {
- mode = (mode != null) ? mode : this.mode;
- noSpin = (noSpin != null) ? noSpin : false;
- noReload = (noReload != null) ? noReload : false;
- var xml = this.createLibraryDataFromImages(images);
-
- var error = mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
-
- if (fn != null)
- {
- fn();
- }
-
- this.handleError(resp, (resp != null) ? mxResources.get('errorSavingFile') : null);
- });
-
- // Handles special case for local libraries
- if (file == null && mode == App.MODE_DEVICE)
- {
- file = new LocalLibrary(this, xml, name);
- }
-
- if (file == null)
- {
- this.pickFolder(mode, mxUtils.bind(this, function(folderId)
- {
- if (mode == App.MODE_GOOGLE && this.drive != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- this.drive.insertFile(name, xml, folderId, mxUtils.bind(this, function(newFile)
- {
- this.spinner.stop();
- this.hideDialog(true);
- this.libraryLoaded(newFile, images);
- }), error, this.drive.libraryMimeType);
- }
- else if (mode == App.MODE_GITHUB && this.gitHub != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- this.gitHub.insertLibrary(name, xml, mxUtils.bind(this, function(newFile)
- {
- this.spinner.stop();
- this.hideDialog(true);
- this.libraryLoaded(newFile, images);
- }), error, folderId);
- }
- else if (mode == App.MODE_GITLAB && this.gitLab != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- this.gitLab.insertLibrary(name, xml, mxUtils.bind(this, function(newFile)
- {
- this.spinner.stop();
- this.hideDialog(true);
- this.libraryLoaded(newFile, images);
- }), error, folderId);
- }
- else if (mode == App.MODE_TRELLO && this.trello != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- this.trello.insertLibrary(name, xml, mxUtils.bind(this, function(newFile)
- {
- this.spinner.stop();
- this.hideDialog(true);
- this.libraryLoaded(newFile, images);
- }), error, folderId);
- }
- else if (mode == App.MODE_DROPBOX && this.dropbox != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- this.dropbox.insertLibrary(name, xml, mxUtils.bind(this, function(newFile)
- {
- this.spinner.stop();
- this.hideDialog(true);
- this.libraryLoaded(newFile, images);
- }), error, folderId);
- }
- else if (mode == App.MODE_ONEDRIVE && this.oneDrive != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- this.oneDrive.insertLibrary(name, xml, mxUtils.bind(this, function(newFile)
- {
- this.spinner.stop();
- this.hideDialog(true);
- this.libraryLoaded(newFile, images);
- }), error, folderId);
- }
- else if (mode == App.MODE_BROWSER)
- {
- var fn = mxUtils.bind(this, function()
- {
- var file = new StorageLibrary(this, xml, name);
-
- // Inserts data into local storage
- file.saveFile(name, false, mxUtils.bind(this, function()
- {
- this.hideDialog(true);
- this.libraryLoaded(file, images);
- }), error);
- });
-
- if (localStorage.getItem(name) == null)
- {
- fn();
- }
- else
- {
- this.confirm(mxResources.get('replaceIt', [name]), fn);
- }
- }
- else
- {
- this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')});
- }
- }));
- }
- else if (noSpin || this.spinner.spin(document.body, mxResources.get('saving')))
- {
- file.setData(xml);
-
- var doSave = mxUtils.bind(this, function()
- {
- file.save(true, mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
- this.hideDialog(true);
-
- if (!noReload)
- {
- this.libraryLoaded(file, images);
- }
-
- if (fn != null)
- {
- fn();
- }
- }), error);
- });
-
- if (name != file.getTitle())
- {
- var oldHash = file.getHash();
-
- file.rename(name, mxUtils.bind(this, function(resp)
- {
- // Change hash in stored settings
- if (file.constructor != LocalLibrary && oldHash != file.getHash())
- {
- mxSettings.removeCustomLibrary(oldHash);
- mxSettings.addCustomLibrary(file.getHash());
- }
-
- // Workaround for library files changing hash so
- // the old library cannot be removed from the
- // sidebar using the updated file in libraryLoaded
- this.removeLibrarySidebar(oldHash);
-
- doSave();
- }), error)
- }
- else
- {
- doSave();
- }
- }
- }
- catch (e)
- {
- this.handleError(e);
- }
- };
- /**
- * Adds the label menu items to the given menu and parent.
- */
- App.prototype.saveFile = function(forceDialog, success)
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- // FIXME: Invoke for local files
- var done = mxUtils.bind(this, function()
- {
- if (EditorUi.enableDrafts)
- {
- file.removeDraft();
- }
-
- if (this.getCurrentFile() != file && !file.isModified())
- {
- // Workaround for possible status update while save as dialog is showing
- // is to show no saved status for device files
- if (file.getMode() != App.MODE_DEVICE)
- {
- this.editor.setStatus(mxUtils.htmlEntities(mxResources.get('allChangesSaved')));
- }
- else
- {
- this.editor.setStatus('');
- }
- }
-
- if (success != null)
- {
- success();
- }
- });
-
- if (!forceDialog && file.getTitle() != null && this.mode != null)
- {
- this.save(file.getTitle(), done);
- }
- else if (file != null && file.constructor == LocalFile && file.fileHandle != null)
- {
- this.chooseFileSystemEntries(mxUtils.bind(this, function(fileHandle, desc)
- {
- file.fileHandle = fileHandle;
- file.title = desc.name;
- file.desc = desc;
- this.save(desc.name, done);
- }), null, this.createFileSystemOptions(file.getTitle()));
- }
- else
- {
- var filename = (file.getTitle() != null) ? file.getTitle() : this.defaultFilename;
- var allowTab = !mxClient.IS_IOS || !navigator.standalone;
- var prev = this.mode;
- var serviceCount = this.getServiceCount(true);
-
- if (isLocalStorage)
- {
- serviceCount++;
- }
-
- var rowLimit = (serviceCount <= 4) ? 2 : (serviceCount > 6 ? 4 : 3);
-
- var dlg = new CreateDialog(this, filename, mxUtils.bind(this, function(name, mode, input)
- {
- if (name != null && name.length > 0)
- {
- // Handles special case where PDF export is detected
- if (/(\.pdf)$/i.test(name))
- {
- this.confirm(mxResources.get('didYouMeanToExportToPdf'), mxUtils.bind(this, function()
- {
- this.hideDialog();
- this.actions.get('exportPdf').funct();
- }), mxUtils.bind(this, function()
- {
- input.value = name.split('.').slice(0, -1).join('.');
- input.focus();
-
- if (mxClient.IS_GC || mxClient.IS_FF || document.documentMode >= 5 || mxClient.IS_QUIRKS)
- {
- input.select();
- }
- else
- {
- document.execCommand('selectAll', false, null);
- }
- }), mxResources.get('yes'), mxResources.get('no'));
- }
- else
- {
- this.hideDialog();
-
- if (prev == null && mode == App.MODE_DEVICE)
- {
- if (file != null && 'chooseFileSystemEntries' in window)
- {
- this.chooseFileSystemEntries(mxUtils.bind(this, function(fileHandle, desc)
- {
- file.fileHandle = fileHandle;
- file.mode = App.MODE_DEVICE;
- file.title = desc.name;
- file.desc = desc;
- this.setMode(App.MODE_DEVICE);
- this.save(desc.name, done);
- }), mxUtils.bind(this, function(e)
- {
- if (e.name != 'AbortError')
- {
- this.handleError(e);
- }
- }), this.createFileSystemOptions(name));
- }
- else
- {
- this.setMode(App.MODE_DEVICE);
- this.save(name, done);
- }
- }
- else if (mode == 'download')
- {
- var tmp = new LocalFile(this, null, name);
- tmp.save();
- }
- else if (mode == '_blank')
- {
- window.openFile = new OpenFile(function()
- {
- window.openFile = null;
- });
-
- // Do not use a filename to use undefined mode
- window.openFile.setData(this.getFileData(true));
- this.openLink(this.getUrl(window.location.pathname), null, true);
- }
- else if (prev != mode)
- {
- this.pickFolder(mode, mxUtils.bind(this, function(folderId)
- {
- this.createFile(name, this.getFileData(/(\.xml)$/i.test(name) ||
- name.indexOf('.') < 0 || /(\.drawio)$/i.test(name),
- /(\.svg)$/i.test(name), /(\.html)$/i.test(name)),
- null, mode, done, this.mode == null, folderId);
- }));
- }
- else if (mode != null)
- {
- this.save(name, done);
- }
- }
- }
- }), mxUtils.bind(this, function()
- {
- this.hideDialog();
- }), mxResources.get('saveAs'), mxResources.get('download'), null, null, allowTab,
- null, true, rowLimit, null, null, null, this.editor.fileExtensions, false);
- this.showDialog(dlg.container, 400, (serviceCount > rowLimit) ? 390 : 270, true, true);
- dlg.init();
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.loadTemplate = function(url, onload, onerror, templateFilename, asLibrary)
- {
- var base64 = false;
- var realUrl = url;
-
- if (!this.editor.isCorsEnabledForUrl(realUrl))
- {
- // Always uses base64 response to check magic numbers for file type
- var nocache = 't=' + new Date().getTime();
- realUrl = PROXY_URL + '?url=' + encodeURIComponent(url) + '&base64=1&' + nocache;
- base64 = true;
- }
- var filterFn = (templateFilename != null) ? templateFilename : url;
-
- this.editor.loadUrl(realUrl, mxUtils.bind(this, function(responseData)
- {
- try
- {
- var data = (!base64) ? responseData : ((window.atob && !mxClient.IS_IE && !mxClient.IS_IE11) ?
- atob(responseData) : Base64.decode(responseData));
- var isVisioFilename = /(\.v(dx|sdx?))($|\?)/i.test(filterFn) ||
- /(\.vs(x|sx?))($|\?)/i.test(filterFn);
-
- if (isVisioFilename || this.isVisioData(data))
- {
- // Adds filename to control converter code
- if (!isVisioFilename)
- {
- if (asLibrary)
- {
- filterFn = this.isRemoteVisioData(data) ? 'raw.vss' : 'raw.vssx';
- }
- else
- {
- filterFn = this.isRemoteVisioData(data) ? 'raw.vsd' : 'raw.vsdx';
- }
- }
-
- this.importVisio(this.base64ToBlob(responseData.substring(responseData.indexOf(',') + 1)), function(xml)
- {
- onload(xml);
- }, onerror, filterFn);
- }
- else if (!this.isOffline() && new XMLHttpRequest().upload && this.isRemoteFileFormat(data, filterFn))
- {
- // Asynchronous parsing via server
- this.parseFile(new Blob([data], {type: 'application/octet-stream'}), mxUtils.bind(this, function(xhr)
- {
- if (xhr.readyState == 4 && xhr.status >= 200 && xhr.status <= 299 &&
- xhr.responseText.substring(0, 13) == '<mxGraphModel')
- {
- onload(xhr.responseText);
- }
- }), url);
- }
- else if (this.isLucidChartData(data))
- {
- this.convertLucidChart(data, mxUtils.bind(this, function(xml)
- {
- onload(xml);
- }), mxUtils.bind(this, function(e)
- {
- onerror(e);
- }));
- }
- else
- {
- if (/(\.png)($|\?)/i.test(filterFn) || this.isPngData(data))
- {
- data = this.extractGraphModelFromPng(responseData);
- }
-
- onload(data);
- }
- }
- catch (e)
- {
- onerror(e);
- }
- }), onerror, /(\.png)($|\?)/i.test(filterFn) || /(\.v(dx|sdx?))($|\?)/i.test(filterFn) ||
- /(\.vs(x|sx?))($|\?)/i.test(filterFn), null, null, base64);
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.getPeerForMode = function(mode)
- {
- if (mode == App.MODE_GOOGLE)
- {
- return this.drive;
- }
- else if (mode == App.MODE_GITHUB)
- {
- return this.gitHub;
- }
- else if (mode == App.MODE_GITLAB)
- {
- return this.gitLab;
- }
- else if (mode == App.MODE_DROPBOX)
- {
- return this.dropbox;
- }
- else if (mode == App.MODE_ONEDRIVE)
- {
- return this.oneDrive;
- }
- else if (mode == App.MODE_TRELLO)
- {
- return this.trello;
- }
- else
- {
- return null;
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.createFile = function(title, data, libs, mode, done, replace, folderId, tempFile, clibs)
- {
- mode = (tempFile) ? null : ((mode != null) ? mode : this.mode);
- if (title != null && this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- data = (data != null) ? data : this.emptyDiagramXml;
-
- var complete = mxUtils.bind(this, function()
- {
- this.spinner.stop();
- });
-
- var error = mxUtils.bind(this, function(resp)
- {
- complete();
-
- if (resp == null && this.getCurrentFile() == null && this.dialog == null)
- {
- this.showSplash();
- }
- else if (resp != null)
- {
- this.handleError(resp);
- }
- });
-
- try
- {
- if (mode == App.MODE_GOOGLE && this.drive != null)
- {
- if (folderId == null && this.stateArg != null && this.stateArg.folderId != null)
- {
- folderId = this.stateArg.folderId;
- }
-
- this.drive.insertFile(title, data, folderId, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error);
- }
- else if (mode == App.MODE_GITHUB && this.gitHub != null)
- {
- this.gitHub.insertFile(title, data, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error, false, folderId);
- }
- else if (mode == App.MODE_GITLAB && this.gitLab != null)
- {
- this.gitLab.insertFile(title, data, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error, false, folderId);
- }
- else if (mode == App.MODE_TRELLO && this.trello != null)
- {
- this.trello.insertFile(title, data, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error, false, folderId);
- }
- else if (mode == App.MODE_DROPBOX && this.dropbox != null)
- {
- this.dropbox.insertFile(title, data, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error);
- }
- else if (mode == App.MODE_ONEDRIVE && this.oneDrive != null)
- {
- this.oneDrive.insertFile(title, data, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error, false, folderId);
- }
- else if (mode == App.MODE_BROWSER)
- {
- StorageFile.insertFile(this, title, data, mxUtils.bind(this, function(file)
- {
- complete();
- this.fileCreated(file, libs, replace, done, clibs);
- }), error);
- }
- else if (!tempFile && mode == App.MODE_DEVICE && 'chooseFileSystemEntries' in window)
- {
- complete();
-
- this.chooseFileSystemEntries(mxUtils.bind(this, function(fileHandle, desc)
- {
- var file = new LocalFile(this, data, desc.name, null, fileHandle, desc);
-
- file.saveFile(desc.name, false, mxUtils.bind(this, function()
- {
- this.fileCreated(file, libs, replace, done, clibs);
- }), error, true);
- }), mxUtils.bind(this, function(e)
- {
- if (e.name != 'AbortError')
- {
- error(e);
- }
- }), this.createFileSystemOptions(title));
- }
- else
- {
- complete();
- this.fileCreated(new LocalFile(this, data, title, mode == null), libs, replace, done, clibs);
- }
- }
- catch (e)
- {
- complete();
- this.handleError(e);
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.fileCreated = function(file, libs, replace, done, clibs)
- {
- var url = window.location.pathname;
-
- if (libs != null && libs.length > 0)
- {
- url += '?libs=' + libs;
- }
- if (clibs != null && clibs.length > 0)
- {
- url += '?clibs=' + clibs;
- }
-
- url = this.getUrl(url);
- // Always opens a new tab for local files to avoid losing changes
- if (file.getMode() != App.MODE_DEVICE)
- {
- url += '#' + file.getHash();
- }
- // Makes sure to produce consistent output with finalized files via createFileData this needs
- // to save the file again since it needs the newly created file ID for redirecting in HTML
- if (this.spinner.spin(document.body, mxResources.get('inserting')))
- {
- var data = file.getData();
- var dataNode = (data.length > 0) ? this.editor.extractGraphModel(
- mxUtils.parseXml(data).documentElement, true) : null;
- var redirect = window.location.protocol + '//' + window.location.hostname + url;
- var node = dataNode;
- var graph = null;
-
- // Handles special case where SVG files need a rendered graph to be saved
- if (dataNode != null && /\.svg$/i.test(file.getTitle()))
- {
- graph = this.createTemporaryGraph(this.editor.graph.getStylesheet());
- document.body.appendChild(graph.container);
- node = this.decodeNodeIntoGraph(node, graph);
- }
-
- file.setData(this.createFileData(dataNode, graph, file, redirect));
- if (graph != null)
- {
- graph.container.parentNode.removeChild(graph.container);
- }
- var complete = mxUtils.bind(this, function()
- {
- this.spinner.stop();
- });
-
- var fn = mxUtils.bind(this, function()
- {
- complete();
-
- var currentFile = this.getCurrentFile();
-
- if (replace == null && currentFile != null)
- {
- replace = !currentFile.isModified() && currentFile.getMode() == null;
- }
-
- var fn3 = mxUtils.bind(this, function()
- {
- window.openFile = null;
- this.fileLoaded(file);
-
- if (replace)
- {
- file.addAllSavedStatus();
- }
-
- if (libs != null)
- {
- this.sidebar.showEntries(libs);
- }
-
- if (clibs != null)
- {
- var temp = [];
- var tokens = clibs.split(';');
-
- for (var i = 0; i < tokens.length; i++)
- {
- temp.push(decodeURIComponent(tokens[i]));
- }
-
- this.loadLibraries(temp);
- }
- });
- var fn2 = mxUtils.bind(this, function()
- {
- if (replace || currentFile == null || !currentFile.isModified())
- {
- fn3();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), null, fn3,
- mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- });
- if (done != null)
- {
- done();
- }
-
- // Opens the file in a new window
- if (replace != null && !replace)
- {
- // Opens local file in a new window
- if (file.constructor == LocalFile)
- {
- window.openFile = new OpenFile(function()
- {
- window.openFile = null;
- });
-
- window.openFile.setData(file.getData(), file.getTitle(), file.getMode() == null);
- }
- if (done != null)
- {
- done();
- }
-
- window.openWindow(url, null, fn2);
- }
- else
- {
- fn2();
- }
- });
-
- // Updates data in memory for local files
- if (file.constructor == LocalFile)
- {
- fn();
- }
- else
- {
- file.saveFile(file.getTitle(), false, mxUtils.bind(this, function()
- {
- fn();
- }), mxUtils.bind(this, function(resp)
- {
- complete();
- this.handleError(resp);
- }));
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.loadFile = function(id, sameWindow, file, success, force)
- {
- this.hideDialog();
-
- var fn2 = mxUtils.bind(this, function()
- {
- if (id == null || id.length == 0)
- {
- this.editor.setStatus('');
- this.fileLoaded(null);
- }
- else if (this.spinner.spin(document.body, mxResources.get('loading')))
- {
- // Handles files from localStorage
- if (id.charAt(0) == 'L')
- {
- this.spinner.stop();
- if (!isLocalStorage)
- {
- this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function()
- {
- var tempFile = this.getCurrentFile();
- window.location.hash = (tempFile != null) ? tempFile.getHash() : '';
- }));
- }
- else
- {
- var error = mxUtils.bind(this, function (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function()
- {
- var tempFile = this.getCurrentFile();
- window.location.hash = (tempFile != null) ? tempFile.getHash() : '';
- }));
- });
-
- id = decodeURIComponent(id.substring(1));
-
- StorageFile.getFileContent(this, id, mxUtils.bind(this, function(data)
- {
- if (data != null)
- {
- this.fileLoaded(new StorageFile(this, data, id));
- if (success != null)
- {
- success();
- }
- }
- else
- {
- error({message: mxResources.get('fileNotFound')});
- }
- }), error);
- }
- }
- else if (file != null)
- {
- // File already loaded
- this.spinner.stop();
- this.fileLoaded(file);
- if (success != null)
- {
- success();
- }
- }
- else if (id.charAt(0) == 'S')
- {
- this.spinner.stop();
-
- try
- {
- this.loadDescriptor(JSON.parse(Graph.decompress(id.substring(1))),
- success, mxUtils.bind(this, function(e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }));
- }
- catch (e)
- {
- this.handleError(e, mxResources.get('errorLoadingFile'));
- }
- }
- else if (id.charAt(0) == 'R')
- {
- // Raw file encoded into URL
- this.spinner.stop();
- var data = decodeURIComponent(id.substring(1));
-
- if (data.charAt(0) != '<')
- {
- data = Graph.decompress(data);
- }
-
- var tempFile = new LocalFile(this, data, (urlParams['title'] != null) ?
- decodeURIComponent(urlParams['title']) : this.defaultFilename, true);
- tempFile.getHash = function()
- {
- return id;
- };
- this.fileLoaded(tempFile);
- if (success != null)
- {
- success();
- }
- }
- else if (id.charAt(0) == 'E') // Embed file
- {
- //Currently we only reload current file. Id is not used!
- var currentFile = this.getCurrentFile();
-
- if (currentFile == null)
- {
- this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile'));
- }
- else
- {
- this.remoteInvoke('getDraftFileContent', null, null, mxUtils.bind(this, function(data, desc)
- {
- this.spinner.stop();
- this.fileLoaded(new EmbedFile(this, data, desc));
-
- if (success != null)
- {
- success();
- }
- }), mxUtils.bind(this, function()
- {
- this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile'));
- }));
- }
- }
- else if (id.charAt(0) == 'U')
- {
- var url = decodeURIComponent(id.substring(1));
-
- var doFallback = mxUtils.bind(this, function()
- {
- // Fallback for non-public Google Drive files
- if (url.substring(0, 31) == 'https://drive.google.com/uc?id=' &&
- (this.drive != null || typeof window.DriveClient === 'function'))
- {
- this.hideDialog();
-
- var fallback = mxUtils.bind(this, function()
- {
- this.spinner.stop();
-
- if (this.drive != null)
- {
- var tempId = url.substring(31, url.lastIndexOf('&ex'));
-
- this.loadFile('G' + tempId, sameWindow, null, mxUtils.bind(this, function()
- {
- var currentFile = this.getCurrentFile();
-
- if (currentFile != null && this.editor.chromeless && !this.editor.editable)
- {
- currentFile.getHash = function()
- {
- return 'G' + tempId;
- };
-
- window.location.hash = '#' + currentFile.getHash();
- }
-
- if (success != null)
- {
- success();
- }
- }));
-
- return true;
- }
- else
- {
- return false;
- }
- });
-
- if (!fallback() && this.spinner.spin(document.body, mxResources.get('loading')))
- {
- this.addListener('clientLoaded', fallback);
- }
-
- return true;
- }
- else
- {
- return false;
- }
- });
-
- this.loadTemplate(url, mxUtils.bind(this, function(text)
- {
- this.spinner.stop();
-
- if (text != null && text.length > 0)
- {
- var filename = this.defaultFilename;
-
- // Tries to find name from URL with valid extensions
- if (urlParams['title'] == null && urlParams['notitle'] != '1')
- {
- var tmp = url;
- var dot = url.lastIndexOf('.');
- var slash = tmp.lastIndexOf('/');
-
- if (dot > slash && slash > 0)
- {
- tmp = tmp.substring(slash + 1, dot);
- var ext = url.substring(dot);
-
- if (!this.useCanvasForExport && ext == '.png')
- {
- ext = '.drawio';
- }
- if (ext === '.svg' || ext === '.xml' ||
- ext === '.html' || ext === '.png' ||
- ext === '.drawio')
- {
- filename = tmp + ext;
- }
- }
- }
-
- var tempFile = new LocalFile(this, text, (urlParams['title'] != null) ?
- decodeURIComponent(urlParams['title']) : filename, true);
- tempFile.getHash = function()
- {
- return id;
- };
-
- if (!this.fileLoaded(tempFile, true) && !doFallback())
- {
- this.handleError({message: mxResources.get('fileNotFound')},
- mxResources.get('errorLoadingFile'));
- }
- }
- else if (!doFallback())
- {
- this.handleError({message: mxResources.get('fileNotFound')},
- mxResources.get('errorLoadingFile'));
- }
- }), mxUtils.bind(this, function()
- {
- if (!doFallback())
- {
- this.spinner.stop();
- this.handleError({message: mxResources.get('fileNotFound')},
- mxResources.get('errorLoadingFile'));
- }
- }), (urlParams['template-filename'] != null) ?
- decodeURIComponent(urlParams['template-filename']) : null);
- }
- else
- {
- // Google Drive files are handled as default file types
- var peer = null;
-
- if (id.charAt(0) == 'G')
- {
- peer = this.drive;
- }
- else if (id.charAt(0) == 'D')
- {
- peer = this.dropbox;
- }
- else if (id.charAt(0) == 'W')
- {
- peer = this.oneDrive;
- }
- else if (id.charAt(0) == 'H')
- {
- peer = this.gitHub;
- }
- else if (id.charAt(0) == 'A')
- {
- peer = this.gitLab;
- }
- else if (id.charAt(0) == 'T')
- {
- peer = this.trello;
- }
-
- if (peer == null)
- {
- this.handleError({message: mxResources.get('serviceUnavailableOrBlocked')}, mxResources.get('errorLoadingFile'), mxUtils.bind(this, function()
- {
- var currentFile = this.getCurrentFile();
- window.location.hash = (currentFile != null) ? currentFile.getHash() : '';
- }));
- }
- else
- {
- var peerChar = id.charAt(0);
- id = decodeURIComponent(id.substring(1));
- peer.getFile(id, mxUtils.bind(this, function(file)
- {
- this.spinner.stop();
- this.fileLoaded(file);
- var currentFile = this.getCurrentFile();
-
- if (currentFile == null)
- {
- window.location.hash = '';
- this.showSplash();
- }
- else if (this.editor.chromeless && !this.editor.editable)
- {
- // Keeps ID even for converted files in chromeless mode for refresh to work
- currentFile.getHash = function()
- {
- return peerChar + id;
- };
-
- window.location.hash = '#' + currentFile.getHash();
- }
- else if (file == currentFile && file.getMode() == null)
- {
- // Shows a warning if a copy was opened which happens
- // eg. for .png files in IE as they cannot be written
- var status = mxResources.get('copyCreated');
- this.editor.setStatus('<div title="'+ status + '" class="geStatusAlert" style="overflow:hidden;">' + status + '</div>');
- }
-
- if (success != null)
- {
- success();
- }
- }), mxUtils.bind(this, function(resp)
- {
- // Makes sure the file does not save the invalid UI model and overwrites anything important
- if (window.console != null && resp != null)
- {
- console.log('error in loadFile:', id, resp);
- }
-
- this.handleError(resp, (resp != null) ? mxResources.get('errorLoadingFile') : null, mxUtils.bind(this, function()
- {
- var currentFile = this.getCurrentFile();
-
- if (currentFile == null)
- {
- window.location.hash = '';
- this.showSplash();
- }
- else
- {
- window.location.hash = '#' + currentFile.getHash();
- }
- }), null, null, '#' + peerChar + id);
- }));
- }
- }
- }
- });
-
- var currentFile = this.getCurrentFile();
-
- var fn = mxUtils.bind(this, function()
- {
- if (force || currentFile == null || !currentFile.isModified())
- {
- fn2();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), mxUtils.bind(this, function()
- {
- if (currentFile != null)
- {
- window.location.hash = currentFile.getHash();
- }
- }), fn2, mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- });
-
- if (id == null || id.length == 0)
- {
- fn();
- }
- else if (currentFile != null && !sameWindow)
- {
- this.showDialog(new PopupDialog(this, this.getUrl() + '#' + id,
- null, fn).container, 320, 140, true, true);
- }
- else
- {
- fn();
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.getLibraryStorageHint = function(file)
- {
- var tip = file.getTitle();
-
- if (file.constructor != LocalLibrary)
- {
- tip += '\n' + file.getHash();
- }
-
- if (file.constructor == DriveLibrary)
- {
- tip += ' (' + mxResources.get('googleDrive') + ')';
- }
- else if (file.constructor == GitHubLibrary)
- {
- tip += ' (' + mxResources.get('github') + ')';
- }
- else if (file.constructor == TrelloLibrary)
- {
- tip += ' (' + mxResources.get('trello') + ')';
- }
- else if (file.constructor == DropboxLibrary)
- {
- tip += ' (' + mxResources.get('dropbox') + ')';
- }
- else if (file.constructor == OneDriveLibrary)
- {
- tip += ' (' + mxResources.get('oneDrive') + ')';
- }
- else if (file.constructor == StorageLibrary)
- {
- tip += ' (' + mxResources.get('browser') + ')';
- }
- else if (file.constructor == LocalLibrary)
- {
- tip += ' (' + mxResources.get('device') + ')';
- }
- return tip;
- };
- /**
- * Updates action states depending on the selection.
- */
- App.prototype.restoreLibraries = function()
- {
- this.loadLibraries(mxSettings.getCustomLibraries(), mxUtils.bind(this, function()
- {
- this.loadLibraries((urlParams['clibs'] || '').split(';'));
- }));
- };
- /**
- * Updates action states depending on the selection.
- */
- App.prototype.loadLibraries = function(libs, done)
- {
- if (this.sidebar != null)
- {
- if (this.pendingLibraries == null)
- {
- this.pendingLibraries = new Object();
- }
-
- // Ignores this library next time
- var ignore = mxUtils.bind(this, function(id, keep)
- {
- if (!keep)
- {
- mxSettings.removeCustomLibrary(id);
- }
-
- delete this.pendingLibraries[id];
- });
-
- var waiting = 0;
- var files = [];
- // Loads in order of libs array
- var checkDone = mxUtils.bind(this, function()
- {
- if (waiting == 0)
- {
- if (libs != null)
- {
- for (var i = libs.length - 1; i >= 0; i--)
- {
- if (files[i] != null)
- {
- this.loadLibrary(files[i]);
- }
- }
- }
-
- if (done != null)
- {
- done();
- }
- }
- });
-
- if (libs != null)
- {
- for (var i = 0; i < libs.length; i++)
- {
- var name = encodeURIComponent(decodeURIComponent(libs[i]));
-
- (mxUtils.bind(this, function(id, index)
- {
- if (id != null && id.length > 0 && this.pendingLibraries[id] == null &&
- this.sidebar.palettes[id] == null)
- {
- // Waits for all libraries to load
- waiting++;
-
- var onload = mxUtils.bind(this, function(file)
- {
- delete this.pendingLibraries[id];
- files[index] = file;
- waiting--;
- checkDone();
- });
-
- var onerror = mxUtils.bind(this, function(keep)
- {
- ignore(id, keep);
- waiting--;
- checkDone();
- });
-
- this.pendingLibraries[id] = true;
- var service = id.substring(0, 1);
-
- if (service == 'L')
- {
- if (isLocalStorage || mxClient.IS_CHROMEAPP)
- {
- // Make asynchronous for barrier to work
- window.setTimeout(mxUtils.bind(this, function()
- {
- try
- {
- var name = decodeURIComponent(id.substring(1));
-
- StorageFile.getFileContent(this, name, mxUtils.bind(this, function(xml)
- {
- if (name == '.scratchpad' && xml == null)
- {
- xml = this.emptyLibraryXml;
- }
-
- if (xml != null)
- {
- onload(new StorageLibrary(this, xml, name));
- }
- else
- {
- onerror();
- }
- }), onerror);
- }
- catch (e)
- {
- onerror();
- }
- }), 0);
- }
- }
- else if (service == 'U')
- {
- var url = decodeURIComponent(id.substring(1));
-
- if (!this.isOffline())
- {
- this.loadTemplate(url, mxUtils.bind(this, function(text)
- {
- if (text != null && text.length > 0)
- {
- // LATER: Convert mxfile to mxlibrary using code from libraryLoaded
- onload(new UrlLibrary(this, text, url));
- }
- else
- {
- onerror();
- }
- }), function()
- {
- onerror();
- }, null, true);
- }
- }
- else if (service == 'R')
- {
- var libDesc = decodeURIComponent(id.substring(1));
-
- if (!this.isOffline())
- {
- try
- {
- libDesc = JSON.parse(libDesc);
- var libObj = {
- id: libDesc[0],
- title: libDesc[1],
- downloadUrl: libDesc[2]
- }
-
- this.remoteInvoke('getFileContent', [libObj.downloadUrl], null, mxUtils.bind(this, function(libContent)
- {
- try
- {
- onload(new RemoteLibrary(this, libContent, libObj));
- }
- catch (e)
- {
- onerror();
- }
- }), function()
- {
- onerror();
- });
- }
- catch (e)
- {
- onerror();
- }
- }
- }
- else if (service == 'S' && this.loadDesktopLib != null)
- {
- try
- {
- this.loadDesktopLib(decodeURIComponent(id.substring(1)), function(desktopLib)
- {
- onload(desktopLib);
- }, onerror);
- }
- catch (e)
- {
- onerror();
- }
- }
- else
- {
- var peer = null;
-
- if (service == 'G')
- {
- if (this.drive != null && this.drive.user != null)
- {
- peer = this.drive;
- }
- }
- else if (service == 'H')
- {
- if (this.gitHub != null && this.gitHub.getUser() != null)
- {
- peer = this.gitHub;
- }
- }
- else if (service == 'T')
- {
- if (this.trello != null && this.trello.isAuthorized())
- {
- peer = this.trello;
- }
- }
- else if (service == 'D')
- {
- if (this.dropbox != null && this.dropbox.getUser() != null)
- {
- peer = this.dropbox;
- }
- }
- else if (service == 'W')
- {
- if (this.oneDrive != null && this.oneDrive.getUser() != null)
- {
- peer = this.oneDrive;
- }
- }
-
- if (peer != null)
- {
- peer.getLibrary(decodeURIComponent(id.substring(1)), mxUtils.bind(this, function(file)
- {
- try
- {
- onload(file);
- }
- catch (e)
- {
- onerror();
- }
- }), function(resp)
- {
- onerror();
- });
- }
- else
- {
- onerror(true);
- }
- }
- }
- }))(name, i);
- }
-
- checkDone();
- }
- else
- {
- checkDone();
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.updateButtonContainer = function()
- {
- if (this.buttonContainer != null)
- {
- var file = this.getCurrentFile();
-
- // Comments
- if (this.commentsSupported())
- {
- if (this.commentButton == null)
- {
- this.commentButton = document.createElement('a');
- this.commentButton.setAttribute('title', mxResources.get('comments'));
- this.commentButton.className = 'geToolbarButton';
- this.commentButton.style.cssText = 'display:inline-block;position:relative;box-sizing:border-box;' +
- 'margin-right:4px;float:left;cursor:pointer;width:24px;height:24px;background-size:24px 24px;' +
- 'background-position:center center;background-repeat:no-repeat;background-image:' +
- 'url(' + Editor.commentImage + ');';
-
- if (uiTheme == 'atlas')
- {
- this.commentButton.style.marginRight = '10px';
- this.commentButton.style.marginTop = '-3px';
- }
- else if (uiTheme == 'min')
- {
- this.commentButton.style.marginTop = '1px';
- }
- else
- {
- this.commentButton.style.marginTop = '-5px';
- }
-
- mxEvent.addListener(this.commentButton, 'click', mxUtils.bind(this, function()
- {
- this.actions.get('comments').funct();
- }));
-
- this.buttonContainer.appendChild(this.commentButton);
-
- if (uiTheme == 'dark' || uiTheme == 'atlas')
- {
- this.commentButton.style.filter = 'invert(100%)';
- }
- }
- }
- else if (this.commentButton != null)
- {
- this.commentButton.parentNode.removeChild(this.commentButton);
- this.commentButton = null;
- }
-
- // Share
- if (urlParams['embed'] != '1' && this.getServiceName() == 'draw.io' &&
- !mxClient.IS_CHROMEAPP && !EditorUi.isElectronApp &&
- !this.isOfflineApp() && file != null)
- {
- if (this.shareButton == null)
- {
- this.shareButton = document.createElement('div');
- this.shareButton.className = 'geBtn gePrimaryBtn';
- this.shareButton.style.display = 'inline-block';
- this.shareButton.style.backgroundColor = '#F2931E';
- this.shareButton.style.borderColor = '#F08705';
- this.shareButton.style.backgroundImage = 'none';
- this.shareButton.style.padding = '2px 10px 0 10px';
- this.shareButton.style.marginTop = '-10px';
- this.shareButton.style.height = '28px';
- this.shareButton.style.lineHeight = '28px';
- this.shareButton.style.minWidth = '0px';
- this.shareButton.style.cssFloat = 'right';
- this.shareButton.setAttribute('title', mxResources.get('share'));
-
- var icon = document.createElement('img');
- icon.setAttribute('src', this.shareImage);
- icon.setAttribute('align', 'absmiddle');
- icon.style.marginRight = '4px';
- icon.style.marginTop = '-3px';
- this.shareButton.appendChild(icon);
-
- if (uiTheme != 'dark' && uiTheme != 'atlas')
- {
- this.shareButton.style.color = 'black';
- icon.style.filter = 'invert(100%)';
- }
-
- mxUtils.write(this.shareButton, mxResources.get('share'));
-
- mxEvent.addListener(this.shareButton, 'click', mxUtils.bind(this, function()
- {
- this.actions.get('share').funct();
- }));
-
- this.buttonContainer.appendChild(this.shareButton);
- }
- }
- else if (this.shareButton != null)
- {
- this.shareButton.parentNode.removeChild(this.shareButton);
- this.shareButton = null;
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.save = function(name, done)
- {
- var file = this.getCurrentFile();
-
- if (file != null && this.spinner.spin(document.body, mxResources.get('saving')))
- {
- this.editor.setStatus('');
-
- if (this.editor.graph.isEditing())
- {
- this.editor.graph.stopEditing();
- }
-
- var success = mxUtils.bind(this, function()
- {
- file.handleFileSuccess(true);
- if (done != null)
- {
- done();
- }
- });
-
- var error = mxUtils.bind(this, function(err)
- {
- if (file.isModified())
- {
- Editor.addRetryToError(err, mxUtils.bind(this, function()
- {
- this.save(name, done);
- }));
- }
-
- file.handleFileError(err, true);
- });
-
- try
- {
- if (name == file.getTitle())
- {
- file.save(true, success, error);
- }
- else
- {
- file.saveAs(name, success, error)
- }
- }
- catch (err)
- {
- error(err);
- }
- }
- };
- /**
- * Invokes callback with null if mode does not support folder or not null
- * if a valid folder was chosen for a mode that supports it. No callback
- * is made if no folder was chosen for a mode that supports it.
- */
- App.prototype.pickFolder = function(mode, fn, enabled, direct, force)
- {
- enabled = (enabled != null) ? enabled : true;
- var resume = this.spinner.pause();
-
- if (enabled && mode == App.MODE_GOOGLE && this.drive != null)
- {
- // Shows a save dialog
- this.drive.pickFolder(mxUtils.bind(this, function(evt)
- {
- resume();
-
- if (evt.action == google.picker.Action.PICKED)
- {
- var folderId = null;
-
- if (evt.docs != null && evt.docs.length > 0 && evt.docs[0].type == 'folder')
- {
- folderId = evt.docs[0].id;
- }
-
- fn(folderId);
- }
- }), force);
- }
- else if (enabled && mode == App.MODE_ONEDRIVE && this.oneDrive != null)
- {
- this.oneDrive.pickFolder(mxUtils.bind(this, function(files)
- {
- var folderId = null;
- resume();
-
- if (files != null && files.value != null && files.value.length > 0)
- {
- folderId = OneDriveFile.prototype.getIdOf(files.value[0]);
- fn(folderId);
- }
- }), direct);
- }
- else if (enabled && mode == App.MODE_GITHUB && this.gitHub != null)
- {
- this.gitHub.pickFolder(mxUtils.bind(this, function(folderPath)
- {
- resume();
- fn(folderPath);
- }));
- }
- else if (enabled && mode == App.MODE_GITLAB && this.gitLab != null)
- {
- this.gitLab.pickFolder(mxUtils.bind(this, function(folderPath)
- {
- resume();
- fn(folderPath);
- }));
- }
- else if (enabled && mode == App.MODE_TRELLO && this.trello != null)
- {
- this.trello.pickFolder(mxUtils.bind(this, function(cardId)
- {
- resume();
- fn(cardId);
- }));
- }
- else
- {
- EditorUi.prototype.pickFolder.apply(this, arguments);
- }
- };
- /**
- *
- */
- App.prototype.exportFile = function(data, filename, mimeType, base64Encoded, mode, folderId)
- {
- if (mode == App.MODE_DROPBOX)
- {
- if (this.dropbox != null && this.spinner.spin(document.body, mxResources.get('saving')))
- {
- // LATER: Add folder picker
- this.dropbox.insertFile(filename, (base64Encoded) ? this.base64ToBlob(data, mimeType) :
- data, mxUtils.bind(this, function()
- {
- this.spinner.stop();
- }), mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
- this.handleError(resp);
- }));
- }
- }
- else if (mode == App.MODE_GOOGLE)
- {
- if (this.drive != null && this.spinner.spin(document.body, mxResources.get('saving')))
- {
- this.drive.insertFile(filename, data, folderId, mxUtils.bind(this, function(resp)
- {
- // TODO: Add callback with url param for clickable status message
- // "File exported. Click here to open folder."
- // this.editor.setStatus('<div class="geStatusMessage" style="cursor:pointer;">' +
- // mxResources.get('saved') + '</div>');
- //
- // // Installs click handler for opening
- // if (this.statusContainer != null)
- // {
- // var links = this.statusContainer.getElementsByTagName('div');
- //
- // if (links.length > 0)
- // {
- // mxEvent.addListener(links[0], 'click', mxUtils.bind(this, function()
- // {
- // if (resp != null && resp.id != null)
- // {
- // window.open('https://drive.google.com/open?id=' + resp.id);
- // }
- // }));
- // }
- // }
-
- this.spinner.stop();
- }), mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
- this.handleError(resp);
- }), mimeType, base64Encoded);
- }
- }
- else if (mode == App.MODE_ONEDRIVE)
- {
- if (this.oneDrive != null && this.spinner.spin(document.body, mxResources.get('saving')))
- {
- // KNOWN: OneDrive does not show .svg extension
- this.oneDrive.insertFile(filename, (base64Encoded) ? this.base64ToBlob(data, mimeType) :
- data, mxUtils.bind(this, function()
- {
- this.spinner.stop();
- }), mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
- this.handleError(resp);
- }), false, folderId);
- }
- }
- else if (mode == App.MODE_GITHUB)
- {
- if (this.gitHub != null && this.spinner.spin(document.body, mxResources.get('saving')))
- {
- // Must insert file as library to force the file to be written
- this.gitHub.insertFile(filename, data, mxUtils.bind(this, function()
- {
- this.spinner.stop();
- }), mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
- this.handleError(resp);
- }), true, folderId, base64Encoded);
- }
- }
- else if (mode == App.MODE_TRELLO)
- {
- if (this.trello != null && this.spinner.spin(document.body, mxResources.get('saving')))
- {
- this.trello.insertFile(filename, (base64Encoded) ? this.base64ToBlob(data, mimeType) :
- data, mxUtils.bind(this, function()
- {
- this.spinner.stop();
- }), mxUtils.bind(this, function(resp)
- {
- this.spinner.stop();
- this.handleError(resp);
- }), false, folderId);
- }
- }
- else if (mode == App.MODE_BROWSER)
- {
- var fn = mxUtils.bind(this, function()
- {
- localStorage.setItem(filename, data);
- });
-
- if (localStorage.getItem(filename) == null)
- {
- fn();
- }
- else
- {
- this.confirm(mxResources.get('replaceIt', [filename]), fn);
- }
- }
- };
- /**
- * Translates this point by the given vector.
- *
- * @param {number} dx X-coordinate of the translation.
- * @param {number} dy Y-coordinate of the translation.
- */
- App.prototype.descriptorChanged = function()
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- if (this.fname != null)
- {
- this.fnameWrapper.style.display = 'block';
- this.fname.innerHTML = '';
- var filename = (file.getTitle() != null) ? file.getTitle() : this.defaultFilename;
- mxUtils.write(this.fname, filename);
- this.fname.setAttribute('title', filename + ' - ' + mxResources.get('rename'));
- }
-
- var graph = this.editor.graph;
- var editable = file.isEditable() && !file.invalidChecksum;
-
- if (graph.isEnabled() && !editable)
- {
- graph.reset();
- }
-
- graph.setEnabled(editable);
-
- // Ignores title and hash for revisions
- if (urlParams['rev'] == null)
- {
- this.updateDocumentTitle();
- var newHash = file.getHash();
-
- if (newHash.length > 0)
- {
- window.location.hash = newHash;
- }
- else if (window.location.hash.length > 0)
- {
- window.location.hash = '';
- }
- }
- }
-
- this.updateUi();
- if (this.format != null && this.editor.graph.isSelectionEmpty())
- {
- this.format.refresh();
- }
- };
- /**
- * Adds the listener for automatically saving the diagram for local changes.
- */
- App.prototype.showAuthDialog = function(peer, showRememberOption, fn, closeFn)
- {
- var resume = this.spinner.pause();
-
- this.showDialog(new AuthDialog(this, peer, showRememberOption, mxUtils.bind(this, function(remember)
- {
- try
- {
- if (fn != null)
- {
- fn(remember, mxUtils.bind(this, function()
- {
- this.hideDialog();
- resume();
- }));
- }
- }
- catch (e)
- {
- this.editor.setStatus(mxUtils.htmlEntities(e.message));
- }
- })).container, 300, (showRememberOption) ? 180 : 140, true, true, mxUtils.bind(this, function(cancel)
- {
- if (closeFn != null)
- {
- closeFn();
- }
-
- if (cancel && this.getCurrentFile() == null && this.dialog == null)
- {
- this.showSplash();
- }
- }));
- };
- /**
- * Checks if the client is authorized and calls the next step. The optional
- * readXml argument is used for import. Default is false. The optional
- * readLibrary argument is used for reading libraries. Default is false.
- */
- App.prototype.convertFile = function(url, filename, mimeType, extension, success, error, executeRequest, headers)
- {
- var name = filename;
-
- // SVG file extensions are valid and needed for image import
- if (!/\.svg$/i.test(name))
- {
- name = name.substring(0, filename.lastIndexOf('.')) + extension;
- }
-
- var gitHubUrl = false;
-
- if (this.gitHub != null && url.substring(0, this.gitHub.baseUrl.length) == this.gitHub.baseUrl)
- {
- gitHubUrl = true;
- }
-
- // Workaround for wrong binary response with VSD(X) & VDX files
- if (/\.v(dx|sdx?)$/i.test(filename) && Graph.fileSupport && new XMLHttpRequest().upload &&
- typeof new XMLHttpRequest().responseType === 'string')
- {
- var req = new XMLHttpRequest();
- req.open('GET', url, true);
-
- if (!gitHubUrl)
- {
- req.responseType = 'blob';
- }
-
- if (headers)
- {
- for (var key in headers)
- {
- req.setRequestHeader(key, headers[key]);
- }
- }
-
- req.onload = mxUtils.bind(this, function()
- {
- if (req.status >= 200 && req.status <= 299)
- {
- var blob = null;
-
- if (gitHubUrl)
- {
- var file = JSON.parse(req.responseText);
- blob = this.base64ToBlob(file.content, 'application/octet-stream');
- }
- else
- {
- blob = new Blob([req.response], {type: 'application/octet-stream'});
- }
-
- this.importVisio(blob, mxUtils.bind(this, function(xml)
- {
- success(new LocalFile(this, xml, name, true));
- }), error, filename)
- }
- else if (error != null)
- {
- error({message: mxResources.get('errorLoadingFile')});
- }
- });
- req.onerror = error;
- req.send();
- }
- else
- {
- var handleData = mxUtils.bind(this, function(data)
- {
- try
- {
- if (/\.pdf$/i.test(filename))
- {
- var temp = Editor.extractGraphModelFromPdf(data);
-
- if (temp != null && temp.length > 0)
- {
- success(new LocalFile(this, temp, name, true));
- }
- }
- else if (/\.png$/i.test(filename))
- {
- var temp = this.extractGraphModelFromPng(data);
-
- if (temp != null)
- {
- success(new LocalFile(this, temp, name, true));
- }
- else
- {
- success(new LocalFile(this, data, filename, true));
- }
- }
- else if (Graph.fileSupport && new XMLHttpRequest().upload && this.isRemoteFileFormat(data, url))
- {
- this.parseFile(new Blob([data], {type: 'application/octet-stream'}), mxUtils.bind(this, function(xhr)
- {
- if (xhr.readyState == 4)
- {
- if (xhr.status >= 200 && xhr.status <= 299)
- {
- success(new LocalFile(this, xhr.responseText, name, true));
- }
- else if (error != null)
- {
- error({message: mxResources.get('errorLoadingFile')});
- }
- }
- }), filename);
- }
- else
- {
- success(new LocalFile(this, data, name, true));
- }
- }
- catch (e)
- {
- if (error != null)
- {
- error(e);
- }
- }
- });
- var binary = /\.png$/i.test(filename) || /\.jpe?g$/i.test(filename) ||
- /\.pdf$/i.test(filename) || (mimeType != null &&
- mimeType.substring(0, 6) == 'image/');
-
- // NOTE: Cannot force non-binary request via loadUrl so needs separate
- // code as decoding twice on content with binary data did not work
- if (gitHubUrl)
- {
- mxUtils.get(url, mxUtils.bind(this, function(req)
- {
- if (req.getStatus() >= 200 && req.getStatus() <= 299)
- {
- if (success != null)
- {
- var file = JSON.parse(req.getText());
- var data = file.content;
-
- if (file.encoding === 'base64')
- {
- if (/\.png$/i.test(filename))
- {
- data = 'data:image/png;base64,' + data;
- }
- else if (/\.pdf$/i.test(filename))
- {
- data = 'data:application/pdf;base64,' + data;
- }
- else
- {
- // Workaround for character encoding issues in IE10/11
- data = (window.atob && !mxClient.IS_IE && !mxClient.IS_IE11) ? atob(data) : Base64.decode(data);
- }
- }
-
- handleData(data);
- }
- }
- else if (error != null)
- {
- error({code: App.ERROR_UNKNOWN});
- }
- }), function()
- {
- if (error != null)
- {
- error({code: App.ERROR_UNKNOWN});
- }
- }, false, this.timeout, function()
- {
- if (error != null)
- {
- error({code: App.ERROR_TIMEOUT, retry: fn});
- }
- }, headers);
- }
- else if (executeRequest != null)
- {
- executeRequest(url, handleData, error, binary);
- }
- else
- {
- this.editor.loadUrl(url, handleData, error, binary, null, null, null, headers);
- }
- }
- };
- /**
- * Adds the listener for automatically saving the diagram for local changes.
- */
- App.prototype.updateHeader = function()
- {
- if (this.menubar != null)
- {
- this.appIcon = document.createElement('a');
- this.appIcon.style.display = 'block';
- this.appIcon.style.position = 'absolute';
- this.appIcon.style.width = '32px';
- this.appIcon.style.height = (this.menubarHeight - 28) + 'px';
- this.appIcon.style.margin = '14px 0px 8px 16px';
- this.appIcon.style.opacity = '0.85';
- this.appIcon.style.borderRadius = '3px';
-
- if (uiTheme != 'dark')
- {
- this.appIcon.style.backgroundColor = '#f08705';
- }
-
- mxEvent.disableContextMenu(this.appIcon);
-
- mxEvent.addListener(this.appIcon, 'click', mxUtils.bind(this, function(evt)
- {
- this.appIconClicked(evt);
- }));
-
- // LATER: Use Alpha image loader in IE6
- // NOTE: This uses the diagram bit of the old logo as it looks better in this case
- //this.appIcon.style.filter = 'progid:DXImageTransform.Microsoft.AlphaImageLoader(src=' + IMAGE_PATH + '/logo-white.png,sizingMethod=\'scale\')';
- var logo = (!mxClient.IS_SVG) ? 'url(\'' + IMAGE_PATH + '/logo-white.png\')' :
- ((uiTheme == 'dark') ? 'url()' :
- 'url()');
- this.appIcon.style.backgroundImage = logo;
- this.appIcon.style.backgroundPosition = 'center center';
- this.appIcon.style.backgroundSize = '100% 100%';
- this.appIcon.style.backgroundRepeat = 'no-repeat';
-
- mxUtils.setPrefixedStyle(this.appIcon.style, 'transition', 'all 125ms linear');
-
- mxEvent.addListener(this.appIcon, 'mouseover', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
-
- if (file != null)
- {
- var mode = file.getMode();
-
- if (mode == App.MODE_GOOGLE)
- {
- this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/google-drive-logo-white.svg)';
- this.appIcon.style.backgroundSize = '70% 70%';
- }
- else if (mode == App.MODE_DROPBOX)
- {
- this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/dropbox-logo-white.svg)';
- this.appIcon.style.backgroundSize = '70% 70%';
- }
- else if (mode == App.MODE_ONEDRIVE)
- {
- this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/onedrive-logo-white.svg)';
- this.appIcon.style.backgroundSize = '70% 70%';
- }
- else if (mode == App.MODE_GITHUB)
- {
- this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/github-logo-white.svg)';
- this.appIcon.style.backgroundSize = '70% 70%';
- }
- else if (mode == App.MODE_GITLAB)
- {
- this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/gitlab-logo-white.svg)';
- this.appIcon.style.backgroundSize = '100% 100%';
- }
- else if (mode == App.MODE_TRELLO)
- {
- this.appIcon.style.backgroundImage = 'url(' + IMAGE_PATH + '/trello-logo-white-orange.svg)';
- this.appIcon.style.backgroundSize = '70% 70%';
- }
- }
- }));
-
- mxEvent.addListener(this.appIcon, 'mouseout', mxUtils.bind(this, function()
- {
- this.appIcon.style.backgroundImage = logo;
- this.appIcon.style.backgroundSize = '90% 90%';
- }));
-
- if (urlParams['embed'] != '1')
- {
- this.menubarContainer.appendChild(this.appIcon);
- }
-
- this.fnameWrapper = document.createElement('div');
- this.fnameWrapper.style.position = 'absolute';
- this.fnameWrapper.style.right = '120px';
- this.fnameWrapper.style.left = '60px';
- this.fnameWrapper.style.top = '9px';
- this.fnameWrapper.style.height = '26px';
- this.fnameWrapper.style.display = 'none';
- this.fnameWrapper.style.overflow = 'hidden';
- this.fnameWrapper.style.textOverflow = 'ellipsis';
-
- this.fname = document.createElement('a');
- this.fname.setAttribute('title', mxResources.get('rename'));
- this.fname.className = 'geItem';
- this.fname.style.padding = '2px 8px 2px 8px';
- this.fname.style.display = 'inline';
- this.fname.style.fontSize = '18px';
- this.fname.style.whiteSpace = 'nowrap';
-
- // Prevents focus
- mxEvent.addListener(this.fname, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown',
- mxUtils.bind(this, function(evt)
- {
- evt.preventDefault();
- }));
-
- mxEvent.addListener(this.fname, 'click', mxUtils.bind(this, function(evt)
- {
- var file = this.getCurrentFile();
-
- if (file != null && file.isRenamable())
- {
- if (this.editor.graph.isEditing())
- {
- this.editor.graph.stopEditing();
- }
- this.actions.get('rename').funct();
- }
-
- mxEvent.consume(evt);
- }));
-
- this.fnameWrapper.appendChild(this.fname);
-
- if (urlParams['embed'] != '1')
- {
- this.menubarContainer.appendChild(this.fnameWrapper);
-
- this.menubar.container.style.position = 'absolute';
- this.menubar.container.style.paddingLeft = '59px';
- this.toolbar.container.style.paddingLeft = '16px';
- this.menubar.container.style.boxSizing = 'border-box';
- this.menubar.container.style.top = '34px';
- }
-
- /**
- * Adds format panel toggle.
- */
- this.toggleFormatElement = document.createElement('a');
- this.toggleFormatElement.setAttribute('title', mxResources.get('formatPanel') + ' (' + Editor.ctrlKey + '+Shift+P)');
- this.toggleFormatElement.style.position = 'absolute';
- this.toggleFormatElement.style.display = 'inline-block';
- this.toggleFormatElement.style.top = (uiTheme == 'atlas') ? '8px' : '6px';
- this.toggleFormatElement.style.right = (uiTheme != 'atlas' && urlParams['embed'] != '1') ? '30px' : '10px';
- this.toggleFormatElement.style.padding = '2px';
- this.toggleFormatElement.style.fontSize = '14px';
- this.toggleFormatElement.className = (uiTheme != 'atlas') ? 'geButton' : '';
- this.toggleFormatElement.style.width = '16px';
- this.toggleFormatElement.style.height = '16px';
- this.toggleFormatElement.style.backgroundPosition = '50% 50%';
- this.toggleFormatElement.style.backgroundRepeat = 'no-repeat';
- this.toolbarContainer.appendChild(this.toggleFormatElement);
-
- if (uiTheme == 'dark')
- {
- this.toggleFormatElement.style.filter = 'invert(100%)';
- }
-
- // Prevents focus
- mxEvent.addListener(this.toggleFormatElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown',
- mxUtils.bind(this, function(evt)
- {
- evt.preventDefault();
- }));
-
- mxEvent.addListener(this.toggleFormatElement, 'click', mxUtils.bind(this, function(evt)
- {
- this.actions.get('formatPanel').funct();
- mxEvent.consume(evt);
- }));
- var toggleFormatPanel = mxUtils.bind(this, function()
- {
- if (this.formatWidth > 0)
- {
- this.toggleFormatElement.style.backgroundImage = 'url(\'' + this.formatShowImage + '\')';
- }
- else
- {
- this.toggleFormatElement.style.backgroundImage = 'url(\'' + this.formatHideImage + '\')';
- }
- });
-
- this.addListener('formatWidthChanged', toggleFormatPanel);
- toggleFormatPanel();
- this.fullscreenElement = document.createElement('a');
- this.fullscreenElement.setAttribute('title', mxResources.get('fullscreen'));
- this.fullscreenElement.style.position = 'absolute';
- this.fullscreenElement.style.display = 'inline-block';
- this.fullscreenElement.style.top = (uiTheme == 'atlas') ? '8px' : '6px';
- this.fullscreenElement.style.right = (uiTheme != 'atlas' && urlParams['embed'] != '1') ? '50px' : '30px';
- this.fullscreenElement.style.padding = '2px';
- this.fullscreenElement.style.fontSize = '14px';
- this.fullscreenElement.className = (uiTheme != 'atlas') ? 'geButton' : '';
- this.fullscreenElement.style.width = '16px';
- this.fullscreenElement.style.height = '16px';
- this.fullscreenElement.style.backgroundPosition = '50% 50%';
- this.fullscreenElement.style.backgroundRepeat = 'no-repeat';
- this.fullscreenElement.style.backgroundImage = 'url(\'' + this.fullscreenImage + '\')';
- this.toolbarContainer.appendChild(this.fullscreenElement);
-
- // Prevents focus
- mxEvent.addListener(this.fullscreenElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown',
- mxUtils.bind(this, function(evt)
- {
- evt.preventDefault();
- }));
-
- // Some style changes in Atlas theme
- if (uiTheme == 'atlas')
- {
- mxUtils.setOpacity(this.toggleFormatElement, 70);
- mxUtils.setOpacity(this.fullscreenElement, 70);
- }
-
- var initialPosition = this.hsplitPosition;
- if (uiTheme == 'dark')
- {
- this.fullscreenElement.style.filter = 'invert(100%)';
- }
-
- mxEvent.addListener(this.fullscreenElement, 'click', mxUtils.bind(this, function(evt)
- {
- var visible = this.fullscreenMode;
-
- if (uiTheme != 'atlas' && urlParams['embed'] != '1')
- {
- this.toggleCompactMode(visible);
- }
- if (!visible)
- {
- initialPosition = this.hsplitPosition;
- }
-
- this.hsplitPosition = (visible) ? initialPosition : 0;
- this.toggleFormatPanel(visible);
- this.fullscreenMode = !visible;
- mxEvent.consume(evt);
- }));
- /**
- * Adds compact UI toggle.
- */
- if (urlParams['embed'] != '1')
- {
- this.toggleElement = document.createElement('a');
- this.toggleElement.setAttribute('title', mxResources.get('collapseExpand'));
- this.toggleElement.className = 'geButton';
- this.toggleElement.style.position = 'absolute';
- this.toggleElement.style.display = 'inline-block';
- this.toggleElement.style.width = '16px';
- this.toggleElement.style.height = '16px';
- this.toggleElement.style.color = '#666';
- this.toggleElement.style.top = (uiTheme == 'atlas') ? '8px' : '6px';
- this.toggleElement.style.right = '10px';
- this.toggleElement.style.padding = '2px';
- this.toggleElement.style.fontSize = '14px';
- this.toggleElement.style.textDecoration = 'none';
- this.toggleElement.style.backgroundImage = 'url(\'' + this.chevronUpImage + '\')';
-
- this.toggleElement.style.backgroundPosition = '50% 50%';
- this.toggleElement.style.backgroundRepeat = 'no-repeat';
-
- if (uiTheme == 'dark')
- {
- this.toggleElement.style.filter = 'invert(100%)';
- }
-
- // Prevents focus
- mxEvent.addListener(this.toggleElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown',
- mxUtils.bind(this, function(evt)
- {
- evt.preventDefault();
- }));
-
- // Toggles compact mode
- mxEvent.addListener(this.toggleElement, 'click', mxUtils.bind(this, function(evt)
- {
- this.toggleCompactMode();
- mxEvent.consume(evt);
- }));
-
- if (uiTheme != 'atlas')
- {
- this.toolbarContainer.appendChild(this.toggleElement);
- }
-
- // Enable compact mode for small screens except for Firefox where the height is wrong
- if (!mxClient.IS_FF && screen.height <= 740 && typeof this.toggleElement.click !== 'undefined')
- {
- window.setTimeout(mxUtils.bind(this, function()
- {
- this.toggleElement.click();
- }), 0);
- }
- }
- }
- };
- /**
- * Adds the listener for automatically saving the diagram for local changes.
- */
- App.prototype.toggleCompactMode = function(visible)
- {
- visible = (visible != null) ? visible : this.compactMode;
-
- if (visible)
- {
- this.menubar.container.style.position = 'absolute';
- this.menubar.container.style.paddingLeft = '59px';
- this.menubar.container.style.paddingTop = '';
- this.menubar.container.style.paddingBottom = '';
- this.menubar.container.style.top = '34px';
- this.toolbar.container.style.paddingLeft = '16px';
- this.buttonContainer.style.visibility = 'visible';
- this.appIcon.style.display = 'block';
- this.fnameWrapper.style.display = 'block';
- this.fnameWrapper.style.visibility = 'visible';
- this.menubarHeight = App.prototype.menubarHeight;
- this.refresh();
- this.toggleElement.style.backgroundImage = 'url(\'' + this.chevronUpImage + '\')';
- }
- else
- {
- this.menubar.container.style.position = 'relative';
- this.menubar.container.style.paddingLeft = '4px';
- this.menubar.container.style.paddingTop = '0px';
- this.menubar.container.style.paddingBottom = '0px';
- this.menubar.container.style.top = '0px';
- this.toolbar.container.style.paddingLeft = '8px';
- this.buttonContainer.style.visibility = 'hidden';
- this.appIcon.style.display = 'none';
- this.fnameWrapper.style.display = 'none';
- this.fnameWrapper.style.visibility = 'hidden';
- this.menubarHeight = EditorUi.prototype.menubarHeight;
- this.refresh();
- this.toggleElement.style.backgroundImage = 'url(\'' + this.chevronDownImage + '\')';
- }
-
- this.compactMode = !visible;
- };
- /**
- * Adds the listener for automatically saving the diagram for local changes.
- */
- App.prototype.updateUserElement = function()
- {
- if ((this.drive == null || this.drive.getUser() == null) &&
- (this.oneDrive == null || this.oneDrive.getUser() == null) &&
- (this.dropbox == null || this.dropbox.getUser() == null) &&
- (this.gitHub == null || this.gitHub.getUser() == null) &&
- (this.gitLab == null || this.gitLab.getUser() == null) &&
- (this.trello == null || !this.trello.isAuthorized())) //TODO Trello no user issue
- {
- if (this.userElement != null)
- {
- this.userElement.parentNode.removeChild(this.userElement);
- this.userElement = null;
- }
- }
- else
- {
- if (this.userElement == null)
- {
- this.userElement = document.createElement('a');
- this.userElement.className = 'geItem';
- this.userElement.style.position = 'absolute';
- this.userElement.style.fontSize = '8pt';
- this.userElement.style.top = (uiTheme == 'atlas') ? '8px' : '2px';
- this.userElement.style.right = '30px';
- this.userElement.style.margin = '4px';
- this.userElement.style.padding = '2px';
- this.userElement.style.paddingRight = '16px';
- this.userElement.style.verticalAlign = 'middle';
- this.userElement.style.backgroundImage = 'url(' + IMAGE_PATH + '/expanded.gif)';
- this.userElement.style.backgroundPosition = '100% 60%';
- this.userElement.style.backgroundRepeat = 'no-repeat';
-
- this.menubarContainer.appendChild(this.userElement);
- // Prevents focus
- mxEvent.addListener(this.userElement, (mxClient.IS_POINTER) ? 'pointerdown' : 'mousedown',
- mxUtils.bind(this, function(evt)
- {
- evt.preventDefault();
- }));
-
- mxEvent.addListener(this.userElement, 'click', mxUtils.bind(this, function(evt)
- {
- if (this.userPanel == null)
- {
- var div = document.createElement('div');
- div.className = 'geDialog';
- div.style.position = 'absolute';
- div.style.top = (this.userElement.clientTop + this.userElement.clientHeight + 6) + 'px';
- div.style.right = '36px';
- div.style.padding = '0px';
- div.style.cursor = 'default';
-
- this.userPanel = div;
- }
-
- if (this.userPanel.parentNode != null)
- {
- this.userPanel.parentNode.removeChild(this.userPanel);
- }
- else
- {
- var connected = false;
- this.userPanel.innerHTML = '';
-
- var img = document.createElement('img');
- img.setAttribute('src', Dialog.prototype.closeImage);
- img.setAttribute('title', mxResources.get('close'));
- img.className = 'geDialogClose';
- img.style.top = '8px';
- img.style.right = '8px';
-
- mxEvent.addListener(img, 'click', mxUtils.bind(this, function()
- {
- if (this.userPanel.parentNode != null)
- {
- this.userPanel.parentNode.removeChild(this.userPanel);
- }
- }));
-
- this.userPanel.appendChild(img);
-
- if (this.drive != null)
- {
- var driveUsers = this.drive.getUsersList();
-
- if (driveUsers.length > 0)
- {
- // LATER: Cannot change user while file is open since close will not work with new
- // credentials and closing the file using fileLoaded(null) will show splash dialog.
- var closeFile = mxUtils.bind(this, function(callback, spinnerMsg)
- {
- var file = this.getCurrentFile();
- if (file != null && file.constructor == DriveFile)
- {
- this.spinner.spin(document.body, spinnerMsg);
-
- // file.close();
- this.fileLoaded(null);
- // LATER: Use callback to wait for thumbnail update
- window.setTimeout(mxUtils.bind(this, function()
- {
- this.spinner.stop();
- callback();
- }), 2000);
- }
- else
- {
- callback();
- }
- });
-
- var createUserRow = mxUtils.bind(this, function (user)
- {
- var tr = document.createElement('tr');
- tr.style.cssText = user.isCurrent? '' : 'background-color: whitesmoke; cursor: pointer';
- tr.setAttribute('title', 'User ID: ' + user.id);
- tr.innerHTML = '<td valign="middle" style="height: 59px;width: 66px;' +
- (user.isCurrent? '' : 'border-top: 1px solid rgb(224, 224, 224);') + '">' +
- '<img width="50" height="50" style="margin: 4px 8px 0 8px;border-radius:50%;" src="' +
- ((user.pictureUrl != null) ? user.pictureUrl : this.defaultUserPicture) + '"/>' +
- '</td><td valign="middle" style="white-space:nowrap;' +
- ((user.pictureUrl != null) ? 'padding-top:4px;' : '') +
- (user.isCurrent? '' : 'border-top: 1px solid rgb(224, 224, 224);') +
- '">' + mxUtils.htmlEntities(user.displayName) + '<br>' +
- '<small style="color:gray;">' + mxUtils.htmlEntities(user.email) +
- '</small><div style="margin-top:4px;"><i>' +
- mxResources.get('googleDrive') + '</i></div>';
-
- if (!user.isCurrent)
- {
- mxEvent.addListener(tr, 'click', mxUtils.bind(this, function(evt)
- {
- closeFile(mxUtils.bind(this, function()
- {
- this.stateArg = null;
- this.drive.setUser(user);
-
- this.drive.authorize(true, mxUtils.bind(this, function()
- {
- this.setMode(App.MODE_GOOGLE);
- this.hideDialog();
- this.showSplash();
- }), mxUtils.bind(this, function(resp)
- {
- this.handleError(resp);
- }), true); //Remember is true since add account imply keeping that account
- }), mxResources.get('closingFile') + '...');
-
- mxEvent.consume(evt);
- }));
- }
-
- return tr;
- });
-
- connected = true;
-
- var driveUserTable = document.createElement('table');
- driveUserTable.style.cssText ='font-size:10pt;padding: 20px 0 0 0;min-width: 300px;border-spacing: 0;';
- for (var i = 0; i < driveUsers.length; i++)
- {
- driveUserTable.appendChild(createUserRow(driveUsers[i]));
- }
-
- this.userPanel.appendChild(driveUserTable);
-
- var div = document.createElement('div');
- div.style.textAlign = 'left';
- div.style.padding = '8px';
- div.style.whiteSpace = 'nowrap';
- div.style.borderTop = '1px solid rgb(224, 224, 224)';
- var btn = mxUtils.button(mxResources.get('signOut'), mxUtils.bind(this, function()
- {
- this.confirm(mxResources.get('areYouSure'), mxUtils.bind(this, function()
- {
- closeFile(mxUtils.bind(this, function()
- {
- this.stateArg = null;
- this.drive.logout();
- this.setMode(App.MODE_GOOGLE);
- this.hideDialog();
- this.showSplash();
- }), mxResources.get('signOut'));
- }));
- }));
- btn.className = 'geBtn';
- btn.style.float = 'right';
- div.appendChild(btn);
-
- var btn = mxUtils.button(mxResources.get('addAccount'), mxUtils.bind(this, function()
- {
- var authWin = this.drive.createAuthWin();
- //FIXME This doean't work to set focus back to main window until closing the file is done
- authWin.blur();
- window.focus();
-
- closeFile(mxUtils.bind(this, function()
- {
- this.stateArg = null;
-
- this.drive.authorize(false, mxUtils.bind(this, function()
- {
- this.setMode(App.MODE_GOOGLE);
- this.hideDialog();
- this.showSplash();
- }), mxUtils.bind(this, function(resp)
- {
- this.handleError(resp);
- }), true, authWin); //Remember is true since add account imply keeping that account
- }), mxResources.get('closingFile') + '...');
- }));
- btn.className = 'geBtn';
- btn.style.margin = '0px';
- div.appendChild(btn);
- this.userPanel.appendChild(div);
- }
- }
-
- var addUser = mxUtils.bind(this, function(user, logo, logout, label)
- {
- if (user != null)
- {
- if (connected)
- {
- this.userPanel.appendChild(document.createElement('hr'));
- }
-
- connected = true;
- var userTable = document.createElement('table');
- userTable.style.cssText = 'font-size:10pt;padding:' + (connected? '10' : '20') + 'px 20px 10px 10px;';
-
- userTable.innerHTML += '<tr><td valign="top">' +
- ((logo != null) ? '<img style="margin-right:6px;" src="' + logo + '" width="40" height="40"/></td>' : '') +
- '<td valign="middle" style="white-space:nowrap;">' + mxUtils.htmlEntities(user.displayName) +
- ((user.email != null) ? '<br><small style="color:gray;">' + mxUtils.htmlEntities(user.email) + '</small>' : '') +
- ((label != null) ? '<div style="margin-top:4px;"><i>' + mxUtils.htmlEntities(label) + '</i></div>' : '') +
- '</td></tr>';
-
- this.userPanel.appendChild(userTable);
- var div = document.createElement('div');
- div.style.textAlign = 'center';
- div.style.paddingBottom = '12px';
- div.style.whiteSpace = 'nowrap';
-
- if (logout != null)
- {
- var btn = mxUtils.button(mxResources.get('signOut'), logout);
- btn.className = 'geBtn';
- div.appendChild(btn);
- }
-
- this.userPanel.appendChild(div);
- }
- });
-
- if (this.dropbox != null)
- {
- addUser(this.dropbox.getUser(), IMAGE_PATH + '/dropbox-logo.svg', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- if (file != null && file.constructor == DropboxFile)
- {
- var doLogout = mxUtils.bind(this, function()
- {
- this.dropbox.logout();
- window.location.hash = '';
- });
-
- if (!file.isModified())
- {
- doLogout();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), null, doLogout,
- mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- }
- else
- {
- this.dropbox.logout();
- }
- }), mxResources.get('dropbox'));
- }
- if (this.oneDrive != null)
- {
- addUser(this.oneDrive.getUser(), IMAGE_PATH + '/onedrive-logo.svg', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- if (file != null && file.constructor == OneDriveFile)
- {
- var doLogout = mxUtils.bind(this, function()
- {
- this.oneDrive.logout();
- window.location.hash = '';
- });
-
- if (!file.isModified())
- {
- doLogout();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), null, doLogout,
- mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- }
- else
- {
- this.oneDrive.logout();
- }
- }), mxResources.get('oneDrive'));
- }
- if (this.gitHub != null)
- {
- addUser(this.gitHub.getUser(), IMAGE_PATH + '/github-logo.svg', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- if (file != null && file.constructor == GitHubFile)
- {
- var doLogout = mxUtils.bind(this, function()
- {
- this.gitHub.logout();
- window.location.hash = '';
- });
-
- if (!file.isModified())
- {
- doLogout();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), null, doLogout,
- mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- }
- else
- {
- this.gitHub.logout();
- }
- }), mxResources.get('github'));
- }
-
- if (this.gitLab != null)
- {
- addUser(this.gitLab.getUser(), IMAGE_PATH + '/gitlab-logo.svg', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- if (file != null && file.constructor == GitLabFile)
- {
- var doLogout = mxUtils.bind(this, function()
- {
- this.gitLab.logout();
- window.location.hash = '';
- });
- if (!file.isModified())
- {
- doLogout();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), null, doLogout,
- mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- }
- else
- {
- this.gitLab.logout();
- }
- }), mxResources.get('gitlab'));
- }
-
- //TODO We have no user info from Trello, how we can create a user?
- if (this.trello != null)
- {
- addUser(this.trello.getUser(), IMAGE_PATH + '/trello-logo.svg', mxUtils.bind(this, function()
- {
- var file = this.getCurrentFile();
- if (file != null && file.constructor == TrelloFile)
- {
- var doLogout = mxUtils.bind(this, function()
- {
- this.trello.logout();
- window.location.hash = '';
- });
-
- if (!file.isModified())
- {
- doLogout();
- }
- else
- {
- this.confirm(mxResources.get('allChangesLost'), null, doLogout,
- mxResources.get('cancel'), mxResources.get('discardChanges'));
- }
- }
- else
- {
- this.trello.logout();
- }
- }), mxResources.get('trello'));
- }
-
- if (!connected)
- {
- var div = document.createElement('div');
- div.style.textAlign = 'center';
- div.style.padding = '20px 20px 10px 10px';
- div.innerHTML = mxResources.get('notConnected');
-
- this.userPanel.appendChild(div);
- }
-
- var div = document.createElement('div');
- div.style.textAlign = 'center';
- div.style.padding = '12px';
- div.style.background = 'whiteSmoke';
- div.style.borderTop = '1px solid #e0e0e0';
- div.style.whiteSpace = 'nowrap';
-
- var btn = mxUtils.button(mxResources.get('close'), mxUtils.bind(this, function()
- {
- if (!mxEvent.isConsumed(evt) && this.userPanel != null && this.userPanel.parentNode != null)
- {
- this.userPanel.parentNode.removeChild(this.userPanel);
- }
- }));
- btn.className = 'geBtn';
- div.appendChild(btn);
- this.userPanel.appendChild(div);
- document.body.appendChild(this.userPanel);
- }
-
- mxEvent.consume(evt);
- }));
-
- mxEvent.addListener(document.body, 'click', mxUtils.bind(this, function(evt)
- {
- if (!mxEvent.isConsumed(evt) && this.userPanel != null && this.userPanel.parentNode != null)
- {
- this.userPanel.parentNode.removeChild(this.userPanel);
- }
- }));
- }
-
- var user = null;
-
- if (this.drive != null && this.drive.getUser() != null)
- {
- user = this.drive.getUser();
- }
- else if (this.oneDrive != null && this.oneDrive.getUser() != null)
- {
- user = this.oneDrive.getUser();
- }
- else if (this.dropbox != null && this.dropbox.getUser() != null)
- {
- user = this.dropbox.getUser();
- }
- else if (this.gitHub != null && this.gitHub.getUser() != null)
- {
- user = this.gitHub.getUser();
- }
- else if (this.gitLab != null && this.gitLab.getUser() != null)
- {
- user = this.gitLab.getUser();
- }
- //TODO Trello no user issue
-
- if (user != null)
- {
- this.userElement.innerHTML = '';
-
- if (screen.width > 560)
- {
- mxUtils.write(this.userElement, user.displayName);
- this.userElement.style.display = 'block';
- }
- }
- else
- {
- this.userElement.style.display = 'none';
- }
- }
- };
- //TODO Use this function to get the currently logged in user
- App.prototype.getCurrentUser = function()
- {
- var user = null;
-
- if (this.drive != null && this.drive.getUser() != null)
- {
- user = this.drive.getUser();
- }
- else if (this.oneDrive != null && this.oneDrive.getUser() != null)
- {
- user = this.oneDrive.getUser();
- }
- else if (this.dropbox != null && this.dropbox.getUser() != null)
- {
- user = this.dropbox.getUser();
- }
- else if (this.gitHub != null && this.gitHub.getUser() != null)
- {
- user = this.gitHub.getUser();
- }
- //TODO Trello no user issue
-
- return user;
- }
- /**
- * Override depends on mxSettings which is not defined in the minified viewer.
- */
- var editorResetGraph = Editor.prototype.resetGraph;
- Editor.prototype.resetGraph = function()
- {
- editorResetGraph.apply(this, arguments);
-
- // Overrides default with persisted value
- this.graph.pageFormat = mxSettings.getPageFormat();
- };
|