1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690769176927693769476957696769776987699770077017702770377047705770677077708770977107711771277137714771577167717771877197720772177227723772477257726772777287729773077317732773377347735773677377738773977407741774277437744774577467747774877497750775177527753775477557756775777587759776077617762776377647765776677677768776977707771777277737774777577767777777877797780778177827783778477857786778777887789779077917792779377947795779677977798779978007801780278037804780578067807780878097810781178127813781478157816781778187819782078217822782378247825782678277828782978307831783278337834783578367837783878397840784178427843784478457846784778487849785078517852785378547855785678577858785978607861786278637864786578667867786878697870787178727873787478757876787778787879788078817882788378847885788678877888788978907891789278937894789578967897789878997900790179027903790479057906790779087909791079117912791379147915791679177918791979207921792279237924792579267927792879297930793179327933793479357936793779387939794079417942794379447945794679477948794979507951795279537954795579567957795879597960796179627963796479657966796779687969797079717972797379747975797679777978797979807981798279837984798579867987798879897990799179927993799479957996799779987999800080018002800380048005800680078008800980108011801280138014801580168017801880198020802180228023802480258026802780288029803080318032803380348035803680378038803980408041804280438044804580468047804880498050805180528053805480558056805780588059806080618062806380648065806680678068806980708071807280738074807580768077807880798080808180828083808480858086808780888089809080918092809380948095809680978098809981008101810281038104810581068107810881098110811181128113811481158116811781188119812081218122812381248125812681278128812981308131813281338134813581368137813881398140814181428143814481458146814781488149815081518152815381548155815681578158815981608161816281638164816581668167816881698170817181728173817481758176817781788179818081818182818381848185818681878188818981908191819281938194819581968197819881998200820182028203820482058206820782088209821082118212821382148215821682178218821982208221822282238224822582268227822882298230823182328233823482358236823782388239824082418242824382448245824682478248824982508251825282538254825582568257825882598260826182628263826482658266826782688269827082718272827382748275827682778278827982808281828282838284828582868287828882898290829182928293829482958296829782988299830083018302830383048305830683078308830983108311831283138314831583168317831883198320832183228323832483258326832783288329833083318332833383348335833683378338833983408341834283438344834583468347834883498350835183528353835483558356835783588359836083618362836383648365836683678368836983708371837283738374837583768377837883798380838183828383838483858386838783888389839083918392839383948395839683978398839984008401840284038404840584068407840884098410841184128413841484158416841784188419842084218422842384248425842684278428842984308431843284338434843584368437843884398440844184428443844484458446844784488449845084518452845384548455845684578458845984608461846284638464846584668467846884698470847184728473847484758476847784788479848084818482848384848485848684878488848984908491849284938494849584968497849884998500850185028503850485058506850785088509851085118512851385148515851685178518851985208521852285238524852585268527852885298530853185328533853485358536853785388539854085418542854385448545854685478548854985508551855285538554855585568557855885598560856185628563856485658566856785688569857085718572857385748575857685778578857985808581858285838584858585868587858885898590859185928593859485958596859785988599860086018602860386048605860686078608860986108611861286138614861586168617861886198620862186228623862486258626862786288629863086318632863386348635863686378638863986408641864286438644864586468647864886498650865186528653865486558656865786588659866086618662866386648665866686678668866986708671867286738674867586768677867886798680868186828683868486858686868786888689869086918692869386948695869686978698869987008701870287038704870587068707870887098710871187128713871487158716871787188719872087218722872387248725872687278728872987308731873287338734873587368737873887398740874187428743874487458746874787488749875087518752875387548755875687578758875987608761876287638764876587668767876887698770 |
- /*
- * svgcanvas.js
- *
- * Licensed under the Apache License, Version 2
- *
- * Copyright(c) 2010 Alexis Deveria
- * Copyright(c) 2010 Pavol Rusnak
- * Copyright(c) 2010 Jeff Schiller
- *
- */
- // Dependencies:
- // 1) jQuery
- // 2) browser.js
- // 3) svgtransformlist.js
- // 4) math.js
- // 5) units.js
- // 6) svgutils.js
- // 7) sanitize.js
- // 8) history.js
- // 9) select.js
- // 10) draw.js
- // 11) path.js
- if(!window.console) {
- window.console = {};
- window.console.log = function(str) {};
- window.console.dir = function(str) {};
- }
- if(window.opera) {
- window.console.log = function(str) { opera.postError(str); };
- window.console.dir = function(str) {};
- }
- (function() {
- // This fixes $(...).attr() to work as expected with SVG elements.
- // Does not currently use *AttributeNS() since we rarely need that.
-
- // See http://api.jquery.com/attr/ for basic documentation of .attr()
-
- // Additional functionality:
- // - When getting attributes, a string that's a number is return as type number.
- // - If an array is supplied as first parameter, multiple values are returned
- // as an object with values for each given attributes
- var proxied = jQuery.fn.attr, svgns = "http://www.w3.org/2000/svg";
- jQuery.fn.attr = function(key, value) {
- var len = this.length;
- if(!len) return this;
- for(var i=0; i<len; i++) {
- var elem = this[i];
- // set/get SVG attribute
- if(elem.namespaceURI === svgns) {
- // Setting attribute
- if(value !== undefined) {
- elem.setAttribute(key, value);
- } else if($.isArray(key)) {
- // Getting attributes from array
- var j = key.length, obj = {};
- while(j--) {
- var aname = key[j];
- var attr = elem.getAttribute(aname);
- // This returns a number when appropriate
- if(attr || attr === "0") {
- attr = isNaN(attr)?attr:attr-0;
- }
- obj[aname] = attr;
- }
- return obj;
-
- } else if(typeof key === "object") {
- // Setting attributes form object
- for(var v in key) {
- elem.setAttribute(v, key[v]);
- }
- // Getting attribute
- } else {
- var attr = elem.getAttribute(key);
- if(attr || attr === "0") {
- attr = isNaN(attr)?attr:attr-0;
- }
- return attr;
- }
- } else {
- return proxied.apply(this, arguments);
- }
- }
- return this;
- };
-
- }());
- // Class: SvgCanvas
- // The main SvgCanvas class that manages all SVG-related functions
- //
- // Parameters:
- // container - The container HTML element that should hold the SVG root element
- // config - An object that contains configuration data
- $.SvgCanvas = function(container, config)
- {
- // Namespace constants
- var svgns = "http://www.w3.org/2000/svg",
- xlinkns = "http://www.w3.org/1999/xlink",
- xmlns = "http://www.w3.org/XML/1998/namespace",
- xmlnsns = "http://www.w3.org/2000/xmlns/", // see http://www.w3.org/TR/REC-xml-names/#xmlReserved
- se_ns = "http://svg-edit.googlecode.com",
- htmlns = "http://www.w3.org/1999/xhtml",
- mathns = "http://www.w3.org/1998/Math/MathML";
- // Default configuration options
- var curConfig = {
- show_outside_canvas: true,
- selectNew: true,
- dimensions: [640, 480]
- };
- // Update config with new one if given
- if(config) {
- $.extend(curConfig, config);
- }
- // Array with width/height of canvas
- var dimensions = curConfig.dimensions;
- var canvas = this;
- // "document" element associated with the container (same as window.document using default svg-editor.js)
- // NOTE: This is not actually a SVG document, but a HTML document.
- var svgdoc = container.ownerDocument;
- // This is a container for the document being edited, not the document itself.
- var svgroot = svgdoc.importNode(svgedit.utilities.text2xml(
- '<svg id="svgroot" xmlns="' + svgns + '" xlinkns="' + xlinkns + '" ' +
- 'width="' + dimensions[0] + '" height="' + dimensions[1] + '" x="' + dimensions[0] + '" y="' + dimensions[1] + '" overflow="visible">' +
- '<defs>' +
- '<filter id="canvashadow" filterUnits="objectBoundingBox">' +
- '<feGaussianBlur in="SourceAlpha" stdDeviation="4" result="blur"/>'+
- '<feOffset in="blur" dx="5" dy="5" result="offsetBlur"/>'+
- '<feMerge>'+
- '<feMergeNode in="offsetBlur"/>'+
- '<feMergeNode in="SourceGraphic"/>'+
- '</feMerge>'+
- '</filter>'+
- '</defs>'+
- '</svg>').documentElement, true);
- container.appendChild(svgroot);
- // The actual element that represents the final output SVG element
- var svgcontent = svgdoc.createElementNS(svgns, "svg");
- // This function resets the svgcontent element while keeping it in the DOM.
- var clearSvgContentElement = canvas.clearSvgContentElement = function() {
- while (svgcontent.firstChild) { svgcontent.removeChild(svgcontent.firstChild); }
- // TODO: Clear out all other attributes first?
- $(svgcontent).attr({
- id: 'svgcontent',
- width: dimensions[0],
- height: dimensions[1],
- x: dimensions[0],
- y: dimensions[1],
- overflow: curConfig.show_outside_canvas ? 'visible' : 'hidden',
- xmlns: svgns,
- "xmlns:se": se_ns,
- "xmlns:xlink": xlinkns
- }).appendTo(svgroot);
- // TODO: make this string optional and set by the client
- var comment = svgdoc.createComment(" Created with SVG-edit - http://svg-edit.googlecode.com/ ");
- svgcontent.appendChild(comment);
- };
- clearSvgContentElement();
- // Prefix string for element IDs
- var idprefix = "svg_";
- // Function: setIdPrefix
- // Changes the ID prefix to the given value
- //
- // Parameters:
- // p - String with the new prefix
- canvas.setIdPrefix = function(p) {
- idprefix = p;
- };
- // Current svgedit.draw.Drawing object
- // @type {svgedit.draw.Drawing}
- canvas.current_drawing_ = new svgedit.draw.Drawing(svgcontent, idprefix);
- // Function: getCurrentDrawing
- // Returns the current Drawing.
- // @return {svgedit.draw.Drawing}
- var getCurrentDrawing = canvas.getCurrentDrawing = function() {
- return canvas.current_drawing_;
- };
- // Float displaying the current zoom level (1 = 100%, .5 = 50%, etc)
- var current_zoom = 1;
- // pointer to current group (for in-group editing)
- var current_group = null;
- // Object containing data for the currently selected styles
- var all_properties = {
- shape: {
- fill: "#" + curConfig.initFill.color,
- fill_paint: null,
- fill_opacity: curConfig.initFill.opacity,
- stroke: "#" + curConfig.initStroke.color,
- stroke_paint: null,
- stroke_opacity: curConfig.initStroke.opacity,
- stroke_width: curConfig.initStroke.width,
- stroke_dasharray: 'none',
- stroke_linejoin: 'miter',
- stroke_linecap: 'butt',
- opacity: curConfig.initOpacity
- }
- };
- all_properties.text = $.extend(true, {}, all_properties.shape);
- $.extend(all_properties.text, {
- fill: "#000000",
- stroke_width: 0,
- font_size: 24,
- font_family: 'serif'
- });
- // Current shape style properties
- var cur_shape = all_properties.shape;
- // Array with all the currently selected elements
- // default size of 1 until it needs to grow bigger
- var selectedElements = new Array(1);
- // Function: addSvgElementFromJson
- // Create a new SVG element based on the given object keys/values and add it to the current layer
- // The element will be ran through cleanupElement before being returned
- //
- // Parameters:
- // data - Object with the following keys/values:
- // * element - tag name of the SVG element to create
- // * attr - Object with attributes key-values to assign to the new element
- // * curStyles - Boolean indicating that current style attributes should be applied first
- //
- // Returns: The new element
- var addSvgElementFromJson = this.addSvgElementFromJson = function(data) {
- var shape = svgedit.utilities.getElem(data.attr.id);
- // if shape is a path but we need to create a rect/ellipse, then remove the path
- var current_layer = getCurrentDrawing().getCurrentLayer();
- if (shape && data.element != shape.tagName) {
- current_layer.removeChild(shape);
- shape = null;
- }
- if (!shape) {
- shape = svgdoc.createElementNS(svgns, data.element);
- if (current_layer) {
- (current_group || current_layer).appendChild(shape);
- }
- }
- if(data.curStyles) {
- svgedit.utilities.assignAttributes(shape, {
- "fill": cur_shape.fill,
- "stroke": cur_shape.stroke,
- "stroke-width": cur_shape.stroke_width,
- "stroke-dasharray": cur_shape.stroke_dasharray,
- "stroke-linejoin": cur_shape.stroke_linejoin,
- "stroke-linecap": cur_shape.stroke_linecap,
- "stroke-opacity": cur_shape.stroke_opacity,
- "fill-opacity": cur_shape.fill_opacity,
- "opacity": cur_shape.opacity / 2,
- "style": "pointer-events:inherit"
- }, 100);
- }
- svgedit.utilities.assignAttributes(shape, data.attr, 100);
- svgedit.utilities.cleanupElement(shape);
- return shape;
- };
- // import svgtransformlist.js
- var getTransformList = canvas.getTransformList = svgedit.transformlist.getTransformList;
- // import from math.js.
- var transformPoint = svgedit.math.transformPoint;
- var matrixMultiply = canvas.matrixMultiply = svgedit.math.matrixMultiply;
- var hasMatrixTransform = canvas.hasMatrixTransform = svgedit.math.hasMatrixTransform;
- var transformListToTransform = canvas.transformListToTransform = svgedit.math.transformListToTransform;
- var snapToAngle = svgedit.math.snapToAngle;
- var getMatrix = svgedit.math.getMatrix;
- // initialize from units.js
- // send in an object implementing the ElementContainer interface (see units.js)
- svgedit.units.init({
- getBaseUnit: function() { return curConfig.baseUnit; },
- getElement: svgedit.utilities.getElem,
- getHeight: function() { return svgcontent.getAttribute("height")/current_zoom; },
- getWidth: function() { return svgcontent.getAttribute("width")/current_zoom; },
- getRoundDigits: function() { return save_options.round_digits; }
- });
- // import from units.js
- var convertToNum = canvas.convertToNum = svgedit.units.convertToNum;
- // import from svgutils.js
- svgedit.utilities.init({
- getDOMDocument: function() { return svgdoc; },
- getDOMContainer: function() { return container; },
- getSVGRoot: function() { return svgroot; },
- // TODO: replace this mostly with a way to get the current drawing.
- getSelectedElements: function() { return selectedElements; },
- getSVGContent: function() { return svgcontent; }
- });
- var getUrlFromAttr = canvas.getUrlFromAttr = svgedit.utilities.getUrlFromAttr;
- var getHref = canvas.getHref = svgedit.utilities.getHref;
- var setHref = canvas.setHref = svgedit.utilities.setHref;
- var getPathBBox = svgedit.utilities.getPathBBox;
- var getBBox = canvas.getBBox = svgedit.utilities.getBBox;
- var getRotationAngle = canvas.getRotationAngle = svgedit.utilities.getRotationAngle;
- var getElem = canvas.getElem = svgedit.utilities.getElem;
- var assignAttributes = canvas.assignAttributes = svgedit.utilities.assignAttributes;
- var cleanupElement = this.cleanupElement = svgedit.utilities.cleanupElement;
- // import from sanitize.js
- var nsMap = svgedit.sanitize.getNSMap();
- var sanitizeSvg = canvas.sanitizeSvg = svgedit.sanitize.sanitizeSvg;
- // import from history.js
- var MoveElementCommand = svgedit.history.MoveElementCommand;
- var InsertElementCommand = svgedit.history.InsertElementCommand;
- var RemoveElementCommand = svgedit.history.RemoveElementCommand;
- var ChangeElementCommand = svgedit.history.ChangeElementCommand;
- var BatchCommand = svgedit.history.BatchCommand;
- // Implement the svgedit.history.HistoryEventHandler interface.
- canvas.undoMgr = new svgedit.history.UndoManager({
- handleHistoryEvent: function(eventType, cmd) {
- var EventTypes = svgedit.history.HistoryEventTypes;
- // TODO: handle setBlurOffsets.
- if (eventType == EventTypes.BEFORE_UNAPPLY || eventType == EventTypes.BEFORE_APPLY) {
- canvas.clearSelection();
- } else if (eventType == EventTypes.AFTER_APPLY || eventType == EventTypes.AFTER_UNAPPLY) {
- var elems = cmd.elements();
- canvas.pathActions.clear();
- call("changed", elems);
-
- var cmdType = cmd.type();
- var isApply = (eventType == EventTypes.AFTER_APPLY);
- if (cmdType == MoveElementCommand.type()) {
- var parent = isApply ? cmd.newParent : cmd.oldParent;
- if (parent == svgcontent) {
- canvas.identifyLayers();
- }
- } else if (cmdType == InsertElementCommand.type() ||
- cmdType == RemoveElementCommand.type()) {
- if (cmd.parent == svgcontent) {
- canvas.identifyLayers();
- }
- if (cmdType == InsertElementCommand.type()) {
- if (isApply) restoreRefElems(cmd.elem);
- } else {
- if (!isApply) restoreRefElems(cmd.elem);
- }
-
- if(cmd.elem.tagName === 'use') {
- setUseData(cmd.elem);
- }
- } else if (cmdType == ChangeElementCommand.type()) {
- // if we are changing layer names, re-identify all layers
- if (cmd.elem.tagName == "title" && cmd.elem.parentNode.parentNode == svgcontent) {
- canvas.identifyLayers();
- }
- var values = isApply ? cmd.newValues : cmd.oldValues;
- // If stdDeviation was changed, update the blur.
- if (values["stdDeviation"]) {
- canvas.setBlurOffsets(cmd.elem.parentNode, values["stdDeviation"]);
- }
-
- // Remove & Re-add hack for Webkit (issue 775)
- if(cmd.elem.tagName === 'use' && svgedit.browser.isWebkit()) {
- var elem = cmd.elem;
- if(!elem.getAttribute('x') && !elem.getAttribute('y')) {
- var parent = elem.parentNode;
- var sib = elem.nextSibling;
- parent.removeChild(elem);
- parent.insertBefore(elem, sib);
- }
- }
- }
- }
- }
- });
- var addCommandToHistory = function(cmd) {
- canvas.undoMgr.addCommandToHistory(cmd);
- };
- // import from select.js
- svgedit.select.init(curConfig, {
- createSVGElement: function(jsonMap) { return canvas.addSvgElementFromJson(jsonMap); },
- svgRoot: function() { return svgroot; },
- svgContent: function() { return svgcontent; },
- currentZoom: function() { return current_zoom; },
- // TODO(codedread): Remove when getStrokedBBox() has been put into svgutils.js.
- getStrokedBBox: function(elems) { return canvas.getStrokedBBox([elems]); }
- });
- // this object manages selectors for us
- var selectorManager = this.selectorManager = svgedit.select.getSelectorManager();
- // Import from path.js
- svgedit.path.init({
- getCurrentZoom: function() { return current_zoom; },
- getSVGRoot: function() { return svgroot; }
- });
- // Function: snapToGrid
- // round value to for snapping
- // NOTE: This function did not move to svgutils.js since it depends on curConfig.
- svgedit.utilities.snapToGrid = function(value){
- var stepSize = curConfig.snappingStep;
- var unit = curConfig.baseUnit;
- if(unit !== "px") {
- stepSize *= svgedit.units.getTypeMap()[unit];
- }
- value = Math.round(value/stepSize)*stepSize;
- return value;
- };
- var snapToGrid = svgedit.utilities.snapToGrid;
- // Interface strings, usually for title elements
- var uiStrings = {
- "exportNoBlur": "Blurred elements will appear as un-blurred",
- "exportNoforeignObject": "foreignObject elements will not appear",
- "exportNoDashArray": "Strokes will appear filled",
- "exportNoText": "Text may not appear as expected"
- };
- var visElems = 'a,circle,ellipse,foreignObject,g,image,line,path,polygon,polyline,rect,svg,text,tspan,use';
- var ref_attrs = ["clip-path", "fill", "filter", "marker-end", "marker-mid", "marker-start", "mask", "stroke"];
- var elData = $.data;
- // Animation element to change the opacity of any newly created element
- var opac_ani = document.createElementNS(svgns, 'animate');
- $(opac_ani).attr({
- attributeName: 'opacity',
- begin: 'indefinite',
- dur: 1,
- fill: 'freeze'
- }).appendTo(svgroot);
- var restoreRefElems = function(elem) {
- // Look for missing reference elements, restore any found
- var attrs = $(elem).attr(ref_attrs);
- for(var o in attrs) {
- var val = attrs[o];
- if (val && val.indexOf('url(') === 0) {
- var id = getUrlFromAttr(val).substr(1);
- var ref = getElem(id);
- if(!ref) {
- findDefs().appendChild(removedElements[id]);
- delete removedElements[id];
- }
- }
- }
-
- var childs = elem.getElementsByTagName('*');
-
- if(childs.length) {
- for(var i = 0, l = childs.length; i < l; i++) {
- restoreRefElems(childs[i]);
- }
- }
- };
- (function() {
- // TODO For Issue 208: this is a start on a thumbnail
- // var svgthumb = svgdoc.createElementNS(svgns, "use");
- // svgthumb.setAttribute('width', '100');
- // svgthumb.setAttribute('height', '100');
- // svgedit.utilities.setHref(svgthumb, '#svgcontent');
- // svgroot.appendChild(svgthumb);
- })();
- // Object to contain image data for raster images that were found encodable
- var encodableImages = {},
-
- // String with image URL of last loadable image
- last_good_img_url = curConfig.imgPath + 'logo.png',
-
- // Array with current disabled elements (for in-group editing)
- disabled_elems = [],
-
- // Object with save options
- save_options = {round_digits: 5},
-
- // Boolean indicating whether or not a draw action has been started
- started = false,
-
- // String with an element's initial transform attribute value
- start_transform = null,
-
- // String indicating the current editor mode
- current_mode = "select",
-
- // String with the current direction in which an element is being resized
- current_resize_mode = "none",
-
- // Object with IDs for imported files, to see if one was already added
- import_ids = {};
- // Current text style properties
- var cur_text = all_properties.text,
-
- // Current general properties
- cur_properties = cur_shape,
-
- // Array with selected elements' Bounding box object
- // selectedBBoxes = new Array(1),
-
- // The DOM element that was just selected
- justSelected = null,
-
- // DOM element for selection rectangle drawn by the user
- rubberBox = null,
-
- // Array of current BBoxes (still needed?)
- curBBoxes = [],
-
- // Object to contain all included extensions
- extensions = {},
-
- // Canvas point for the most recent right click
- lastClickPoint = null,
-
- // Map of deleted reference elements
- removedElements = {}
- // Clipboard for cut, copy&pasted elements
- canvas.clipBoard = [];
- // Should this return an array by default, so extension results aren't overwritten?
- var runExtensions = this.runExtensions = function(action, vars, returnArray) {
- var result = false;
- if(returnArray) result = [];
- $.each(extensions, function(name, opts) {
- if(action in opts) {
- if(returnArray) {
- result.push(opts[action](vars))
- } else {
- result = opts[action](vars);
- }
- }
- });
- return result;
- }
- // Function: addExtension
- // Add an extension to the editor
- //
- // Parameters:
- // name - String with the ID of the extension
- // ext_func - Function supplied by the extension with its data
- this.addExtension = function(name, ext_func) {
- if(!(name in extensions)) {
- // Provide private vars/funcs here. Is there a better way to do this?
-
- if($.isFunction(ext_func)) {
- var ext = ext_func($.extend(canvas.getPrivateMethods(), {
- svgroot: svgroot,
- svgcontent: svgcontent,
- nonce: getCurrentDrawing().getNonce(),
- selectorManager: selectorManager
- }));
- } else {
- var ext = ext_func;
- }
- extensions[name] = ext;
- call("extension_added", ext);
- } else {
- console.log('Cannot add extension "' + name + '", an extension by that name already exists"');
- }
- };
-
- // This method rounds the incoming value to the nearest value based on the current_zoom
- var round = this.round = function(val) {
- return parseInt(val*current_zoom)/current_zoom;
- };
- // This method sends back an array or a NodeList full of elements that
- // intersect the multi-select rubber-band-box on the current_layer only.
- //
- // Since the only browser that supports the SVG DOM getIntersectionList is Opera,
- // we need to provide an implementation here. We brute-force it for now.
- //
- // Reference:
- // Firefox does not implement getIntersectionList(), see https://bugzilla.mozilla.org/show_bug.cgi?id=501421
- // Webkit does not implement getIntersectionList(), see https://bugs.webkit.org/show_bug.cgi?id=11274
- var getIntersectionList = this.getIntersectionList = function(rect) {
- if (rubberBox == null) { return null; }
- var parent = current_group || getCurrentDrawing().getCurrentLayer();
-
- if(!curBBoxes.length) {
- // Cache all bboxes
- curBBoxes = getVisibleElementsAndBBoxes(parent);
- }
-
- var resultList = null;
- try {
- resultList = parent.getIntersectionList(rect, null);
- } catch(e) { }
- if (resultList == null || typeof(resultList.item) != "function") {
- resultList = [];
-
- if(!rect) {
- var rubberBBox = rubberBox.getBBox();
- var bb = {};
-
- for(var o in rubberBBox) {
- bb[o] = rubberBBox[o] / current_zoom;
- }
- rubberBBox = bb;
-
- } else {
- var rubberBBox = rect;
- }
- var i = curBBoxes.length;
- while (i--) {
- if(!rubberBBox.width || !rubberBBox.width) continue;
- if (svgedit.math.rectsIntersect(rubberBBox, curBBoxes[i].bbox)) {
- resultList.push(curBBoxes[i].elem);
- }
- }
- }
- // addToSelection expects an array, but it's ok to pass a NodeList
- // because using square-bracket notation is allowed:
- // http://www.w3.org/TR/DOM-Level-2-Core/ecma-script-binding.html
- return resultList;
- };
- // TODO(codedread): Migrate this into svgutils.js
- // Function: getStrokedBBox
- // Get the bounding box for one or more stroked and/or transformed elements
- //
- // Parameters:
- // elems - Array with DOM elements to check
- //
- // Returns:
- // A single bounding box object
- getStrokedBBox = this.getStrokedBBox = function(elems) {
- if(!elems) elems = getVisibleElements();
- if(!elems.length) return false;
- // Make sure the expected BBox is returned if the element is a group
- var getCheckedBBox = function(elem) {
-
- try {
- // TODO: Fix issue with rotated groups. Currently they work
- // fine in FF, but not in other browsers (same problem mentioned
- // in Issue 339 comment #2).
-
- var bb = svgedit.utilities.getBBox(elem);
-
- var angle = svgedit.utilities.getRotationAngle(elem);
- if ((angle && angle % 90) ||
- svgedit.math.hasMatrixTransform(svgedit.transformlist.getTransformList(elem))) {
- // Accurate way to get BBox of rotated element in Firefox:
- // Put element in group and get its BBox
-
- var good_bb = false;
-
- // Get the BBox from the raw path for these elements
- var elemNames = ['ellipse','path','line','polyline','polygon'];
- if(elemNames.indexOf(elem.tagName) >= 0) {
- bb = good_bb = canvas.convertToPath(elem, true);
- } else if(elem.tagName == 'rect') {
- // Look for radius
- var rx = elem.getAttribute('rx');
- var ry = elem.getAttribute('ry');
- if(rx || ry) {
- bb = good_bb = canvas.convertToPath(elem, true);
- }
- }
-
- if(!good_bb) {
- // Must use clone else FF freaks out
- var clone = elem.cloneNode(true);
- var g = document.createElementNS(svgns, "g");
- var parent = elem.parentNode;
- parent.appendChild(g);
- g.appendChild(clone);
- bb = svgedit.utilities.bboxToObj(g.getBBox());
- parent.removeChild(g);
- }
-
- // Old method: Works by giving the rotated BBox,
- // this is (unfortunately) what Opera and Safari do
- // natively when getting the BBox of the parent group
- // var angle = angle * Math.PI / 180.0;
- // var rminx = Number.MAX_VALUE, rminy = Number.MAX_VALUE,
- // rmaxx = Number.MIN_VALUE, rmaxy = Number.MIN_VALUE;
- // var cx = round(bb.x + bb.width/2),
- // cy = round(bb.y + bb.height/2);
- // var pts = [ [bb.x - cx, bb.y - cy],
- // [bb.x + bb.width - cx, bb.y - cy],
- // [bb.x + bb.width - cx, bb.y + bb.height - cy],
- // [bb.x - cx, bb.y + bb.height - cy] ];
- // var j = 4;
- // while (j--) {
- // var x = pts[j][0],
- // y = pts[j][1],
- // r = Math.sqrt( x*x + y*y );
- // var theta = Math.atan2(y,x) + angle;
- // x = round(r * Math.cos(theta) + cx);
- // y = round(r * Math.sin(theta) + cy);
- //
- // // now set the bbox for the shape after it's been rotated
- // if (x < rminx) rminx = x;
- // if (y < rminy) rminy = y;
- // if (x > rmaxx) rmaxx = x;
- // if (y > rmaxy) rmaxy = y;
- // }
- //
- // bb.x = rminx;
- // bb.y = rminy;
- // bb.width = rmaxx - rminx;
- // bb.height = rmaxy - rminy;
- }
- return bb;
- } catch(e) {
- console.log(elem, e);
- return null;
- }
- };
- var full_bb;
- $.each(elems, function() {
- if(full_bb) return;
- if(!this.parentNode) return;
- full_bb = getCheckedBBox(this);
- });
-
- // This shouldn't ever happen...
- if(full_bb == null) return null;
-
- // full_bb doesn't include the stoke, so this does no good!
- // if(elems.length == 1) return full_bb;
-
- var max_x = full_bb.x + full_bb.width;
- var max_y = full_bb.y + full_bb.height;
- var min_x = full_bb.x;
- var min_y = full_bb.y;
-
- // FIXME: same re-creation problem with this function as getCheckedBBox() above
- var getOffset = function(elem) {
- var sw = elem.getAttribute("stroke-width");
- var offset = 0;
- if (elem.getAttribute("stroke") != "none" && !isNaN(sw)) {
- offset += sw/2;
- }
- return offset;
- }
- var bboxes = [];
- $.each(elems, function(i, elem) {
- var cur_bb = getCheckedBBox(elem);
- if(cur_bb) {
- var offset = getOffset(elem);
- min_x = Math.min(min_x, cur_bb.x - offset);
- min_y = Math.min(min_y, cur_bb.y - offset);
- bboxes.push(cur_bb);
- }
- });
-
- full_bb.x = min_x;
- full_bb.y = min_y;
-
- $.each(elems, function(i, elem) {
- var cur_bb = bboxes[i];
- // ensure that elem is really an element node
- if (cur_bb && elem.nodeType == 1) {
- var offset = getOffset(elem);
- max_x = Math.max(max_x, cur_bb.x + cur_bb.width + offset);
- max_y = Math.max(max_y, cur_bb.y + cur_bb.height + offset);
- }
- });
-
- full_bb.width = max_x - min_x;
- full_bb.height = max_y - min_y;
- return full_bb;
- }
- // Function: getVisibleElements
- // Get all elements that have a BBox (excludes <defs>, <title>, etc).
- // Note that 0-opacity, off-screen etc elements are still considered "visible"
- // for this function
- //
- // Parameters:
- // parent - The parent DOM element to search within
- //
- // Returns:
- // An array with all "visible" elements.
- var getVisibleElements = this.getVisibleElements = function(parent) {
- if(!parent) parent = $(svgcontent).children(); // Prevent layers from being included
-
- var contentElems = [];
- $(parent).children().each(function(i, elem) {
- try {
- if (elem.getBBox()) {
- contentElems.push(elem);
- }
- } catch(e) {}
- });
- return contentElems.reverse();
- };
- // Function: getVisibleElementsAndBBoxes
- // Get all elements that have a BBox (excludes <defs>, <title>, etc).
- // Note that 0-opacity, off-screen etc elements are still considered "visible"
- // for this function
- //
- // Parameters:
- // parent - The parent DOM element to search within
- //
- // Returns:
- // An array with objects that include:
- // * elem - The element
- // * bbox - The element's BBox as retrieved from getStrokedBBox
- var getVisibleElementsAndBBoxes = this.getVisibleElementsAndBBoxes = function(parent) {
- if(!parent) parent = $(svgcontent).children(); // Prevent layers from being included
-
- var contentElems = [];
- $(parent).children().each(function(i, elem) {
- try {
- if (elem.getBBox()) {
- contentElems.push({'elem':elem, 'bbox':getStrokedBBox([elem])});
- }
- } catch(e) {}
- });
- return contentElems.reverse();
- };
- // Function: groupSvgElem
- // Wrap an SVG element into a group element, mark the group as 'gsvg'
- //
- // Parameters:
- // elem - SVG element to wrap
- var groupSvgElem = this.groupSvgElem = function(elem) {
- var g = document.createElementNS(svgns, "g");
- elem.parentNode.replaceChild(g, elem);
- $(g).append(elem).data('gsvg', elem)[0].id = getNextId();
- }
- // Function: copyElem
- // Create a clone of an element, updating its ID and its children's IDs when needed
- //
- // Parameters:
- // el - DOM element to clone
- //
- // Returns: The cloned element
- var copyElem = function(el) {
- // manually create a copy of the element
- var new_el = document.createElementNS(el.namespaceURI, el.nodeName);
- $.each(el.attributes, function(i, attr) {
- if (attr.localName != '-moz-math-font-style') {
- new_el.setAttributeNS(attr.namespaceURI, attr.nodeName, attr.nodeValue);
- }
- });
- // set the copied element's new id
- new_el.removeAttribute("id");
- new_el.id = getNextId();
-
- // Opera's "d" value needs to be reset for Opera/Win/non-EN
- // Also needed for webkit (else does not keep curved segments on clone)
- if(svgedit.browser.isWebkit() && el.nodeName == 'path') {
- var fixed_d = pathActions.convertPath(el);
- new_el.setAttribute('d', fixed_d);
- }
- // now create copies of all children
- $.each(el.childNodes, function(i, child) {
- switch(child.nodeType) {
- case 1: // element node
- new_el.appendChild(copyElem(child));
- break;
- case 3: // text node
- new_el.textContent = child.nodeValue;
- break;
- default:
- break;
- }
- });
-
- if($(el).data('gsvg')) {
- $(new_el).data('gsvg', new_el.firstChild);
- } else if($(el).data('symbol')) {
- var ref = $(el).data('symbol');
- $(new_el).data('ref', ref).data('symbol', ref);
- }
-
- else if(new_el.tagName == 'image') {
- preventClickDefault(new_el);
- }
- return new_el;
- };
- // Set scope for these functions
- var getId, getNextId, call;
- (function(c) {
- // Object to contain editor event names and callback functions
- var events = {};
- getId = c.getId = function() { return getCurrentDrawing().getId(); };
- getNextId = c.getNextId = function() { return getCurrentDrawing().getNextId(); };
-
- // Function: call
- // Run the callback function associated with the given event
- //
- // Parameters:
- // event - String with the event name
- // arg - Argument to pass through to the callback function
- call = c.call = function(event, arg) {
- if (events[event]) {
- return events[event](this, arg);
- }
- };
-
- // Function: bind
- // Attaches a callback function to an event
- //
- // Parameters:
- // event - String indicating the name of the event
- // f - The callback function to bind to the event
- //
- // Return:
- // The previous event
- c.bind = function(event, f) {
- var old = events[event];
- events[event] = f;
- return old;
- };
-
- }(canvas));
- // Function: canvas.prepareSvg
- // Runs the SVG Document through the sanitizer and then updates its paths.
- //
- // Parameters:
- // newDoc - The SVG DOM document
- this.prepareSvg = function(newDoc) {
- this.sanitizeSvg(newDoc.documentElement);
- // convert paths into absolute commands
- var paths = newDoc.getElementsByTagNameNS(svgns, "path");
- for (var i = 0, len = paths.length; i < len; ++i) {
- var path = paths[i];
- path.setAttribute('d', pathActions.convertPath(path));
- pathActions.fixEnd(path);
- }
- };
- // Function getRefElem
- // Get the reference element associated with the given attribute value
- //
- // Parameters:
- // attrVal - The attribute value as a string
- var getRefElem = this.getRefElem = function(attrVal) {
- return getElem(getUrlFromAttr(attrVal).substr(1));
- }
- // Function: ffClone
- // Hack for Firefox bugs where text element features aren't updated or get
- // messed up. See issue 136 and issue 137.
- // This function clones the element and re-selects it
- // TODO: Test for this bug on load and add it to "support" object instead of
- // browser sniffing
- //
- // Parameters:
- // elem - The (text) DOM element to clone
- var ffClone = function(elem) {
- if(!svgedit.browser.isGecko()) return elem;
- var clone = elem.cloneNode(true)
- elem.parentNode.insertBefore(clone, elem);
- elem.parentNode.removeChild(elem);
- selectorManager.releaseSelector(elem);
- selectedElements[0] = clone;
- selectorManager.requestSelector(clone).showGrips(true);
- return clone;
- }
- // this.each is deprecated, if any extension used this it can be recreated by doing this:
- // $(canvas.getRootElem()).children().each(...)
- // this.each = function(cb) {
- // $(svgroot).children().each(cb);
- // };
- // Function: setRotationAngle
- // Removes any old rotations if present, prepends a new rotation at the
- // transformed center
- //
- // Parameters:
- // val - The new rotation angle in degrees
- // preventUndo - Boolean indicating whether the action should be undoable or not
- this.setRotationAngle = function(val, preventUndo) {
- // ensure val is the proper type
- val = parseFloat(val);
- var elem = selectedElements[0];
- var oldTransform = elem.getAttribute("transform");
- var bbox = svgedit.utilities.getBBox(elem);
- var cx = bbox.x+bbox.width/2, cy = bbox.y+bbox.height/2;
- var tlist = getTransformList(elem);
-
- // only remove the real rotational transform if present (i.e. at index=0)
- if (tlist.numberOfItems > 0) {
- var xform = tlist.getItem(0);
- if (xform.type == 4) {
- tlist.removeItem(0);
- }
- }
- // find R_nc and insert it
- if (val != 0) {
- var center = transformPoint(cx,cy,transformListToTransform(tlist).matrix);
- var R_nc = svgroot.createSVGTransform();
- R_nc.setRotate(val, center.x, center.y);
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(R_nc, 0);
- } else {
- tlist.appendItem(R_nc);
- }
- }
- else if (tlist.numberOfItems == 0) {
- elem.removeAttribute("transform");
- }
-
- if (!preventUndo) {
- // we need to undo it, then redo it so it can be undo-able! :)
- // TODO: figure out how to make changes to transform list undo-able cross-browser?
- var newTransform = elem.getAttribute("transform");
- elem.setAttribute("transform", oldTransform);
- changeSelectedAttribute("transform",newTransform,selectedElements);
- call("changed", selectedElements);
- }
- var pointGripContainer = getElem("pathpointgrip_container");
- // if(elem.nodeName == "path" && pointGripContainer) {
- // pathActions.setPointContainerTransform(elem.getAttribute("transform"));
- // }
- var selector = selectorManager.requestSelector(selectedElements[0]);
- selector.resize();
- selector.updateGripCursors(val);
- };
- // Function: recalculateAllSelectedDimensions
- // Runs recalculateDimensions on the selected elements,
- // adding the changes to a single batch command
- var recalculateAllSelectedDimensions = this.recalculateAllSelectedDimensions = function() {
- var text = (current_resize_mode == "none" ? "position" : "size");
- var batchCmd = new BatchCommand(text);
- var i = selectedElements.length;
- while(i--) {
- var elem = selectedElements[i];
- // if(getRotationAngle(elem) && !hasMatrixTransform(getTransformList(elem))) continue;
- var cmd = recalculateDimensions(elem);
- if (cmd) {
- batchCmd.addSubCommand(cmd);
- }
- }
- if (!batchCmd.isEmpty()) {
- addCommandToHistory(batchCmd);
- call("changed", selectedElements);
- }
- };
- // this is how we map paths to our preferred relative segment types
- var pathMap = [0, 'z', 'M', 'm', 'L', 'l', 'C', 'c', 'Q', 'q', 'A', 'a',
- 'H', 'h', 'V', 'v', 'S', 's', 'T', 't'];
-
- // Debug tool to easily see the current matrix in the browser's console
- var logMatrix = function(m) {
- console.log([m.a,m.b,m.c,m.d,m.e,m.f]);
- };
- // Function: remapElement
- // Applies coordinate changes to an element based on the given matrix
- //
- // Parameters:
- // selected - DOM element to be changed
- // changes - Object with changes to be remapped
- // m - Matrix object to use for remapping coordinates
- var remapElement = this.remapElement = function(selected,changes,m) {
- var remap = function(x,y) { return transformPoint(x,y,m); },
- scalew = function(w) { return m.a*w; },
- scaleh = function(h) { return m.d*h; },
- doSnapping = curConfig.gridSnapping && selected.parentNode.parentNode.localName === "svg",
- finishUp = function() {
- if(doSnapping) for(var o in changes) changes[o] = snapToGrid(changes[o]);
- assignAttributes(selected, changes, 1000, true);
- }
- box = svgedit.utilities.getBBox(selected);
-
- for(var i = 0; i < 2; i++) {
- var type = i === 0 ? 'fill' : 'stroke';
- var attrVal = selected.getAttribute(type);
- if(attrVal && attrVal.indexOf('url(') === 0) {
- if(m.a < 0 || m.d < 0) {
- var grad = getRefElem(attrVal);
- var newgrad = grad.cloneNode(true);
-
- if(m.a < 0) {
- //flip x
- var x1 = newgrad.getAttribute('x1');
- var x2 = newgrad.getAttribute('x2');
- newgrad.setAttribute('x1', -(x1 - 1));
- newgrad.setAttribute('x2', -(x2 - 1));
- }
-
- if(m.d < 0) {
- //flip y
- var y1 = newgrad.getAttribute('y1');
- var y2 = newgrad.getAttribute('y2');
- newgrad.setAttribute('y1', -(y1 - 1));
- newgrad.setAttribute('y2', -(y2 - 1));
- }
- newgrad.id = getNextId();
- findDefs().appendChild(newgrad);
- selected.setAttribute(type, 'url(#' + newgrad.id + ')');
- }
-
- // Not really working :(
- // if(selected.tagName === 'path') {
- // reorientGrads(selected, m);
- // }
- }
- }
- var elName = selected.tagName;
- if(elName === "g" || elName === "text" || elName === "use") {
- // if it was a translate, then just update x,y
- if (m.a == 1 && m.b == 0 && m.c == 0 && m.d == 1 &&
- (m.e != 0 || m.f != 0) )
- {
- // [T][M] = [M][T']
- // therefore [T'] = [M_inv][T][M]
- var existing = transformListToTransform(selected).matrix,
- t_new = matrixMultiply(existing.inverse(), m, existing);
- changes.x = parseFloat(changes.x) + t_new.e;
- changes.y = parseFloat(changes.y) + t_new.f;
- }
- else {
- // we just absorb all matrices into the element and don't do any remapping
- var chlist = getTransformList(selected);
- var mt = svgroot.createSVGTransform();
- mt.setMatrix(matrixMultiply(transformListToTransform(chlist).matrix,m));
- chlist.clear();
- chlist.appendItem(mt);
- }
- }
-
- // now we have a set of changes and an applied reduced transform list
- // we apply the changes directly to the DOM
- switch (elName)
- {
- case "foreignObject":
- case "rect":
- case "image":
-
- // Allow images to be inverted (give them matrix when flipped)
- if(elName === 'image' && (m.a < 0 || m.d < 0)) {
- // Convert to matrix
- var chlist = getTransformList(selected);
- var mt = svgroot.createSVGTransform();
- mt.setMatrix(matrixMultiply(transformListToTransform(chlist).matrix,m));
- chlist.clear();
- chlist.appendItem(mt);
- } else {
- var pt1 = remap(changes.x,changes.y);
-
- changes.width = scalew(changes.width);
- changes.height = scaleh(changes.height);
-
- changes.x = pt1.x + Math.min(0,changes.width);
- changes.y = pt1.y + Math.min(0,changes.height);
- changes.width = Math.abs(changes.width);
- changes.height = Math.abs(changes.height);
- }
- finishUp();
- break;
- case "ellipse":
- var c = remap(changes.cx,changes.cy);
- changes.cx = c.x;
- changes.cy = c.y;
- changes.rx = scalew(changes.rx);
- changes.ry = scaleh(changes.ry);
-
- changes.rx = Math.abs(changes.rx);
- changes.ry = Math.abs(changes.ry);
- finishUp();
- break;
- case "circle":
- var c = remap(changes.cx,changes.cy);
- changes.cx = c.x;
- changes.cy = c.y;
- // take the minimum of the new selected box's dimensions for the new circle radius
- var tbox = svgedit.math.transformBox(box.x, box.y, box.width, box.height, m);
- var w = tbox.tr.x - tbox.tl.x, h = tbox.bl.y - tbox.tl.y;
- changes.r = Math.min(w/2, h/2);
- if(changes.r) changes.r = Math.abs(changes.r);
- finishUp();
- break;
- case "line":
- var pt1 = remap(changes.x1,changes.y1),
- pt2 = remap(changes.x2,changes.y2);
- changes.x1 = pt1.x;
- changes.y1 = pt1.y;
- changes.x2 = pt2.x;
- changes.y2 = pt2.y;
-
- case "text":
- case "use":
- finishUp();
- break;
- case "g":
- var gsvg = $(selected).data('gsvg');
- if(gsvg) {
- assignAttributes(gsvg, changes, 1000, true);
- }
- break;
- case "polyline":
- case "polygon":
- var len = changes.points.length;
- for (var i = 0; i < len; ++i) {
- var pt = changes.points[i];
- pt = remap(pt.x,pt.y);
- changes.points[i].x = pt.x;
- changes.points[i].y = pt.y;
- }
- var len = changes.points.length;
- var pstr = "";
- for (var i = 0; i < len; ++i) {
- var pt = changes.points[i];
- pstr += pt.x + "," + pt.y + " ";
- }
- selected.setAttribute("points", pstr);
- break;
- case "path":
-
- var segList = selected.pathSegList;
- var len = segList.numberOfItems;
- changes.d = new Array(len);
- for (var i = 0; i < len; ++i) {
- var seg = segList.getItem(i);
- changes.d[i] = {
- type: seg.pathSegType,
- x: seg.x,
- y: seg.y,
- x1: seg.x1,
- y1: seg.y1,
- x2: seg.x2,
- y2: seg.y2,
- r1: seg.r1,
- r2: seg.r2,
- angle: seg.angle,
- largeArcFlag: seg.largeArcFlag,
- sweepFlag: seg.sweepFlag
- };
- }
-
- var len = changes.d.length,
- firstseg = changes.d[0],
- currentpt = remap(firstseg.x,firstseg.y);
- changes.d[0].x = currentpt.x;
- changes.d[0].y = currentpt.y;
- for (var i = 1; i < len; ++i) {
- var seg = changes.d[i];
- var type = seg.type;
- // if absolute or first segment, we want to remap x, y, x1, y1, x2, y2
- // if relative, we want to scalew, scaleh
- if (type % 2 == 0) { // absolute
- var thisx = (seg.x != undefined) ? seg.x : currentpt.x, // for V commands
- thisy = (seg.y != undefined) ? seg.y : currentpt.y, // for H commands
- pt = remap(thisx,thisy),
- pt1 = remap(seg.x1,seg.y1),
- pt2 = remap(seg.x2,seg.y2);
- seg.x = pt.x;
- seg.y = pt.y;
- seg.x1 = pt1.x;
- seg.y1 = pt1.y;
- seg.x2 = pt2.x;
- seg.y2 = pt2.y;
- seg.r1 = scalew(seg.r1),
- seg.r2 = scaleh(seg.r2);
- }
- else { // relative
- seg.x = scalew(seg.x);
- seg.y = scaleh(seg.y);
- seg.x1 = scalew(seg.x1);
- seg.y1 = scaleh(seg.y1);
- seg.x2 = scalew(seg.x2);
- seg.y2 = scaleh(seg.y2);
- seg.r1 = scalew(seg.r1),
- seg.r2 = scaleh(seg.r2);
- }
- } // for each segment
-
- var dstr = "";
- var len = changes.d.length;
- for (var i = 0; i < len; ++i) {
- var seg = changes.d[i];
- var type = seg.type;
- dstr += pathMap[type];
- switch(type) {
- case 13: // relative horizontal line (h)
- case 12: // absolute horizontal line (H)
- dstr += seg.x + " ";
- break;
- case 15: // relative vertical line (v)
- case 14: // absolute vertical line (V)
- dstr += seg.y + " ";
- break;
- case 3: // relative move (m)
- case 5: // relative line (l)
- case 19: // relative smooth quad (t)
- case 2: // absolute move (M)
- case 4: // absolute line (L)
- case 18: // absolute smooth quad (T)
- dstr += seg.x + "," + seg.y + " ";
- break;
- case 7: // relative cubic (c)
- case 6: // absolute cubic (C)
- dstr += seg.x1 + "," + seg.y1 + " " + seg.x2 + "," + seg.y2 + " " +
- seg.x + "," + seg.y + " ";
- break;
- case 9: // relative quad (q)
- case 8: // absolute quad (Q)
- dstr += seg.x1 + "," + seg.y1 + " " + seg.x + "," + seg.y + " ";
- break;
- case 11: // relative elliptical arc (a)
- case 10: // absolute elliptical arc (A)
- dstr += seg.r1 + "," + seg.r2 + " " + seg.angle + " " + (+seg.largeArcFlag) +
- " " + (+seg.sweepFlag) + " " + seg.x + "," + seg.y + " ";
- break;
- case 17: // relative smooth cubic (s)
- case 16: // absolute smooth cubic (S)
- dstr += seg.x2 + "," + seg.y2 + " " + seg.x + "," + seg.y + " ";
- break;
- }
- }
- selected.setAttribute("d", dstr);
- break;
- }
- };
- // Function: updateClipPath
- // Updates a <clipPath>s values based on the given translation of an element
- //
- // Parameters:
- // attr - The clip-path attribute value with the clipPath's ID
- // tx - The translation's x value
- // ty - The translation's y value
- var updateClipPath = function(attr, tx, ty) {
- var path = getRefElem(attr).firstChild;
-
- var cp_xform = getTransformList(path);
-
- var newxlate = svgroot.createSVGTransform();
- newxlate.setTranslate(tx, ty);
- cp_xform.appendItem(newxlate);
-
- // Update clipPath's dimensions
- recalculateDimensions(path);
- }
- // Function: recalculateDimensions
- // Decides the course of action based on the element's transform list
- //
- // Parameters:
- // selected - The DOM element to recalculate
- //
- // Returns:
- // Undo command object with the resulting change
- var recalculateDimensions = this.recalculateDimensions = function(selected) {
- if (selected == null) return null;
-
- var tlist = getTransformList(selected);
-
- // remove any unnecessary transforms
- if (tlist && tlist.numberOfItems > 0) {
- var k = tlist.numberOfItems;
- while (k--) {
- var xform = tlist.getItem(k);
- if (xform.type === 0) {
- tlist.removeItem(k);
- }
- // remove identity matrices
- else if (xform.type === 1) {
- if (svgedit.math.isIdentity(xform.matrix)) {
- tlist.removeItem(k);
- }
- }
- // remove zero-degree rotations
- else if (xform.type === 4) {
- if (xform.angle === 0) {
- tlist.removeItem(k);
- }
- }
- }
- // End here if all it has is a rotation
- if(tlist.numberOfItems === 1 && getRotationAngle(selected)) return null;
- }
-
- // if this element had no transforms, we are done
- if (!tlist || tlist.numberOfItems == 0) {
- selected.removeAttribute("transform");
- return null;
- }
-
- // TODO: Make this work for more than 2
- if (tlist) {
- var k = tlist.numberOfItems;
- var mxs = [];
- while (k--) {
- var xform = tlist.getItem(k);
- if (xform.type === 1) {
- mxs.push([xform.matrix, k]);
- } else if(mxs.length) {
- mxs = [];
- }
- }
- if(mxs.length === 2) {
- var m_new = svgroot.createSVGTransformFromMatrix(matrixMultiply(mxs[1][0], mxs[0][0]));
- tlist.removeItem(mxs[0][1]);
- tlist.removeItem(mxs[1][1]);
- tlist.insertItemBefore(m_new, mxs[1][1]);
- }
-
- // combine matrix + translate
- k = tlist.numberOfItems;
- if(k >= 2 && tlist.getItem(k-2).type === 1 && tlist.getItem(k-1).type === 2) {
- var mt = svgroot.createSVGTransform();
-
- var m = matrixMultiply(
- tlist.getItem(k-2).matrix,
- tlist.getItem(k-1).matrix
- );
- mt.setMatrix(m);
- tlist.removeItem(k-2);
- tlist.removeItem(k-2);
- tlist.appendItem(mt);
- }
- }
-
- // If it still has a single [M] or [R][M], return null too (prevents BatchCommand from being returned).
- switch ( selected.tagName ) {
- // Ignore these elements, as they can absorb the [M]
- case 'line':
- case 'polyline':
- case 'polygon':
- case 'path':
- break;
- default:
- if(
- (tlist.numberOfItems === 1 && tlist.getItem(0).type === 1)
- || (tlist.numberOfItems === 2 && tlist.getItem(0).type === 1 && tlist.getItem(0).type === 4)
- ) {
- return null;
- }
- }
-
- // Grouped SVG element
- var gsvg = $(selected).data('gsvg');
-
- // we know we have some transforms, so set up return variable
- var batchCmd = new BatchCommand("Transform");
-
- // store initial values that will be affected by reducing the transform list
- var changes = {}, initial = null, attrs = [];
- switch (selected.tagName)
- {
- case "line":
- attrs = ["x1", "y1", "x2", "y2"];
- break;
- case "circle":
- attrs = ["cx", "cy", "r"];
- break;
- case "ellipse":
- attrs = ["cx", "cy", "rx", "ry"];
- break;
- case "foreignObject":
- case "rect":
- case "image":
- attrs = ["width", "height", "x", "y"];
- break;
- case "use":
- case "text":
- attrs = ["x", "y"];
- break;
- case "polygon":
- case "polyline":
- initial = {};
- initial["points"] = selected.getAttribute("points");
- var list = selected.points;
- var len = list.numberOfItems;
- changes["points"] = new Array(len);
- for (var i = 0; i < len; ++i) {
- var pt = list.getItem(i);
- changes["points"][i] = {x:pt.x,y:pt.y};
- }
- break;
- case "path":
- initial = {};
- initial["d"] = selected.getAttribute("d");
- changes["d"] = selected.getAttribute("d");
- break;
- } // switch on element type to get initial values
-
- if(attrs.length) {
- changes = $(selected).attr(attrs);
- $.each(changes, function(attr, val) {
- changes[attr] = convertToNum(attr, val);
- });
- } else if(gsvg) {
- // GSVG exception
- changes = {
- x: $(gsvg).attr('x') || 0,
- y: $(gsvg).attr('y') || 0
- };
- }
-
- // if we haven't created an initial array in polygon/polyline/path, then
- // make a copy of initial values and include the transform
- if (initial == null) {
- initial = $.extend(true, {}, changes);
- $.each(initial, function(attr, val) {
- initial[attr] = convertToNum(attr, val);
- });
- }
- // save the start transform value too
- initial["transform"] = start_transform ? start_transform : "";
-
- // if it's a regular group, we have special processing to flatten transforms
- if ((selected.tagName == "g" && !gsvg) || selected.tagName == "a") {
- var box = svgedit.utilities.getBBox(selected),
- oldcenter = {x: box.x+box.width/2, y: box.y+box.height/2},
- newcenter = transformPoint(box.x+box.width/2, box.y+box.height/2,
- transformListToTransform(tlist).matrix),
- m = svgroot.createSVGMatrix();
-
-
- // temporarily strip off the rotate and save the old center
- var gangle = getRotationAngle(selected);
- if (gangle) {
- var a = gangle * Math.PI / 180;
- if ( Math.abs(a) > (1.0e-10) ) {
- var s = Math.sin(a)/(1 - Math.cos(a));
- } else {
- // FIXME: This blows up if the angle is exactly 0!
- var s = 2/a;
- }
- for (var i = 0; i < tlist.numberOfItems; ++i) {
- var xform = tlist.getItem(i);
- if (xform.type == 4) {
- // extract old center through mystical arts
- var rm = xform.matrix;
- oldcenter.y = (s*rm.e + rm.f)/2;
- oldcenter.x = (rm.e - s*rm.f)/2;
- tlist.removeItem(i);
- break;
- }
- }
- }
- var tx = 0, ty = 0,
- operation = 0,
- N = tlist.numberOfItems;
- if(N) {
- var first_m = tlist.getItem(0).matrix;
- }
- // first, if it was a scale then the second-last transform will be it
- if (N >= 3 && tlist.getItem(N-2).type == 3 &&
- tlist.getItem(N-3).type == 2 && tlist.getItem(N-1).type == 2)
- {
- operation = 3; // scale
-
- // if the children are unrotated, pass the scale down directly
- // otherwise pass the equivalent matrix() down directly
- var tm = tlist.getItem(N-3).matrix,
- sm = tlist.getItem(N-2).matrix,
- tmn = tlist.getItem(N-1).matrix;
-
- var children = selected.childNodes;
- var c = children.length;
- while (c--) {
- var child = children.item(c);
- tx = 0;
- ty = 0;
- if (child.nodeType == 1) {
- var childTlist = getTransformList(child);
- // some children might not have a transform (<metadata>, <defs>, etc)
- if (!childTlist) continue;
- var m = transformListToTransform(childTlist).matrix;
- // Convert a matrix to a scale if applicable
- // if(hasMatrixTransform(childTlist) && childTlist.numberOfItems == 1) {
- // if(m.b==0 && m.c==0 && m.e==0 && m.f==0) {
- // childTlist.removeItem(0);
- // var translateOrigin = svgroot.createSVGTransform(),
- // scale = svgroot.createSVGTransform(),
- // translateBack = svgroot.createSVGTransform();
- // translateOrigin.setTranslate(0, 0);
- // scale.setScale(m.a, m.d);
- // translateBack.setTranslate(0, 0);
- // childTlist.appendItem(translateBack);
- // childTlist.appendItem(scale);
- // childTlist.appendItem(translateOrigin);
- // }
- // }
-
- var angle = getRotationAngle(child);
- var old_start_transform = start_transform;
- var childxforms = [];
- start_transform = child.getAttribute("transform");
- if(angle || hasMatrixTransform(childTlist)) {
- var e2t = svgroot.createSVGTransform();
- e2t.setMatrix(matrixMultiply(tm, sm, tmn, m));
- childTlist.clear();
- childTlist.appendItem(e2t);
- childxforms.push(e2t);
- }
- // if not rotated or skewed, push the [T][S][-T] down to the child
- else {
- // update the transform list with translate,scale,translate
-
- // slide the [T][S][-T] from the front to the back
- // [T][S][-T][M] = [M][T2][S2][-T2]
-
- // (only bringing [-T] to the right of [M])
- // [T][S][-T][M] = [T][S][M][-T2]
- // [-T2] = [M_inv][-T][M]
- var t2n = matrixMultiply(m.inverse(), tmn, m);
- // [T2] is always negative translation of [-T2]
- var t2 = svgroot.createSVGMatrix();
- t2.e = -t2n.e;
- t2.f = -t2n.f;
-
- // [T][S][-T][M] = [M][T2][S2][-T2]
- // [S2] = [T2_inv][M_inv][T][S][-T][M][-T2_inv]
- var s2 = matrixMultiply(t2.inverse(), m.inverse(), tm, sm, tmn, m, t2n.inverse());
- var translateOrigin = svgroot.createSVGTransform(),
- scale = svgroot.createSVGTransform(),
- translateBack = svgroot.createSVGTransform();
- translateOrigin.setTranslate(t2n.e, t2n.f);
- scale.setScale(s2.a, s2.d);
- translateBack.setTranslate(t2.e, t2.f);
- childTlist.appendItem(translateBack);
- childTlist.appendItem(scale);
- childTlist.appendItem(translateOrigin);
- childxforms.push(translateBack);
- childxforms.push(scale);
- childxforms.push(translateOrigin);
- // logMatrix(translateBack.matrix);
- // logMatrix(scale.matrix);
- } // not rotated
- batchCmd.addSubCommand( recalculateDimensions(child) );
- // TODO: If any <use> have this group as a parent and are
- // referencing this child, then we need to impose a reverse
- // scale on it so that when it won't get double-translated
- // var uses = selected.getElementsByTagNameNS(svgns, "use");
- // var href = "#"+child.id;
- // var u = uses.length;
- // while (u--) {
- // var useElem = uses.item(u);
- // if(href == getHref(useElem)) {
- // var usexlate = svgroot.createSVGTransform();
- // usexlate.setTranslate(-tx,-ty);
- // getTransformList(useElem).insertItemBefore(usexlate,0);
- // batchCmd.addSubCommand( recalculateDimensions(useElem) );
- // }
- // }
- start_transform = old_start_transform;
- } // element
- } // for each child
- // Remove these transforms from group
- tlist.removeItem(N-1);
- tlist.removeItem(N-2);
- tlist.removeItem(N-3);
- }
- else if (N >= 3 && tlist.getItem(N-1).type == 1)
- {
- operation = 3; // scale
- m = transformListToTransform(tlist).matrix;
- var e2t = svgroot.createSVGTransform();
- e2t.setMatrix(m);
- tlist.clear();
- tlist.appendItem(e2t);
- }
- // next, check if the first transform was a translate
- // if we had [ T1 ] [ M ] we want to transform this into [ M ] [ T2 ]
- // therefore [ T2 ] = [ M_inv ] [ T1 ] [ M ]
- else if ( (N == 1 || (N > 1 && tlist.getItem(1).type != 3)) &&
- tlist.getItem(0).type == 2)
- {
- operation = 2; // translate
- var T_M = transformListToTransform(tlist).matrix;
- tlist.removeItem(0);
- var M_inv = transformListToTransform(tlist).matrix.inverse();
- var M2 = matrixMultiply( M_inv, T_M );
-
- tx = M2.e;
- ty = M2.f;
- if (tx != 0 || ty != 0) {
- // we pass the translates down to the individual children
- var children = selected.childNodes;
- var c = children.length;
-
- var clipPaths_done = [];
-
- while (c--) {
- var child = children.item(c);
- if (child.nodeType == 1) {
-
- // Check if child has clip-path
- if(child.getAttribute('clip-path')) {
- // tx, ty
- var attr = child.getAttribute('clip-path');
- if(clipPaths_done.indexOf(attr) === -1) {
- updateClipPath(attr, tx, ty);
- clipPaths_done.push(attr);
- }
- }
- var old_start_transform = start_transform;
- start_transform = child.getAttribute("transform");
-
- var childTlist = getTransformList(child);
- // some children might not have a transform (<metadata>, <defs>, etc)
- if (childTlist) {
- var newxlate = svgroot.createSVGTransform();
- newxlate.setTranslate(tx,ty);
- if(childTlist.numberOfItems) {
- childTlist.insertItemBefore(newxlate, 0);
- } else {
- childTlist.appendItem(newxlate);
- }
- batchCmd.addSubCommand( recalculateDimensions(child) );
- // If any <use> have this group as a parent and are
- // referencing this child, then impose a reverse translate on it
- // so that when it won't get double-translated
- var uses = selected.getElementsByTagNameNS(svgns, "use");
- var href = "#"+child.id;
- var u = uses.length;
- while (u--) {
- var useElem = uses.item(u);
- if(href == getHref(useElem)) {
- var usexlate = svgroot.createSVGTransform();
- usexlate.setTranslate(-tx,-ty);
- getTransformList(useElem).insertItemBefore(usexlate,0);
- batchCmd.addSubCommand( recalculateDimensions(useElem) );
- }
- }
- start_transform = old_start_transform;
- }
- }
- }
-
- clipPaths_done = [];
-
- start_transform = old_start_transform;
- }
- }
- // else, a matrix imposition from a parent group
- // keep pushing it down to the children
- else if (N == 1 && tlist.getItem(0).type == 1 && !gangle) {
- operation = 1;
- var m = tlist.getItem(0).matrix,
- children = selected.childNodes,
- c = children.length;
- while (c--) {
- var child = children.item(c);
- if (child.nodeType == 1) {
- var old_start_transform = start_transform;
- start_transform = child.getAttribute("transform");
- var childTlist = getTransformList(child);
-
- if (!childTlist) continue;
-
- var em = matrixMultiply(m, transformListToTransform(childTlist).matrix);
- var e2m = svgroot.createSVGTransform();
- e2m.setMatrix(em);
- childTlist.clear();
- childTlist.appendItem(e2m,0);
-
- batchCmd.addSubCommand( recalculateDimensions(child) );
- start_transform = old_start_transform;
-
- // Convert stroke
- // TODO: Find out if this should actually happen somewhere else
- var sw = child.getAttribute("stroke-width");
- if (child.getAttribute("stroke") !== "none" && !isNaN(sw)) {
- var avg = (Math.abs(em.a) + Math.abs(em.d)) / 2;
- child.setAttribute('stroke-width', sw * avg);
- }
- }
- }
- tlist.clear();
- }
- // else it was just a rotate
- else {
- if (gangle) {
- var newRot = svgroot.createSVGTransform();
- newRot.setRotate(gangle,newcenter.x,newcenter.y);
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(newRot, 0);
- } else {
- tlist.appendItem(newRot);
- }
- }
- if (tlist.numberOfItems == 0) {
- selected.removeAttribute("transform");
- }
- return null;
- }
-
- // if it was a translate, put back the rotate at the new center
- if (operation == 2) {
- if (gangle) {
- newcenter = {
- x: oldcenter.x + first_m.e,
- y: oldcenter.y + first_m.f
- };
-
- var newRot = svgroot.createSVGTransform();
- newRot.setRotate(gangle,newcenter.x,newcenter.y);
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(newRot, 0);
- } else {
- tlist.appendItem(newRot);
- }
- }
- }
- // if it was a resize
- else if (operation == 3) {
- var m = transformListToTransform(tlist).matrix;
- var roldt = svgroot.createSVGTransform();
- roldt.setRotate(gangle, oldcenter.x, oldcenter.y);
- var rold = roldt.matrix;
- var rnew = svgroot.createSVGTransform();
- rnew.setRotate(gangle, newcenter.x, newcenter.y);
- var rnew_inv = rnew.matrix.inverse(),
- m_inv = m.inverse(),
- extrat = matrixMultiply(m_inv, rnew_inv, rold, m);
- tx = extrat.e;
- ty = extrat.f;
- if (tx != 0 || ty != 0) {
- // now push this transform down to the children
- // we pass the translates down to the individual children
- var children = selected.childNodes;
- var c = children.length;
- while (c--) {
- var child = children.item(c);
- if (child.nodeType == 1) {
- var old_start_transform = start_transform;
- start_transform = child.getAttribute("transform");
- var childTlist = getTransformList(child);
- var newxlate = svgroot.createSVGTransform();
- newxlate.setTranslate(tx,ty);
- if(childTlist.numberOfItems) {
- childTlist.insertItemBefore(newxlate, 0);
- } else {
- childTlist.appendItem(newxlate);
- }
- batchCmd.addSubCommand( recalculateDimensions(child) );
- start_transform = old_start_transform;
- }
- }
- }
-
- if (gangle) {
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(rnew, 0);
- } else {
- tlist.appendItem(rnew);
- }
- }
- }
- }
- // else, it's a non-group
- else {
- // FIXME: box might be null for some elements (<metadata> etc), need to handle this
- var box = svgedit.utilities.getBBox(selected);
- // Paths (and possbly other shapes) will have no BBox while still in <defs>,
- // but we still may need to recalculate them (see issue 595).
- // TODO: Figure out how to get BBox from these elements in case they
- // have a rotation transform
-
- if(!box && selected.tagName != 'path') return null;
-
- var m = svgroot.createSVGMatrix(),
- // temporarily strip off the rotate and save the old center
- angle = getRotationAngle(selected);
- if (angle) {
- var oldcenter = {x: box.x+box.width/2, y: box.y+box.height/2},
- newcenter = transformPoint(box.x+box.width/2, box.y+box.height/2,
- transformListToTransform(tlist).matrix);
-
- var a = angle * Math.PI / 180;
- if ( Math.abs(a) > (1.0e-10) ) {
- var s = Math.sin(a)/(1 - Math.cos(a));
- } else {
- // FIXME: This blows up if the angle is exactly 0!
- var s = 2/a;
- }
- for (var i = 0; i < tlist.numberOfItems; ++i) {
- var xform = tlist.getItem(i);
- if (xform.type == 4) {
- // extract old center through mystical arts
- var rm = xform.matrix;
- oldcenter.y = (s*rm.e + rm.f)/2;
- oldcenter.x = (rm.e - s*rm.f)/2;
- tlist.removeItem(i);
- break;
- }
- }
- }
-
- // 2 = translate, 3 = scale, 4 = rotate, 1 = matrix imposition
- var operation = 0;
- var N = tlist.numberOfItems;
-
- // Check if it has a gradient with userSpaceOnUse, in which case
- // adjust it by recalculating the matrix transform.
- // TODO: Make this work in Webkit using svgedit.transformlist.SVGTransformList
- if(!svgedit.browser.isWebkit()) {
- var fill = selected.getAttribute('fill');
- if(fill && fill.indexOf('url(') === 0) {
- var paint = getRefElem(fill);
- var type = 'pattern';
- if(paint.tagName !== type) type = 'gradient';
- var attrVal = paint.getAttribute(type + 'Units');
- if(attrVal === 'userSpaceOnUse') {
- //Update the userSpaceOnUse element
- m = transformListToTransform(tlist).matrix;
- var gtlist = getTransformList(paint);
- var gmatrix = transformListToTransform(gtlist).matrix;
- m = matrixMultiply(m, gmatrix);
- var m_str = "matrix(" + [m.a,m.b,m.c,m.d,m.e,m.f].join(",") + ")";
- paint.setAttribute(type + 'Transform', m_str);
- }
- }
- }
- // first, if it was a scale of a non-skewed element, then the second-last
- // transform will be the [S]
- // if we had [M][T][S][T] we want to extract the matrix equivalent of
- // [T][S][T] and push it down to the element
- if (N >= 3 && tlist.getItem(N-2).type == 3 &&
- tlist.getItem(N-3).type == 2 && tlist.getItem(N-1).type == 2)
-
- // Removed this so a <use> with a given [T][S][T] would convert to a matrix.
- // Is that bad?
- // && selected.nodeName != "use"
- {
- operation = 3; // scale
- m = transformListToTransform(tlist,N-3,N-1).matrix;
- tlist.removeItem(N-1);
- tlist.removeItem(N-2);
- tlist.removeItem(N-3);
- } // if we had [T][S][-T][M], then this was a skewed element being resized
- // Thus, we simply combine it all into one matrix
- else if(N == 4 && tlist.getItem(N-1).type == 1) {
- operation = 3; // scale
- m = transformListToTransform(tlist).matrix;
- var e2t = svgroot.createSVGTransform();
- e2t.setMatrix(m);
- tlist.clear();
- tlist.appendItem(e2t);
- // reset the matrix so that the element is not re-mapped
- m = svgroot.createSVGMatrix();
- } // if we had [R][T][S][-T][M], then this was a rotated matrix-element
- // if we had [T1][M] we want to transform this into [M][T2]
- // therefore [ T2 ] = [ M_inv ] [ T1 ] [ M ] and we can push [T2]
- // down to the element
- else if ( (N == 1 || (N > 1 && tlist.getItem(1).type != 3)) &&
- tlist.getItem(0).type == 2)
- {
- operation = 2; // translate
- var oldxlate = tlist.getItem(0).matrix,
- meq = transformListToTransform(tlist,1).matrix,
- meq_inv = meq.inverse();
- m = matrixMultiply( meq_inv, oldxlate, meq );
- tlist.removeItem(0);
- }
- // else if this child now has a matrix imposition (from a parent group)
- // we might be able to simplify
- else if (N == 1 && tlist.getItem(0).type == 1 && !angle) {
- // Remap all point-based elements
- m = transformListToTransform(tlist).matrix;
- switch (selected.tagName) {
- case 'line':
- changes = $(selected).attr(["x1","y1","x2","y2"]);
- case 'polyline':
- case 'polygon':
- changes.points = selected.getAttribute("points");
- if(changes.points) {
- var list = selected.points;
- var len = list.numberOfItems;
- changes.points = new Array(len);
- for (var i = 0; i < len; ++i) {
- var pt = list.getItem(i);
- changes.points[i] = {x:pt.x,y:pt.y};
- }
- }
- case 'path':
- changes.d = selected.getAttribute("d");
- operation = 1;
- tlist.clear();
- break;
- default:
- break;
- }
- }
- // if it was a rotation, put the rotate back and return without a command
- // (this function has zero work to do for a rotate())
- else {
- operation = 4; // rotation
- if (angle) {
- var newRot = svgroot.createSVGTransform();
- newRot.setRotate(angle,newcenter.x,newcenter.y);
-
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(newRot, 0);
- } else {
- tlist.appendItem(newRot);
- }
- }
- if (tlist.numberOfItems == 0) {
- selected.removeAttribute("transform");
- }
- return null;
- }
-
- // if it was a translate or resize, we need to remap the element and absorb the xform
- if (operation == 1 || operation == 2 || operation == 3) {
- remapElement(selected,changes,m);
- } // if we are remapping
-
- // if it was a translate, put back the rotate at the new center
- if (operation == 2) {
- if (angle) {
- if(!hasMatrixTransform(tlist)) {
- newcenter = {
- x: oldcenter.x + m.e,
- y: oldcenter.y + m.f
- };
- }
- var newRot = svgroot.createSVGTransform();
- newRot.setRotate(angle, newcenter.x, newcenter.y);
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(newRot, 0);
- } else {
- tlist.appendItem(newRot);
- }
- }
- }
- // [Rold][M][T][S][-T] became [Rold][M]
- // we want it to be [Rnew][M][Tr] where Tr is the
- // translation required to re-center it
- // Therefore, [Tr] = [M_inv][Rnew_inv][Rold][M]
- else if (operation == 3 && angle) {
- var m = transformListToTransform(tlist).matrix;
- var roldt = svgroot.createSVGTransform();
- roldt.setRotate(angle, oldcenter.x, oldcenter.y);
- var rold = roldt.matrix;
- var rnew = svgroot.createSVGTransform();
- rnew.setRotate(angle, newcenter.x, newcenter.y);
- var rnew_inv = rnew.matrix.inverse();
- var m_inv = m.inverse();
- var extrat = matrixMultiply(m_inv, rnew_inv, rold, m);
-
- remapElement(selected,changes,extrat);
- if (angle) {
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(rnew, 0);
- } else {
- tlist.appendItem(rnew);
- }
- }
- }
- } // a non-group
- // if the transform list has been emptied, remove it
- if (tlist.numberOfItems == 0) {
- selected.removeAttribute("transform");
- }
-
- batchCmd.addSubCommand(new ChangeElementCommand(selected, initial));
-
- return batchCmd;
- };
- // Root Current Transformation Matrix in user units
- var root_sctm = null;
- // Group: Selection
- // Function: clearSelection
- // Clears the selection. The 'selected' handler is then called.
- // Parameters:
- // noCall - Optional boolean that when true does not call the "selected" handler
- var clearSelection = this.clearSelection = function(noCall) {
- if (selectedElements[0] != null) {
- var len = selectedElements.length;
- for (var i = 0; i < len; ++i) {
- var elem = selectedElements[i];
- if (elem == null) break;
- selectorManager.releaseSelector(elem);
- selectedElements[i] = null;
- }
- // selectedBBoxes[0] = null;
- }
- if(!noCall) call("selected", selectedElements);
- };
- // TODO: do we need to worry about selectedBBoxes here?
- // Function: addToSelection
- // Adds a list of elements to the selection. The 'selected' handler is then called.
- //
- // Parameters:
- // elemsToAdd - an array of DOM elements to add to the selection
- // showGrips - a boolean flag indicating whether the resize grips should be shown
- var addToSelection = this.addToSelection = function(elemsToAdd, showGrips) {
- if (elemsToAdd.length == 0) { return; }
- // find the first null in our selectedElements array
- var j = 0;
-
- while (j < selectedElements.length) {
- if (selectedElements[j] == null) {
- break;
- }
- ++j;
- }
- // now add each element consecutively
- var i = elemsToAdd.length;
- while (i--) {
- var elem = elemsToAdd[i];
- if (!elem || !svgedit.utilities.getBBox(elem)) continue;
- if(elem.tagName === 'a' && elem.childNodes.length === 1) {
- // Make "a" element's child be the selected element
- elem = elem.firstChild;
- }
- // if it's not already there, add it
- if (selectedElements.indexOf(elem) == -1) {
- selectedElements[j] = elem;
- // only the first selectedBBoxes element is ever used in the codebase these days
- // if (j == 0) selectedBBoxes[0] = svgedit.utilities.getBBox(elem);
- j++;
- var sel = selectorManager.requestSelector(elem);
-
- if (selectedElements.length > 1) {
- sel.showGrips(false);
- }
- }
- }
- call("selected", selectedElements);
-
- if (showGrips || selectedElements.length == 1) {
- selectorManager.requestSelector(selectedElements[0]).showGrips(true);
- }
- else {
- selectorManager.requestSelector(selectedElements[0]).showGrips(false);
- }
- // make sure the elements are in the correct order
- // See: http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-compareDocumentPosition
- selectedElements.sort(function(a,b) {
- if(a && b && a.compareDocumentPosition) {
- return 3 - (b.compareDocumentPosition(a) & 6);
- } else if(a == null) {
- return 1;
- }
- });
-
- // Make sure first elements are not null
- while(selectedElements[0] == null) selectedElements.shift(0);
- };
- // Function: selectOnly()
- // Selects only the given elements, shortcut for clearSelection(); addToSelection()
- //
- // Parameters:
- // elems - an array of DOM elements to be selected
- var selectOnly = this.selectOnly = function(elems, showGrips) {
- clearSelection(true);
- addToSelection(elems, showGrips);
- }
- // TODO: could use slice here to make this faster?
- // TODO: should the 'selected' handler
- // Function: removeFromSelection
- // Removes elements from the selection.
- //
- // Parameters:
- // elemsToRemove - an array of elements to remove from selection
- var removeFromSelection = this.removeFromSelection = function(elemsToRemove) {
- if (selectedElements[0] == null) { return; }
- if (elemsToRemove.length == 0) { return; }
- // find every element and remove it from our array copy
- var newSelectedItems = new Array(selectedElements.length);
- j = 0,
- len = selectedElements.length;
- for (var i = 0; i < len; ++i) {
- var elem = selectedElements[i];
- if (elem) {
- // keep the item
- if (elemsToRemove.indexOf(elem) == -1) {
- newSelectedItems[j] = elem;
- j++;
- }
- else { // remove the item and its selector
- selectorManager.releaseSelector(elem);
- }
- }
- }
- // the copy becomes the master now
- selectedElements = newSelectedItems;
- };
- // Function: selectAllInCurrentLayer
- // Clears the selection, then adds all elements in the current layer to the selection.
- this.selectAllInCurrentLayer = function() {
- var current_layer = getCurrentDrawing().getCurrentLayer();
- if (current_layer) {
- current_mode = "select";
- selectOnly($(current_group || current_layer).children());
- }
- };
- // Function: getMouseTarget
- // Gets the desired element from a mouse event
- //
- // Parameters:
- // evt - Event object from the mouse event
- //
- // Returns:
- // DOM element we want
- var getMouseTarget = this.getMouseTarget = function(evt) {
- if (evt == null) {
- return null;
- }
- var mouse_target = evt.target;
-
- // if it was a <use>, Opera and WebKit return the SVGElementInstance
- if (mouse_target.correspondingUseElement) mouse_target = mouse_target.correspondingUseElement;
-
- // for foreign content, go up until we find the foreignObject
- // WebKit browsers set the mouse target to the svgcanvas div
- if ([mathns, htmlns].indexOf(mouse_target.namespaceURI) >= 0 &&
- mouse_target.id != "svgcanvas")
- {
- while (mouse_target.nodeName != "foreignObject") {
- mouse_target = mouse_target.parentNode;
- if(!mouse_target) return svgroot;
- }
- }
-
- // Get the desired mouse_target with jQuery selector-fu
- // If it's root-like, select the root
- var current_layer = getCurrentDrawing().getCurrentLayer();
- if([svgroot, container, svgcontent, current_layer].indexOf(mouse_target) >= 0) {
- return svgroot;
- }
-
- var $target = $(mouse_target);
- // If it's a selection grip, return the grip parent
- if($target.closest('#selectorParentGroup').length) {
- // While we could instead have just returned mouse_target,
- // this makes it easier to indentify as being a selector grip
- return selectorManager.selectorParentGroup;
- }
- while (mouse_target.parentNode !== (current_group || current_layer)) {
- mouse_target = mouse_target.parentNode;
- }
-
- //
- // // go up until we hit a child of a layer
- // while (mouse_target.parentNode.parentNode.tagName == 'g') {
- // mouse_target = mouse_target.parentNode;
- // }
- // Webkit bubbles the mouse event all the way up to the div, so we
- // set the mouse_target to the svgroot like the other browsers
- // if (mouse_target.nodeName.toLowerCase() == "div") {
- // mouse_target = svgroot;
- // }
-
- return mouse_target;
- };
- // Mouse events
- (function() {
- var d_attr = null,
- start_x = null,
- start_y = null,
- r_start_x = null,
- r_start_y = null,
- init_bbox = {},
- freehand = {
- minx: null,
- miny: null,
- maxx: null,
- maxy: null
- };
-
- // - when we are in a create mode, the element is added to the canvas
- // but the action is not recorded until mousing up
- // - when we are in select mode, select the element, remember the position
- // and do nothing else
- var mouseDown = function(evt)
- {
- if(canvas.spaceKey || evt.button === 1) return;
-
- var right_click = evt.button === 2;
-
- if(evt.altKey) { // duplicate when dragging
- svgCanvas.cloneSelectedElements(0,0);
- }
-
- root_sctm = svgcontent.getScreenCTM().inverse();
-
- var pt = transformPoint( evt.pageX, evt.pageY, root_sctm ),
- mouse_x = pt.x * current_zoom,
- mouse_y = pt.y * current_zoom;
-
- evt.preventDefault();
- if(right_click) {
- current_mode = "select";
- lastClickPoint = pt;
- }
-
- // This would seem to be unnecessary...
- // if(['select', 'resize'].indexOf(current_mode) == -1) {
- // setGradient();
- // }
-
- var x = mouse_x / current_zoom,
- y = mouse_y / current_zoom,
- mouse_target = getMouseTarget(evt);
-
- if(mouse_target.tagName === 'a' && mouse_target.childNodes.length === 1) {
- mouse_target = mouse_target.firstChild;
- }
-
- // real_x/y ignores grid-snap value
- var real_x = r_start_x = start_x = x;
- var real_y = r_start_y = start_y = y;
- if(curConfig.gridSnapping){
- x = snapToGrid(x);
- y = snapToGrid(y);
- start_x = snapToGrid(start_x);
- start_y = snapToGrid(start_y);
- }
- // if it is a selector grip, then it must be a single element selected,
- // set the mouse_target to that and update the mode to rotate/resize
-
- if (mouse_target == selectorManager.selectorParentGroup && selectedElements[0] != null) {
- var grip = evt.target;
- var griptype = elData(grip, "type");
- // rotating
- if (griptype == "rotate") {
- current_mode = "rotate";
- }
- // resizing
- else if(griptype == "resize") {
- current_mode = "resize";
- current_resize_mode = elData(grip, "dir");
- }
- mouse_target = selectedElements[0];
- }
-
- start_transform = mouse_target.getAttribute("transform");
- var tlist = getTransformList(mouse_target);
- switch (current_mode) {
- case "select":
- started = true;
- current_resize_mode = "none";
- if(right_click) started = false;
-
- if (mouse_target != svgroot) {
- // if this element is not yet selected, clear selection and select it
- if (selectedElements.indexOf(mouse_target) == -1) {
- // only clear selection if shift is not pressed (otherwise, add
- // element to selection)
- if (!evt.shiftKey) {
- // No need to do the call here as it will be done on addToSelection
- clearSelection(true);
- }
- addToSelection([mouse_target]);
- justSelected = mouse_target;
- pathActions.clear();
- }
- // else if it's a path, go into pathedit mode in mouseup
-
- if(!right_click) {
- // insert a dummy transform so if the element(s) are moved it will have
- // a transform to use for its translate
- for (var i = 0; i < selectedElements.length; ++i) {
- if(selectedElements[i] == null) continue;
- var slist = getTransformList(selectedElements[i]);
- if(slist.numberOfItems) {
- slist.insertItemBefore(svgroot.createSVGTransform(), 0);
- } else {
- slist.appendItem(svgroot.createSVGTransform());
- }
- }
- }
- }
- else if(!right_click){
- clearSelection();
- current_mode = "multiselect";
- if (rubberBox == null) {
- rubberBox = selectorManager.getRubberBandBox();
- }
- r_start_x *= current_zoom;
- r_start_y *= current_zoom;
- // console.log('p',[evt.pageX, evt.pageY]);
- // console.log('c',[evt.clientX, evt.clientY]);
- // console.log('o',[evt.offsetX, evt.offsetY]);
- // console.log('s',[start_x, start_y]);
-
- assignAttributes(rubberBox, {
- 'x': r_start_x,
- 'y': r_start_y,
- 'width': 0,
- 'height': 0,
- 'display': 'inline'
- }, 100);
- }
- break;
- case "zoom":
- started = true;
- if (rubberBox == null) {
- rubberBox = selectorManager.getRubberBandBox();
- }
- assignAttributes(rubberBox, {
- 'x': real_x * current_zoom,
- 'y': real_x * current_zoom,
- 'width': 0,
- 'height': 0,
- 'display': 'inline'
- }, 100);
- break;
- case "resize":
- started = true;
- start_x = x;
- start_y = y;
-
- // Getting the BBox from the selection box, since we know we
- // want to orient around it
- init_bbox = svgedit.utilities.getBBox($('#selectedBox0')[0]);
- var bb = {};
- $.each(init_bbox, function(key, val) {
- bb[key] = val/current_zoom;
- });
- init_bbox = bb;
-
- // append three dummy transforms to the tlist so that
- // we can translate,scale,translate in mousemove
- var pos = getRotationAngle(mouse_target)?1:0;
-
- if(hasMatrixTransform(tlist)) {
- tlist.insertItemBefore(svgroot.createSVGTransform(), pos);
- tlist.insertItemBefore(svgroot.createSVGTransform(), pos);
- tlist.insertItemBefore(svgroot.createSVGTransform(), pos);
- } else {
- tlist.appendItem(svgroot.createSVGTransform());
- tlist.appendItem(svgroot.createSVGTransform());
- tlist.appendItem(svgroot.createSVGTransform());
-
- if(svgedit.browser.supportsNonScalingStroke()) {
- //Handle crash for newer Chrome + Windows: https://code.google.com/p/svg-edit/issues/detail?id=904
- // TODO: Remove this workaround (all isChromeWindows blocks) once vendor fixes the issue
- var isChromeWindows = svgedit.browser.isChrome() && svgedit.browser.isWindows();
- if(isChromeWindows) {
- var delayedStroke = function(ele) {
- var _stroke = ele.getAttributeNS(null, 'stroke');
- ele.removeAttributeNS(null, 'stroke');
- //Re-apply stroke after delay. Anything higher than 1 seems to cause flicker
- setTimeout(function() { ele.setAttributeNS(null, 'stroke', _stroke) }, 1);
- }
- }
- mouse_target.style.vectorEffect = 'non-scaling-stroke';
- if(isChromeWindows) delayedStroke(mouse_target);
- var all = mouse_target.getElementsByTagName('*'),
- len = all.length;
- for(var i = 0; i < len; i++) {
- all[i].style.vectorEffect = 'non-scaling-stroke';
- if(isChromeWindows) delayedStroke(all[i]);
- }
- }
- }
- break;
- case "fhellipse":
- case "fhrect":
- case "fhpath":
- started = true;
- d_attr = real_x + "," + real_y + " ";
- var stroke_w = cur_shape.stroke_width == 0?1:cur_shape.stroke_width;
- addSvgElementFromJson({
- "element": "polyline",
- "curStyles": true,
- "attr": {
- "points": d_attr,
- "id": getNextId(),
- "fill": "none",
- "opacity": cur_shape.opacity / 2,
- "stroke-linecap": "round",
- "style": "pointer-events:none"
- }
- });
- freehand.minx = real_x;
- freehand.maxx = real_x;
- freehand.miny = real_y;
- freehand.maxy = real_y;
- break;
- case "image":
- started = true;
- var newImage = addSvgElementFromJson({
- "element": "image",
- "attr": {
- "x": x,
- "y": y,
- "width": 0,
- "height": 0,
- "id": getNextId(),
- "opacity": cur_shape.opacity / 2,
- "style": "pointer-events:inherit"
- }
- });
- setHref(newImage, last_good_img_url);
- preventClickDefault(newImage);
- break;
- case "square":
- // FIXME: once we create the rect, we lose information that this was a square
- // (for resizing purposes this could be important)
- case "rect":
- started = true;
- start_x = x;
- start_y = y;
- addSvgElementFromJson({
- "element": "rect",
- "curStyles": true,
- "attr": {
- "x": x,
- "y": y,
- "width": 0,
- "height": 0,
- "id": getNextId(),
- "opacity": cur_shape.opacity / 2
- }
- });
- break;
- case "line":
- started = true;
- var stroke_w = cur_shape.stroke_width == 0?1:cur_shape.stroke_width;
- addSvgElementFromJson({
- "element": "line",
- "curStyles": true,
- "attr": {
- "x1": x,
- "y1": y,
- "x2": x,
- "y2": y,
- "id": getNextId(),
- "stroke": cur_shape.stroke,
- "stroke-width": stroke_w,
- "stroke-dasharray": cur_shape.stroke_dasharray,
- "stroke-linejoin": cur_shape.stroke_linejoin,
- "stroke-linecap": cur_shape.stroke_linecap,
- "stroke-opacity": cur_shape.stroke_opacity,
- "fill": "none",
- "opacity": cur_shape.opacity / 2,
- "style": "pointer-events:none"
- }
- });
- break;
- case "circle":
- started = true;
- addSvgElementFromJson({
- "element": "circle",
- "curStyles": true,
- "attr": {
- "cx": x,
- "cy": y,
- "r": 0,
- "id": getNextId(),
- "opacity": cur_shape.opacity / 2
- }
- });
- break;
- case "ellipse":
- started = true;
- addSvgElementFromJson({
- "element": "ellipse",
- "curStyles": true,
- "attr": {
- "cx": x,
- "cy": y,
- "rx": 0,
- "ry": 0,
- "id": getNextId(),
- "opacity": cur_shape.opacity / 2
- }
- });
- break;
- case "text":
- started = true;
- var newText = addSvgElementFromJson({
- "element": "text",
- "curStyles": true,
- "attr": {
- "x": x,
- "y": y,
- "id": getNextId(),
- "fill": cur_text.fill,
- "stroke-width": cur_text.stroke_width,
- "font-size": cur_text.font_size,
- "font-family": cur_text.font_family,
- "text-anchor": "middle",
- "xml:space": "preserve",
- "opacity": cur_shape.opacity
- }
- });
- // newText.textContent = "text";
- break;
- case "path":
- // Fall through
- case "pathedit":
- start_x *= current_zoom;
- start_y *= current_zoom;
- pathActions.mouseDown(evt, mouse_target, start_x, start_y);
- started = true;
- break;
- case "textedit":
- start_x *= current_zoom;
- start_y *= current_zoom;
- textActions.mouseDown(evt, mouse_target, start_x, start_y);
- started = true;
- break;
- case "rotate":
- started = true;
- // we are starting an undoable change (a drag-rotation)
- canvas.undoMgr.beginUndoableChange("transform", selectedElements);
- break;
- default:
- // This could occur in an extension
- break;
- }
-
- var ext_result = runExtensions("mouseDown", {
- event: evt,
- start_x: start_x,
- start_y: start_y,
- selectedElements: selectedElements
- }, true);
-
- $.each(ext_result, function(i, r) {
- if(r && r.started) {
- started = true;
- }
- });
- };
-
- // in this function we do not record any state changes yet (but we do update
- // any elements that are still being created, moved or resized on the canvas)
- var mouseMove = function(evt)
- {
- if (!started) return;
- if(evt.button === 1 || canvas.spaceKey) return;
- var selected = selectedElements[0],
- pt = transformPoint( evt.pageX, evt.pageY, root_sctm ),
- mouse_x = pt.x * current_zoom,
- mouse_y = pt.y * current_zoom,
- shape = getElem(getId());
- var real_x = x = mouse_x / current_zoom;
- var real_y = y = mouse_y / current_zoom;
- if(curConfig.gridSnapping){
- x = snapToGrid(x);
- y = snapToGrid(y);
- }
- evt.preventDefault();
-
- switch (current_mode)
- {
- case "select":
- // we temporarily use a translate on the element(s) being dragged
- // this transform is removed upon mousing up and the element is
- // relocated to the new location
- if (selectedElements[0] !== null) {
- var dx = x - start_x;
- var dy = y - start_y;
-
- if(curConfig.gridSnapping){
- dx = snapToGrid(dx);
- dy = snapToGrid(dy);
- }
- if(evt.shiftKey) { var xya = snapToAngle(start_x,start_y,x,y); x=xya.x; y=xya.y; }
- if (dx != 0 || dy != 0) {
- var len = selectedElements.length;
- for (var i = 0; i < len; ++i) {
- var selected = selectedElements[i];
- if (selected == null) break;
- // if (i==0) {
- // var box = svgedit.utilities.getBBox(selected);
- // selectedBBoxes[i].x = box.x + dx;
- // selectedBBoxes[i].y = box.y + dy;
- // }
- // update the dummy transform in our transform list
- // to be a translate
- var xform = svgroot.createSVGTransform();
- var tlist = getTransformList(selected);
- // Note that if Webkit and there's no ID for this
- // element, the dummy transform may have gotten lost.
- // This results in unexpected behaviour
-
- xform.setTranslate(dx,dy);
- if(tlist.numberOfItems) {
- tlist.replaceItem(xform, 0);
- } else {
- tlist.appendItem(xform);
- }
-
- // update our internal bbox that we're tracking while dragging
- selectorManager.requestSelector(selected).resize();
- }
-
- call("transition", selectedElements);
- }
- }
- break;
- case "multiselect":
- real_x *= current_zoom;
- real_y *= current_zoom;
- assignAttributes(rubberBox, {
- 'x': Math.min(r_start_x, real_x),
- 'y': Math.min(r_start_y, real_y),
- 'width': Math.abs(real_x - r_start_x),
- 'height': Math.abs(real_y - r_start_y)
- },100);
- // for each selected:
- // - if newList contains selected, do nothing
- // - if newList doesn't contain selected, remove it from selected
- // - for any newList that was not in selectedElements, add it to selected
- var elemsToRemove = [], elemsToAdd = [],
- newList = getIntersectionList(),
- len = selectedElements.length;
-
- for (var i = 0; i < len; ++i) {
- var ind = newList.indexOf(selectedElements[i]);
- if (ind == -1) {
- elemsToRemove.push(selectedElements[i]);
- }
- else {
- newList[ind] = null;
- }
- }
-
- len = newList.length;
- for (i = 0; i < len; ++i) { if (newList[i]) elemsToAdd.push(newList[i]); }
-
- if (elemsToRemove.length > 0)
- canvas.removeFromSelection(elemsToRemove);
-
- if (elemsToAdd.length > 0)
- addToSelection(elemsToAdd);
-
- break;
- case "resize":
- // we track the resize bounding box and translate/scale the selected element
- // while the mouse is down, when mouse goes up, we use this to recalculate
- // the shape's coordinates
- var tlist = getTransformList(selected),
- hasMatrix = hasMatrixTransform(tlist),
- box = hasMatrix ? init_bbox : svgedit.utilities.getBBox(selected),
- left=box.x, top=box.y, width=box.width,
- height=box.height, dx=(x-start_x), dy=(y-start_y);
-
- if(curConfig.gridSnapping){
- dx = snapToGrid(dx);
- dy = snapToGrid(dy);
- height = snapToGrid(height);
- width = snapToGrid(width);
- }
- // if rotated, adjust the dx,dy values
- var angle = getRotationAngle(selected);
- if (angle) {
- var r = Math.sqrt( dx*dx + dy*dy ),
- theta = Math.atan2(dy,dx) - angle * Math.PI / 180.0;
- dx = r * Math.cos(theta);
- dy = r * Math.sin(theta);
- }
- // if not stretching in y direction, set dy to 0
- // if not stretching in x direction, set dx to 0
- if(current_resize_mode.indexOf("n")==-1 && current_resize_mode.indexOf("s")==-1) {
- dy = 0;
- }
- if(current_resize_mode.indexOf("e")==-1 && current_resize_mode.indexOf("w")==-1) {
- dx = 0;
- }
-
- var ts = null,
- tx = 0, ty = 0,
- sy = height ? (height+dy)/height : 1,
- sx = width ? (width+dx)/width : 1;
- // if we are dragging on the north side, then adjust the scale factor and ty
- if(current_resize_mode.indexOf("n") >= 0) {
- sy = height ? (height-dy)/height : 1;
- ty = height;
- }
-
- // if we dragging on the east side, then adjust the scale factor and tx
- if(current_resize_mode.indexOf("w") >= 0) {
- sx = width ? (width-dx)/width : 1;
- tx = width;
- }
-
- // update the transform list with translate,scale,translate
- var translateOrigin = svgroot.createSVGTransform(),
- scale = svgroot.createSVGTransform(),
- translateBack = svgroot.createSVGTransform();
- if(curConfig.gridSnapping){
- left = snapToGrid(left);
- tx = snapToGrid(tx);
- top = snapToGrid(top);
- ty = snapToGrid(ty);
- }
- translateOrigin.setTranslate(-(left+tx),-(top+ty));
- if(evt.shiftKey) {
- if(sx == 1) sx = sy
- else sy = sx;
- }
- scale.setScale(sx,sy);
-
- translateBack.setTranslate(left+tx,top+ty);
- if(hasMatrix) {
- var diff = angle?1:0;
- tlist.replaceItem(translateOrigin, 2+diff);
- tlist.replaceItem(scale, 1+diff);
- tlist.replaceItem(translateBack, 0+diff);
- } else {
- var N = tlist.numberOfItems;
- tlist.replaceItem(translateBack, N-3);
- tlist.replaceItem(scale, N-2);
- tlist.replaceItem(translateOrigin, N-1);
- }
- selectorManager.requestSelector(selected).resize();
-
- call("transition", selectedElements);
-
- break;
- case "zoom":
- real_x *= current_zoom;
- real_y *= current_zoom;
- assignAttributes(rubberBox, {
- 'x': Math.min(r_start_x*current_zoom, real_x),
- 'y': Math.min(r_start_y*current_zoom, real_y),
- 'width': Math.abs(real_x - r_start_x*current_zoom),
- 'height': Math.abs(real_y - r_start_y*current_zoom)
- },100);
- break;
- case "text":
- assignAttributes(shape,{
- 'x': x,
- 'y': y
- },1000);
- break;
- case "line":
- // Opera has a problem with suspendRedraw() apparently
- var handle = null;
- if (!window.opera) svgroot.suspendRedraw(1000);
- if(curConfig.gridSnapping){
- x = snapToGrid(x);
- y = snapToGrid(y);
- }
- var x2 = x;
- var y2 = y;
- if(evt.shiftKey) { var xya = snapToAngle(start_x,start_y,x2,y2); x2=xya.x; y2=xya.y; }
-
- shape.setAttributeNS(null, "x2", x2);
- shape.setAttributeNS(null, "y2", y2);
- if (!window.opera) svgroot.unsuspendRedraw(handle);
- break;
- case "foreignObject":
- // fall through
- case "square":
- // fall through
- case "rect":
- // fall through
- case "image":
- var square = (current_mode == 'square') || evt.shiftKey,
- w = Math.abs(x - start_x),
- h = Math.abs(y - start_y),
- new_x, new_y;
- if(square) {
- w = h = Math.max(w, h);
- new_x = start_x < x ? start_x : start_x - w;
- new_y = start_y < y ? start_y : start_y - h;
- } else {
- new_x = Math.min(start_x,x);
- new_y = Math.min(start_y,y);
- }
-
- if(curConfig.gridSnapping){
- w = snapToGrid(w);
- h = snapToGrid(h);
- new_x = snapToGrid(new_x);
- new_y = snapToGrid(new_y);
- }
- assignAttributes(shape,{
- 'width': w,
- 'height': h,
- 'x': new_x,
- 'y': new_y
- },1000);
-
- break;
- case "circle":
- var c = $(shape).attr(["cx", "cy"]);
- var cx = c.cx, cy = c.cy,
- rad = Math.sqrt( (x-cx)*(x-cx) + (y-cy)*(y-cy) );
- if(curConfig.gridSnapping){
- rad = snapToGrid(rad);
- }
- shape.setAttributeNS(null, "r", rad);
- break;
- case "ellipse":
- var c = $(shape).attr(["cx", "cy"]);
- var cx = c.cx, cy = c.cy;
- // Opera has a problem with suspendRedraw() apparently
- handle = null;
- if (!window.opera) svgroot.suspendRedraw(1000);
- if(curConfig.gridSnapping){
- x = snapToGrid(x);
- cx = snapToGrid(cx);
- y = snapToGrid(y);
- cy = snapToGrid(cy);
- }
- shape.setAttributeNS(null, "rx", Math.abs(x - cx) );
- var ry = Math.abs(evt.shiftKey?(x - cx):(y - cy));
- shape.setAttributeNS(null, "ry", ry );
- if (!window.opera) svgroot.unsuspendRedraw(handle);
- break;
- case "fhellipse":
- case "fhrect":
- freehand.minx = Math.min(real_x, freehand.minx);
- freehand.maxx = Math.max(real_x, freehand.maxx);
- freehand.miny = Math.min(real_y, freehand.miny);
- freehand.maxy = Math.max(real_y, freehand.maxy);
- // break; missing on purpose
- case "fhpath":
- d_attr += + real_x + "," + real_y + " ";
- shape.setAttributeNS(null, "points", d_attr);
- break;
- // update path stretch line coordinates
- case "path":
- // fall through
- case "pathedit":
- x *= current_zoom;
- y *= current_zoom;
-
- if(curConfig.gridSnapping){
- x = snapToGrid(x);
- y = snapToGrid(y);
- start_x = snapToGrid(start_x);
- start_y = snapToGrid(start_y);
- }
- if(evt.shiftKey) {
- var path = svgedit.path.path;
- if(path) {
- var x1 = path.dragging?path.dragging[0]:start_x;
- var y1 = path.dragging?path.dragging[1]:start_y;
- } else {
- var x1 = start_x;
- var y1 = start_y;
- }
- var xya = snapToAngle(x1,y1,x,y);
- x=xya.x; y=xya.y;
- }
-
- if(rubberBox && rubberBox.getAttribute('display') !== 'none') {
- real_x *= current_zoom;
- real_y *= current_zoom;
- assignAttributes(rubberBox, {
- 'x': Math.min(r_start_x*current_zoom, real_x),
- 'y': Math.min(r_start_y*current_zoom, real_y),
- 'width': Math.abs(real_x - r_start_x*current_zoom),
- 'height': Math.abs(real_y - r_start_y*current_zoom)
- },100);
- }
- pathActions.mouseMove(x, y);
-
- break;
- case "textedit":
- x *= current_zoom;
- y *= current_zoom;
- // if(rubberBox && rubberBox.getAttribute('display') != 'none') {
- // assignAttributes(rubberBox, {
- // 'x': Math.min(start_x,x),
- // 'y': Math.min(start_y,y),
- // 'width': Math.abs(x-start_x),
- // 'height': Math.abs(y-start_y)
- // },100);
- // }
-
- textActions.mouseMove(mouse_x, mouse_y);
-
- break;
- case "rotate":
- var box = svgedit.utilities.getBBox(selected),
- cx = box.x + box.width/2,
- cy = box.y + box.height/2,
- m = getMatrix(selected),
- center = transformPoint(cx,cy,m);
- cx = center.x;
- cy = center.y;
- var angle = ((Math.atan2(cy-y,cx-x) * (180/Math.PI))-90) % 360;
- if(curConfig.gridSnapping){
- angle = snapToGrid(angle);
- }
- if(evt.shiftKey) { // restrict rotations to nice angles (WRS)
- var snap = 45;
- angle= Math.round(angle/snap)*snap;
- }
- canvas.setRotationAngle(angle<-180?(360+angle):angle, true);
- call("transition", selectedElements);
- break;
- default:
- break;
- }
-
- runExtensions("mouseMove", {
- event: evt,
- mouse_x: mouse_x,
- mouse_y: mouse_y,
- selected: selected
- });
- }; // mouseMove()
-
- // - in create mode, the element's opacity is set properly, we create an InsertElementCommand
- // and store it on the Undo stack
- // - in move/resize mode, the element's attributes which were affected by the move/resize are
- // identified, a ChangeElementCommand is created and stored on the stack for those attrs
- // this is done in when we recalculate the selected dimensions()
- var mouseUp = function(evt)
- {
- if(evt.button === 2) return;
- var tempJustSelected = justSelected;
- justSelected = null;
- if (!started) return;
- var pt = transformPoint( evt.pageX, evt.pageY, root_sctm ),
- mouse_x = pt.x * current_zoom,
- mouse_y = pt.y * current_zoom,
- x = mouse_x / current_zoom,
- y = mouse_y / current_zoom,
- element = getElem(getId()),
- keep = false;
- var real_x = x;
- var real_y = y;
- // TODO: Make true when in multi-unit mode
- var useUnit = false; // (curConfig.baseUnit !== 'px');
- started = false;
- switch (current_mode)
- {
- // intentionally fall-through to select here
- case "resize":
- case "multiselect":
- if (rubberBox != null) {
- rubberBox.setAttribute("display", "none");
- curBBoxes = [];
- }
- current_mode = "select";
- case "select":
- if (selectedElements[0] != null) {
- // if we only have one selected element
- if (selectedElements[1] == null) {
- // set our current stroke/fill properties to the element's
- var selected = selectedElements[0];
- switch ( selected.tagName ) {
- case "g":
- case "use":
- case "image":
- case "foreignObject":
- break;
- default:
- cur_properties.fill = selected.getAttribute("fill");
- cur_properties.fill_opacity = selected.getAttribute("fill-opacity");
- cur_properties.stroke = selected.getAttribute("stroke");
- cur_properties.stroke_opacity = selected.getAttribute("stroke-opacity");
- cur_properties.stroke_width = selected.getAttribute("stroke-width");
- cur_properties.stroke_dasharray = selected.getAttribute("stroke-dasharray");
- cur_properties.stroke_linejoin = selected.getAttribute("stroke-linejoin");
- cur_properties.stroke_linecap = selected.getAttribute("stroke-linecap");
- }
- if (selected.tagName == "text") {
- cur_text.font_size = selected.getAttribute("font-size");
- cur_text.font_family = selected.getAttribute("font-family");
- }
- selectorManager.requestSelector(selected).showGrips(true);
-
- // This shouldn't be necessary as it was done on mouseDown...
- // call("selected", [selected]);
- }
- // always recalculate dimensions to strip off stray identity transforms
- recalculateAllSelectedDimensions();
- // if it was being dragged/resized
- if (real_x != r_start_x || real_y != r_start_y) {
- var len = selectedElements.length;
- for (var i = 0; i < len; ++i) {
- if (selectedElements[i] == null) break;
- if(!selectedElements[i].firstChild) {
- // Not needed for groups (incorrectly resizes elems), possibly not needed at all?
- selectorManager.requestSelector(selectedElements[i]).resize();
- }
- }
- }
- // no change in position/size, so maybe we should move to pathedit
- else {
- var t = evt.target;
- if (selectedElements[0].nodeName === "path" && selectedElements[1] == null) {
- pathActions.select(selectedElements[0]);
- } // if it was a path
- // else, if it was selected and this is a shift-click, remove it from selection
- else if (evt.shiftKey) {
- if(tempJustSelected != t) {
- canvas.removeFromSelection([t]);
- }
- }
- } // no change in mouse position
-
- // Remove non-scaling stroke
- if(svgedit.browser.supportsNonScalingStroke()) {
- var elem = selectedElements[0];
- if (elem) {
- elem.removeAttribute('style');
- svgedit.utilities.walkTree(elem, function(elem) {
- elem.removeAttribute('style');
- });
- }
- }
- }
- return;
- break;
- case "zoom":
- if (rubberBox != null) {
- rubberBox.setAttribute("display", "none");
- }
- var factor = evt.shiftKey?.5:2;
- call("zoomed", {
- 'x': Math.min(r_start_x, real_x),
- 'y': Math.min(r_start_y, real_y),
- 'width': Math.abs(real_x - r_start_x),
- 'height': Math.abs(real_y - r_start_y),
- 'factor': factor
- });
- return;
- case "fhpath":
- // Check that the path contains at least 2 points; a degenerate one-point path
- // causes problems.
- // Webkit ignores how we set the points attribute with commas and uses space
- // to separate all coordinates, see https://bugs.webkit.org/show_bug.cgi?id=29870
- var coords = element.getAttribute('points');
- var commaIndex = coords.indexOf(',');
- if (commaIndex >= 0) {
- keep = coords.indexOf(',', commaIndex+1) >= 0;
- } else {
- keep = coords.indexOf(' ', coords.indexOf(' ')+1) >= 0;
- }
- if (keep) {
- element = pathActions.smoothPolylineIntoPath(element);
- }
- break;
- case "line":
- var attrs = $(element).attr(["x1", "x2", "y1", "y2"]);
- keep = (attrs.x1 != attrs.x2 || attrs.y1 != attrs.y2);
- break;
- case "foreignObject":
- case "square":
- case "rect":
- case "image":
- var attrs = $(element).attr(["width", "height"]);
- // Image should be kept regardless of size (use inherit dimensions later)
- keep = (attrs.width != 0 || attrs.height != 0) || current_mode === "image";
- break;
- case "circle":
- keep = (element.getAttribute('r') != 0);
- break;
- case "ellipse":
- var attrs = $(element).attr(["rx", "ry"]);
- keep = (attrs.rx != null || attrs.ry != null);
- break;
- case "fhellipse":
- if ((freehand.maxx - freehand.minx) > 0 &&
- (freehand.maxy - freehand.miny) > 0) {
- element = addSvgElementFromJson({
- "element": "ellipse",
- "curStyles": true,
- "attr": {
- "cx": (freehand.minx + freehand.maxx) / 2,
- "cy": (freehand.miny + freehand.maxy) / 2,
- "rx": (freehand.maxx - freehand.minx) / 2,
- "ry": (freehand.maxy - freehand.miny) / 2,
- "id": getId()
- }
- });
- call("changed",[element]);
- keep = true;
- }
- break;
- case "fhrect":
- if ((freehand.maxx - freehand.minx) > 0 &&
- (freehand.maxy - freehand.miny) > 0) {
- element = addSvgElementFromJson({
- "element": "rect",
- "curStyles": true,
- "attr": {
- "x": freehand.minx,
- "y": freehand.miny,
- "width": (freehand.maxx - freehand.minx),
- "height": (freehand.maxy - freehand.miny),
- "id": getId()
- }
- });
- call("changed",[element]);
- keep = true;
- }
- break;
- case "text":
- keep = true;
- selectOnly([element]);
- textActions.start(element);
- break;
- case "path":
- // set element to null here so that it is not removed nor finalized
- element = null;
- // continue to be set to true so that mouseMove happens
- started = true;
-
- var res = pathActions.mouseUp(evt, element, mouse_x, mouse_y);
- element = res.element
- keep = res.keep;
- break;
- case "pathedit":
- keep = true;
- element = null;
- pathActions.mouseUp(evt);
- break;
- case "textedit":
- keep = false;
- element = null;
- textActions.mouseUp(evt, mouse_x, mouse_y);
- break;
- case "rotate":
- keep = true;
- element = null;
- current_mode = "select";
- var batchCmd = canvas.undoMgr.finishUndoableChange();
- if (!batchCmd.isEmpty()) {
- addCommandToHistory(batchCmd);
- }
- // perform recalculation to weed out any stray identity transforms that might get stuck
- recalculateAllSelectedDimensions();
- call("changed", selectedElements);
- break;
- default:
- // This could occur in an extension
- break;
- }
-
- var ext_result = runExtensions("mouseUp", {
- event: evt,
- mouse_x: mouse_x,
- mouse_y: mouse_y
- }, true);
-
- $.each(ext_result, function(i, r) {
- if(r) {
- keep = r.keep || keep;
- element = r.element;
- started = r.started || started;
- }
- });
-
- if (!keep && element != null) {
- getCurrentDrawing().releaseId(getId());
- element.parentNode.removeChild(element);
- element = null;
-
- var t = evt.target;
-
- // if this element is in a group, go up until we reach the top-level group
- // just below the layer groups
- // TODO: once we implement links, we also would have to check for <a> elements
- while (t.parentNode.parentNode.tagName == "g") {
- t = t.parentNode;
- }
- // if we are not in the middle of creating a path, and we've clicked on some shape,
- // then go to Select mode.
- // WebKit returns <div> when the canvas is clicked, Firefox/Opera return <svg>
- if ( (current_mode != "path" || !drawn_path) &&
- t.parentNode.id != "selectorParentGroup" &&
- t.id != "svgcanvas" && t.id != "svgroot")
- {
- // switch into "select" mode if we've clicked on an element
- canvas.setMode("select");
- selectOnly([t], true);
- }
-
- } else if (element != null) {
- canvas.addedNew = true;
-
- if(useUnit) svgedit.units.convertAttrs(element);
-
- var ani_dur = .2, c_ani;
- if(opac_ani.beginElement && element.getAttribute('opacity') != cur_shape.opacity) {
- c_ani = $(opac_ani).clone().attr({
- to: cur_shape.opacity,
- dur: ani_dur
- }).appendTo(element);
- try {
- // Fails in FF4 on foreignObject
- c_ani[0].beginElement();
- } catch(e){}
- } else {
- ani_dur = 0;
- }
-
- // Ideally this would be done on the endEvent of the animation,
- // but that doesn't seem to be supported in Webkit
- setTimeout(function() {
- if(c_ani) c_ani.remove();
- element.setAttribute("opacity", cur_shape.opacity);
- element.setAttribute("style", "pointer-events:inherit");
- cleanupElement(element);
- if(current_mode === "path") {
- pathActions.toEditMode(element);
- } else {
- if(curConfig.selectNew) {
- selectOnly([element], true);
- }
- }
- // we create the insert command that is stored on the stack
- // undo means to call cmd.unapply(), redo means to call cmd.apply()
- addCommandToHistory(new InsertElementCommand(element));
-
- call("changed",[element]);
- }, ani_dur * 1000);
- }
-
- start_transform = null;
- };
-
- var dblClick = function(evt) {
- var evt_target = evt.target;
- var parent = evt_target.parentNode;
-
- // Do nothing if already in current group
- if(parent === current_group) return;
-
- var mouse_target = getMouseTarget(evt);
- var tagName = mouse_target.tagName;
-
- if(tagName === 'text' && current_mode !== 'textedit') {
- var pt = transformPoint( evt.pageX, evt.pageY, root_sctm );
- textActions.select(mouse_target, pt.x, pt.y);
- }
-
- if((tagName === "g" || tagName === "a") && getRotationAngle(mouse_target)) {
- // TODO: Allow method of in-group editing without having to do
- // this (similar to editing rotated paths)
-
- // Ungroup and regroup
- pushGroupProperties(mouse_target);
- mouse_target = selectedElements[0];
- clearSelection(true);
- }
- // Reset context
- if(current_group) {
- leaveContext();
- }
-
- if((parent.tagName !== 'g' && parent.tagName !== 'a') ||
- parent === getCurrentDrawing().getCurrentLayer() ||
- mouse_target === selectorManager.selectorParentGroup)
- {
- // Escape from in-group edit
- return;
- }
- setContext(mouse_target);
- }
- // prevent links from being followed in the canvas
- var handleLinkInCanvas = function(e) {
- e.preventDefault();
- return false;
- };
-
- // Added mouseup to the container here.
- // TODO(codedread): Figure out why after the Closure compiler, the window mouseup is ignored.
- $(container).mousedown(mouseDown).mousemove(mouseMove).click(handleLinkInCanvas).dblclick(dblClick).mouseup(mouseUp);
- // $(window).mouseup(mouseUp);
-
- $(container).bind("mousewheel DOMMouseScroll", function(e){
- if(!e.shiftKey) return;
- e.preventDefault();
- root_sctm = svgcontent.getScreenCTM().inverse();
- var pt = transformPoint( e.pageX, e.pageY, root_sctm );
- var bbox = {
- 'x': pt.x,
- 'y': pt.y,
- 'width': 0,
- 'height': 0
- };
- // Respond to mouse wheel in IE/Webkit/Opera.
- // (It returns up/dn motion in multiples of 120)
- if(e.wheelDelta) {
- if (e.wheelDelta >= 120) {
- bbox.factor = 2;
- } else if (e.wheelDelta <= -120) {
- bbox.factor = .5;
- }
- } else if(e.detail) {
- if (e.detail > 0) {
- bbox.factor = .5;
- } else if (e.detail < 0) {
- bbox.factor = 2;
- }
- }
-
- if(!bbox.factor) return;
- call("zoomed", bbox);
- });
-
- }());
- // Function: preventClickDefault
- // Prevents default browser click behaviour on the given element
- //
- // Parameters:
- // img - The DOM element to prevent the cilck on
- var preventClickDefault = function(img) {
- $(img).click(function(e){e.preventDefault()});
- }
- // Group: Text edit functions
- // Functions relating to editing text elements
- var textActions = canvas.textActions = function() {
- var curtext;
- var textinput;
- var cursor;
- var selblock;
- var blinker;
- var chardata = [];
- var textbb, transbb;
- var matrix;
- var last_x, last_y;
- var allow_dbl;
-
- function setCursor(index) {
- var empty = (textinput.value === "");
- $(textinput).focus();
-
- if(!arguments.length) {
- if(empty) {
- index = 0;
- } else {
- if(textinput.selectionEnd !== textinput.selectionStart) return;
- index = textinput.selectionEnd;
- }
- }
-
- var charbb;
- charbb = chardata[index];
- if(!empty) {
- textinput.setSelectionRange(index, index);
- }
- cursor = getElem("text_cursor");
- if (!cursor) {
- cursor = document.createElementNS(svgns, "line");
- assignAttributes(cursor, {
- 'id': "text_cursor",
- 'stroke': "#333",
- 'stroke-width': 1
- });
- cursor = getElem("selectorParentGroup").appendChild(cursor);
- }
-
- if(!blinker) {
- blinker = setInterval(function() {
- var show = (cursor.getAttribute('display') === 'none');
- cursor.setAttribute('display', show?'inline':'none');
- }, 600);
- }
-
-
- var start_pt = ptToScreen(charbb.x, textbb.y);
- var end_pt = ptToScreen(charbb.x, (textbb.y + textbb.height));
-
- assignAttributes(cursor, {
- x1: start_pt.x,
- y1: start_pt.y,
- x2: end_pt.x,
- y2: end_pt.y,
- visibility: 'visible',
- display: 'inline'
- });
-
- if(selblock) selblock.setAttribute('d', '');
- }
-
- function setSelection(start, end, skipInput) {
- if(start === end) {
- setCursor(end);
- return;
- }
-
- if(!skipInput) {
- textinput.setSelectionRange(start, end);
- }
-
- selblock = getElem("text_selectblock");
- if (!selblock) {
- selblock = document.createElementNS(svgns, "path");
- assignAttributes(selblock, {
- 'id': "text_selectblock",
- 'fill': "green",
- 'opacity': .5,
- 'style': "pointer-events:none"
- });
- getElem("selectorParentGroup").appendChild(selblock);
- }
-
- var startbb = chardata[start];
-
- var endbb = chardata[end];
-
- cursor.setAttribute('visibility', 'hidden');
-
- var tl = ptToScreen(startbb.x, textbb.y),
- tr = ptToScreen(startbb.x + (endbb.x - startbb.x), textbb.y),
- bl = ptToScreen(startbb.x, textbb.y + textbb.height),
- br = ptToScreen(startbb.x + (endbb.x - startbb.x), textbb.y + textbb.height);
-
-
- var dstr = "M" + tl.x + "," + tl.y
- + " L" + tr.x + "," + tr.y
- + " " + br.x + "," + br.y
- + " " + bl.x + "," + bl.y + "z";
-
- assignAttributes(selblock, {
- d: dstr,
- 'display': 'inline'
- });
- }
-
- function getIndexFromPoint(mouse_x, mouse_y) {
- // Position cursor here
- var pt = svgroot.createSVGPoint();
- pt.x = mouse_x;
- pt.y = mouse_y;
- // No content, so return 0
- if(chardata.length == 1) return 0;
- // Determine if cursor should be on left or right of character
- var charpos = curtext.getCharNumAtPosition(pt);
- if(charpos < 0) {
- // Out of text range, look at mouse coords
- charpos = chardata.length - 2;
- if(mouse_x <= chardata[0].x) {
- charpos = 0;
- }
- } else if(charpos >= chardata.length - 2) {
- charpos = chardata.length - 2;
- }
- var charbb = chardata[charpos];
- var mid = charbb.x + (charbb.width/2);
- if(mouse_x > mid) {
- charpos++;
- }
- return charpos;
- }
-
- function setCursorFromPoint(mouse_x, mouse_y) {
- setCursor(getIndexFromPoint(mouse_x, mouse_y));
- }
-
- function setEndSelectionFromPoint(x, y, apply) {
- var i1 = textinput.selectionStart;
- var i2 = getIndexFromPoint(x, y);
-
- var start = Math.min(i1, i2);
- var end = Math.max(i1, i2);
- setSelection(start, end, !apply);
- }
-
- function screenToPt(x_in, y_in) {
- var out = {
- x: x_in,
- y: y_in
- }
-
- out.x /= current_zoom;
- out.y /= current_zoom;
- if(matrix) {
- var pt = transformPoint(out.x, out.y, matrix.inverse());
- out.x = pt.x;
- out.y = pt.y;
- }
-
- return out;
- }
-
- function ptToScreen(x_in, y_in) {
- var out = {
- x: x_in,
- y: y_in
- }
-
- if(matrix) {
- var pt = transformPoint(out.x, out.y, matrix);
- out.x = pt.x;
- out.y = pt.y;
- }
-
- out.x *= current_zoom;
- out.y *= current_zoom;
-
- return out;
- }
-
- function hideCursor() {
- if(cursor) {
- cursor.setAttribute('visibility', 'hidden');
- }
- }
-
- function selectAll(evt) {
- setSelection(0, curtext.textContent.length);
- $(this).unbind(evt);
- }
- function selectWord(evt) {
- if(!allow_dbl || !curtext) return;
-
- var ept = transformPoint( evt.pageX, evt.pageY, root_sctm ),
- mouse_x = ept.x * current_zoom,
- mouse_y = ept.y * current_zoom;
- var pt = screenToPt(mouse_x, mouse_y);
-
- var index = getIndexFromPoint(pt.x, pt.y);
- var str = curtext.textContent;
- var first = str.substr(0, index).replace(/[a-z0-9]+$/i, '').length;
- var m = str.substr(index).match(/^[a-z0-9]+/i);
- var last = (m?m[0].length:0) + index;
- setSelection(first, last);
-
- // Set tripleclick
- $(evt.target).click(selectAll);
- setTimeout(function() {
- $(evt.target).unbind('click', selectAll);
- }, 300);
-
- }
- return {
- select: function(target, x, y) {
- curtext = target;
- textActions.toEditMode(x, y);
- },
- start: function(elem) {
- curtext = elem;
- textActions.toEditMode();
- },
- mouseDown: function(evt, mouse_target, start_x, start_y) {
- var pt = screenToPt(start_x, start_y);
-
- textinput.focus();
- setCursorFromPoint(pt.x, pt.y);
- last_x = start_x;
- last_y = start_y;
-
- // TODO: Find way to block native selection
- },
- mouseMove: function(mouse_x, mouse_y) {
- var pt = screenToPt(mouse_x, mouse_y);
- setEndSelectionFromPoint(pt.x, pt.y);
- },
- mouseUp: function(evt, mouse_x, mouse_y) {
- var pt = screenToPt(mouse_x, mouse_y);
-
- setEndSelectionFromPoint(pt.x, pt.y, true);
-
- // TODO: Find a way to make this work: Use transformed BBox instead of evt.target
- // if(last_x === mouse_x && last_y === mouse_y
- // && !svgedit.math.rectsIntersect(transbb, {x: pt.x, y: pt.y, width:0, height:0})) {
- // textActions.toSelectMode(true);
- // }
- if(
- evt.target !== curtext
- && mouse_x < last_x + 2
- && mouse_x > last_x - 2
- && mouse_y < last_y + 2
- && mouse_y > last_y - 2) {
- textActions.toSelectMode(true);
- }
- },
- setCursor: setCursor,
- toEditMode: function(x, y) {
- allow_dbl = false;
- current_mode = "textedit";
- selectorManager.requestSelector(curtext).showGrips(false);
- // Make selector group accept clicks
- var sel = selectorManager.requestSelector(curtext).selectorRect;
-
- textActions.init();
- $(curtext).css('cursor', 'text');
-
- // if(svgedit.browser.supportsEditableText()) {
- // curtext.setAttribute('editable', 'simple');
- // return;
- // }
-
- if(!arguments.length) {
- setCursor();
- } else {
- var pt = screenToPt(x, y);
- setCursorFromPoint(pt.x, pt.y);
- }
-
- setTimeout(function() {
- allow_dbl = true;
- }, 300);
- },
- toSelectMode: function(selectElem) {
- current_mode = "select";
- clearInterval(blinker);
- blinker = null;
- if(selblock) $(selblock).attr('display','none');
- if(cursor) $(cursor).attr('visibility','hidden');
- $(curtext).css('cursor', 'move');
-
- if(selectElem) {
- clearSelection();
- $(curtext).css('cursor', 'move');
-
- call("selected", [curtext]);
- addToSelection([curtext], true);
- }
- if(curtext && !curtext.textContent.length) {
- // No content, so delete
- canvas.deleteSelectedElements();
- }
-
- $(textinput).blur();
-
- curtext = false;
-
- // if(svgedit.browser.supportsEditableText()) {
- // curtext.removeAttribute('editable');
- // }
- },
- setInputElem: function(elem) {
- textinput = elem;
- // $(textinput).blur(hideCursor);
- },
- clear: function() {
- if(current_mode == "textedit") {
- textActions.toSelectMode();
- }
- },
- init: function(inputElem) {
- if(!curtext) return;
- // if(svgedit.browser.supportsEditableText()) {
- // curtext.select();
- // return;
- // }
-
- if(!curtext.parentNode) {
- // Result of the ffClone, need to get correct element
- curtext = selectedElements[0];
- selectorManager.requestSelector(curtext).showGrips(false);
- }
-
- var str = curtext.textContent;
- var len = str.length;
-
- var xform = curtext.getAttribute('transform');
- textbb = svgedit.utilities.getBBox(curtext);
-
- matrix = xform?getMatrix(curtext):null;
- chardata = Array(len);
- textinput.focus();
-
- $(curtext).unbind('dblclick', selectWord).dblclick(selectWord);
-
- if(!len) {
- var end = {x: textbb.x + (textbb.width/2), width: 0};
- }
-
- for(var i=0; i<len; i++) {
- var start = curtext.getStartPositionOfChar(i);
- var end = curtext.getEndPositionOfChar(i);
-
- if(!svgedit.browser.supportsGoodTextCharPos()) {
- var offset = canvas.contentW * current_zoom;
- start.x -= offset;
- end.x -= offset;
-
- start.x /= current_zoom;
- end.x /= current_zoom;
- }
-
- // Get a "bbox" equivalent for each character. Uses the
- // bbox data of the actual text for y, height purposes
-
- // TODO: Decide if y, width and height are actually necessary
- chardata[i] = {
- x: start.x,
- y: textbb.y, // start.y?
- width: end.x - start.x,
- height: textbb.height
- };
- }
-
- // Add a last bbox for cursor at end of text
- chardata.push({
- x: end.x,
- width: 0
- });
- setSelection(textinput.selectionStart, textinput.selectionEnd, true);
- }
- }
- }();
- // TODO: Migrate all of this code into path.js
- // Group: Path edit functions
- // Functions relating to editing path elements
- var pathActions = canvas.pathActions = function() {
-
- var subpath = false;
- var current_path;
- var newPoint, firstCtrl;
-
- function resetD(p) {
- p.setAttribute("d", pathActions.convertPath(p));
- }
- // TODO: Move into path.js
- svgedit.path.Path.prototype.endChanges = function(text) {
- if(svgedit.browser.isWebkit()) resetD(this.elem);
- var cmd = new ChangeElementCommand(this.elem, {d: this.last_d}, text);
- addCommandToHistory(cmd);
- call("changed", [this.elem]);
- }
- svgedit.path.Path.prototype.addPtsToSelection = function(indexes) {
- if(!$.isArray(indexes)) indexes = [indexes];
- for(var i=0; i< indexes.length; i++) {
- var index = indexes[i];
- var seg = this.segs[index];
- if(seg.ptgrip) {
- if(this.selected_pts.indexOf(index) == -1 && index >= 0) {
- this.selected_pts.push(index);
- }
- }
- };
- this.selected_pts.sort();
- var i = this.selected_pts.length,
- grips = new Array(i);
- // Loop through points to be selected and highlight each
- while(i--) {
- var pt = this.selected_pts[i];
- var seg = this.segs[pt];
- seg.select(true);
- grips[i] = seg.ptgrip;
- }
- // TODO: Correct this:
- pathActions.canDeleteNodes = true;
-
- pathActions.closed_subpath = this.subpathIsClosed(this.selected_pts[0]);
-
- call("selected", grips);
- }
- var current_path = null,
- drawn_path = null,
- hasMoved = false;
-
- // This function converts a polyline (created by the fh_path tool) into
- // a path element and coverts every three line segments into a single bezier
- // curve in an attempt to smooth out the free-hand
- var smoothPolylineIntoPath = function(element) {
- var points = element.points;
- var N = points.numberOfItems;
- if (N >= 4) {
- // loop through every 3 points and convert to a cubic bezier curve segment
- //
- // NOTE: this is cheating, it means that every 3 points has the potential to
- // be a corner instead of treating each point in an equal manner. In general,
- // this technique does not look that good.
- //
- // I am open to better ideas!
- //
- // Reading:
- // - http://www.efg2.com/Lab/Graphics/Jean-YvesQueinecBezierCurves.htm
- // - http://www.codeproject.com/KB/graphics/BezierSpline.aspx?msg=2956963
- // - http://www.ian-ko.com/ET_GeoWizards/UserGuide/smooth.htm
- // - http://www.cs.mtu.edu/~shene/COURSES/cs3621/NOTES/spline/Bezier/bezier-der.html
- var curpos = points.getItem(0), prevCtlPt = null;
- var d = [];
- d.push(["M",curpos.x,",",curpos.y," C"].join(""));
- for (var i = 1; i <= (N-4); i += 3) {
- var ct1 = points.getItem(i);
- var ct2 = points.getItem(i+1);
- var end = points.getItem(i+2);
-
- // if the previous segment had a control point, we want to smooth out
- // the control points on both sides
- if (prevCtlPt) {
- var newpts = svgedit.path.smoothControlPoints( prevCtlPt, ct1, curpos );
- if (newpts && newpts.length == 2) {
- var prevArr = d[d.length-1].split(',');
- prevArr[2] = newpts[0].x;
- prevArr[3] = newpts[0].y;
- d[d.length-1] = prevArr.join(',');
- ct1 = newpts[1];
- }
- }
-
- d.push([ct1.x,ct1.y,ct2.x,ct2.y,end.x,end.y].join(','));
-
- curpos = end;
- prevCtlPt = ct2;
- }
- // handle remaining line segments
- d.push("L");
- for(;i < N;++i) {
- var pt = points.getItem(i);
- d.push([pt.x,pt.y].join(","));
- }
- d = d.join(" ");
- // create new path element
- element = addSvgElementFromJson({
- "element": "path",
- "curStyles": true,
- "attr": {
- "id": getId(),
- "d": d,
- "fill": "none"
- }
- });
- // No need to call "changed", as this is already done under mouseUp
- }
- return element;
- };
- return {
- mouseDown: function(evt, mouse_target, start_x, start_y) {
- if(current_mode === "path") {
- mouse_x = start_x;
- mouse_y = start_y;
-
- var x = mouse_x/current_zoom,
- y = mouse_y/current_zoom,
- stretchy = getElem("path_stretch_line");
- newPoint = [x, y];
-
- if(curConfig.gridSnapping){
- x = snapToGrid(x);
- y = snapToGrid(y);
- mouse_x = snapToGrid(mouse_x);
- mouse_y = snapToGrid(mouse_y);
- }
- if (!stretchy) {
- stretchy = document.createElementNS(svgns, "path");
- assignAttributes(stretchy, {
- 'id': "path_stretch_line",
- 'stroke': "#22C",
- 'stroke-width': "0.5",
- 'fill': 'none'
- });
- stretchy = getElem("selectorParentGroup").appendChild(stretchy);
- }
- stretchy.setAttribute("display", "inline");
-
- var keep = null;
-
- // if pts array is empty, create path element with M at current point
- if (!drawn_path) {
- d_attr = "M" + x + "," + y + " ";
- drawn_path = addSvgElementFromJson({
- "element": "path",
- "curStyles": true,
- "attr": {
- "d": d_attr,
- "id": getNextId(),
- "opacity": cur_shape.opacity / 2
- }
- });
- // set stretchy line to first point
- stretchy.setAttribute('d', ['M', mouse_x, mouse_y, mouse_x, mouse_y].join(' '));
- var index = subpath ? svgedit.path.path.segs.length : 0;
- svgedit.path.addPointGrip(index, mouse_x, mouse_y);
- }
- else {
- // determine if we clicked on an existing point
- var seglist = drawn_path.pathSegList;
- var i = seglist.numberOfItems;
- var FUZZ = 6/current_zoom;
- var clickOnPoint = false;
- while(i) {
- i --;
- var item = seglist.getItem(i);
- var px = item.x, py = item.y;
- // found a matching point
- if ( x >= (px-FUZZ) && x <= (px+FUZZ) && y >= (py-FUZZ) && y <= (py+FUZZ) ) {
- clickOnPoint = true;
- break;
- }
- }
-
- // get path element that we are in the process of creating
- var id = getId();
-
- // Remove previous path object if previously created
- svgedit.path.removePath_(id);
-
- var newpath = getElem(id);
-
- var len = seglist.numberOfItems;
- // if we clicked on an existing point, then we are done this path, commit it
- // (i,i+1) are the x,y that were clicked on
- if (clickOnPoint) {
- // if clicked on any other point but the first OR
- // the first point was clicked on and there are less than 3 points
- // then leave the path open
- // otherwise, close the path
- if (i <= 1 && len >= 2) {
- // Create end segment
- var abs_x = seglist.getItem(0).x;
- var abs_y = seglist.getItem(0).y;
-
- var s_seg = stretchy.pathSegList.getItem(1);
- if(s_seg.pathSegType === 4) {
- var newseg = drawn_path.createSVGPathSegLinetoAbs(abs_x, abs_y);
- } else {
- var newseg = drawn_path.createSVGPathSegCurvetoCubicAbs(
- abs_x,
- abs_y,
- s_seg.x1 / current_zoom,
- s_seg.y1 / current_zoom,
- abs_x,
- abs_y
- );
- }
-
- var endseg = drawn_path.createSVGPathSegClosePath();
- seglist.appendItem(newseg);
- seglist.appendItem(endseg);
- } else if(len < 3) {
- keep = false;
- return keep;
- }
- $(stretchy).remove();
-
- // this will signal to commit the path
- element = newpath;
- drawn_path = null;
- started = false;
-
- if(subpath) {
- if(svgedit.path.path.matrix) {
- remapElement(newpath, {}, svgedit.path.path.matrix.inverse());
- }
-
- var new_d = newpath.getAttribute("d");
- var orig_d = $(svgedit.path.path.elem).attr("d");
- $(svgedit.path.path.elem).attr("d", orig_d + new_d);
- $(newpath).remove();
- if(svgedit.path.path.matrix) {
- svgedit.path.recalcRotatedPath();
- }
- svgedit.path.path.init();
- pathActions.toEditMode(svgedit.path.path.elem);
- svgedit.path.path.selectPt();
- return false;
- }
- }
- // else, create a new point, update path element
- else {
- // Checks if current target or parents are #svgcontent
- if(!$.contains(container, getMouseTarget(evt))) {
- // Clicked outside canvas, so don't make point
- console.log("Clicked outside canvas");
- return false;
- }
- var num = drawn_path.pathSegList.numberOfItems;
- var last = drawn_path.pathSegList.getItem(num -1);
- var lastx = last.x, lasty = last.y;
- if(evt.shiftKey) { var xya = snapToAngle(lastx,lasty,x,y); x=xya.x; y=xya.y; }
-
- // Use the segment defined by stretchy
- var s_seg = stretchy.pathSegList.getItem(1);
- if(s_seg.pathSegType === 4) {
- var newseg = drawn_path.createSVGPathSegLinetoAbs(round(x), round(y));
- } else {
- var newseg = drawn_path.createSVGPathSegCurvetoCubicAbs(
- round(x),
- round(y),
- s_seg.x1 / current_zoom,
- s_seg.y1 / current_zoom,
- s_seg.x2 / current_zoom,
- s_seg.y2 / current_zoom
- );
- }
-
- drawn_path.pathSegList.appendItem(newseg);
-
- x *= current_zoom;
- y *= current_zoom;
-
- // set stretchy line to latest point
- stretchy.setAttribute('d', ['M', x, y, x, y].join(' '));
- var index = num;
- if(subpath) index += svgedit.path.path.segs.length;
- svgedit.path.addPointGrip(index, x, y);
- }
- // keep = true;
- }
-
- return;
- }
-
- // TODO: Make sure current_path isn't null at this point
- if(!svgedit.path.path) return;
-
- svgedit.path.path.storeD();
-
- var id = evt.target.id;
- if (id.substr(0,14) == "pathpointgrip_") {
- // Select this point
- var cur_pt = svgedit.path.path.cur_pt = parseInt(id.substr(14));
- svgedit.path.path.dragging = [start_x, start_y];
- var seg = svgedit.path.path.segs[cur_pt];
-
- // only clear selection if shift is not pressed (otherwise, add
- // node to selection)
- if (!evt.shiftKey) {
- if(svgedit.path.path.selected_pts.length <= 1 || !seg.selected) {
- svgedit.path.path.clearSelection();
- }
- svgedit.path.path.addPtsToSelection(cur_pt);
- } else if(seg.selected) {
- svgedit.path.path.removePtFromSelection(cur_pt);
- } else {
- svgedit.path.path.addPtsToSelection(cur_pt);
- }
- } else if(id.indexOf("ctrlpointgrip_") == 0) {
- svgedit.path.path.dragging = [start_x, start_y];
-
- var parts = id.split('_')[1].split('c');
- var cur_pt = parts[0]-0;
- var ctrl_num = parts[1]-0;
- svgedit.path.path.selectPt(cur_pt, ctrl_num);
- }
- // Start selection box
- if(!svgedit.path.path.dragging) {
- if (rubberBox == null) {
- rubberBox = selectorManager.getRubberBandBox();
- }
- assignAttributes(rubberBox, {
- 'x': start_x * current_zoom,
- 'y': start_y * current_zoom,
- 'width': 0,
- 'height': 0,
- 'display': 'inline'
- }, 100);
- }
- },
- mouseMove: function(mouse_x, mouse_y) {
- hasMoved = true;
- if(current_mode === "path") {
- if(!drawn_path) return;
- var seglist = drawn_path.pathSegList;
- var index = seglist.numberOfItems - 1;
- if(newPoint) {
- // First point
- // if(!index) return;
- // Set control points
- var pointGrip1 = svgedit.path.addCtrlGrip('1c1');
- var pointGrip2 = svgedit.path.addCtrlGrip('0c2');
-
- // dragging pointGrip1
- pointGrip1.setAttribute('cx', mouse_x);
- pointGrip1.setAttribute('cy', mouse_y);
- pointGrip1.setAttribute('display', 'inline');
- var pt_x = newPoint[0];
- var pt_y = newPoint[1];
-
- // set curve
- var seg = seglist.getItem(index);
- var cur_x = mouse_x / current_zoom;
- var cur_y = mouse_y / current_zoom;
- var alt_x = (pt_x + (pt_x - cur_x));
- var alt_y = (pt_y + (pt_y - cur_y));
-
- pointGrip2.setAttribute('cx', alt_x * current_zoom);
- pointGrip2.setAttribute('cy', alt_y * current_zoom);
- pointGrip2.setAttribute('display', 'inline');
-
- var ctrlLine = svgedit.path.getCtrlLine(1);
- assignAttributes(ctrlLine, {
- x1: mouse_x,
- y1: mouse_y,
- x2: alt_x * current_zoom,
- y2: alt_y * current_zoom,
- display: 'inline'
- });
- if(index === 0) {
- firstCtrl = [mouse_x, mouse_y];
- } else {
- var last_x, last_y;
-
- var last = seglist.getItem(index - 1);
- var last_x = last.x;
- var last_y = last.y
-
- if(last.pathSegType === 6) {
- last_x += (last_x - last.x2);
- last_y += (last_y - last.y2);
- } else if(firstCtrl) {
- last_x = firstCtrl[0]/current_zoom;
- last_y = firstCtrl[1]/current_zoom;
- }
- svgedit.path.replacePathSeg(6, index, [pt_x, pt_y, last_x, last_y, alt_x, alt_y], drawn_path);
- }
- } else {
- var stretchy = getElem("path_stretch_line");
- if (stretchy) {
- var prev = seglist.getItem(index);
- if(prev.pathSegType === 6) {
- var prev_x = prev.x + (prev.x - prev.x2);
- var prev_y = prev.y + (prev.y - prev.y2);
- svgedit.path.replacePathSeg(6, 1, [mouse_x, mouse_y, prev_x * current_zoom, prev_y * current_zoom, mouse_x, mouse_y], stretchy);
- } else if(firstCtrl) {
- svgedit.path.replacePathSeg(6, 1, [mouse_x, mouse_y, firstCtrl[0], firstCtrl[1], mouse_x, mouse_y], stretchy);
- } else {
- svgedit.path.replacePathSeg(4, 1, [mouse_x, mouse_y], stretchy);
- }
- }
- }
- return;
- }
- // if we are dragging a point, let's move it
- if (svgedit.path.path.dragging) {
- var pt = svgedit.path.getPointFromGrip({
- x: svgedit.path.path.dragging[0],
- y: svgedit.path.path.dragging[1]
- }, svgedit.path.path);
- var mpt = svgedit.path.getPointFromGrip({
- x: mouse_x,
- y: mouse_y
- }, svgedit.path.path);
- var diff_x = mpt.x - pt.x;
- var diff_y = mpt.y - pt.y;
- svgedit.path.path.dragging = [mouse_x, mouse_y];
-
- if(svgedit.path.path.dragctrl) {
- svgedit.path.path.moveCtrl(diff_x, diff_y);
- } else {
- svgedit.path.path.movePts(diff_x, diff_y);
- }
- } else {
- svgedit.path.path.selected_pts = [];
- svgedit.path.path.eachSeg(function(i) {
- var seg = this;
- if(!seg.next && !seg.prev) return;
-
- var item = seg.item;
- var rbb = rubberBox.getBBox();
-
- var pt = svgedit.path.getGripPt(seg);
- var pt_bb = {
- x: pt.x,
- y: pt.y,
- width: 0,
- height: 0
- };
-
- var sel = svgedit.math.rectsIntersect(rbb, pt_bb);
- this.select(sel);
- //Note that addPtsToSelection is not being run
- if(sel) svgedit.path.path.selected_pts.push(seg.index);
- });
- }
- },
- mouseUp: function(evt, element, mouse_x, mouse_y) {
-
- // Create mode
- if(current_mode === "path") {
- newPoint = null;
- if(!drawn_path) {
- element = getElem(getId());
- started = false;
- firstCtrl = null;
- }
- return {
- keep: true,
- element: element
- }
- }
-
- // Edit mode
-
- if (svgedit.path.path.dragging) {
- var last_pt = svgedit.path.path.cur_pt;
- svgedit.path.path.dragging = false;
- svgedit.path.path.dragctrl = false;
- svgedit.path.path.update();
-
-
- if(hasMoved) {
- svgedit.path.path.endChanges("Move path point(s)");
- }
-
- if(!evt.shiftKey && !hasMoved) {
- svgedit.path.path.selectPt(last_pt);
- }
- }
- else if(rubberBox && rubberBox.getAttribute('display') != 'none') {
- // Done with multi-node-select
- rubberBox.setAttribute("display", "none");
-
- if(rubberBox.getAttribute('width') <= 2 && rubberBox.getAttribute('height') <= 2) {
- pathActions.toSelectMode(evt.target);
- }
-
- // else, move back to select mode
- } else {
- pathActions.toSelectMode(evt.target);
- }
- hasMoved = false;
- },
- toEditMode: function(element) {
- svgedit.path.path = svgedit.path.getPath_(element);
- current_mode = "pathedit";
- clearSelection();
- svgedit.path.path.show(true).update();
- svgedit.path.path.oldbbox = svgedit.utilities.getBBox(svgedit.path.path.elem);
- subpath = false;
- },
- toSelectMode: function(elem) {
- var selPath = (elem == svgedit.path.path.elem);
- current_mode = "select";
- svgedit.path.path.show(false);
- current_path = false;
- clearSelection();
-
- if(svgedit.path.path.matrix) {
- // Rotated, so may need to re-calculate the center
- svgedit.path.recalcRotatedPath();
- }
-
- if(selPath) {
- call("selected", [elem]);
- addToSelection([elem], true);
- }
- },
- addSubPath: function(on) {
- if(on) {
- // Internally we go into "path" mode, but in the UI it will
- // still appear as if in "pathedit" mode.
- current_mode = "path";
- subpath = true;
- } else {
- pathActions.clear(true);
- pathActions.toEditMode(svgedit.path.path.elem);
- }
- },
- select: function(target) {
- if (current_path === target) {
- pathActions.toEditMode(target);
- current_mode = "pathedit";
- } // going into pathedit mode
- else {
- current_path = target;
- }
- },
- reorient: function() {
- var elem = selectedElements[0];
- if(!elem) return;
- var angle = getRotationAngle(elem);
- if(angle == 0) return;
-
- var batchCmd = new BatchCommand("Reorient path");
- var changes = {
- d: elem.getAttribute('d'),
- transform: elem.getAttribute('transform')
- };
- batchCmd.addSubCommand(new ChangeElementCommand(elem, changes));
- clearSelection();
- this.resetOrientation(elem);
-
- addCommandToHistory(batchCmd);
- // Set matrix to null
- svgedit.path.getPath_(elem).show(false).matrix = null;
- this.clear();
-
- addToSelection([elem], true);
- call("changed", selectedElements);
- },
-
- clear: function(remove) {
- current_path = null;
- if (drawn_path) {
- var elem = getElem(getId());
- $(getElem("path_stretch_line")).remove();
- $(elem).remove();
- $(getElem("pathpointgrip_container")).find('*').attr('display', 'none');
- drawn_path = firstCtrl = null;
- started = false;
- } else if (current_mode == "pathedit") {
- this.toSelectMode();
- }
- if(svgedit.path.path) svgedit.path.path.init().show(false);
- },
- resetOrientation: function(path) {
- if(path == null || path.nodeName != 'path') return false;
- var tlist = getTransformList(path);
- var m = transformListToTransform(tlist).matrix;
- tlist.clear();
- path.removeAttribute("transform");
- var segList = path.pathSegList;
-
- // Opera/win/non-EN throws an error here.
- // TODO: Find out why!
- // Presumed fixed in Opera 10.5, so commented out for now
-
- // try {
- var len = segList.numberOfItems;
- // } catch(err) {
- // var fixed_d = pathActions.convertPath(path);
- // path.setAttribute('d', fixed_d);
- // segList = path.pathSegList;
- // var len = segList.numberOfItems;
- // }
- var last_x, last_y;
- for (var i = 0; i < len; ++i) {
- var seg = segList.getItem(i);
- var type = seg.pathSegType;
- if(type == 1) continue;
- var pts = [];
- $.each(['',1,2], function(j, n) {
- var x = seg['x'+n], y = seg['y'+n];
- if(x !== undefined && y !== undefined) {
- var pt = transformPoint(x, y, m);
- pts.splice(pts.length, 0, pt.x, pt.y);
- }
- });
- svgedit.path.replacePathSeg(type, i, pts, path);
- }
-
- reorientGrads(path, m);
- },
- zoomChange: function() {
- if(current_mode == "pathedit") {
- svgedit.path.path.update();
- }
- },
- getNodePoint: function() {
- var sel_pt = svgedit.path.path.selected_pts.length ? svgedit.path.path.selected_pts[0] : 1;
- var seg = svgedit.path.path.segs[sel_pt];
- return {
- x: seg.item.x,
- y: seg.item.y,
- type: seg.type
- };
- },
- linkControlPoints: function(linkPoints) {
- svgedit.path.setLinkControlPoints(linkPoints);
- },
- clonePathNode: function() {
- svgedit.path.path.storeD();
-
- var sel_pts = svgedit.path.path.selected_pts;
- var segs = svgedit.path.path.segs;
-
- var i = sel_pts.length;
- var nums = [];
- while(i--) {
- var pt = sel_pts[i];
- svgedit.path.path.addSeg(pt);
-
- nums.push(pt + i);
- nums.push(pt + i + 1);
- }
- svgedit.path.path.init().addPtsToSelection(nums);
- svgedit.path.path.endChanges("Clone path node(s)");
- },
- opencloseSubPath: function() {
- var sel_pts = svgedit.path.path.selected_pts;
- // Only allow one selected node for now
- if(sel_pts.length !== 1) return;
-
- var elem = svgedit.path.path.elem;
- var list = elem.pathSegList;
- var len = list.numberOfItems;
- var index = sel_pts[0];
-
- var open_pt = null;
- var start_item = null;
- // Check if subpath is already open
- svgedit.path.path.eachSeg(function(i) {
- if(this.type === 2 && i <= index) {
- start_item = this.item;
- }
- if(i <= index) return true;
- if(this.type === 2) {
- // Found M first, so open
- open_pt = i;
- return false;
- } else if(this.type === 1) {
- // Found Z first, so closed
- open_pt = false;
- return false;
- }
- });
-
- if(open_pt == null) {
- // Single path, so close last seg
- open_pt = svgedit.path.path.segs.length - 1;
- }
- if(open_pt !== false) {
- // Close this path
-
- // Create a line going to the previous "M"
- var newseg = elem.createSVGPathSegLinetoAbs(start_item.x, start_item.y);
-
- var closer = elem.createSVGPathSegClosePath();
- if(open_pt == svgedit.path.path.segs.length - 1) {
- list.appendItem(newseg);
- list.appendItem(closer);
- } else {
- svgedit.path.insertItemBefore(elem, closer, open_pt);
- svgedit.path.insertItemBefore(elem, newseg, open_pt);
- }
-
- svgedit.path.path.init().selectPt(open_pt+1);
- return;
- }
-
-
- // M 1,1 L 2,2 L 3,3 L 1,1 z // open at 2,2
- // M 2,2 L 3,3 L 1,1
-
- // M 1,1 L 2,2 L 1,1 z M 4,4 L 5,5 L6,6 L 5,5 z
- // M 1,1 L 2,2 L 1,1 z [M 4,4] L 5,5 L(M)6,6 L 5,5 z
-
- var seg = svgedit.path.path.segs[index];
-
- if(seg.mate) {
- list.removeItem(index); // Removes last "L"
- list.removeItem(index); // Removes the "Z"
- svgedit.path.path.init().selectPt(index - 1);
- return;
- }
-
- var last_m, z_seg;
-
- // Find this sub-path's closing point and remove
- for(var i=0; i<list.numberOfItems; i++) {
- var item = list.getItem(i);
- if(item.pathSegType === 2) {
- // Find the preceding M
- last_m = i;
- } else if(i === index) {
- // Remove it
- list.removeItem(last_m);
- // index--;
- } else if(item.pathSegType === 1 && index < i) {
- // Remove the closing seg of this subpath
- z_seg = i-1;
- list.removeItem(i);
- break;
- }
- }
-
- var num = (index - last_m) - 1;
-
- while(num--) {
- svgedit.path.insertItemBefore(elem, list.getItem(last_m), z_seg);
- }
-
- var pt = list.getItem(last_m);
-
- // Make this point the new "M"
- svgedit.path.replacePathSeg(2, last_m, [pt.x, pt.y]);
-
- var i = index
-
- svgedit.path.path.init().selectPt(0);
- },
- deletePathNode: function() {
- if(!pathActions.canDeleteNodes) return;
- svgedit.path.path.storeD();
-
- var sel_pts = svgedit.path.path.selected_pts;
- var i = sel_pts.length;
- while(i--) {
- var pt = sel_pts[i];
- svgedit.path.path.deleteSeg(pt);
- }
-
- // Cleanup
- var cleanup = function() {
- var segList = svgedit.path.path.elem.pathSegList;
- var len = segList.numberOfItems;
-
- var remItems = function(pos, count) {
- while(count--) {
- segList.removeItem(pos);
- }
- }
- if(len <= 1) return true;
-
- while(len--) {
- var item = segList.getItem(len);
- if(item.pathSegType === 1) {
- var prev = segList.getItem(len-1);
- var nprev = segList.getItem(len-2);
- if(prev.pathSegType === 2) {
- remItems(len-1, 2);
- cleanup();
- break;
- } else if(nprev.pathSegType === 2) {
- remItems(len-2, 3);
- cleanup();
- break;
- }
- } else if(item.pathSegType === 2) {
- if(len > 0) {
- var prev_type = segList.getItem(len-1).pathSegType;
- // Path has M M
- if(prev_type === 2) {
- remItems(len-1, 1);
- cleanup();
- break;
- // Entire path ends with Z M
- } else if(prev_type === 1 && segList.numberOfItems-1 === len) {
- remItems(len, 1);
- cleanup();
- break;
- }
- }
- }
- }
- return false;
- }
-
- cleanup();
-
- // Completely delete a path with 1 or 0 segments
- if(svgedit.path.path.elem.pathSegList.numberOfItems <= 1) {
- pathActions.toSelectMode(svgedit.path.path.elem);
- canvas.deleteSelectedElements();
- return;
- }
-
- svgedit.path.path.init();
-
- svgedit.path.path.clearSelection();
-
- // TODO: Find right way to select point now
- // path.selectPt(sel_pt);
- if(window.opera) { // Opera repaints incorrectly
- var cp = $(svgedit.path.path.elem); cp.attr('d',cp.attr('d'));
- }
- svgedit.path.path.endChanges("Delete path node(s)");
- },
- smoothPolylineIntoPath: smoothPolylineIntoPath,
- setSegType: function(v) {
- svgedit.path.path.setSegType(v);
- },
- moveNode: function(attr, newValue) {
- var sel_pts = svgedit.path.path.selected_pts;
- if(!sel_pts.length) return;
-
- svgedit.path.path.storeD();
-
- // Get first selected point
- var seg = svgedit.path.path.segs[sel_pts[0]];
- var diff = {x:0, y:0};
- diff[attr] = newValue - seg.item[attr];
-
- seg.move(diff.x, diff.y);
- svgedit.path.path.endChanges("Move path point");
- },
- fixEnd: function(elem) {
- // Adds an extra segment if the last seg before a Z doesn't end
- // at its M point
- // M0,0 L0,100 L100,100 z
- var segList = elem.pathSegList;
- var len = segList.numberOfItems;
- var last_m;
- for (var i = 0; i < len; ++i) {
- var item = segList.getItem(i);
- if(item.pathSegType === 2) {
- last_m = item;
- }
-
- if(item.pathSegType === 1) {
- var prev = segList.getItem(i-1);
- if(prev.x != last_m.x || prev.y != last_m.y) {
- // Add an L segment here
- var newseg = elem.createSVGPathSegLinetoAbs(last_m.x, last_m.y);
- svgedit.path.insertItemBefore(elem, newseg, i);
- // Can this be done better?
- pathActions.fixEnd(elem);
- break;
- }
-
- }
- }
- if(svgedit.browser.isWebkit()) resetD(elem);
- },
- // Convert a path to one with only absolute or relative values
- convertPath: function(path, toRel) {
- var segList = path.pathSegList;
- var len = segList.numberOfItems;
- var curx = 0, cury = 0;
- var d = "";
- var last_m = null;
-
- for (var i = 0; i < len; ++i) {
- var seg = segList.getItem(i);
- // if these properties are not in the segment, set them to zero
- var x = seg.x || 0,
- y = seg.y || 0,
- x1 = seg.x1 || 0,
- y1 = seg.y1 || 0,
- x2 = seg.x2 || 0,
- y2 = seg.y2 || 0;
-
- var type = seg.pathSegType;
- var letter = pathMap[type]['to'+(toRel?'Lower':'Upper')+'Case']();
-
- var addToD = function(pnts, more, last) {
- var str = '';
- var more = more?' '+more.join(' '):'';
- var last = last?' '+svgedit.units.shortFloat(last):'';
- $.each(pnts, function(i, pnt) {
- pnts[i] = svgedit.units.shortFloat(pnt);
- });
- d += letter + pnts.join(' ') + more + last;
- }
-
- switch (type) {
- case 1: // z,Z closepath (Z/z)
- d += "z";
- break;
- case 12: // absolute horizontal line (H)
- x -= curx;
- case 13: // relative horizontal line (h)
- if(toRel) {
- curx += x;
- letter = 'l';
- } else {
- x += curx;
- curx = x;
- letter = 'L';
- }
- // Convert to "line" for easier editing
- addToD([[x, cury]]);
- break;
- case 14: // absolute vertical line (V)
- y -= cury;
- case 15: // relative vertical line (v)
- if(toRel) {
- cury += y;
- letter = 'l';
- } else {
- y += cury;
- cury = y;
- letter = 'L';
- }
- // Convert to "line" for easier editing
- addToD([[curx, y]]);
- break;
- case 2: // absolute move (M)
- case 4: // absolute line (L)
- case 18: // absolute smooth quad (T)
- x -= curx;
- y -= cury;
- case 5: // relative line (l)
- case 3: // relative move (m)
- // If the last segment was a "z", this must be relative to
- if(last_m && segList.getItem(i-1).pathSegType === 1 && !toRel) {
- curx = last_m[0];
- cury = last_m[1];
- }
-
- case 19: // relative smooth quad (t)
- if(toRel) {
- curx += x;
- cury += y;
- } else {
- x += curx;
- y += cury;
- curx = x;
- cury = y;
- }
- if(type === 3) last_m = [curx, cury];
-
- addToD([[x,y]]);
- break;
- case 6: // absolute cubic (C)
- x -= curx; x1 -= curx; x2 -= curx;
- y -= cury; y1 -= cury; y2 -= cury;
- case 7: // relative cubic (c)
- if(toRel) {
- curx += x;
- cury += y;
- } else {
- x += curx; x1 += curx; x2 += curx;
- y += cury; y1 += cury; y2 += cury;
- curx = x;
- cury = y;
- }
- addToD([[x1,y1],[x2,y2],[x,y]]);
- break;
- case 8: // absolute quad (Q)
- x -= curx; x1 -= curx;
- y -= cury; y1 -= cury;
- case 9: // relative quad (q)
- if(toRel) {
- curx += x;
- cury += y;
- } else {
- x += curx; x1 += curx;
- y += cury; y1 += cury;
- curx = x;
- cury = y;
- }
- addToD([[x1,y1],[x,y]]);
- break;
- case 10: // absolute elliptical arc (A)
- x -= curx;
- y -= cury;
- case 11: // relative elliptical arc (a)
- if(toRel) {
- curx += x;
- cury += y;
- } else {
- x += curx;
- y += cury;
- curx = x;
- cury = y;
- }
- addToD([[seg.r1,seg.r2]], [
- seg.angle,
- (seg.largeArcFlag ? 1 : 0),
- (seg.sweepFlag ? 1 : 0)
- ],[x,y]
- );
- break;
- case 16: // absolute smooth cubic (S)
- x -= curx; x2 -= curx;
- y -= cury; y2 -= cury;
- case 17: // relative smooth cubic (s)
- if(toRel) {
- curx += x;
- cury += y;
- } else {
- x += curx; x2 += curx;
- y += cury; y2 += cury;
- curx = x;
- cury = y;
- }
- addToD([[x2,y2],[x,y]]);
- break;
- } // switch on path segment type
- } // for each segment
- return d;
- }
- }
- }();
- // end pathActions
- // Group: Serialization
- // Function: removeUnusedDefElems
- // Looks at DOM elements inside the <defs> to see if they are referred to,
- // removes them from the DOM if they are not.
- //
- // Returns:
- // The amount of elements that were removed
- var removeUnusedDefElems = this.removeUnusedDefElems = function() {
- var defs = svgcontent.getElementsByTagNameNS(svgns, "defs");
- if(!defs || !defs.length) return 0;
-
- // if(!defs.firstChild) return;
-
- var defelem_uses = [],
- numRemoved = 0;
- var attrs = ['fill', 'stroke', 'filter', 'marker-start', 'marker-mid', 'marker-end'];
- var alen = attrs.length;
-
- var all_els = svgcontent.getElementsByTagNameNS(svgns, '*');
- var all_len = all_els.length;
-
- for(var i=0; i<all_len; i++) {
- var el = all_els[i];
- for(var j = 0; j < alen; j++) {
- var ref = getUrlFromAttr(el.getAttribute(attrs[j]));
- if(ref) {
- defelem_uses.push(ref.substr(1));
- }
- }
-
- // gradients can refer to other gradients
- var href = getHref(el);
- if (href && href.indexOf('#') === 0) {
- defelem_uses.push(href.substr(1));
- }
- };
-
- var defelems = $(svgcontent).find("linearGradient, radialGradient, filter, marker, svg, symbol");
- defelem_ids = [],
- i = defelems.length;
- while (i--) {
- var defelem = defelems[i];
- var id = defelem.id;
- if(defelem_uses.indexOf(id) < 0) {
- // Not found, so remove (but remember)
- removedElements[id] = defelem;
- defelem.parentNode.removeChild(defelem);
- numRemoved++;
- }
- }
- return numRemoved;
- }
- // Function: svgCanvasToString
- // Main function to set up the SVG content for output
- //
- // Returns:
- // String containing the SVG image for output
- this.svgCanvasToString = function() {
- // keep calling it until there are none to remove
- while (removeUnusedDefElems() > 0) {};
-
- pathActions.clear(true);
-
- // Keep SVG-Edit comment on top
- $.each(svgcontent.childNodes, function(i, node) {
- if(i && node.nodeType === 8 && node.data.indexOf('Created with') >= 0) {
- svgcontent.insertBefore(node, svgcontent.firstChild);
- }
- });
-
- // Move out of in-group editing mode
- if(current_group) {
- leaveContext();
- selectOnly([current_group]);
- }
-
- var naked_svgs = [];
-
- // Unwrap gsvg if it has no special attributes (only id and style)
- $(svgcontent).find('g:data(gsvg)').each(function() {
- var attrs = this.attributes;
- var len = attrs.length;
- for(var i=0; i<len; i++) {
- if(attrs[i].nodeName == 'id' || attrs[i].nodeName == 'style') {
- len--;
- }
- }
- // No significant attributes, so ungroup
- if(len <= 0) {
- var svg = this.firstChild;
- naked_svgs.push(svg);
- $(this).replaceWith(svg);
- }
- });
- var output = this.svgToString(svgcontent, 0);
-
- // Rewrap gsvg
- if(naked_svgs.length) {
- $(naked_svgs).each(function() {
- groupSvgElem(this);
- });
- }
-
- return output;
- };
- // Function: svgToString
- // Sub function ran on each SVG element to convert it to a string as desired
- //
- // Parameters:
- // elem - The SVG element to convert
- // indent - Integer with the amount of spaces to indent this tag
- //
- // Returns:
- // String with the given element as an SVG tag
- this.svgToString = function(elem, indent) {
- var out = new Array(), toXml = svgedit.utilities.toXml;
- var unit = curConfig.baseUnit;
- var unit_re = new RegExp('^-?[\\d\\.]+' + unit + '$');
- if (elem) {
- cleanupElement(elem);
- var attrs = elem.attributes,
- attr,
- i,
- childs = elem.childNodes;
-
- for (var i=0; i<indent; i++) out.push(" ");
- out.push("<"); out.push(elem.nodeName);
- if(elem.id === 'svgcontent') {
- // Process root element separately
- var res = getResolution();
-
- var vb = "";
- // TODO: Allow this by dividing all values by current baseVal
- // Note that this also means we should properly deal with this on import
- // if(curConfig.baseUnit !== "px") {
- // var unit = curConfig.baseUnit;
- // var unit_m = svgedit.units.getTypeMap()[unit];
- // res.w = svgedit.units.shortFloat(res.w / unit_m)
- // res.h = svgedit.units.shortFloat(res.h / unit_m)
- // vb = ' viewBox="' + [0, 0, res.w, res.h].join(' ') + '"';
- // res.w += unit;
- // res.h += unit;
- // }
-
- if(unit !== "px") {
- res.w = svgedit.units.convertUnit(res.w, unit) + unit;
- res.h = svgedit.units.convertUnit(res.h, unit) + unit;
- }
-
- out.push(' width="' + res.w + '" height="' + res.h + '"' + vb + ' xmlns="'+svgns+'"');
-
- var nsuris = {};
-
- // Check elements for namespaces, add if found
- $(elem).find('*').andSelf().each(function() {
- var el = this;
- $.each(this.attributes, function(i, attr) {
- var uri = attr.namespaceURI;
- if(uri && !nsuris[uri] && nsMap[uri] !== 'xmlns' && nsMap[uri] !== 'xml' ) {
- nsuris[uri] = true;
- out.push(" xmlns:" + nsMap[uri] + '="' + uri +'"');
- }
- });
- });
-
- var i = attrs.length;
- var attr_names = ['width','height','xmlns','x','y','viewBox','id','overflow'];
- while (i--) {
- attr = attrs.item(i);
- var attrVal = toXml(attr.nodeValue);
-
- // Namespaces have already been dealt with, so skip
- if(attr.nodeName.indexOf('xmlns:') === 0) continue;
- // only serialize attributes we don't use internally
- if (attrVal != "" && attr_names.indexOf(attr.localName) == -1)
- {
- if(!attr.namespaceURI || nsMap[attr.namespaceURI]) {
- out.push(' ');
- out.push(attr.nodeName); out.push("=\"");
- out.push(attrVal); out.push("\"");
- }
- }
- }
- } else {
- // Skip empty defs
- if(elem.nodeName === 'defs' && !elem.firstChild) return;
-
- var moz_attrs = ['-moz-math-font-style', '_moz-math-font-style'];
- for (var i=attrs.length-1; i>=0; i--) {
- attr = attrs.item(i);
- var attrVal = toXml(attr.nodeValue);
- //remove bogus attributes added by Gecko
- if (moz_attrs.indexOf(attr.localName) >= 0) continue;
- if (attrVal != "") {
- if(attrVal.indexOf('pointer-events') === 0) continue;
- if(attr.localName === "class" && attrVal.indexOf('se_') === 0) continue;
- out.push(" ");
- if(attr.localName === 'd') attrVal = pathActions.convertPath(elem, true);
- if(!isNaN(attrVal)) {
- attrVal = svgedit.units.shortFloat(attrVal);
- } else if(unit_re.test(attrVal)) {
- attrVal = svgedit.units.shortFloat(attrVal) + unit;
- }
-
- // Embed images when saving
- if(save_options.apply
- && elem.nodeName === 'image'
- && attr.localName === 'href'
- && save_options.images
- && save_options.images === 'embed')
- {
- var img = encodableImages[attrVal];
- if(img) attrVal = img;
- }
-
- // map various namespaces to our fixed namespace prefixes
- // (the default xmlns attribute itself does not get a prefix)
- if(!attr.namespaceURI || attr.namespaceURI == svgns || nsMap[attr.namespaceURI]) {
- out.push(attr.nodeName); out.push("=\"");
- out.push(attrVal); out.push("\"");
- }
- }
- }
- }
- if (elem.hasChildNodes()) {
- out.push(">");
- indent++;
- var bOneLine = false;
-
- for (var i=0; i<childs.length; i++)
- {
- var child = childs.item(i);
- switch(child.nodeType) {
- case 1: // element node
- out.push("\n");
- out.push(this.svgToString(childs.item(i), indent));
- break;
- case 3: // text node
- var str = child.nodeValue.replace(/^\s+|\s+$/g, "");
- if (str != "") {
- bOneLine = true;
- out.push(toXml(str) + "");
- }
- break;
- case 4: // cdata node
- out.push("\n");
- out.push(new Array(indent+1).join(" "));
- out.push("<![CDATA[");
- out.push(child.nodeValue);
- out.push("]]>");
- break;
- case 8: // comment
- out.push("\n");
- out.push(new Array(indent+1).join(" "));
- out.push("<!--");
- out.push(child.data);
- out.push("-->");
- break;
- } // switch on node type
- }
- indent--;
- if (!bOneLine) {
- out.push("\n");
- for (var i=0; i<indent; i++) out.push(" ");
- }
- out.push("</"); out.push(elem.nodeName); out.push(">");
- } else {
- out.push("/>");
- }
- }
- return out.join('');
- }; // end svgToString()
- // Function: embedImage
- // Converts a given image file to a data URL when possible, then runs a given callback
- //
- // Parameters:
- // val - String with the path/URL of the image
- // callback - Optional function to run when image data is found, supplies the
- // result (data URL or false) as first parameter.
- this.embedImage = function(val, callback) {
- // load in the image and once it's loaded, get the dimensions
- $(new Image()).load(function() {
- // create a canvas the same size as the raster image
- var canvas = document.createElement("canvas");
- canvas.width = this.width;
- canvas.height = this.height;
- // load the raster image into the canvas
- canvas.getContext("2d").drawImage(this,0,0);
- // retrieve the data: URL
- try {
- var urldata = ';svgedit_url=' + encodeURIComponent(val);
- urldata = canvas.toDataURL().replace(';base64',urldata+';base64');
- encodableImages[val] = urldata;
- } catch(e) {
- encodableImages[val] = false;
- }
- last_good_img_url = val;
- if(callback) callback(encodableImages[val]);
- }).attr('src',val);
- }
- // Function: setGoodImage
- // Sets a given URL to be a "last good image" URL
- this.setGoodImage = function(val) {
- last_good_img_url = val;
- }
- this.open = function() {
- // Nothing by default, handled by optional widget/extension
- };
- // Function: save
- // Serializes the current drawing into SVG XML text and returns it to the 'saved' handler.
- // This function also includes the XML prolog. Clients of the SvgCanvas bind their save
- // function to the 'saved' event.
- //
- // Returns:
- // Nothing
- this.save = function(opts) {
- // remove the selected outline before serializing
- clearSelection();
- // Update save options if provided
- if(opts) $.extend(save_options, opts);
- save_options.apply = true;
-
- // no need for doctype, see http://jwatt.org/svg/authoring/#doctype-declaration
- var str = this.svgCanvasToString();
- call("saved", str);
- };
- // Function: rasterExport
- // Generates a PNG Data URL based on the current image, then calls "exported"
- // with an object including the string and any issues found
- this.rasterExport = function() {
- // remove the selected outline before serializing
- clearSelection();
-
- // Check for known CanVG issues
- var issues = [];
-
- // Selector and notice
- var issue_list = {
- 'feGaussianBlur': uiStrings.exportNoBlur,
- 'foreignObject': uiStrings.exportNoforeignObject,
- '[stroke-dasharray]': uiStrings.exportNoDashArray
- };
- var content = $(svgcontent);
-
- // Add font/text check if Canvas Text API is not implemented
- if(!("font" in $('<canvas>')[0].getContext('2d'))) {
- issue_list['text'] = uiStrings.exportNoText;
- }
-
- $.each(issue_list, function(sel, descr) {
- if(content.find(sel).length) {
- issues.push(descr);
- }
- });
- var str = this.svgCanvasToString();
- call("exported", {svg: str, issues: issues});
- };
- // Function: getSvgString
- // Returns the current drawing as raw SVG XML text.
- //
- // Returns:
- // The current drawing as raw SVG XML text.
- this.getSvgString = function() {
- save_options.apply = false;
- return this.svgCanvasToString();
- };
- // Function: randomizeIds
- // This function determines whether to use a nonce in the prefix, when
- // generating IDs for future documents in SVG-Edit.
- //
- // Parameters:
- // an opional boolean, which, if true, adds a nonce to the prefix. Thus
- // svgCanvas.randomizeIds() <==> svgCanvas.randomizeIds(true)
- //
- // if you're controlling SVG-Edit externally, and want randomized IDs, call
- // this BEFORE calling svgCanvas.setSvgString
- //
- this.randomizeIds = function() {
- if (arguments.length > 0 && arguments[0] == false) {
- svgedit.draw.randomizeIds(false, getCurrentDrawing());
- } else {
- svgedit.draw.randomizeIds(true, getCurrentDrawing());
- }
- };
- // Function: uniquifyElems
- // Ensure each element has a unique ID
- //
- // Parameters:
- // g - The parent element of the tree to give unique IDs
- var uniquifyElems = this.uniquifyElems = function(g) {
- var ids = {};
- // TODO: Handle markers and connectors. These are not yet re-identified properly
- // as their referring elements do not get remapped.
- //
- // <marker id='se_marker_end_svg_7'/>
- // <polyline id='svg_7' se:connector='svg_1 svg_6' marker-end='url(#se_marker_end_svg_7)'/>
- //
- // Problem #1: if svg_1 gets renamed, we do not update the polyline's se:connector attribute
- // Problem #2: if the polyline svg_7 gets renamed, we do not update the marker id nor the polyline's marker-end attribute
- var ref_elems = ["filter", "linearGradient", "pattern", "radialGradient", "symbol", "textPath", "use"];
-
- svgedit.utilities.walkTree(g, function(n) {
- // if it's an element node
- if (n.nodeType == 1) {
- // and the element has an ID
- if (n.id) {
- // and we haven't tracked this ID yet
- if (!(n.id in ids)) {
- // add this id to our map
- ids[n.id] = {elem:null, attrs:[], hrefs:[]};
- }
- ids[n.id]["elem"] = n;
- }
-
- // now search for all attributes on this element that might refer
- // to other elements
- $.each(ref_attrs,function(i,attr) {
- var attrnode = n.getAttributeNode(attr);
- if (attrnode) {
- // the incoming file has been sanitized, so we should be able to safely just strip off the leading #
- var url = svgedit.utilities.getUrlFromAttr(attrnode.value),
- refid = url ? url.substr(1) : null;
- if (refid) {
- if (!(refid in ids)) {
- // add this id to our map
- ids[refid] = {elem:null, attrs:[], hrefs:[]};
- }
- ids[refid]["attrs"].push(attrnode);
- }
- }
- });
-
- // check xlink:href now
- var href = svgedit.utilities.getHref(n);
- // TODO: what if an <image> or <a> element refers to an element internally?
- if(href && ref_elems.indexOf(n.nodeName) >= 0)
- {
- var refid = href.substr(1);
- if (refid) {
- if (!(refid in ids)) {
- // add this id to our map
- ids[refid] = {elem:null, attrs:[], hrefs:[]};
- }
- ids[refid]["hrefs"].push(n);
- }
- }
- }
- });
-
- // in ids, we now have a map of ids, elements and attributes, let's re-identify
- for (var oldid in ids) {
- if (!oldid) continue;
- var elem = ids[oldid]["elem"];
- if (elem) {
- var newid = getNextId();
-
- // assign element its new id
- elem.id = newid;
-
- // remap all url() attributes
- var attrs = ids[oldid]["attrs"];
- var j = attrs.length;
- while (j--) {
- var attr = attrs[j];
- attr.ownerElement.setAttribute(attr.name, "url(#" + newid + ")");
- }
-
- // remap all href attributes
- var hreffers = ids[oldid]["hrefs"];
- var k = hreffers.length;
- while (k--) {
- var hreffer = hreffers[k];
- svgedit.utilities.setHref(hreffer, "#"+newid);
- }
- }
- }
- }
- // Function setUseData
- // Assigns reference data for each use element
- var setUseData = this.setUseData = function(parent) {
- var elems = $(parent);
-
- if(parent.tagName !== 'use') {
- elems = elems.find('use');
- }
-
- elems.each(function() {
- var id = getHref(this).substr(1);
- var ref_elem = getElem(id);
- if(!ref_elem) return;
- $(this).data('ref', ref_elem);
- if(ref_elem.tagName == 'symbol' || ref_elem.tagName == 'svg') {
- $(this).data('symbol', ref_elem).data('ref', ref_elem);
- }
- });
- }
- // Function convertGradients
- // Converts gradients from userSpaceOnUse to objectBoundingBox
- var convertGradients = this.convertGradients = function(elem) {
- var elems = $(elem).find('linearGradient, radialGradient');
- if(!elems.length && svgedit.browser.isWebkit()) {
- // Bug in webkit prevents regular *Gradient selector search
- elems = $(elem).find('*').filter(function() {
- return (this.tagName.indexOf('Gradient') >= 0);
- });
- }
-
- elems.each(function() {
- var grad = this;
- if($(grad).attr('gradientUnits') === 'userSpaceOnUse') {
- // TODO: Support more than one element with this ref by duplicating parent grad
- var elems = $(svgcontent).find('[fill=url(#' + grad.id + ')],[stroke=url(#' + grad.id + ')]');
- if(!elems.length) return;
-
- // get object's bounding box
- var bb = svgedit.utilities.getBBox(elems[0]);
-
- // This will occur if the element is inside a <defs> or a <symbol>,
- // in which we shouldn't need to convert anyway.
- if(!bb) return;
-
- if(grad.tagName === 'linearGradient') {
- var g_coords = $(grad).attr(['x1', 'y1', 'x2', 'y2']);
-
- // If has transform, convert
- var tlist = grad.gradientTransform.baseVal;
- if(tlist && tlist.numberOfItems > 0) {
- var m = transformListToTransform(tlist).matrix;
- var pt1 = transformPoint(g_coords.x1, g_coords.y1, m);
- var pt2 = transformPoint(g_coords.x2, g_coords.y2, m);
-
- g_coords.x1 = pt1.x;
- g_coords.y1 = pt1.y;
- g_coords.x2 = pt2.x;
- g_coords.y2 = pt2.y;
- grad.removeAttribute('gradientTransform');
- }
-
- $(grad).attr({
- x1: (g_coords.x1 - bb.x) / bb.width,
- y1: (g_coords.y1 - bb.y) / bb.height,
- x2: (g_coords.x2 - bb.x) / bb.width,
- y2: (g_coords.y2 - bb.y) / bb.height
- });
- grad.removeAttribute('gradientUnits');
- } else {
- // Note: radialGradient elements cannot be easily converted
- // because userSpaceOnUse will keep circular gradients, while
- // objectBoundingBox will x/y scale the gradient according to
- // its bbox.
-
- // For now we'll do nothing, though we should probably have
- // the gradient be updated as the element is moved, as
- // inkscape/illustrator do.
-
- // var g_coords = $(grad).attr(['cx', 'cy', 'r']);
- //
- // $(grad).attr({
- // cx: (g_coords.cx - bb.x) / bb.width,
- // cy: (g_coords.cy - bb.y) / bb.height,
- // r: g_coords.r
- // });
- //
- // grad.removeAttribute('gradientUnits');
- }
-
- }
- });
- }
- // Function: convertToGroup
- // Converts selected/given <use> or child SVG element to a group
- var convertToGroup = this.convertToGroup = function(elem) {
- if(!elem) {
- elem = selectedElements[0];
- }
- var $elem = $(elem);
-
- var batchCmd = new BatchCommand();
-
- var ts;
-
- if($elem.data('gsvg')) {
- // Use the gsvg as the new group
- var svg = elem.firstChild;
- var pt = $(svg).attr(['x', 'y']);
-
- $(elem.firstChild.firstChild).unwrap();
- $(elem).removeData('gsvg');
-
- var tlist = getTransformList(elem);
- var xform = svgroot.createSVGTransform();
- xform.setTranslate(pt.x, pt.y);
- tlist.appendItem(xform);
- recalculateDimensions(elem);
- call("selected", [elem]);
- } else if($elem.data('symbol')) {
- elem = $elem.data('symbol');
-
- ts = $elem.attr('transform');
- var pos = $elem.attr(['x','y']);
- var vb = elem.getAttribute('viewBox');
-
- if(vb) {
- var nums = vb.split(' ');
- pos.x -= +nums[0];
- pos.y -= +nums[1];
- }
-
- // Not ideal, but works
- ts += " translate(" + (pos.x || 0) + "," + (pos.y || 0) + ")";
-
- var prev = $elem.prev();
-
- // Remove <use> element
- batchCmd.addSubCommand(new RemoveElementCommand($elem[0], $elem[0].nextSibling, $elem[0].parentNode));
- $elem.remove();
-
- // See if other elements reference this symbol
- var has_more = $(svgcontent).find('use:data(symbol)').length;
-
- var g = svgdoc.createElementNS(svgns, "g");
- var childs = elem.childNodes;
-
- for(var i = 0; i < childs.length; i++) {
- g.appendChild(childs[i].cloneNode(true));
- }
-
- // Duplicate the gradients for Gecko, since they weren't included in the <symbol>
- if(svgedit.browser.isGecko()) {
- var dupeGrads = $(findDefs()).children('linearGradient,radialGradient,pattern').clone();
- $(g).append(dupeGrads);
- }
-
- if (ts) {
- g.setAttribute("transform", ts);
- }
-
- var parent = elem.parentNode;
-
- uniquifyElems(g);
-
- // Put the dupe gradients back into <defs> (after uniquifying them)
- if(svgedit.browser.isGecko()) {
- $(findDefs()).append( $(g).find('linearGradient,radialGradient,pattern') );
- }
-
- // now give the g itself a new id
- g.id = getNextId();
-
- prev.after(g);
-
- if(parent) {
- if(!has_more) {
- // remove symbol/svg element
- var nextSibling = elem.nextSibling;
- parent.removeChild(elem);
- batchCmd.addSubCommand(new RemoveElementCommand(elem, nextSibling, parent));
- }
- batchCmd.addSubCommand(new InsertElementCommand(g));
- }
-
- setUseData(g);
-
- if(svgedit.browser.isGecko()) {
- convertGradients(findDefs());
- } else {
- convertGradients(g);
- }
-
- // recalculate dimensions on the top-level children so that unnecessary transforms
- // are removed
- svgedit.utilities.walkTreePost(g, function(n){try{recalculateDimensions(n)}catch(e){console.log(e)}});
-
- // Give ID for any visible element missing one
- $(g).find(visElems).each(function() {
- if(!this.id) this.id = getNextId();
- });
-
- selectOnly([g]);
-
- var cm = pushGroupProperties(g, true);
- if(cm) {
- batchCmd.addSubCommand(cm);
- }
- addCommandToHistory(batchCmd);
-
- } else {
- console.log('Unexpected element to ungroup:', elem);
- }
- }
- //
- // Function: setSvgString
- // This function sets the current drawing as the input SVG XML.
- //
- // Parameters:
- // xmlString - The SVG as XML text.
- //
- // Returns:
- // This function returns false if the set was unsuccessful, true otherwise.
- this.setSvgString = function(xmlString) {
- try {
- // convert string into XML document
- var newDoc = svgedit.utilities.text2xml(xmlString);
- this.prepareSvg(newDoc);
- var batchCmd = new BatchCommand("Change Source");
- // remove old svg document
- var nextSibling = svgcontent.nextSibling;
- var oldzoom = svgroot.removeChild(svgcontent);
- batchCmd.addSubCommand(new RemoveElementCommand(oldzoom, nextSibling, svgroot));
-
- // set new svg document
- // If DOM3 adoptNode() available, use it. Otherwise fall back to DOM2 importNode()
- if(svgdoc.adoptNode) {
- svgcontent = svgdoc.adoptNode(newDoc.documentElement);
- }
- else {
- svgcontent = svgdoc.importNode(newDoc.documentElement, true);
- }
-
- svgroot.appendChild(svgcontent);
- var content = $(svgcontent);
-
- canvas.current_drawing_ = new svgedit.draw.Drawing(svgcontent, idprefix);
-
- // retrieve or set the nonce
- var nonce = getCurrentDrawing().getNonce();
- if (nonce) {
- call("setnonce", nonce);
- } else {
- call("unsetnonce");
- }
-
- // change image href vals if possible
- content.find('image').each(function() {
- var image = this;
- preventClickDefault(image);
- var val = getHref(this);
- if(val.indexOf('data:') === 0) {
- // Check if an SVG-edit data URI
- var m = val.match(/svgedit_url=(.*?);/);
- if(m) {
- var url = decodeURIComponent(m[1]);
- $(new Image()).load(function() {
- image.setAttributeNS(xlinkns,'xlink:href',url);
- }).attr('src',url);
- }
- }
- // Add to encodableImages if it loads
- canvas.embedImage(val);
- });
-
- // Wrap child SVGs in group elements
- content.find('svg').each(function() {
- // Skip if it's in a <defs>
- if($(this).closest('defs').length) return;
-
- uniquifyElems(this);
-
- // Check if it already has a gsvg group
- var pa = this.parentNode;
- if(pa.childNodes.length === 1 && pa.nodeName === 'g') {
- $(pa).data('gsvg', this);
- pa.id = pa.id || getNextId();
- } else {
- groupSvgElem(this);
- }
- });
-
- // For Firefox: Put all paint elems in defs
- if(svgedit.browser.isGecko()) {
- content.find('linearGradient, radialGradient, pattern').appendTo(findDefs());
- }
-
- // Set ref element for <use> elements
-
- // TODO: This should also be done if the object is re-added through "redo"
- setUseData(content);
-
- convertGradients(content[0]);
-
- // recalculate dimensions on the top-level children so that unnecessary transforms
- // are removed
- svgedit.utilities.walkTreePost(svgcontent, function(n){try{recalculateDimensions(n)}catch(e){console.log(e)}});
-
- var attrs = {
- id: 'svgcontent',
- overflow: curConfig.show_outside_canvas?'visible':'hidden'
- };
-
- var percs = false;
-
- // determine proper size
- if (content.attr("viewBox")) {
- var vb = content.attr("viewBox").split(' ');
- attrs.width = vb[2];
- attrs.height = vb[3];
- }
- // handle content that doesn't have a viewBox
- else {
- $.each(['width', 'height'], function(i, dim) {
- // Set to 100 if not given
- var val = content.attr(dim);
-
- if(!val) val = '100%';
-
- if((val+'').substr(-1) === "%") {
- // Use user units if percentage given
- percs = true;
- } else {
- attrs[dim] = convertToNum(dim, val);
- }
- });
- }
-
- // identify layers
- identifyLayers();
-
- // Give ID for any visible layer children missing one
- content.children().find(visElems).each(function() {
- if(!this.id) this.id = getNextId();
- });
-
- // Percentage width/height, so let's base it on visible elements
- if(percs) {
- var bb = getStrokedBBox();
- attrs.width = bb.width + bb.x;
- attrs.height = bb.height + bb.y;
- }
-
- // Just in case negative numbers are given or
- // result from the percs calculation
- if(attrs.width <= 0) attrs.width = 100;
- if(attrs.height <= 0) attrs.height = 100;
-
- content.attr(attrs);
- this.contentW = attrs['width'];
- this.contentH = attrs['height'];
-
- batchCmd.addSubCommand(new InsertElementCommand(svgcontent));
- // update root to the correct size
- var changes = content.attr(["width", "height"]);
- batchCmd.addSubCommand(new ChangeElementCommand(svgroot, changes));
-
- // reset zoom
- current_zoom = 1;
-
- // reset transform lists
- svgedit.transformlist.resetListMap();
- clearSelection();
- svgedit.path.clearData();
- svgroot.appendChild(selectorManager.selectorParentGroup);
-
- addCommandToHistory(batchCmd);
- call("changed", [svgcontent]);
- } catch(e) {
- console.log(e);
- return false;
- }
- return true;
- };
- // Function: importSvgString
- // This function imports the input SVG XML as a <symbol> in the <defs>, then adds a
- // <use> to the current layer.
- //
- // Parameters:
- // xmlString - The SVG as XML text.
- //
- // Returns:
- // This function returns false if the import was unsuccessful, true otherwise.
- // TODO:
- // * properly handle if namespace is introduced by imported content (must add to svgcontent
- // and update all prefixes in the imported node)
- // * properly handle recalculating dimensions, recalculateDimensions() doesn't handle
- // arbitrary transform lists, but makes some assumptions about how the transform list
- // was obtained
- // * import should happen in top-left of current zoomed viewport
- this.importSvgString = function(xmlString) {
- try {
- // Get unique ID
- var uid = svgedit.utilities.encode64(xmlString.length + xmlString).substr(0,32);
-
- var useExisting = false;
- // Look for symbol and make sure symbol exists in image
- if(import_ids[uid]) {
- if( $(import_ids[uid].symbol).parents('#svgroot').length ) {
- useExisting = true;
- }
- }
-
- var batchCmd = new BatchCommand("Import SVG");
-
- if(useExisting) {
- var symbol = import_ids[uid].symbol;
- var ts = import_ids[uid].xform;
- } else {
- // convert string into XML document
- var newDoc = svgedit.utilities.text2xml(xmlString);
-
- this.prepareSvg(newDoc);
-
- // import new svg document into our document
- var svg;
- // If DOM3 adoptNode() available, use it. Otherwise fall back to DOM2 importNode()
- if(svgdoc.adoptNode) {
- svg = svgdoc.adoptNode(newDoc.documentElement);
- }
- else {
- svg = svgdoc.importNode(newDoc.documentElement, true);
- }
-
- uniquifyElems(svg);
-
- var innerw = convertToNum('width', svg.getAttribute("width")),
- innerh = convertToNum('height', svg.getAttribute("height")),
- innervb = svg.getAttribute("viewBox"),
- // if no explicit viewbox, create one out of the width and height
- vb = innervb ? innervb.split(" ") : [0,0,innerw,innerh];
- for (var j = 0; j < 4; ++j)
- vb[j] = +(vb[j]);
-
- // TODO: properly handle preserveAspectRatio
- var canvasw = +svgcontent.getAttribute("width"),
- canvash = +svgcontent.getAttribute("height");
- // imported content should be 1/3 of the canvas on its largest dimension
-
- if (innerh > innerw) {
- var ts = "scale(" + (canvash/3)/vb[3] + ")";
- }
- else {
- var ts = "scale(" + (canvash/3)/vb[2] + ")";
- }
-
- // Hack to make recalculateDimensions understand how to scale
- ts = "translate(0) " + ts + " translate(0)";
-
- var symbol = svgdoc.createElementNS(svgns, "symbol");
- var defs = findDefs();
-
- if(svgedit.browser.isGecko()) {
- // Move all gradients into root for Firefox, workaround for this bug:
- // https://bugzilla.mozilla.org/show_bug.cgi?id=353575
- // TODO: Make this properly undo-able.
- $(svg).find('linearGradient, radialGradient, pattern').appendTo(defs);
- }
-
- while (svg.firstChild) {
- var first = svg.firstChild;
- symbol.appendChild(first);
- }
- var attrs = svg.attributes;
- for(var i=0; i < attrs.length; i++) {
- var attr = attrs[i];
- symbol.setAttribute(attr.nodeName, attr.nodeValue);
- }
- symbol.id = getNextId();
-
- // Store data
- import_ids[uid] = {
- symbol: symbol,
- xform: ts
- }
-
- findDefs().appendChild(symbol);
- batchCmd.addSubCommand(new InsertElementCommand(symbol));
- }
-
-
- var use_el = svgdoc.createElementNS(svgns, "use");
- use_el.id = getNextId();
- setHref(use_el, "#" + symbol.id);
-
- (current_group || getCurrentDrawing().getCurrentLayer()).appendChild(use_el);
- batchCmd.addSubCommand(new InsertElementCommand(use_el));
- clearSelection();
-
- use_el.setAttribute("transform", ts);
- recalculateDimensions(use_el);
- $(use_el).data('symbol', symbol).data('ref', symbol);
- addToSelection([use_el]);
-
- // TODO: Find way to add this in a recalculateDimensions-parsable way
- // if (vb[0] != 0 || vb[1] != 0)
- // ts = "translate(" + (-vb[0]) + "," + (-vb[1]) + ") " + ts;
- addCommandToHistory(batchCmd);
- call("changed", [svgcontent]);
- } catch(e) {
- console.log(e);
- return false;
- }
- return true;
- };
- // TODO(codedread): Move all layer/context functions in draw.js
- // Layer API Functions
- // Group: Layers
- // Function: identifyLayers
- // Updates layer system
- var identifyLayers = canvas.identifyLayers = function() {
- leaveContext();
- getCurrentDrawing().identifyLayers();
- };
- // Function: createLayer
- // Creates a new top-level layer in the drawing with the given name, sets the current layer
- // to it, and then clears the selection This function then calls the 'changed' handler.
- // This is an undoable action.
- //
- // Parameters:
- // name - The given name
- this.createLayer = function(name) {
- var batchCmd = new BatchCommand("Create Layer");
- var new_layer = getCurrentDrawing().createLayer(name);
- batchCmd.addSubCommand(new InsertElementCommand(new_layer));
- addCommandToHistory(batchCmd);
- clearSelection();
- call("changed", [new_layer]);
- };
- // Function: cloneLayer
- // Creates a new top-level layer in the drawing with the given name, copies all the current layer's contents
- // to it, and then clears the selection This function then calls the 'changed' handler.
- // This is an undoable action.
- //
- // Parameters:
- // name - The given name
- this.cloneLayer = function(name) {
- var batchCmd = new BatchCommand("Duplicate Layer");
- var new_layer = svgdoc.createElementNS(svgns, "g");
- var layer_title = svgdoc.createElementNS(svgns, "title");
- layer_title.textContent = name;
- new_layer.appendChild(layer_title);
- var current_layer = getCurrentDrawing().getCurrentLayer();
- $(current_layer).after(new_layer);
- var childs = current_layer.childNodes;
- for(var i = 0; i < childs.length; i++) {
- var ch = childs[i];
- if(ch.localName == 'title') continue;
- new_layer.appendChild(copyElem(ch));
- }
-
- clearSelection();
- identifyLayers();
- batchCmd.addSubCommand(new InsertElementCommand(new_layer));
- addCommandToHistory(batchCmd);
- canvas.setCurrentLayer(name);
- call("changed", [new_layer]);
- };
- // Function: deleteCurrentLayer
- // Deletes the current layer from the drawing and then clears the selection. This function
- // then calls the 'changed' handler. This is an undoable action.
- this.deleteCurrentLayer = function() {
- var current_layer = getCurrentDrawing().getCurrentLayer();
- var nextSibling = current_layer.nextSibling;
- var parent = current_layer.parentNode;
- current_layer = getCurrentDrawing().deleteCurrentLayer();
- if (current_layer) {
- var batchCmd = new BatchCommand("Delete Layer");
- // store in our Undo History
- batchCmd.addSubCommand(new RemoveElementCommand(current_layer, nextSibling, parent));
- addCommandToHistory(batchCmd);
- clearSelection();
- call("changed", [parent]);
- return true;
- }
- return false;
- };
- // Function: setCurrentLayer
- // Sets the current layer. If the name is not a valid layer name, then this function returns
- // false. Otherwise it returns true. This is not an undo-able action.
- //
- // Parameters:
- // name - the name of the layer you want to switch to.
- //
- // Returns:
- // true if the current layer was switched, otherwise false
- this.setCurrentLayer = function(name) {
- var result = getCurrentDrawing().setCurrentLayer(svgedit.utilities.toXml(name));
- if (result) {
- clearSelection();
- }
- return result;
- };
- // Function: renameCurrentLayer
- // Renames the current layer. If the layer name is not valid (i.e. unique), then this function
- // does nothing and returns false, otherwise it returns true. This is an undo-able action.
- //
- // Parameters:
- // newname - the new name you want to give the current layer. This name must be unique
- // among all layer names.
- //
- // Returns:
- // true if the rename succeeded, false otherwise.
- this.renameCurrentLayer = function(newname) {
- var drawing = getCurrentDrawing();
- if (drawing.current_layer) {
- var oldLayer = drawing.current_layer;
- // setCurrentLayer will return false if the name doesn't already exist
- // this means we are free to rename our oldLayer
- if (!canvas.setCurrentLayer(newname)) {
- var batchCmd = new BatchCommand("Rename Layer");
- // find the index of the layer
- for (var i = 0; i < drawing.getNumLayers(); ++i) {
- if (drawing.all_layers[i][1] == oldLayer) break;
- }
- var oldname = drawing.getLayerName(i);
- drawing.all_layers[i][0] = svgedit.utilities.toXml(newname);
-
- // now change the underlying title element contents
- var len = oldLayer.childNodes.length;
- for (var i = 0; i < len; ++i) {
- var child = oldLayer.childNodes.item(i);
- // found the <title> element, now append all the
- if (child && child.tagName == "title") {
- // wipe out old name
- while (child.firstChild) { child.removeChild(child.firstChild); }
- child.textContent = newname;
- batchCmd.addSubCommand(new ChangeElementCommand(child, {"#text":oldname}));
- addCommandToHistory(batchCmd);
- call("changed", [oldLayer]);
- return true;
- }
- }
- }
- drawing.current_layer = oldLayer;
- }
- return false;
- };
- // Function: setCurrentLayerPosition
- // Changes the position of the current layer to the new value. If the new index is not valid,
- // this function does nothing and returns false, otherwise it returns true. This is an
- // undo-able action.
- //
- // Parameters:
- // newpos - The zero-based index of the new position of the layer. This should be between
- // 0 and (number of layers - 1)
- //
- // Returns:
- // true if the current layer position was changed, false otherwise.
- this.setCurrentLayerPosition = function(newpos) {
- var drawing = getCurrentDrawing();
- if (drawing.current_layer && newpos >= 0 && newpos < drawing.getNumLayers()) {
- for (var oldpos = 0; oldpos < drawing.getNumLayers(); ++oldpos) {
- if (drawing.all_layers[oldpos][1] == drawing.current_layer) break;
- }
- // some unknown error condition (current_layer not in all_layers)
- if (oldpos == drawing.getNumLayers()) { return false; }
-
- if (oldpos != newpos) {
- // if our new position is below us, we need to insert before the node after newpos
- var refLayer = null;
- var oldNextSibling = drawing.current_layer.nextSibling;
- if (newpos > oldpos ) {
- if (newpos < drawing.getNumLayers()-1) {
- refLayer = drawing.all_layers[newpos+1][1];
- }
- }
- // if our new position is above us, we need to insert before the node at newpos
- else {
- refLayer = drawing.all_layers[newpos][1];
- }
- svgcontent.insertBefore(drawing.current_layer, refLayer);
- addCommandToHistory(new MoveElementCommand(drawing.current_layer, oldNextSibling, svgcontent));
-
- identifyLayers();
- canvas.setCurrentLayer(drawing.getLayerName(newpos));
-
- return true;
- }
- }
-
- return false;
- };
- // Function: setLayerVisibility
- // Sets the visibility of the layer. If the layer name is not valid, this function return
- // false, otherwise it returns true. This is an undo-able action.
- //
- // Parameters:
- // layername - the name of the layer to change the visibility
- // bVisible - true/false, whether the layer should be visible
- //
- // Returns:
- // true if the layer's visibility was set, false otherwise
- this.setLayerVisibility = function(layername, bVisible) {
- var drawing = getCurrentDrawing();
- var prevVisibility = drawing.getLayerVisibility(layername);
- var layer = drawing.setLayerVisibility(layername, bVisible);
- if (layer) {
- var oldDisplay = prevVisibility ? 'inline' : 'none';
- addCommandToHistory(new ChangeElementCommand(layer, {'display':oldDisplay}, 'Layer Visibility'));
- } else {
- return false;
- }
-
- if (layer == drawing.getCurrentLayer()) {
- clearSelection();
- pathActions.clear();
- }
- // call("changed", [selected]);
- return true;
- };
- // Function: moveSelectedToLayer
- // Moves the selected elements to layername. If the name is not a valid layer name, then false
- // is returned. Otherwise it returns true. This is an undo-able action.
- //
- // Parameters:
- // layername - the name of the layer you want to which you want to move the selected elements
- //
- // Returns:
- // true if the selected elements were moved to the layer, false otherwise.
- this.moveSelectedToLayer = function(layername) {
- // find the layer
- var layer = null;
- var drawing = getCurrentDrawing();
- for (var i = 0; i < drawing.getNumLayers(); ++i) {
- if (drawing.getLayerName(i) == layername) {
- layer = drawing.all_layers[i][1];
- break;
- }
- }
- if (!layer) return false;
-
- var batchCmd = new BatchCommand("Move Elements to Layer");
-
- // loop for each selected element and move it
- var selElems = selectedElements;
- var i = selElems.length;
- while (i--) {
- var elem = selElems[i];
- if (!elem) continue;
- var oldNextSibling = elem.nextSibling;
- // TODO: this is pretty brittle!
- var oldLayer = elem.parentNode;
- layer.appendChild(elem);
- batchCmd.addSubCommand(new MoveElementCommand(elem, oldNextSibling, oldLayer));
- }
-
- addCommandToHistory(batchCmd);
-
- return true;
- };
- this.mergeLayer = function(skipHistory) {
- var batchCmd = new BatchCommand("Merge Layer");
- var drawing = getCurrentDrawing();
- var prev = $(drawing.current_layer).prev()[0];
- if(!prev) return;
- var childs = drawing.current_layer.childNodes;
- var len = childs.length;
- var layerNextSibling = drawing.current_layer.nextSibling;
- batchCmd.addSubCommand(new RemoveElementCommand(drawing.current_layer, layerNextSibling, svgcontent));
- while(drawing.current_layer.firstChild) {
- var ch = drawing.current_layer.firstChild;
- if(ch.localName == 'title') {
- var chNextSibling = ch.nextSibling;
- batchCmd.addSubCommand(new RemoveElementCommand(ch, chNextSibling, drawing.current_layer));
- drawing.current_layer.removeChild(ch);
- continue;
- }
- var oldNextSibling = ch.nextSibling;
- prev.appendChild(ch);
- batchCmd.addSubCommand(new MoveElementCommand(ch, oldNextSibling, drawing.current_layer));
- }
-
- // Remove current layer
- svgcontent.removeChild(drawing.current_layer);
-
- if(!skipHistory) {
- clearSelection();
- identifyLayers();
- call("changed", [svgcontent]);
-
- addCommandToHistory(batchCmd);
- }
-
- drawing.current_layer = prev;
- return batchCmd;
- }
- this.mergeAllLayers = function() {
- var batchCmd = new BatchCommand("Merge all Layers");
- var drawing = getCurrentDrawing();
- drawing.current_layer = drawing.all_layers[drawing.getNumLayers()-1][1];
- while($(svgcontent).children('g').length > 1) {
- batchCmd.addSubCommand(canvas.mergeLayer(true));
- }
-
- clearSelection();
- identifyLayers();
- call("changed", [svgcontent]);
- addCommandToHistory(batchCmd);
- }
- // Function: leaveContext
- // Return from a group context to the regular kind, make any previously
- // disabled elements enabled again
- var leaveContext = this.leaveContext = function() {
- var len = disabled_elems.length;
- if(len) {
- for(var i = 0; i < len; i++) {
- var elem = disabled_elems[i];
-
- var orig = elData(elem, 'orig_opac');
- if(orig !== 1) {
- elem.setAttribute('opacity', orig);
- } else {
- elem.removeAttribute('opacity');
- }
- elem.setAttribute('style', 'pointer-events: inherit');
- }
- disabled_elems = [];
- clearSelection(true);
- call("contextset", null);
- }
- current_group = null;
- }
- // Function: setContext
- // Set the current context (for in-group editing)
- var setContext = this.setContext = function(elem) {
- leaveContext();
- if(typeof elem === 'string') {
- elem = getElem(elem);
- }
- // Edit inside this group
- current_group = elem;
-
- // Disable other elements
- $(elem).parentsUntil('#svgcontent').andSelf().siblings().each(function() {
- var opac = this.getAttribute('opacity') || 1;
- // Store the original's opacity
- elData(this, 'orig_opac', opac);
- this.setAttribute('opacity', opac * .33);
- this.setAttribute('style', 'pointer-events: none');
- disabled_elems.push(this);
- });
- clearSelection();
- call("contextset", current_group);
- }
- // Group: Document functions
- // Function: clear
- // Clears the current document. This is not an undoable action.
- this.clear = function() {
- pathActions.clear();
- clearSelection();
- // clear the svgcontent node
- canvas.clearSvgContentElement();
- // create new document
- canvas.current_drawing_ = new svgedit.draw.Drawing(svgcontent);
- // create empty first layer
- canvas.createLayer("Layer 1");
-
- // clear the undo stack
- canvas.undoMgr.resetUndoStack();
- // reset the selector manager
- selectorManager.initGroup();
- // reset the rubber band box
- rubberBox = selectorManager.getRubberBandBox();
- call("cleared");
- };
- // Function: linkControlPoints
- // Alias function
- this.linkControlPoints = pathActions.linkControlPoints;
- // Function: getContentElem
- // Returns the content DOM element
- this.getContentElem = function() { return svgcontent; };
- // Function: getRootElem
- // Returns the root DOM element
- this.getRootElem = function() { return svgroot; };
- // Function: getSelectedElems
- // Returns the array with selected DOM elements
- this.getSelectedElems = function() { return selectedElements; };
- // Function: getResolution
- // Returns the current dimensions and zoom level in an object
- var getResolution = this.getResolution = function() {
- // var vb = svgcontent.getAttribute("viewBox").split(' ');
- // return {'w':vb[2], 'h':vb[3], 'zoom': current_zoom};
-
- var width = svgcontent.getAttribute("width")/current_zoom;
- var height = svgcontent.getAttribute("height")/current_zoom;
-
- return {
- 'w': width,
- 'h': height,
- 'zoom': current_zoom
- };
- };
- // Function: getZoom
- // Returns the current zoom level
- this.getZoom = function(){return current_zoom;};
- // Function: getVersion
- // Returns a string which describes the revision number of SvgCanvas.
- this.getVersion = function() {
- return "svgcanvas.js ($Rev: 2059 $)";
- };
- // Function: setUiStrings
- // Update interface strings with given values
- //
- // Parameters:
- // strs - Object with strings (see uiStrings for examples)
- this.setUiStrings = function(strs) {
- $.extend(uiStrings, strs.notification);
- }
- // Function: setConfig
- // Update configuration options with given values
- //
- // Parameters:
- // opts - Object with options (see curConfig for examples)
- this.setConfig = function(opts) {
- $.extend(curConfig, opts);
- }
- // Function: getTitle
- // Returns the current group/SVG's title contents
- this.getTitle = function(elem) {
- elem = elem || selectedElements[0];
- if(!elem) return;
- elem = $(elem).data('gsvg') || $(elem).data('symbol') || elem;
- var childs = elem.childNodes;
- for (var i=0; i<childs.length; i++) {
- if(childs[i].nodeName == 'title') {
- return childs[i].textContent;
- }
- }
- return 'image';//hack for Chamilo. Avoid missing the title inside a svg document
- }
- // Function: setGroupTitle
- // Sets the group/SVG's title content
- // TODO: Combine this with setDocumentTitle
- this.setGroupTitle = function(val) {
- var elem = selectedElements[0];
- elem = $(elem).data('gsvg') || elem;
-
- var ts = $(elem).children('title');
-
- var batchCmd = new BatchCommand("Set Label");
-
- if(!val.length) {
- // Remove title element
- var tsNextSibling = ts.nextSibling;
- batchCmd.addSubCommand(new RemoveElementCommand(ts[0], tsNextSibling, elem));
- ts.remove();
- } else if(ts.length) {
- // Change title contents
- var title = ts[0];
- batchCmd.addSubCommand(new ChangeElementCommand(title, {'#text': title.textContent}));
- title.textContent = val;
- } else {
- // Add title element
- title = svgdoc.createElementNS(svgns, "title");
- title.textContent = val;
- $(elem).prepend(title);
- batchCmd.addSubCommand(new InsertElementCommand(title));
- }
- addCommandToHistory(batchCmd);
- }
- // Function: getDocumentTitle
- // Returns the current document title or an empty string if not found
- this.getDocumentTitle = function() {
- return canvas.getTitle(svgcontent);
- }
- // Function: setDocumentTitle
- // Adds/updates a title element for the document with the given name.
- // This is an undoable action
- //
- // Parameters:
- // newtitle - String with the new title
- this.setDocumentTitle = function(newtitle) {
- var childs = svgcontent.childNodes, doc_title = false, old_title = '';
-
- var batchCmd = new BatchCommand("Change Image Title");
-
- for (var i=0; i<childs.length; i++) {
- if(childs[i].nodeName == 'title') {
- doc_title = childs[i];
- old_title = doc_title.textContent;
- break;
- }
- }
- if(!doc_title) {
- doc_title = svgdoc.createElementNS(svgns, "title");
- svgcontent.insertBefore(doc_title, svgcontent.firstChild);
- }
-
- if(newtitle.length) {
- doc_title.textContent = newtitle;
- } else {
- // No title given, so element is not necessary
- doc_title.parentNode.removeChild(doc_title);
- }
- batchCmd.addSubCommand(new ChangeElementCommand(doc_title, {'#text': old_title}));
- addCommandToHistory(batchCmd);
- }
- // Function: getEditorNS
- // Returns the editor's namespace URL, optionally adds it to root element
- //
- // Parameters:
- // add - Boolean to indicate whether or not to add the namespace value
- this.getEditorNS = function(add) {
- if(add) {
- svgcontent.setAttribute('xmlns:se', se_ns);
- }
- return se_ns;
- }
- // Function: setResolution
- // Changes the document's dimensions to the given size
- //
- // Parameters:
- // x - Number with the width of the new dimensions in user units.
- // Can also be the string "fit" to indicate "fit to content"
- // y - Number with the height of the new dimensions in user units.
- //
- // Returns:
- // Boolean to indicate if resolution change was succesful.
- // It will fail on "fit to content" option with no content to fit to.
- this.setResolution = function(x, y) {
- var res = getResolution();
- var w = res.w, h = res.h;
- var batchCmd;
- if(x == 'fit') {
- // Get bounding box
- var bbox = getStrokedBBox();
-
- if(bbox) {
- batchCmd = new BatchCommand("Fit Canvas to Content");
- var visEls = getVisibleElements();
- addToSelection(visEls);
- var dx = [], dy = [];
- $.each(visEls, function(i, item) {
- dx.push(bbox.x*-1);
- dy.push(bbox.y*-1);
- });
-
- var cmd = canvas.moveSelectedElements(dx, dy, true);
- batchCmd.addSubCommand(cmd);
- clearSelection();
-
- x = Math.round(bbox.width);
- y = Math.round(bbox.height);
- } else {
- return false;
- }
- }
- if (x != w || y != h) {
- var handle = svgroot.suspendRedraw(1000);
- if(!batchCmd) {
- batchCmd = new BatchCommand("Change Image Dimensions");
- }
- x = convertToNum('width', x);
- y = convertToNum('height', y);
-
- svgcontent.setAttribute('width', x);
- svgcontent.setAttribute('height', y);
-
- this.contentW = x;
- this.contentH = y;
- batchCmd.addSubCommand(new ChangeElementCommand(svgcontent, {"width":w, "height":h}));
- svgcontent.setAttribute("viewBox", [0, 0, x/current_zoom, y/current_zoom].join(' '));
- batchCmd.addSubCommand(new ChangeElementCommand(svgcontent, {"viewBox": ["0 0", w, h].join(' ')}));
-
- addCommandToHistory(batchCmd);
- svgroot.unsuspendRedraw(handle);
- call("changed", [svgcontent]);
- }
- return true;
- };
- // Function: getOffset
- // Returns an object with x, y values indicating the svgcontent element's
- // position in the editor's canvas.
- this.getOffset = function() {
- return $(svgcontent).attr(['x', 'y']);
- }
- // Function: setBBoxZoom
- // Sets the zoom level on the canvas-side based on the given value
- //
- // Parameters:
- // val - Bounding box object to zoom to or string indicating zoom option
- // editor_w - Integer with the editor's workarea box's width
- // editor_h - Integer with the editor's workarea box's height
- this.setBBoxZoom = function(val, editor_w, editor_h) {
- var spacer = .85;
- var bb;
- var calcZoom = function(bb) {
- if(!bb) return false;
- var w_zoom = Math.round((editor_w / bb.width)*100 * spacer)/100;
- var h_zoom = Math.round((editor_h / bb.height)*100 * spacer)/100;
- var zoomlevel = Math.min(w_zoom,h_zoom);
- canvas.setZoom(zoomlevel);
- return {'zoom': zoomlevel, 'bbox': bb};
- }
-
- if(typeof val == 'object') {
- bb = val;
- if(bb.width == 0 || bb.height == 0) {
- var newzoom = bb.zoom?bb.zoom:current_zoom * bb.factor;
- canvas.setZoom(newzoom);
- return {'zoom': current_zoom, 'bbox': bb};
- }
- return calcZoom(bb);
- }
- switch (val) {
- case 'selection':
- if(!selectedElements[0]) return;
- var sel_elems = $.map(selectedElements, function(n){ if(n) return n; });
- bb = getStrokedBBox(sel_elems);
- break;
- case 'canvas':
- var res = getResolution();
- spacer = .95;
- bb = {width:res.w, height:res.h ,x:0, y:0};
- break;
- case 'content':
- bb = getStrokedBBox();
- break;
- case 'layer':
- bb = getStrokedBBox(getVisibleElements(getCurrentDrawing().getCurrentLayer()));
- break;
- default:
- return;
- }
- return calcZoom(bb);
- }
- // Function: setZoom
- // Sets the zoom to the given level
- //
- // Parameters:
- // zoomlevel - Float indicating the zoom level to change to
- this.setZoom = function(zoomlevel) {
- var res = getResolution();
- svgcontent.setAttribute("viewBox", "0 0 " + res.w/zoomlevel + " " + res.h/zoomlevel);
- current_zoom = zoomlevel;
- $.each(selectedElements, function(i, elem) {
- if(!elem) return;
- selectorManager.requestSelector(elem).resize();
- });
- pathActions.zoomChange();
- runExtensions("zoomChanged", zoomlevel);
- }
- // Function: getMode
- // Returns the current editor mode string
- this.getMode = function() {
- return current_mode;
- };
- // Function: setMode
- // Sets the editor's mode to the given string
- //
- // Parameters:
- // name - String with the new mode to change to
- this.setMode = function(name) {
- pathActions.clear(true);
- textActions.clear();
- cur_properties = (selectedElements[0] && selectedElements[0].nodeName == 'text') ? cur_text : cur_shape;
- current_mode = name;
- };
- // Group: Element Styling
- // Function: getColor
- // Returns the current fill/stroke option
- this.getColor = function(type) {
- return cur_properties[type];
- };
- // Function: setColor
- // Change the current stroke/fill color/gradient value
- //
- // Parameters:
- // type - String indicating fill or stroke
- // val - The value to set the stroke attribute to
- // preventUndo - Boolean indicating whether or not this should be and undoable option
- this.setColor = function(type, val, preventUndo) {
- cur_shape[type] = val;
- cur_properties[type + '_paint'] = {type:"solidColor"};
- var elems = [];
- var i = selectedElements.length;
- while (i--) {
- var elem = selectedElements[i];
- if (elem) {
- if (elem.tagName == "g")
- svgedit.utilities.walkTree(elem, function(e){if(e.nodeName!="g") elems.push(e);});
- else {
- if(type == 'fill') {
- if(elem.tagName != "polyline" && elem.tagName != "line") {
- elems.push(elem);
- }
- } else {
- elems.push(elem);
- }
- }
- }
- }
- if (elems.length > 0) {
- if (!preventUndo) {
- changeSelectedAttribute(type, val, elems);
- call("changed", elems);
- } else
- changeSelectedAttributeNoUndo(type, val, elems);
- }
- }
- // Function: findDefs
- // Return the document's <defs> element, create it first if necessary
- var findDefs = function() {
- var defs = svgcontent.getElementsByTagNameNS(svgns, "defs");
- if (defs.length > 0) {
- defs = defs[0];
- }
- else {
- defs = svgdoc.createElementNS(svgns, "defs" );
- if(svgcontent.firstChild) {
- // first child is a comment, so call nextSibling
- svgcontent.insertBefore( defs, svgcontent.firstChild.nextSibling);
- } else {
- svgcontent.appendChild(defs);
- }
- }
- return defs;
- };
- // Function: setGradient
- // Apply the current gradient to selected element's fill or stroke
- //
- // Parameters
- // type - String indicating "fill" or "stroke" to apply to an element
- var setGradient = this.setGradient = function(type) {
- if(!cur_properties[type + '_paint'] || cur_properties[type + '_paint'].type == "solidColor") return;
- var grad = canvas[type + 'Grad'];
- // find out if there is a duplicate gradient already in the defs
- var duplicate_grad = findDuplicateGradient(grad);
- var defs = findDefs();
- // no duplicate found, so import gradient into defs
- if (!duplicate_grad) {
- var orig_grad = grad;
- grad = defs.appendChild( svgdoc.importNode(grad, true) );
- // get next id and set it on the grad
- grad.id = getNextId();
- }
- else { // use existing gradient
- grad = duplicate_grad;
- }
- canvas.setColor(type, "url(#" + grad.id + ")");
- }
- // Function: findDuplicateGradient
- // Check if exact gradient already exists
- //
- // Parameters:
- // grad - The gradient DOM element to compare to others
- //
- // Returns:
- // The existing gradient if found, null if not
- var findDuplicateGradient = function(grad) {
- var defs = findDefs();
- var existing_grads = $(defs).find("linearGradient, radialGradient");
- var i = existing_grads.length;
- var rad_attrs = ['r','cx','cy','fx','fy'];
- while (i--) {
- var og = existing_grads[i];
- if(grad.tagName == "linearGradient") {
- if (grad.getAttribute('x1') != og.getAttribute('x1') ||
- grad.getAttribute('y1') != og.getAttribute('y1') ||
- grad.getAttribute('x2') != og.getAttribute('x2') ||
- grad.getAttribute('y2') != og.getAttribute('y2'))
- {
- continue;
- }
- } else {
- var grad_attrs = $(grad).attr(rad_attrs);
- var og_attrs = $(og).attr(rad_attrs);
-
- var diff = false;
- $.each(rad_attrs, function(i, attr) {
- if(grad_attrs[attr] != og_attrs[attr]) diff = true;
- });
-
- if(diff) continue;
- }
-
- // else could be a duplicate, iterate through stops
- var stops = grad.getElementsByTagNameNS(svgns, "stop");
- var ostops = og.getElementsByTagNameNS(svgns, "stop");
- if (stops.length != ostops.length) {
- continue;
- }
- var j = stops.length;
- while(j--) {
- var stop = stops[j];
- var ostop = ostops[j];
- if (stop.getAttribute('offset') != ostop.getAttribute('offset') ||
- stop.getAttribute('stop-opacity') != ostop.getAttribute('stop-opacity') ||
- stop.getAttribute('stop-color') != ostop.getAttribute('stop-color'))
- {
- break;
- }
- }
- if (j == -1) {
- return og;
- }
- } // for each gradient in defs
- return null;
- };
- function reorientGrads(elem, m) {
- var bb = svgedit.utilities.getBBox(elem);
- for(var i = 0; i < 2; i++) {
- var type = i === 0 ? 'fill' : 'stroke';
- var attrVal = elem.getAttribute(type);
- if(attrVal && attrVal.indexOf('url(') === 0) {
- var grad = getRefElem(attrVal);
- if(grad.tagName === 'linearGradient') {
- var x1 = grad.getAttribute('x1') || 0;
- var y1 = grad.getAttribute('y1') || 0;
- var x2 = grad.getAttribute('x2') || 1;
- var y2 = grad.getAttribute('y2') || 0;
-
- // Convert to USOU points
- x1 = (bb.width * x1) + bb.x;
- y1 = (bb.height * y1) + bb.y;
- x2 = (bb.width * x2) + bb.x;
- y2 = (bb.height * y2) + bb.y;
-
- // Transform those points
- var pt1 = transformPoint(x1, y1, m);
- var pt2 = transformPoint(x2, y2, m);
-
- // Convert back to BB points
- var g_coords = {};
-
- g_coords.x1 = (pt1.x - bb.x) / bb.width;
- g_coords.y1 = (pt1.y - bb.y) / bb.height;
- g_coords.x2 = (pt2.x - bb.x) / bb.width;
- g_coords.y2 = (pt2.y - bb.y) / bb.height;
-
- var newgrad = grad.cloneNode(true);
- $(newgrad).attr(g_coords);
-
- newgrad.id = getNextId();
- findDefs().appendChild(newgrad);
- elem.setAttribute(type, 'url(#' + newgrad.id + ')');
- }
- }
- }
- }
- // Function: setPaint
- // Set a color/gradient to a fill/stroke
- //
- // Parameters:
- // type - String with "fill" or "stroke"
- // paint - The jGraduate paint object to apply
- this.setPaint = function(type, paint) {
- // make a copy
- var p = new $.jGraduate.Paint(paint);
- this.setPaintOpacity(type, p.alpha/100, true);
- // now set the current paint object
- cur_properties[type + '_paint'] = p;
- switch ( p.type ) {
- case "solidColor":
- this.setColor(type, p.solidColor != "none" ? "#"+p.solidColor : "none");;
- break;
- case "linearGradient":
- case "radialGradient":
- canvas[type + 'Grad'] = p[p.type];
- setGradient(type);
- break;
- default:
- // console.log("none!");
- }
- };
- // this.setStrokePaint = function(p) {
- // // make a copy
- // var p = new $.jGraduate.Paint(p);
- // this.setStrokeOpacity(p.alpha/100);
- //
- // // now set the current paint object
- // cur_properties.stroke_paint = p;
- // switch ( p.type ) {
- // case "solidColor":
- // this.setColor('stroke', p.solidColor != "none" ? "#"+p.solidColor : "none");;
- // break;
- // case "linearGradient"
- // case "radialGradient"
- // canvas.strokeGrad = p[p.type];
- // setGradient(type);
- // default:
- // // console.log("none!");
- // }
- // };
- //
- // this.setFillPaint = function(p, addGrad) {
- // // make a copy
- // var p = new $.jGraduate.Paint(p);
- // this.setFillOpacity(p.alpha/100, true);
- //
- // // now set the current paint object
- // cur_properties.fill_paint = p;
- // if (p.type == "solidColor") {
- // this.setColor('fill', p.solidColor != "none" ? "#"+p.solidColor : "none");
- // }
- // else if(p.type == "linearGradient") {
- // canvas.fillGrad = p.linearGradient;
- // if(addGrad) setGradient();
- // }
- // else if(p.type == "radialGradient") {
- // canvas.fillGrad = p.radialGradient;
- // if(addGrad) setGradient();
- // }
- // else {
- // // console.log("none!");
- // }
- // };
- // Function: getStrokeWidth
- // Returns the current stroke-width value
- this.getStrokeWidth = function() {
- return cur_properties.stroke_width;
- };
- // Function: setStrokeWidth
- // Sets the stroke width for the current selected elements
- // When attempting to set a line's width to 0, this changes it to 1 instead
- //
- // Parameters:
- // val - A Float indicating the new stroke width value
- this.setStrokeWidth = function(val) {
- if(val == 0 && ['line', 'path'].indexOf(current_mode) >= 0) {
- canvas.setStrokeWidth(1);
- return;
- }
- cur_properties.stroke_width = val;
-
- var elems = [];
- var i = selectedElements.length;
- while (i--) {
- var elem = selectedElements[i];
- if (elem) {
- if (elem.tagName == "g")
- svgedit.utilities.walkTree(elem, function(e){if(e.nodeName!="g") elems.push(e);});
- else
- elems.push(elem);
- }
- }
- if (elems.length > 0) {
- changeSelectedAttribute("stroke-width", val, elems);
- call("changed", selectedElements);
- }
- };
- // Function: setStrokeAttr
- // Set the given stroke-related attribute the given value for selected elements
- //
- // Parameters:
- // attr - String with the attribute name
- // val - String or number with the attribute value
- this.setStrokeAttr = function(attr, val) {
- cur_shape[attr.replace('-','_')] = val;
- var elems = [];
- var i = selectedElements.length;
- while (i--) {
- var elem = selectedElements[i];
- if (elem) {
- if (elem.tagName == "g")
- svgedit.utilities.walkTree(elem, function(e){if(e.nodeName!="g") elems.push(e);});
- else
- elems.push(elem);
- }
- }
- if (elems.length > 0) {
- changeSelectedAttribute(attr, val, elems);
- call("changed", selectedElements);
- }
- };
- // Function: getStyle
- // Returns current style options
- this.getStyle = function() {
- return cur_shape;
- }
- // Function: getOpacity
- // Returns the current opacity
- this.getOpacity = function() {
- return cur_shape.opacity;
- };
- // Function: setOpacity
- // Sets the given opacity to the current selected elements
- this.setOpacity = function(val) {
- cur_shape.opacity = val;
- changeSelectedAttribute("opacity", val);
- };
- // Function: getOpacity
- // Returns the current fill opacity
- this.getFillOpacity = function() {
- return cur_shape.fill_opacity;
- };
- // Function: getStrokeOpacity
- // Returns the current stroke opacity
- this.getStrokeOpacity = function() {
- return cur_shape.stroke_opacity;
- };
- // Function: setPaintOpacity
- // Sets the current fill/stroke opacity
- //
- // Parameters:
- // type - String with "fill" or "stroke"
- // val - Float with the new opacity value
- // preventUndo - Boolean indicating whether or not this should be an undoable action
- this.setPaintOpacity = function(type, val, preventUndo) {
- cur_shape[type + '_opacity'] = val;
- if (!preventUndo)
- changeSelectedAttribute(type + "-opacity", val);
- else
- changeSelectedAttributeNoUndo(type + "-opacity", val);
- };
- // Function: getBlur
- // Gets the stdDeviation blur value of the given element
- //
- // Parameters:
- // elem - The element to check the blur value for
- this.getBlur = function(elem) {
- var val = 0;
- // var elem = selectedElements[0];
-
- if(elem) {
- var filter_url = elem.getAttribute('filter');
- if(filter_url) {
- var blur = getElem(elem.id + '_blur');
- if(blur) {
- val = blur.firstChild.getAttribute('stdDeviation');
- }
- }
- }
- return val;
- };
- (function() {
- var cur_command = null;
- var filter = null;
- var filterHidden = false;
-
- // Function: setBlurNoUndo
- // Sets the stdDeviation blur value on the selected element without being undoable
- //
- // Parameters:
- // val - The new stdDeviation value
- canvas.setBlurNoUndo = function(val) {
- if(!filter) {
- canvas.setBlur(val);
- return;
- }
- if(val === 0) {
- // Don't change the StdDev, as that will hide the element.
- // Instead, just remove the value for "filter"
- changeSelectedAttributeNoUndo("filter", "");
- filterHidden = true;
- } else {
- var elem = selectedElements[0];
- if(filterHidden) {
- changeSelectedAttributeNoUndo("filter", 'url(#' + elem.id + '_blur)');
- }
- if(svgedit.browser.isWebkit()) {
- console.log('e', elem);
- elem.removeAttribute('filter');
- elem.setAttribute('filter', 'url(#' + elem.id + '_blur)');
- }
- changeSelectedAttributeNoUndo("stdDeviation", val, [filter.firstChild]);
- canvas.setBlurOffsets(filter, val);
- }
- }
-
- function finishChange() {
- var bCmd = canvas.undoMgr.finishUndoableChange();
- cur_command.addSubCommand(bCmd);
- addCommandToHistory(cur_command);
- cur_command = null;
- filter = null;
- }
- // Function: setBlurOffsets
- // Sets the x, y, with, height values of the filter element in order to
- // make the blur not be clipped. Removes them if not neeeded
- //
- // Parameters:
- // filter - The filter DOM element to update
- // stdDev - The standard deviation value on which to base the offset size
- canvas.setBlurOffsets = function(filter, stdDev) {
- if(stdDev > 3) {
- // TODO: Create algorithm here where size is based on expected blur
- assignAttributes(filter, {
- x: '-50%',
- y: '-50%',
- width: '200%',
- height: '200%'
- }, 100);
- } else {
- // Removing these attributes hides text in Chrome (see Issue 579)
- if(!svgedit.browser.isWebkit()) {
- filter.removeAttribute('x');
- filter.removeAttribute('y');
- filter.removeAttribute('width');
- filter.removeAttribute('height');
- }
- }
- }
- // Function: setBlur
- // Adds/updates the blur filter to the selected element
- //
- // Parameters:
- // val - Float with the new stdDeviation blur value
- // complete - Boolean indicating whether or not the action should be completed (to add to the undo manager)
- canvas.setBlur = function(val, complete) {
- if(cur_command) {
- finishChange();
- return;
- }
-
- // Looks for associated blur, creates one if not found
- var elem = selectedElements[0];
- var elem_id = elem.id;
- filter = getElem(elem_id + '_blur');
-
- val -= 0;
-
- var batchCmd = new BatchCommand();
-
- // Blur found!
- if(filter) {
- if(val === 0) {
- filter = null;
- }
- } else {
- // Not found, so create
- var newblur = addSvgElementFromJson({ "element": "feGaussianBlur",
- "attr": {
- "in": 'SourceGraphic',
- "stdDeviation": val
- }
- });
-
- filter = addSvgElementFromJson({ "element": "filter",
- "attr": {
- "id": elem_id + '_blur'
- }
- });
-
- filter.appendChild(newblur);
- findDefs().appendChild(filter);
-
- batchCmd.addSubCommand(new InsertElementCommand(filter));
- }
- var changes = {filter: elem.getAttribute('filter')};
-
- if(val === 0) {
- elem.removeAttribute("filter");
- batchCmd.addSubCommand(new ChangeElementCommand(elem, changes));
- return;
- } else {
- changeSelectedAttribute("filter", 'url(#' + elem_id + '_blur)');
-
- batchCmd.addSubCommand(new ChangeElementCommand(elem, changes));
-
- canvas.setBlurOffsets(filter, val);
- }
-
- cur_command = batchCmd;
- canvas.undoMgr.beginUndoableChange("stdDeviation", [filter?filter.firstChild:null]);
- if(complete) {
- canvas.setBlurNoUndo(val);
- finishChange();
- }
- };
- }());
- // Function: getBold
- // Check whether selected element is bold or not
- //
- // Returns:
- // Boolean indicating whether or not element is bold
- this.getBold = function() {
- // should only have one element selected
- var selected = selectedElements[0];
- if (selected != null && selected.tagName == "text" &&
- selectedElements[1] == null)
- {
- return (selected.getAttribute("font-weight") == "bold");
- }
- return false;
- };
- // Function: setBold
- // Make the selected element bold or normal
- //
- // Parameters:
- // b - Boolean indicating bold (true) or normal (false)
- this.setBold = function(b) {
- var selected = selectedElements[0];
- if (selected != null && selected.tagName == "text" &&
- selectedElements[1] == null)
- {
- changeSelectedAttribute("font-weight", b ? "bold" : "normal");
- }
- if(!selectedElements[0].textContent) {
- textActions.setCursor();
- }
- };
- // Function: getItalic
- // Check whether selected element is italic or not
- //
- // Returns:
- // Boolean indicating whether or not element is italic
- this.getItalic = function() {
- var selected = selectedElements[0];
- if (selected != null && selected.tagName == "text" &&
- selectedElements[1] == null)
- {
- return (selected.getAttribute("font-style") == "italic");
- }
- return false;
- };
- // Function: setItalic
- // Make the selected element italic or normal
- //
- // Parameters:
- // b - Boolean indicating italic (true) or normal (false)
- this.setItalic = function(i) {
- var selected = selectedElements[0];
- if (selected != null && selected.tagName == "text" &&
- selectedElements[1] == null)
- {
- changeSelectedAttribute("font-style", i ? "italic" : "normal");
- }
- if(!selectedElements[0].textContent) {
- textActions.setCursor();
- }
- };
- // Function: getFontFamily
- // Returns the current font family
- this.getFontFamily = function() {
- return cur_text.font_family;
- };
- // Function: setFontFamily
- // Set the new font family
- //
- // Parameters:
- // val - String with the new font family
- this.setFontFamily = function(val) {
- cur_text.font_family = val;
- changeSelectedAttribute("font-family", val);
- if(selectedElements[0] && !selectedElements[0].textContent) {
- textActions.setCursor();
- }
- };
- // Function: setFontColor
- // Set the new font color
- //
- // Parameters:
- // val - String with the new font color
- this.setFontColor = function(val) {
- cur_text.fill = val;
- changeSelectedAttribute("fill", val);
- };
- // Function: getFontColor
- // Returns the current font color
- this.getFontSize = function() {
- return cur_text.fill;
- };
- // Function: getFontSize
- // Returns the current font size
- this.getFontSize = function() {
- return cur_text.font_size;
- };
- // Function: setFontSize
- // Applies the given font size to the selected element
- //
- // Parameters:
- // val - Float with the new font size
- this.setFontSize = function(val) {
- cur_text.font_size = val;
- changeSelectedAttribute("font-size", val);
- if(!selectedElements[0].textContent) {
- textActions.setCursor();
- }
- };
- // Function: getText
- // Returns the current text (textContent) of the selected element
- this.getText = function() {
- var selected = selectedElements[0];
- if (selected == null) { return ""; }
- return selected.textContent;
- };
- // Function: setTextContent
- // Updates the text element with the given string
- //
- // Parameters:
- // val - String with the new text
- this.setTextContent = function(val) {
- changeSelectedAttribute("#text", val);
- textActions.init(val);
- textActions.setCursor();
- };
- // Function: setImageURL
- // Sets the new image URL for the selected image element. Updates its size if
- // a new URL is given
- //
- // Parameters:
- // val - String with the image URL/path
- this.setImageURL = function(val) {
- var elem = selectedElements[0];
- if(!elem) return;
-
- var attrs = $(elem).attr(['width', 'height']);
- var setsize = (!attrs.width || !attrs.height);
- var cur_href = getHref(elem);
-
- // Do nothing if no URL change or size change
- if(cur_href !== val) {
- setsize = true;
- } else if(!setsize) return;
- var batchCmd = new BatchCommand("Change Image URL");
- setHref(elem, val);
- batchCmd.addSubCommand(new ChangeElementCommand(elem, {
- "#href": cur_href
- }));
- if(setsize) {
- $(new Image()).load(function() {
- var changes = $(elem).attr(['width', 'height']);
-
- $(elem).attr({
- width: this.width,
- height: this.height
- });
-
- selectorManager.requestSelector(elem).resize();
-
- batchCmd.addSubCommand(new ChangeElementCommand(elem, changes));
- addCommandToHistory(batchCmd);
- call("changed", [elem]);
- }).attr('src',val);
- } else {
- addCommandToHistory(batchCmd);
- }
- };
- // Function: setLinkURL
- // Sets the new link URL for the selected anchor element.
- //
- // Parameters:
- // val - String with the link URL/path
- this.setLinkURL = function(val) {
- var elem = selectedElements[0];
- if(!elem) return;
- if(elem.tagName !== 'a') {
- // See if parent is an anchor
- var parents_a = $(elem).parents('a');
- if(parents_a.length) {
- elem = parents_a[0];
- } else {
- return;
- }
- }
-
- var cur_href = getHref(elem);
-
- if(cur_href === val) return;
-
- var batchCmd = new BatchCommand("Change Link URL");
- setHref(elem, val);
- batchCmd.addSubCommand(new ChangeElementCommand(elem, {
- "#href": cur_href
- }));
- addCommandToHistory(batchCmd);
- };
- // Function: setRectRadius
- // Sets the rx & ry values to the selected rect element to change its corner radius
- //
- // Parameters:
- // val - The new radius
- this.setRectRadius = function(val) {
- var selected = selectedElements[0];
- if (selected != null && selected.tagName == "rect") {
- var r = selected.getAttribute("rx");
- if (r != val) {
- selected.setAttribute("rx", val);
- selected.setAttribute("ry", val);
- addCommandToHistory(new ChangeElementCommand(selected, {"rx":r, "ry":r}, "Radius"));
- call("changed", [selected]);
- }
- }
- };
- // Function: makeHyperlink
- // Wraps the selected element(s) in an anchor element or converts group to one
- this.makeHyperlink = function(url) {
- canvas.groupSelectedElements('a', url);
-
- // TODO: If element is a single "g", convert to "a"
- // if(selectedElements.length > 1 && selectedElements[1]) {
- }
- // Function: removeHyperlink
- this.removeHyperlink = function() {
- canvas.ungroupSelectedElement();
- }
- // Group: Element manipulation
- // Function: setSegType
- // Sets the new segment type to the selected segment(s).
- //
- // Parameters:
- // new_type - Integer with the new segment type
- // See http://www.w3.org/TR/SVG/paths.html#InterfaceSVGPathSeg for list
- this.setSegType = function(new_type) {
- pathActions.setSegType(new_type);
- }
- // TODO(codedread): Remove the getBBox argument and split this function into two.
- // Function: convertToPath
- // Convert selected element to a path, or get the BBox of an element-as-path
- //
- // Parameters:
- // elem - The DOM element to be converted
- // getBBox - Boolean on whether or not to only return the path's BBox
- //
- // Returns:
- // If the getBBox flag is true, the resulting path's bounding box object.
- // Otherwise the resulting path element is returned.
- this.convertToPath = function(elem, getBBox) {
- if(elem == null) {
- var elems = selectedElements;
- $.each(selectedElements, function(i, elem) {
- if(elem) canvas.convertToPath(elem);
- });
- return;
- }
-
- if(!getBBox) {
- var batchCmd = new BatchCommand("Convert element to Path");
- }
-
- var attrs = getBBox?{}:{
- "fill": cur_shape.fill,
- "fill-opacity": cur_shape.fill_opacity,
- "stroke": cur_shape.stroke,
- "stroke-width": cur_shape.stroke_width,
- "stroke-dasharray": cur_shape.stroke_dasharray,
- "stroke-linejoin": cur_shape.stroke_linejoin,
- "stroke-linecap": cur_shape.stroke_linecap,
- "stroke-opacity": cur_shape.stroke_opacity,
- "opacity": cur_shape.opacity,
- "visibility":"hidden"
- };
-
- // any attribute on the element not covered by the above
- // TODO: make this list global so that we can properly maintain it
- // TODO: what about @transform, @clip-rule, @fill-rule, etc?
- $.each(['marker-start', 'marker-end', 'marker-mid', 'filter', 'clip-path'], function() {
- if (elem.getAttribute(this)) {
- attrs[this] = elem.getAttribute(this);
- }
- });
-
- var path = addSvgElementFromJson({
- "element": "path",
- "attr": attrs
- });
-
- var eltrans = elem.getAttribute("transform");
- if(eltrans) {
- path.setAttribute("transform",eltrans);
- }
-
- var id = elem.id;
- var parent = elem.parentNode;
- if(elem.nextSibling) {
- parent.insertBefore(path, elem);
- } else {
- parent.appendChild(path);
- }
-
- var d = '';
-
- var joinSegs = function(segs) {
- $.each(segs, function(j, seg) {
- var l = seg[0], pts = seg[1];
- d += l;
- for(var i=0; i < pts.length; i+=2) {
- d += (pts[i] +','+pts[i+1]) + ' ';
- }
- });
- }
- // Possibly the cubed root of 6, but 1.81 works best
- var num = 1.81;
- switch (elem.tagName) {
- case 'ellipse':
- case 'circle':
- var a = $(elem).attr(['rx', 'ry', 'cx', 'cy']);
- var cx = a.cx, cy = a.cy, rx = a.rx, ry = a.ry;
- if(elem.tagName == 'circle') {
- rx = ry = $(elem).attr('r');
- }
-
- joinSegs([
- ['M',[(cx-rx),(cy)]],
- ['C',[(cx-rx),(cy-ry/num), (cx-rx/num),(cy-ry), (cx),(cy-ry)]],
- ['C',[(cx+rx/num),(cy-ry), (cx+rx),(cy-ry/num), (cx+rx),(cy)]],
- ['C',[(cx+rx),(cy+ry/num), (cx+rx/num),(cy+ry), (cx),(cy+ry)]],
- ['C',[(cx-rx/num),(cy+ry), (cx-rx),(cy+ry/num), (cx-rx),(cy)]],
- ['Z',[]]
- ]);
- break;
- case 'path':
- d = elem.getAttribute('d');
- break;
- case 'line':
- var a = $(elem).attr(["x1", "y1", "x2", "y2"]);
- d = "M"+a.x1+","+a.y1+"L"+a.x2+","+a.y2;
- break;
- case 'polyline':
- case 'polygon':
- d = "M" + elem.getAttribute('points');
- break;
- case 'rect':
- var r = $(elem).attr(['rx', 'ry']);
- var rx = r.rx, ry = r.ry;
- var b = elem.getBBox();
- var x = b.x, y = b.y, w = b.width, h = b.height;
- var num = 4-num; // Why? Because!
-
- if(!rx && !ry) {
- // Regular rect
- joinSegs([
- ['M',[x, y]],
- ['L',[x+w, y]],
- ['L',[x+w, y+h]],
- ['L',[x, y+h]],
- ['L',[x, y]],
- ['Z',[]]
- ]);
- } else {
- joinSegs([
- ['M',[x, y+ry]],
- ['C',[x,y+ry/num, x+rx/num,y, x+rx,y]],
- ['L',[x+w-rx, y]],
- ['C',[x+w-rx/num,y, x+w,y+ry/num, x+w,y+ry]],
- ['L',[x+w, y+h-ry]],
- ['C',[x+w, y+h-ry/num, x+w-rx/num,y+h, x+w-rx,y+h]],
- ['L',[x+rx, y+h]],
- ['C',[x+rx/num, y+h, x,y+h-ry/num, x,y+h-ry]],
- ['L',[x, y+ry]],
- ['Z',[]]
- ]);
- }
- break;
- default:
- path.parentNode.removeChild(path);
- break;
- }
-
- if(d) {
- path.setAttribute('d',d);
- }
-
- if(!getBBox) {
- // Replace the current element with the converted one
-
- // Reorient if it has a matrix
- if(eltrans) {
- var tlist = getTransformList(path);
- if(hasMatrixTransform(tlist)) {
- pathActions.resetOrientation(path);
- }
- }
-
- var nextSibling = elem.nextSibling;
- batchCmd.addSubCommand(new RemoveElementCommand(elem, nextSibling, parent));
- batchCmd.addSubCommand(new InsertElementCommand(path));
- clearSelection();
- elem.parentNode.removeChild(elem)
- path.setAttribute('id', id);
- path.removeAttribute("visibility");
- addToSelection([path], true);
-
- addCommandToHistory(batchCmd);
-
- } else {
- // Get the correct BBox of the new path, then discard it
- pathActions.resetOrientation(path);
- var bb = false;
- try {
- bb = path.getBBox();
- } catch(e) {
- // Firefox fails
- }
- path.parentNode.removeChild(path);
- return bb;
- }
- };
- // Function: changeSelectedAttributeNoUndo
- // This function makes the changes to the elements. It does not add the change
- // to the history stack.
- //
- // Parameters:
- // attr - String with the attribute name
- // newValue - String or number with the new attribute value
- // elems - The DOM elements to apply the change to
- var changeSelectedAttributeNoUndo = function(attr, newValue, elems) {
- var handle = svgroot.suspendRedraw(1000);
- if(current_mode == 'pathedit') {
- // Editing node
- pathActions.moveNode(attr, newValue);
- }
- var elems = elems || selectedElements;
- var i = elems.length;
- var no_xy_elems = ['g', 'polyline', 'path'];
- var good_g_attrs = ['transform', 'opacity', 'filter'];
-
- while (i--) {
- var elem = elems[i];
- if (elem == null) continue;
-
- // Go into "select" mode for text changes
- if(current_mode === "textedit" && attr !== "#text" && elem.textContent.length) {
- textActions.toSelectMode(elem);
- }
-
- // Set x,y vals on elements that don't have them
- if((attr === 'x' || attr === 'y') && no_xy_elems.indexOf(elem.tagName) >= 0) {
- var bbox = getStrokedBBox([elem]);
- var diff_x = attr === 'x' ? newValue - bbox.x : 0;
- var diff_y = attr === 'y' ? newValue - bbox.y : 0;
- canvas.moveSelectedElements(diff_x*current_zoom, diff_y*current_zoom, true);
- continue;
- }
-
- // only allow the transform/opacity/filter attribute to change on <g> elements, slightly hacky
- // TODO: FIXME: This doesn't seem right. Where's the body of this if statement?
- if (elem.tagName === "g" && good_g_attrs.indexOf(attr) >= 0);
- var oldval = attr === "#text" ? elem.textContent : elem.getAttribute(attr);
- if (oldval == null) oldval = "";
- if (oldval !== String(newValue)) {
- if (attr == "#text") {
- var old_w = svgedit.utilities.getBBox(elem).width;
- elem.textContent = newValue;
-
- // FF bug occurs on on rotated elements
- if(/rotate/.test(elem.getAttribute('transform'))) {
- elem = ffClone(elem);
- }
-
- // Hoped to solve the issue of moving text with text-anchor="start",
- // but this doesn't actually fix it. Hopefully on the right track, though. -Fyrd
-
- // var box=getBBox(elem), left=box.x, top=box.y, width=box.width,
- // height=box.height, dx = width - old_w, dy=0;
- // var angle = getRotationAngle(elem, true);
- // if (angle) {
- // var r = Math.sqrt( dx*dx + dy*dy );
- // var theta = Math.atan2(dy,dx) - angle;
- // dx = r * Math.cos(theta);
- // dy = r * Math.sin(theta);
- //
- // elem.setAttribute('x', elem.getAttribute('x')-dx);
- // elem.setAttribute('y', elem.getAttribute('y')-dy);
- // }
-
- } else if (attr == "#href") {
- setHref(elem, newValue);
- }
- else elem.setAttribute(attr, newValue);
- // if (i==0)
- // selectedBBoxes[0] = svgedit.utilities.getBBox(elem);
- // Use the Firefox ffClone hack for text elements with gradients or
- // where other text attributes are changed.
- if(svgedit.browser.isGecko() && elem.nodeName === 'text' && /rotate/.test(elem.getAttribute('transform'))) {
- if((newValue+'').indexOf('url') === 0 || ['font-size','font-family','x','y'].indexOf(attr) >= 0 && elem.textContent) {
- elem = ffClone(elem);
- }
- }
- // Timeout needed for Opera & Firefox
- // codedread: it is now possible for this function to be called with elements
- // that are not in the selectedElements array, we need to only request a
- // selector if the element is in that array
- if (selectedElements.indexOf(elem) >= 0) {
- setTimeout(function() {
- // Due to element replacement, this element may no longer
- // be part of the DOM
- if(!elem.parentNode) return;
- selectorManager.requestSelector(elem).resize();
- },0);
- }
- // if this element was rotated, and we changed the position of this element
- // we need to update the rotational transform attribute
- var angle = getRotationAngle(elem);
- if (angle != 0 && attr != "transform") {
- var tlist = getTransformList(elem);
- var n = tlist.numberOfItems;
- while (n--) {
- var xform = tlist.getItem(n);
- if (xform.type == 4) {
- // remove old rotate
- tlist.removeItem(n);
-
- var box = svgedit.utilities.getBBox(elem);
- var center = transformPoint(box.x+box.width/2, box.y+box.height/2, transformListToTransform(tlist).matrix);
- var cx = center.x,
- cy = center.y;
- var newrot = svgroot.createSVGTransform();
- newrot.setRotate(angle, cx, cy);
- tlist.insertItemBefore(newrot, n);
- break;
- }
- }
- }
- } // if oldValue != newValue
- } // for each elem
- svgroot.unsuspendRedraw(handle);
- };
- // Function: changeSelectedAttribute
- // Change the given/selected element and add the original value to the history stack
- // If you want to change all selectedElements, ignore the elems argument.
- // If you want to change only a subset of selectedElements, then send the
- // subset to this function in the elems argument.
- //
- // Parameters:
- // attr - String with the attribute name
- // newValue - String or number with the new attribute value
- // elems - The DOM elements to apply the change to
- var changeSelectedAttribute = this.changeSelectedAttribute = function(attr, val, elems) {
- var elems = elems || selectedElements;
- canvas.undoMgr.beginUndoableChange(attr, elems);
- var i = elems.length;
- changeSelectedAttributeNoUndo(attr, val, elems);
- var batchCmd = canvas.undoMgr.finishUndoableChange();
- if (!batchCmd.isEmpty()) {
- addCommandToHistory(batchCmd);
- }
- };
- // Function: deleteSelectedElements
- // Removes all selected elements from the DOM and adds the change to the
- // history stack
- this.deleteSelectedElements = function() {
- var batchCmd = new BatchCommand("Delete Elements");
- var len = selectedElements.length;
- var selectedCopy = []; //selectedElements is being deleted
- for (var i = 0; i < len; ++i) {
- var selected = selectedElements[i];
- if (selected == null) break;
- var parent = selected.parentNode;
- var t = selected;
-
- // this will unselect the element and remove the selectedOutline
- selectorManager.releaseSelector(t);
-
- // Remove the path if present.
- svgedit.path.removePath_(t.id);
-
- // Get the parent if it's a single-child anchor
- if(parent.tagName === 'a' && parent.childNodes.length === 1) {
- t = parent;
- parent = parent.parentNode;
- }
-
- var nextSibling = t.nextSibling;
- var elem = parent.removeChild(t);
- selectedCopy.push(selected); //for the copy
- selectedElements[i] = null;
- batchCmd.addSubCommand(new RemoveElementCommand(elem, nextSibling, parent));
- }
- if (!batchCmd.isEmpty()) addCommandToHistory(batchCmd);
- call("changed", selectedCopy);
- clearSelection();
- };
- // Function: cutSelectedElements
- // Removes all selected elements from the DOM and adds the change to the
- // history stack. Remembers removed elements on the clipboard
- // TODO: Combine similar code with deleteSelectedElements
- this.cutSelectedElements = function() {
- var batchCmd = new BatchCommand("Cut Elements");
- var len = selectedElements.length;
- var selectedCopy = []; //selectedElements is being deleted
- for (var i = 0; i < len; ++i) {
- var selected = selectedElements[i];
- if (selected == null) break;
- var parent = selected.parentNode;
- var t = selected;
- // this will unselect the element and remove the selectedOutline
- selectorManager.releaseSelector(t);
- // Remove the path if present.
- svgedit.path.removePath_(t.id);
- var nextSibling = t.nextSibling;
- var elem = parent.removeChild(t);
- selectedCopy.push(selected); //for the copy
- selectedElements[i] = null;
- batchCmd.addSubCommand(new RemoveElementCommand(elem, nextSibling, parent));
- }
- if (!batchCmd.isEmpty()) addCommandToHistory(batchCmd);
- call("changed", selectedCopy);
- clearSelection();
-
- canvas.clipBoard = selectedCopy;
- };
- // Function: copySelectedElements
- // Remembers the current selected elements on the clipboard
- this.copySelectedElements = function() {
- canvas.clipBoard = $.merge([], selectedElements);
- };
- this.pasteElements = function(type, x, y) {
- var cb = canvas.clipBoard;
- var len = cb.length;
- if(!len) return;
-
- var pasted = [];
- var batchCmd = new BatchCommand('Paste elements');
-
- // Move elements to lastClickPoint
- while (len--) {
- var elem = cb[len];
- if(!elem) continue;
- var copy = copyElem(elem);
- // See if elem with elem ID is in the DOM already
- if(!getElem(elem.id)) copy.id = elem.id;
-
- pasted.push(copy);
- (current_group || getCurrentDrawing().getCurrentLayer()).appendChild(copy);
- batchCmd.addSubCommand(new InsertElementCommand(copy));
- }
-
- selectOnly(pasted);
-
- if(type !== 'in_place') {
-
- var ctr_x, ctr_y;
-
- if(!type) {
- ctr_x = lastClickPoint.x;
- ctr_y = lastClickPoint.y;
- } else if(type === 'point') {
- ctr_x = x;
- ctr_y = y;
- }
-
- var bbox = getStrokedBBox(pasted);
- var cx = ctr_x - (bbox.x + bbox.width/2),
- cy = ctr_y - (bbox.y + bbox.height/2),
- dx = [],
- dy = [];
-
- $.each(pasted, function(i, item) {
- dx.push(cx);
- dy.push(cy);
- });
-
- var cmd = canvas.moveSelectedElements(dx, dy, false);
- batchCmd.addSubCommand(cmd);
- }
-
-
- addCommandToHistory(batchCmd);
- call("changed", pasted);
- }
- // Function: groupSelectedElements
- // Wraps all the selected elements in a group (g) element
- // Parameters:
- // type - type of element to group into, defaults to <g>
- this.groupSelectedElements = function(type) {
- if(!type) type = 'g';
- var cmd_str = '';
-
- switch ( type ) {
- case "a":
- cmd_str = "Make hyperlink";
- var url = '';
- if(arguments.length > 1) {
- url = arguments[1];
- }
- break;
- default:
- type = 'g';
- cmd_str = "Group Elements";
- break;
- }
-
- var batchCmd = new BatchCommand(cmd_str);
-
- // create and insert the group element
- var g = addSvgElementFromJson({
- "element": type,
- "attr": {
- "id": getNextId()
- }
- });
- if(type === 'a') {
- setHref(g, url);
- }
- batchCmd.addSubCommand(new InsertElementCommand(g));
-
- // now move all children into the group
- var i = selectedElements.length;
- while (i--) {
- var elem = selectedElements[i];
- if (elem == null) continue;
-
- if (elem.parentNode.tagName === 'a' && elem.parentNode.childNodes.length === 1) {
- elem = elem.parentNode;
- }
-
- var oldNextSibling = elem.nextSibling;
- var oldParent = elem.parentNode;
- g.appendChild(elem);
- batchCmd.addSubCommand(new MoveElementCommand(elem, oldNextSibling, oldParent));
- }
- if (!batchCmd.isEmpty()) addCommandToHistory(batchCmd);
-
- // update selection
- selectOnly([g], true);
- };
- // Function: pushGroupProperties
- // Pushes all appropriate parent group properties down to its children, then
- // removes them from the group
- var pushGroupProperties = this.pushGroupProperties = function(g, undoable) {
- var children = g.childNodes;
- var len = children.length;
- var xform = g.getAttribute("transform");
- var glist = getTransformList(g);
- var m = transformListToTransform(glist).matrix;
-
- var batchCmd = new BatchCommand("Push group properties");
- // TODO: get all fill/stroke properties from the group that we are about to destroy
- // "fill", "fill-opacity", "fill-rule", "stroke", "stroke-dasharray", "stroke-dashoffset",
- // "stroke-linecap", "stroke-linejoin", "stroke-miterlimit", "stroke-opacity",
- // "stroke-width"
- // and then for each child, if they do not have the attribute (or the value is 'inherit')
- // then set the child's attribute
-
- var i = 0;
- var gangle = getRotationAngle(g);
-
- var gattrs = $(g).attr(['filter', 'opacity']);
- var gfilter, gblur;
-
- for(var i = 0; i < len; i++) {
- var elem = children[i];
-
- if(elem.nodeType !== 1) continue;
-
- if(gattrs.opacity !== null && gattrs.opacity !== 1) {
- var c_opac = elem.getAttribute('opacity') || 1;
- var new_opac = Math.round((elem.getAttribute('opacity') || 1) * gattrs.opacity * 100)/100;
- changeSelectedAttribute('opacity', new_opac, [elem]);
- }
- if(gattrs.filter) {
- var cblur = this.getBlur(elem);
- var orig_cblur = cblur;
- if(!gblur) gblur = this.getBlur(g);
- if(cblur) {
- // Is this formula correct?
- cblur = (gblur-0) + (cblur-0);
- } else if(cblur === 0) {
- cblur = gblur;
- }
-
- // If child has no current filter, get group's filter or clone it.
- if(!orig_cblur) {
- // Set group's filter to use first child's ID
- if(!gfilter) {
- gfilter = getRefElem(gattrs.filter);
- } else {
- // Clone the group's filter
- gfilter = copyElem(gfilter);
- findDefs().appendChild(gfilter);
- }
- } else {
- gfilter = getRefElem(elem.getAttribute('filter'));
- }
- // Change this in future for different filters
- var suffix = (gfilter.firstChild.tagName === 'feGaussianBlur')?'blur':'filter';
- gfilter.id = elem.id + '_' + suffix;
- changeSelectedAttribute('filter', 'url(#' + gfilter.id + ')', [elem]);
-
- // Update blur value
- if(cblur) {
- changeSelectedAttribute('stdDeviation', cblur, [gfilter.firstChild]);
- canvas.setBlurOffsets(gfilter, cblur);
- }
- }
-
- var chtlist = getTransformList(elem);
- // Don't process gradient transforms
- if(~elem.tagName.indexOf('Gradient')) chtlist = null;
-
- // Hopefully not a problem to add this. Necessary for elements like <desc/>
- if(!chtlist) continue;
-
- // Apparently <defs> can get get a transformlist, but we don't want it to have one!
- if(elem.tagName === 'defs') continue;
-
- if (glist.numberOfItems) {
- // TODO: if the group's transform is just a rotate, we can always transfer the
- // rotate() down to the children (collapsing consecutive rotates and factoring
- // out any translates)
- if (gangle && glist.numberOfItems == 1) {
- // [Rg] [Rc] [Mc]
- // we want [Tr] [Rc2] [Mc] where:
- // - [Rc2] is at the child's current center but has the
- // sum of the group and child's rotation angles
- // - [Tr] is the equivalent translation that this child
- // undergoes if the group wasn't there
-
- // [Tr] = [Rg] [Rc] [Rc2_inv]
-
- // get group's rotation matrix (Rg)
- var rgm = glist.getItem(0).matrix;
-
- // get child's rotation matrix (Rc)
- var rcm = svgroot.createSVGMatrix();
- var cangle = getRotationAngle(elem);
- if (cangle) {
- rcm = chtlist.getItem(0).matrix;
- }
-
- // get child's old center of rotation
- var cbox = svgedit.utilities.getBBox(elem);
- var ceqm = transformListToTransform(chtlist).matrix;
- var coldc = transformPoint(cbox.x+cbox.width/2, cbox.y+cbox.height/2,ceqm);
-
- // sum group and child's angles
- var sangle = gangle + cangle;
-
- // get child's rotation at the old center (Rc2_inv)
- var r2 = svgroot.createSVGTransform();
- r2.setRotate(sangle, coldc.x, coldc.y);
-
- // calculate equivalent translate
- var trm = matrixMultiply(rgm, rcm, r2.matrix.inverse());
-
- // set up tlist
- if (cangle) {
- chtlist.removeItem(0);
- }
-
- if (sangle) {
- if(chtlist.numberOfItems) {
- chtlist.insertItemBefore(r2, 0);
- } else {
- chtlist.appendItem(r2);
- }
- }
- if (trm.e || trm.f) {
- var tr = svgroot.createSVGTransform();
- tr.setTranslate(trm.e, trm.f);
- if(chtlist.numberOfItems) {
- chtlist.insertItemBefore(tr, 0);
- } else {
- chtlist.appendItem(tr);
- }
- }
- }
- else { // more complicated than just a rotate
-
- // transfer the group's transform down to each child and then
- // call recalculateDimensions()
- var oldxform = elem.getAttribute("transform");
- var changes = {};
- changes["transform"] = oldxform ? oldxform : "";
- var newxform = svgroot.createSVGTransform();
- // [ gm ] [ chm ] = [ chm ] [ gm' ]
- // [ gm' ] = [ chm_inv ] [ gm ] [ chm ]
- var chm = transformListToTransform(chtlist).matrix,
- chm_inv = chm.inverse();
- var gm = matrixMultiply( chm_inv, m, chm );
- newxform.setMatrix(gm);
- chtlist.appendItem(newxform);
- }
- var cmd = recalculateDimensions(elem);
- if(cmd) batchCmd.addSubCommand(cmd);
- }
- }
-
- // remove transform and make it undo-able
- if (xform) {
- var changes = {};
- changes["transform"] = xform;
- g.setAttribute("transform", "");
- g.removeAttribute("transform");
- batchCmd.addSubCommand(new ChangeElementCommand(g, changes));
- }
-
- if (undoable && !batchCmd.isEmpty()) {
- return batchCmd;
- }
- }
- // Function: ungroupSelectedElement
- // Unwraps all the elements in a selected group (g) element. This requires
- // significant recalculations to apply group's transforms, etc to its children
- this.ungroupSelectedElement = function() {
- var g = selectedElements[0];
- if($(g).data('gsvg') || $(g).data('symbol')) {
- // Is svg, so actually convert to group
- convertToGroup(g);
- return;
- } else if(g.tagName === 'use') {
- // Somehow doesn't have data set, so retrieve
- var symbol = getElem(getHref(g).substr(1));
- $(g).data('symbol', symbol).data('ref', symbol);
- convertToGroup(g);
- return;
- }
- var parents_a = $(g).parents('a');
- if(parents_a.length) {
- g = parents_a[0];
- }
-
- // Look for parent "a"
- if (g.tagName === "g" || g.tagName === "a") {
-
- var batchCmd = new BatchCommand("Ungroup Elements");
- var cmd = pushGroupProperties(g, true);
- if(cmd) batchCmd.addSubCommand(cmd);
-
- var parent = g.parentNode;
- var anchor = g.nextSibling;
- var children = new Array(g.childNodes.length);
-
- var i = 0;
-
- while (g.firstChild) {
- var elem = g.firstChild;
- var oldNextSibling = elem.nextSibling;
- var oldParent = elem.parentNode;
-
- // Remove child title elements
- if(elem.tagName === 'title') {
- var nextSibling = elem.nextSibling;
- batchCmd.addSubCommand(new RemoveElementCommand(elem, nextSibling, oldParent));
- oldParent.removeChild(elem);
- continue;
- }
-
- children[i++] = elem = parent.insertBefore(elem, anchor);
- batchCmd.addSubCommand(new MoveElementCommand(elem, oldNextSibling, oldParent));
- }
- // remove the group from the selection
- clearSelection();
-
- // delete the group element (but make undo-able)
- var gNextSibling = g.nextSibling;
- g = parent.removeChild(g);
- batchCmd.addSubCommand(new RemoveElementCommand(g, gNextSibling, parent));
- if (!batchCmd.isEmpty()) addCommandToHistory(batchCmd);
-
- // update selection
- addToSelection(children);
- }
- };
- // Function: moveToTopSelectedElement
- // Repositions the selected element to the bottom in the DOM to appear on top of
- // other elements
- this.moveToTopSelectedElement = function() {
- var selected = selectedElements[0];
- if (selected != null) {
- var t = selected;
- var oldParent = t.parentNode;
- var oldNextSibling = t.nextSibling;
- t = t.parentNode.appendChild(t);
- // If the element actually moved position, add the command and fire the changed
- // event handler.
- if (oldNextSibling != t.nextSibling) {
- addCommandToHistory(new MoveElementCommand(t, oldNextSibling, oldParent, "top"));
- call("changed", [t]);
- }
- }
- };
- // Function: moveToBottomSelectedElement
- // Repositions the selected element to the top in the DOM to appear under
- // other elements
- this.moveToBottomSelectedElement = function() {
- var selected = selectedElements[0];
- if (selected != null) {
- var t = selected;
- var oldParent = t.parentNode;
- var oldNextSibling = t.nextSibling;
- var firstChild = t.parentNode.firstChild;
- if (firstChild.tagName == 'title') {
- firstChild = firstChild.nextSibling;
- }
- // This can probably be removed, as the defs should not ever apppear
- // inside a layer group
- if (firstChild.tagName == 'defs') {
- firstChild = firstChild.nextSibling;
- }
- t = t.parentNode.insertBefore(t, firstChild);
- // If the element actually moved position, add the command and fire the changed
- // event handler.
- if (oldNextSibling != t.nextSibling) {
- addCommandToHistory(new MoveElementCommand(t, oldNextSibling, oldParent, "bottom"));
- call("changed", [t]);
- }
- }
- };
- // Function: moveUpDownSelected
- // Moves the select element up or down the stack, based on the visibly
- // intersecting elements
- //
- // Parameters:
- // dir - String that's either 'Up' or 'Down'
- this.moveUpDownSelected = function(dir) {
- var selected = selectedElements[0];
- if (!selected) return;
-
- curBBoxes = [];
- var closest, found_cur;
- // jQuery sorts this list
- var list = $(getIntersectionList(getStrokedBBox([selected]))).toArray();
- if(dir == 'Down') list.reverse();
- $.each(list, function() {
- if(!found_cur) {
- if(this == selected) {
- found_cur = true;
- }
- return;
- }
- closest = this;
- return false;
- });
- if(!closest) return;
-
- var t = selected;
- var oldParent = t.parentNode;
- var oldNextSibling = t.nextSibling;
- $(closest)[dir == 'Down'?'before':'after'](t);
- // If the element actually moved position, add the command and fire the changed
- // event handler.
- if (oldNextSibling != t.nextSibling) {
- addCommandToHistory(new MoveElementCommand(t, oldNextSibling, oldParent, "Move " + dir));
- call("changed", [t]);
- }
- };
- // Function: moveSelectedElements
- // Moves selected elements on the X/Y axis
- //
- // Parameters:
- // dx - Float with the distance to move on the x-axis
- // dy - Float with the distance to move on the y-axis
- // undoable - Boolean indicating whether or not the action should be undoable
- //
- // Returns:
- // Batch command for the move
- this.moveSelectedElements = function(dx, dy, undoable) {
- // if undoable is not sent, default to true
- // if single values, scale them to the zoom
- if (dx.constructor != Array) {
- dx /= current_zoom;
- dy /= current_zoom;
- }
- var undoable = undoable || true;
- var batchCmd = new BatchCommand("position");
- var i = selectedElements.length;
- while (i--) {
- var selected = selectedElements[i];
- if (selected != null) {
- // if (i==0)
- // selectedBBoxes[0] = svgedit.utilities.getBBox(selected);
-
- // var b = {};
- // for(var j in selectedBBoxes[i]) b[j] = selectedBBoxes[i][j];
- // selectedBBoxes[i] = b;
-
- var xform = svgroot.createSVGTransform();
- var tlist = getTransformList(selected);
-
- // dx and dy could be arrays
- if (dx.constructor == Array) {
- // if (i==0) {
- // selectedBBoxes[0].x += dx[0];
- // selectedBBoxes[0].y += dy[0];
- // }
- xform.setTranslate(dx[i],dy[i]);
- } else {
- // if (i==0) {
- // selectedBBoxes[0].x += dx;
- // selectedBBoxes[0].y += dy;
- // }
- xform.setTranslate(dx,dy);
- }
- if(tlist.numberOfItems) {
- tlist.insertItemBefore(xform, 0);
- } else {
- tlist.appendItem(xform);
- }
-
- var cmd = recalculateDimensions(selected);
- if (cmd) {
- batchCmd.addSubCommand(cmd);
- }
-
- selectorManager.requestSelector(selected).resize();
- }
- }
- if (!batchCmd.isEmpty()) {
- if (undoable)
- addCommandToHistory(batchCmd);
- call("changed", selectedElements);
- return batchCmd;
- }
- };
- // Function: cloneSelectedElements
- // Create deep DOM copies (clones) of all selected elements and move them slightly
- // from their originals
- this.cloneSelectedElements = function(x,y) {
- var batchCmd = new BatchCommand("Clone Elements");
- // find all the elements selected (stop at first null)
- var len = selectedElements.length;
- for (var i = 0; i < len; ++i) {
- var elem = selectedElements[i];
- if (elem == null) break;
- }
- // use slice to quickly get the subset of elements we need
- var copiedElements = selectedElements.slice(0,i);
- this.clearSelection(true);
- // note that we loop in the reverse way because of the way elements are added
- // to the selectedElements array (top-first)
- var i = copiedElements.length;
- while (i--) {
- // clone each element and replace it within copiedElements
- var elem = copiedElements[i] = copyElem(copiedElements[i]);
- (current_group || getCurrentDrawing().getCurrentLayer()).appendChild(elem);
- batchCmd.addSubCommand(new InsertElementCommand(elem));
- }
-
- if (!batchCmd.isEmpty()) {
- addToSelection(copiedElements.reverse()); // Need to reverse for correct selection-adding
- this.moveSelectedElements(x,y,false);
- addCommandToHistory(batchCmd);
- }
- };
- // Function: alignSelectedElements
- // Aligns selected elements
- //
- // Parameters:
- // type - String with single character indicating the alignment type
- // relative_to - String that must be one of the following:
- // "selected", "largest", "smallest", "page"
- this.alignSelectedElements = function(type, relative_to) {
- var bboxes = [], angles = [];
- var minx = Number.MAX_VALUE, maxx = Number.MIN_VALUE, miny = Number.MAX_VALUE, maxy = Number.MIN_VALUE;
- var curwidth = Number.MIN_VALUE, curheight = Number.MIN_VALUE;
- var len = selectedElements.length;
- if (!len) return;
- for (var i = 0; i < len; ++i) {
- if (selectedElements[i] == null) break;
- var elem = selectedElements[i];
- bboxes[i] = getStrokedBBox([elem]);
-
- // now bbox is axis-aligned and handles rotation
- switch (relative_to) {
- case 'smallest':
- if ( (type == 'l' || type == 'c' || type == 'r') && (curwidth == Number.MIN_VALUE || curwidth > bboxes[i].width) ||
- (type == 't' || type == 'm' || type == 'b') && (curheight == Number.MIN_VALUE || curheight > bboxes[i].height) ) {
- minx = bboxes[i].x;
- miny = bboxes[i].y;
- maxx = bboxes[i].x + bboxes[i].width;
- maxy = bboxes[i].y + bboxes[i].height;
- curwidth = bboxes[i].width;
- curheight = bboxes[i].height;
- }
- break;
- case 'largest':
- if ( (type == 'l' || type == 'c' || type == 'r') && (curwidth == Number.MIN_VALUE || curwidth < bboxes[i].width) ||
- (type == 't' || type == 'm' || type == 'b') && (curheight == Number.MIN_VALUE || curheight < bboxes[i].height) ) {
- minx = bboxes[i].x;
- miny = bboxes[i].y;
- maxx = bboxes[i].x + bboxes[i].width;
- maxy = bboxes[i].y + bboxes[i].height;
- curwidth = bboxes[i].width;
- curheight = bboxes[i].height;
- }
- break;
- default: // 'selected'
- if (bboxes[i].x < minx) minx = bboxes[i].x;
- if (bboxes[i].y < miny) miny = bboxes[i].y;
- if (bboxes[i].x + bboxes[i].width > maxx) maxx = bboxes[i].x + bboxes[i].width;
- if (bboxes[i].y + bboxes[i].height > maxy) maxy = bboxes[i].y + bboxes[i].height;
- break;
- }
- } // loop for each element to find the bbox and adjust min/max
- if (relative_to == 'page') {
- minx = 0;
- miny = 0;
- maxx = canvas.contentW;
- maxy = canvas.contentH;
- }
- var dx = new Array(len);
- var dy = new Array(len);
- for (var i = 0; i < len; ++i) {
- if (selectedElements[i] == null) break;
- var elem = selectedElements[i];
- var bbox = bboxes[i];
- dx[i] = 0;
- dy[i] = 0;
- switch (type) {
- case 'l': // left (horizontal)
- dx[i] = minx - bbox.x;
- break;
- case 'c': // center (horizontal)
- dx[i] = (minx+maxx)/2 - (bbox.x + bbox.width/2);
- break;
- case 'r': // right (horizontal)
- dx[i] = maxx - (bbox.x + bbox.width);
- break;
- case 't': // top (vertical)
- dy[i] = miny - bbox.y;
- break;
- case 'm': // middle (vertical)
- dy[i] = (miny+maxy)/2 - (bbox.y + bbox.height/2);
- break;
- case 'b': // bottom (vertical)
- dy[i] = maxy - (bbox.y + bbox.height);
- break;
- }
- }
- this.moveSelectedElements(dx,dy);
- };
- // Group: Additional editor tools
- this.contentW = getResolution().w;
- this.contentH = getResolution().h;
- // Function: updateCanvas
- // Updates the editor canvas width/height/position after a zoom has occurred
- //
- // Parameters:
- // w - Float with the new width
- // h - Float with the new height
- //
- // Returns:
- // Object with the following values:
- // * x - The canvas' new x coordinate
- // * y - The canvas' new y coordinate
- // * old_x - The canvas' old x coordinate
- // * old_y - The canvas' old y coordinate
- // * d_x - The x position difference
- // * d_y - The y position difference
- this.updateCanvas = function(w, h) {
- svgroot.setAttribute("width", w);
- svgroot.setAttribute("height", h);
- var bg = $('#canvasBackground')[0];
- var old_x = svgcontent.getAttribute('x');
- var old_y = svgcontent.getAttribute('y');
- var x = (w/2 - this.contentW*current_zoom/2);
- var y = (h/2 - this.contentH*current_zoom/2);
- assignAttributes(svgcontent, {
- width: this.contentW*current_zoom,
- height: this.contentH*current_zoom,
- 'x': x,
- 'y': y,
- "viewBox" : "0 0 " + this.contentW + " " + this.contentH
- });
-
- assignAttributes(bg, {
- width: svgcontent.getAttribute('width'),
- height: svgcontent.getAttribute('height'),
- x: x,
- y: y
- });
- var bg_img = getElem('background_image');
- if (bg_img) {
- assignAttributes(bg_img, {
- 'width': '100%',
- 'height': '100%'
- });
- }
-
- selectorManager.selectorParentGroup.setAttribute("transform","translate(" + x + "," + y + ")");
-
- return {x:x, y:y, old_x:old_x, old_y:old_y, d_x:x - old_x, d_y:y - old_y};
- }
- // Function: setBackground
- // Set the background of the editor (NOT the actual document)
- //
- // Parameters:
- // color - String with fill color to apply
- // url - URL or path to image to use
- this.setBackground = function(color, url) {
- var bg = getElem('canvasBackground');
- var border = $(bg).find('rect')[0];
- var bg_img = getElem('background_image');
- border.setAttribute('fill',color);
- if(url) {
- if(!bg_img) {
- bg_img = svgdoc.createElementNS(svgns, "image");
- assignAttributes(bg_img, {
- 'id': 'background_image',
- 'width': '100%',
- 'height': '100%',
- 'preserveAspectRatio': 'xMinYMin',
- 'style':'pointer-events:none'
- });
- }
- setHref(bg_img, url);
- bg.appendChild(bg_img);
- } else if(bg_img) {
- bg_img.parentNode.removeChild(bg_img);
- }
- }
- // Function: cycleElement
- // Select the next/previous element within the current layer
- //
- // Parameters:
- // next - Boolean where true = next and false = previous element
- this.cycleElement = function(next) {
- var cur_elem = selectedElements[0];
- var elem = false;
- var all_elems = getVisibleElements(current_group || getCurrentDrawing().getCurrentLayer());
- if(!all_elems.length) return;
- if (cur_elem == null) {
- var num = next?all_elems.length-1:0;
- elem = all_elems[num];
- } else {
- var i = all_elems.length;
- while(i--) {
- if(all_elems[i] == cur_elem) {
- var num = next?i-1:i+1;
- if(num >= all_elems.length) {
- num = 0;
- } else if(num < 0) {
- num = all_elems.length-1;
- }
- elem = all_elems[num];
- break;
- }
- }
- }
- selectOnly([elem], true);
- call("selected", selectedElements);
- }
- this.clear();
- // DEPRECATED: getPrivateMethods
- // Since all methods are/should be public somehow, this function should be removed
- // Being able to access private methods publicly seems wrong somehow,
- // but currently appears to be the best way to allow testing and provide
- // access to them to plugins.
- this.getPrivateMethods = function() {
- var obj = {
- addCommandToHistory: addCommandToHistory,
- setGradient: setGradient,
- addSvgElementFromJson: addSvgElementFromJson,
- assignAttributes: assignAttributes,
- BatchCommand: BatchCommand,
- call: call,
- ChangeElementCommand: ChangeElementCommand,
- copyElem: copyElem,
- ffClone: ffClone,
- findDefs: findDefs,
- findDuplicateGradient: findDuplicateGradient,
- getElem: getElem,
- getId: getId,
- getIntersectionList: getIntersectionList,
- getMouseTarget: getMouseTarget,
- getNextId: getNextId,
- getPathBBox: getPathBBox,
- getUrlFromAttr: getUrlFromAttr,
- hasMatrixTransform: hasMatrixTransform,
- identifyLayers: identifyLayers,
- InsertElementCommand: InsertElementCommand,
- isIdentity: svgedit.math.isIdentity,
- logMatrix: logMatrix,
- matrixMultiply: matrixMultiply,
- MoveElementCommand: MoveElementCommand,
- preventClickDefault: preventClickDefault,
- recalculateAllSelectedDimensions: recalculateAllSelectedDimensions,
- recalculateDimensions: recalculateDimensions,
- remapElement: remapElement,
- RemoveElementCommand: RemoveElementCommand,
- removeUnusedDefElems: removeUnusedDefElems,
- round: round,
- runExtensions: runExtensions,
- sanitizeSvg: sanitizeSvg,
- SVGEditTransformList: svgedit.transformlist.SVGTransformList,
- toString: toString,
- transformBox: svgedit.math.transformBox,
- transformListToTransform: transformListToTransform,
- transformPoint: transformPoint,
- walkTree: svgedit.utilities.walkTree
- }
- return obj;
- };
- }
|