1
0

OverlayScrollbars.js 360 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639
  1. /*!
  2. * OverlayScrollbars
  3. * https://github.com/KingSora/OverlayScrollbars
  4. *
  5. * Version: 1.10.3
  6. *
  7. * Copyright KingSora | Rene Haas.
  8. * https://github.com/KingSora
  9. *
  10. * Released under the MIT license.
  11. * Date: 02.02.2020
  12. */
  13. (function (global, factory) {
  14. if (typeof define === 'function' && define.amd)
  15. define(function () { return factory(global, global.document, undefined); });
  16. else if (typeof module === 'object' && typeof module.exports === 'object')
  17. module.exports = factory(global, global.document, undefined);
  18. else
  19. factory(global, global.document, undefined);
  20. }(typeof window !== 'undefined' ? window : this,
  21. function (window, document, undefined) {
  22. 'use strict';
  23. var PLUGINNAME = 'OverlayScrollbars';
  24. var TYPES = {
  25. o: 'object',
  26. f: 'function',
  27. a: 'array',
  28. s: 'string',
  29. b: 'boolean',
  30. n: 'number',
  31. u: 'undefined',
  32. z: 'null'
  33. //d : 'date',
  34. //e : 'error',
  35. //r : 'regexp',
  36. //y : 'symbol'
  37. };
  38. var LEXICON = {
  39. c: 'class',
  40. s: 'style',
  41. i: 'id',
  42. l: 'length',
  43. p: 'prototype',
  44. oH: 'offsetHeight',
  45. cH: 'clientHeight',
  46. sH: 'scrollHeight',
  47. oW: 'offsetWidth',
  48. cW: 'clientWidth',
  49. sW: 'scrollWidth',
  50. hOP: 'hasOwnProperty',
  51. bCR: 'getBoundingClientRect'
  52. };
  53. var VENDORS = (function () {
  54. //https://developer.mozilla.org/en-US/docs/Glossary/Vendor_Prefix
  55. var jsCache = {};
  56. var cssCache = {};
  57. var cssPrefixes = ['-webkit-', '-moz-', '-o-', '-ms-'];
  58. var jsPrefixes = ['WebKit', 'Moz', 'O', 'MS'];
  59. function firstLetterToUpper(str) {
  60. return str.charAt(0).toUpperCase() + str.slice(1);
  61. }
  62. return {
  63. _cssPrefixes: cssPrefixes,
  64. _jsPrefixes: jsPrefixes,
  65. _cssProperty: function (name) {
  66. var result = cssCache[name];
  67. if (cssCache[LEXICON.hOP](name))
  68. return result;
  69. var uppercasedName = firstLetterToUpper(name);
  70. var elmStyle = document.createElement('div')[LEXICON.s];
  71. var resultPossibilities;
  72. var i = 0;
  73. var v;
  74. var currVendorWithoutDashes;
  75. for (; i < cssPrefixes.length; i++) {
  76. currVendorWithoutDashes = cssPrefixes[i].replace(/-/g, '');
  77. resultPossibilities = [
  78. name, //transition
  79. cssPrefixes[i] + name, //-webkit-transition
  80. currVendorWithoutDashes + uppercasedName, //webkitTransition
  81. firstLetterToUpper(currVendorWithoutDashes) + uppercasedName //WebkitTransition
  82. ];
  83. for (v = 0; v < resultPossibilities[LEXICON.l]; v++) {
  84. if (elmStyle[resultPossibilities[v]] !== undefined) {
  85. result = resultPossibilities[v];
  86. break;
  87. }
  88. }
  89. }
  90. cssCache[name] = result;
  91. return result;
  92. },
  93. _jsAPI: function (name, isInterface, fallback) {
  94. var i = 0;
  95. var result = jsCache[name];
  96. if (!jsCache[LEXICON.hOP](name)) {
  97. result = window[name];
  98. for (; i < jsPrefixes[LEXICON.l]; i++)
  99. result = result || window[(isInterface ? jsPrefixes[i] : jsPrefixes[i].toLowerCase()) + firstLetterToUpper(name)];
  100. jsCache[name] = result;
  101. }
  102. return result || fallback;
  103. }
  104. }
  105. })();
  106. var COMPATIBILITY = (function () {
  107. function windowSize(x) {
  108. return x ? window.innerWidth || document.documentElement[LEXICON.cW] || document.body[LEXICON.cW] : window.innerHeight || document.documentElement[LEXICON.cH] || document.body[LEXICON.cH];
  109. }
  110. function bind(func, thisObj) {
  111. if (typeof func != TYPES.f) {
  112. throw "Can't bind function!";
  113. // closest thing possible to the ECMAScript 5
  114. // internal IsCallable function
  115. //throw new TypeError('Function.prototype.bind - what is trying to be bound is not callable');
  116. }
  117. var proto = LEXICON.p;
  118. var aArgs = Array[proto].slice.call(arguments, 2);
  119. var fNOP = function () { };
  120. var fBound = function () { return func.apply(this instanceof fNOP ? this : thisObj, aArgs.concat(Array[proto].slice.call(arguments))); };
  121. if (func[proto])
  122. fNOP[proto] = func[proto]; // Function.prototype doesn't have a prototype property
  123. fBound[proto] = new fNOP();
  124. return fBound;
  125. }
  126. return {
  127. /**
  128. * Gets the current window width.
  129. * @returns {Number|number} The current window width in pixel.
  130. */
  131. wW: bind(windowSize, 0, true),
  132. /**
  133. * Gets the current window height.
  134. * @returns {Number|number} The current window height in pixel.
  135. */
  136. wH: bind(windowSize, 0),
  137. /**
  138. * Gets the MutationObserver Object or undefined if not supported.
  139. * @returns {MutationObserver|*|undefined} The MutationsObserver Object or undefined.
  140. */
  141. mO: bind(VENDORS._jsAPI, 0, 'MutationObserver', true),
  142. /**
  143. * Gets the ResizeObserver Object or undefined if not supported.
  144. * @returns {MutationObserver|*|undefined} The ResizeObserver Object or undefined.
  145. */
  146. rO: bind(VENDORS._jsAPI, 0, 'ResizeObserver', true),
  147. /**
  148. * Gets the RequestAnimationFrame method or it's corresponding polyfill.
  149. * @returns {*|Function} The RequestAnimationFrame method or it's corresponding polyfill.
  150. */
  151. rAF: bind(VENDORS._jsAPI, 0, 'requestAnimationFrame', false, function (func) { return window.setTimeout(func, 1000 / 60); }),
  152. /**
  153. * Gets the CancelAnimationFrame method or it's corresponding polyfill.
  154. * @returns {*|Function} The CancelAnimationFrame method or it's corresponding polyfill.
  155. */
  156. cAF: bind(VENDORS._jsAPI, 0, 'cancelAnimationFrame', false, function (id) { return window.clearTimeout(id); }),
  157. /**
  158. * Gets the current time.
  159. * @returns {number} The current time.
  160. */
  161. now: function () {
  162. return Date.now && Date.now() || new Date().getTime();
  163. },
  164. /**
  165. * Stops the propagation of the given event.
  166. * @param event The event of which the propagation shall be stoped.
  167. */
  168. stpP: function (event) {
  169. if (event.stopPropagation)
  170. event.stopPropagation();
  171. else
  172. event.cancelBubble = true;
  173. },
  174. /**
  175. * Prevents the default action of the given event.
  176. * @param event The event of which the default action shall be prevented.
  177. */
  178. prvD: function (event) {
  179. if (event.preventDefault && event.cancelable)
  180. event.preventDefault();
  181. else
  182. event.returnValue = false;
  183. },
  184. /**
  185. * Gets the pageX and pageY values of the given mouse event.
  186. * @param event The mouse event of which the pageX and pageX shall be got.
  187. * @returns {{x: number, y: number}} x = pageX value, y = pageY value.
  188. */
  189. page: function (event) {
  190. event = event.originalEvent || event;
  191. var strPage = 'page';
  192. var strClient = 'client';
  193. var strX = 'X';
  194. var strY = 'Y';
  195. var target = event.target || event.srcElement || document;
  196. var eventDoc = target.ownerDocument || document;
  197. var doc = eventDoc.documentElement;
  198. var body = eventDoc.body;
  199. //if touch event return return pageX/Y of it
  200. if (event.touches !== undefined) {
  201. var touch = event.touches[0];
  202. return {
  203. x: touch[strPage + strX],
  204. y: touch[strPage + strY]
  205. }
  206. }
  207. // Calculate pageX/Y if not native supported
  208. if (!event[strPage + strX] && event[strClient + strX] && event[strClient + strX] != null) {
  209. return {
  210. x: event[strClient + strX] +
  211. (doc && doc.scrollLeft || body && body.scrollLeft || 0) -
  212. (doc && doc.clientLeft || body && body.clientLeft || 0),
  213. y: event[strClient + strY] +
  214. (doc && doc.scrollTop || body && body.scrollTop || 0) -
  215. (doc && doc.clientTop || body && body.clientTop || 0)
  216. }
  217. }
  218. return {
  219. x: event[strPage + strX],
  220. y: event[strPage + strY]
  221. };
  222. },
  223. /**
  224. * Gets the clicked mouse button of the given mouse event.
  225. * @param event The mouse event of which the clicked button shal be got.
  226. * @returns {number} The number of the clicked mouse button. (0 : none | 1 : leftButton | 2 : middleButton | 3 : rightButton)
  227. */
  228. mBtn: function (event) {
  229. var button = event.button;
  230. if (!event.which && button !== undefined)
  231. return (button & 1 ? 1 : (button & 2 ? 3 : (button & 4 ? 2 : 0)));
  232. else
  233. return event.which;
  234. },
  235. /**
  236. * Checks whether a item is in the given array and returns its index.
  237. * @param item The item of which the position in the array shall be determined.
  238. * @param arr The array.
  239. * @returns {number} The zero based index of the item or -1 if the item isn't in the array.
  240. */
  241. inA: function (item, arr) {
  242. for (var i = 0; i < arr[LEXICON.l]; i++)
  243. //Sometiems in IE a "SCRIPT70" Permission denied error occurs if HTML elements in a iFrame are compared
  244. try {
  245. if (arr[i] === item)
  246. return i;
  247. }
  248. catch (e) { }
  249. return -1;
  250. },
  251. /**
  252. * Returns true if the given value is a array.
  253. * @param arr The potential array.
  254. * @returns {boolean} True if the given value is a array, false otherwise.
  255. */
  256. isA: function (arr) {
  257. var def = Array.isArray;
  258. return def ? def(arr) : this.type(arr) == TYPES.a;
  259. },
  260. /**
  261. * Determine the internal JavaScript [[Class]] of the given object.
  262. * @param obj The object of which the type shall be determined.
  263. * @returns {string} The type of the given object.
  264. */
  265. type: function (obj) {
  266. if (obj === undefined)
  267. return obj + '';
  268. if (obj === null)
  269. return obj + '';
  270. return Object[LEXICON.p].toString.call(obj).replace(/^\[object (.+)\]$/, '$1').toLowerCase();
  271. },
  272. bind: bind
  273. /**
  274. * Gets the vendor-prefixed CSS property by the given name.
  275. * For example the given name is "transform" and you're using a old Firefox browser then the returned value would be "-moz-transform".
  276. * If the browser doesn't need a vendor-prefix, then the returned string is the given name.
  277. * If the browser doesn't support the given property name at all (not even with a vendor-prefix) the returned value is null.
  278. * @param propName The unprefixed CSS property name.
  279. * @returns {string|null} The vendor-prefixed CSS property or null if the browser doesn't support the given CSS property.
  280. cssProp: function(propName) {
  281. return VENDORS._cssProperty(propName);
  282. }
  283. */
  284. }
  285. })();
  286. var MATH = Math;
  287. var JQUERY = window.jQuery;
  288. var EASING = (function () {
  289. var _easingsMath = {
  290. p: MATH.PI,
  291. c: MATH.cos,
  292. s: MATH.sin,
  293. w: MATH.pow,
  294. t: MATH.sqrt,
  295. n: MATH.asin,
  296. a: MATH.abs,
  297. o: 1.70158
  298. };
  299. /*
  300. x : current percent (0 - 1),
  301. t : current time (duration * percent),
  302. b : start value (from),
  303. c : end value (to),
  304. d : duration
  305. easingName : function(x, t, b, c, d) { return easedValue; }
  306. */
  307. return {
  308. swing: function (x, t, b, c, d) {
  309. return 0.5 - _easingsMath.c(x * _easingsMath.p) / 2;
  310. },
  311. linear: function (x, t, b, c, d) {
  312. return x;
  313. },
  314. easeInQuad: function (x, t, b, c, d) {
  315. return c * (t /= d) * t + b;
  316. },
  317. easeOutQuad: function (x, t, b, c, d) {
  318. return -c * (t /= d) * (t - 2) + b;
  319. },
  320. easeInOutQuad: function (x, t, b, c, d) {
  321. return ((t /= d / 2) < 1) ? c / 2 * t * t + b : -c / 2 * ((--t) * (t - 2) - 1) + b;
  322. },
  323. easeInCubic: function (x, t, b, c, d) {
  324. return c * (t /= d) * t * t + b;
  325. },
  326. easeOutCubic: function (x, t, b, c, d) {
  327. return c * ((t = t / d - 1) * t * t + 1) + b;
  328. },
  329. easeInOutCubic: function (x, t, b, c, d) {
  330. return ((t /= d / 2) < 1) ? c / 2 * t * t * t + b : c / 2 * ((t -= 2) * t * t + 2) + b;
  331. },
  332. easeInQuart: function (x, t, b, c, d) {
  333. return c * (t /= d) * t * t * t + b;
  334. },
  335. easeOutQuart: function (x, t, b, c, d) {
  336. return -c * ((t = t / d - 1) * t * t * t - 1) + b;
  337. },
  338. easeInOutQuart: function (x, t, b, c, d) {
  339. return ((t /= d / 2) < 1) ? c / 2 * t * t * t * t + b : -c / 2 * ((t -= 2) * t * t * t - 2) + b;
  340. },
  341. easeInQuint: function (x, t, b, c, d) {
  342. return c * (t /= d) * t * t * t * t + b;
  343. },
  344. easeOutQuint: function (x, t, b, c, d) {
  345. return c * ((t = t / d - 1) * t * t * t * t + 1) + b;
  346. },
  347. easeInOutQuint: function (x, t, b, c, d) {
  348. return ((t /= d / 2) < 1) ? c / 2 * t * t * t * t * t + b : c / 2 * ((t -= 2) * t * t * t * t + 2) + b;
  349. },
  350. easeInSine: function (x, t, b, c, d) {
  351. return -c * _easingsMath.c(t / d * (_easingsMath.p / 2)) + c + b;
  352. },
  353. easeOutSine: function (x, t, b, c, d) {
  354. return c * _easingsMath.s(t / d * (_easingsMath.p / 2)) + b;
  355. },
  356. easeInOutSine: function (x, t, b, c, d) {
  357. return -c / 2 * (_easingsMath.c(_easingsMath.p * t / d) - 1) + b;
  358. },
  359. easeInExpo: function (x, t, b, c, d) {
  360. return (t == 0) ? b : c * _easingsMath.w(2, 10 * (t / d - 1)) + b;
  361. },
  362. easeOutExpo: function (x, t, b, c, d) {
  363. return (t == d) ? b + c : c * (-_easingsMath.w(2, -10 * t / d) + 1) + b;
  364. },
  365. easeInOutExpo: function (x, t, b, c, d) {
  366. if (t == 0) return b;
  367. if (t == d) return b + c;
  368. if ((t /= d / 2) < 1) return c / 2 * _easingsMath.w(2, 10 * (t - 1)) + b;
  369. return c / 2 * (-_easingsMath.w(2, -10 * --t) + 2) + b;
  370. },
  371. easeInCirc: function (x, t, b, c, d) {
  372. return -c * (_easingsMath.t(1 - (t /= d) * t) - 1) + b;
  373. },
  374. easeOutCirc: function (x, t, b, c, d) {
  375. return c * _easingsMath.t(1 - (t = t / d - 1) * t) + b;
  376. },
  377. easeInOutCirc: function (x, t, b, c, d) {
  378. return ((t /= d / 2) < 1) ? -c / 2 * (_easingsMath.t(1 - t * t) - 1) + b : c / 2 * (_easingsMath.t(1 - (t -= 2) * t) + 1) + b;
  379. },
  380. easeInElastic: function (x, t, b, c, d) {
  381. var s = _easingsMath.o; var p = 0; var a = c;
  382. if (t == 0) return b; if ((t /= d) == 1) return b + c; if (!p) p = d * .3;
  383. if (a < _easingsMath.a(c)) { a = c; s = p / 4; }
  384. else s = p / (2 * _easingsMath.p) * _easingsMath.n(c / a);
  385. return -(a * _easingsMath.w(2, 10 * (t -= 1)) * _easingsMath.s((t * d - s) * (2 * _easingsMath.p) / p)) + b;
  386. },
  387. easeOutElastic: function (x, t, b, c, d) {
  388. var s = _easingsMath.o; var p = 0; var a = c;
  389. if (t == 0) return b;
  390. if ((t /= d) == 1) return b + c;
  391. if (!p) p = d * .3;
  392. if (a < _easingsMath.a(c)) { a = c; s = p / 4; }
  393. else s = p / (2 * _easingsMath.p) * _easingsMath.n(c / a);
  394. return a * _easingsMath.w(2, -10 * t) * _easingsMath.s((t * d - s) * (2 * _easingsMath.p) / p) + c + b;
  395. },
  396. easeInOutElastic: function (x, t, b, c, d) {
  397. var s = _easingsMath.o; var p = 0; var a = c;
  398. if (t == 0) return b;
  399. if ((t /= d / 2) == 2) return b + c;
  400. if (!p) p = d * (.3 * 1.5);
  401. if (a < _easingsMath.a(c)) { a = c; s = p / 4; }
  402. else s = p / (2 * _easingsMath.p) * _easingsMath.n(c / a);
  403. if (t < 1) return -.5 * (a * _easingsMath.w(2, 10 * (t -= 1)) * _easingsMath.s((t * d - s) * (2 * _easingsMath.p) / p)) + b;
  404. return a * _easingsMath.w(2, -10 * (t -= 1)) * _easingsMath.s((t * d - s) * (2 * _easingsMath.p) / p) * .5 + c + b;
  405. },
  406. easeInBack: function (x, t, b, c, d, s) {
  407. s = s || _easingsMath.o;
  408. return c * (t /= d) * t * ((s + 1) * t - s) + b;
  409. },
  410. easeOutBack: function (x, t, b, c, d, s) {
  411. s = s || _easingsMath.o;
  412. return c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;
  413. },
  414. easeInOutBack: function (x, t, b, c, d, s) {
  415. s = s || _easingsMath.o;
  416. return ((t /= d / 2) < 1) ? c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b : c / 2 * ((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
  417. },
  418. easeInBounce: function (x, t, b, c, d) {
  419. return c - this.easeOutBounce(x, d - t, 0, c, d) + b;
  420. },
  421. easeOutBounce: function (x, t, b, c, d) {
  422. var o = 7.5625;
  423. if ((t /= d) < (1 / 2.75)) {
  424. return c * (o * t * t) + b;
  425. } else if (t < (2 / 2.75)) {
  426. return c * (o * (t -= (1.5 / 2.75)) * t + .75) + b;
  427. } else if (t < (2.5 / 2.75)) {
  428. return c * (o * (t -= (2.25 / 2.75)) * t + .9375) + b;
  429. } else {
  430. return c * (o * (t -= (2.625 / 2.75)) * t + .984375) + b;
  431. }
  432. },
  433. easeInOutBounce: function (x, t, b, c, d) {
  434. return (t < d / 2) ? this.easeInBounce(x, t * 2, 0, c, d) * .5 + b : this.easeOutBounce(x, t * 2 - d, 0, c, d) * .5 + c * .5 + b;
  435. }
  436. };
  437. /*
  438. *
  439. * TERMS OF USE - EASING EQUATIONS
  440. *
  441. * Open source under the BSD License.
  442. *
  443. * Copyright © 2001 Robert Penner
  444. * All rights reserved.
  445. *
  446. * Redistribution and use in source and binary forms, with or without modification,
  447. * are permitted provided that the following conditions are met:
  448. *
  449. * Redistributions of source code must retain the above copyright notice, this list of
  450. * conditions and the following disclaimer.
  451. * Redistributions in binary form must reproduce the above copyright notice, this list
  452. * of conditions and the following disclaimer in the documentation and/or other materials
  453. * provided with the distribution.
  454. *
  455. * Neither the name of the author nor the names of contributors may be used to endorse
  456. * or promote products derived from this software without specific prior written permission.
  457. *
  458. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
  459. * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
  460. * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
  461. * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  462. * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
  463. * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
  464. * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  465. * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
  466. * OF THE POSSIBILITY OF SUCH DAMAGE.
  467. *
  468. */
  469. })();
  470. var FRAMEWORK = (function () {
  471. var _rnothtmlwhite = (/[^\x20\t\r\n\f]+/g);
  472. var _strSpace = ' ';
  473. var _strEmpty = '';
  474. var _strScrollLeft = 'scrollLeft';
  475. var _strScrollTop = 'scrollTop';
  476. var _animations = [];
  477. var _type = COMPATIBILITY.type;
  478. var _cssNumber = {
  479. animationIterationCount: true,
  480. columnCount: true,
  481. fillOpacity: true,
  482. flexGrow: true,
  483. flexShrink: true,
  484. fontWeight: true,
  485. lineHeight: true,
  486. opacity: true,
  487. order: true,
  488. orphans: true,
  489. widows: true,
  490. zIndex: true,
  491. zoom: true
  492. };
  493. function extend() {
  494. var src, copyIsArray, copy, name, options, clone, target = arguments[0] || {},
  495. i = 1,
  496. length = arguments[LEXICON.l],
  497. deep = false;
  498. // Handle a deep copy situation
  499. if (_type(target) == TYPES.b) {
  500. deep = target;
  501. target = arguments[1] || {};
  502. // skip the boolean and the target
  503. i = 2;
  504. }
  505. // Handle case when target is a string or something (possible in deep copy)
  506. if (_type(target) != TYPES.o && !_type(target) == TYPES.f) {
  507. target = {};
  508. }
  509. // extend jQuery itself if only one argument is passed
  510. if (length === i) {
  511. target = FakejQuery;
  512. --i;
  513. }
  514. for (; i < length; i++) {
  515. // Only deal with non-null/undefined values
  516. if ((options = arguments[i]) != null) {
  517. // Extend the base object
  518. for (name in options) {
  519. src = target[name];
  520. copy = options[name];
  521. // Prevent never-ending loop
  522. if (target === copy) {
  523. continue;
  524. }
  525. // Recurse if we're merging plain objects or arrays
  526. if (deep && copy && (isPlainObject(copy) || (copyIsArray = COMPATIBILITY.isA(copy)))) {
  527. if (copyIsArray) {
  528. copyIsArray = false;
  529. clone = src && COMPATIBILITY.isA(src) ? src : [];
  530. } else {
  531. clone = src && isPlainObject(src) ? src : {};
  532. }
  533. // Never move original objects, clone them
  534. target[name] = extend(deep, clone, copy);
  535. // Don't bring in undefined values
  536. } else if (copy !== undefined) {
  537. target[name] = copy;
  538. }
  539. }
  540. }
  541. }
  542. // Return the modified object
  543. return target;
  544. };
  545. function inArray(item, arr, fromIndex) {
  546. for (var i = fromIndex || 0; i < arr[LEXICON.l]; i++)
  547. if (arr[i] === item)
  548. return i;
  549. return -1;
  550. }
  551. function isFunction(obj) {
  552. return _type(obj) == TYPES.f;
  553. };
  554. function isEmptyObject(obj) {
  555. for (var name in obj)
  556. return false;
  557. return true;
  558. };
  559. function isPlainObject(obj) {
  560. if (!obj || _type(obj) != TYPES.o)
  561. return false;
  562. var key;
  563. var proto = LEXICON.p;
  564. var hasOwnProperty = Object[proto].hasOwnProperty;
  565. var hasOwnConstructor = hasOwnProperty.call(obj, 'constructor');
  566. var hasIsPrototypeOf = obj.constructor && obj.constructor[proto] && hasOwnProperty.call(obj.constructor[proto], 'isPrototypeOf');
  567. if (obj.constructor && !hasOwnConstructor && !hasIsPrototypeOf) {
  568. return false;
  569. }
  570. for (key in obj) { /**/ }
  571. return _type(key) == TYPES.u || hasOwnProperty.call(obj, key);
  572. };
  573. function each(obj, callback) {
  574. var i = 0;
  575. if (isArrayLike(obj)) {
  576. for (; i < obj[LEXICON.l]; i++) {
  577. if (callback.call(obj[i], i, obj[i]) === false)
  578. break;
  579. }
  580. }
  581. else {
  582. for (i in obj) {
  583. if (callback.call(obj[i], i, obj[i]) === false)
  584. break;
  585. }
  586. }
  587. return obj;
  588. };
  589. function isArrayLike(obj) {
  590. var length = !!obj && [LEXICON.l] in obj && obj[LEXICON.l];
  591. var t = _type(obj);
  592. return isFunction(t) ? false : (t == TYPES.a || length === 0 || _type(length) == TYPES.n && length > 0 && (length - 1) in obj);
  593. }
  594. function stripAndCollapse(value) {
  595. var tokens = value.match(_rnothtmlwhite) || [];
  596. return tokens.join(_strSpace);
  597. }
  598. function matches(elem, selector) {
  599. var nodeList = (elem.parentNode || document).querySelectorAll(selector) || [];
  600. var i = nodeList[LEXICON.l];
  601. while (i--)
  602. if (nodeList[i] == elem)
  603. return true;
  604. return false;
  605. }
  606. function insertAdjacentElement(el, strategy, child) {
  607. if (_type(child) == TYPES.a) {
  608. for (var i = 0; i < child[LEXICON.l]; i++)
  609. insertAdjacentElement(el, strategy, child[i]);
  610. }
  611. else if (_type(child) == TYPES.s)
  612. el.insertAdjacentHTML(strategy, child);
  613. else
  614. el.insertAdjacentElement(strategy, child.nodeType ? child : child[0]);
  615. }
  616. function setCSSVal(el, prop, val) {
  617. try {
  618. if (el[LEXICON.s][prop] !== undefined)
  619. el[LEXICON.s][prop] = parseCSSVal(prop, val);
  620. } catch (e) { }
  621. }
  622. function parseCSSVal(prop, val) {
  623. if (!_cssNumber[prop.toLowerCase()] && _type(val) == TYPES.n)
  624. val += 'px';
  625. return val;
  626. }
  627. function startNextAnimationInQ(animObj, removeFromQ) {
  628. var index;
  629. var nextAnim;
  630. if (removeFromQ !== false)
  631. animObj.q.splice(0, 1);
  632. if (animObj.q[LEXICON.l] > 0) {
  633. nextAnim = animObj.q[0];
  634. animate(animObj.el, nextAnim.props, nextAnim.duration, nextAnim.easing, nextAnim.complete, true);
  635. }
  636. else {
  637. index = inArray(animObj, _animations);
  638. if (index > -1)
  639. _animations.splice(index, 1);
  640. }
  641. }
  642. function setAnimationValue(el, prop, value) {
  643. if (prop === _strScrollLeft || prop === _strScrollTop)
  644. el[prop] = value;
  645. else
  646. setCSSVal(el, prop, value);
  647. }
  648. function animate(el, props, options, easing, complete, guaranteedNext) {
  649. var hasOptions = isPlainObject(options);
  650. var from = {};
  651. var to = {};
  652. var i = 0;
  653. var key;
  654. var animObj;
  655. var start;
  656. var progress;
  657. var step;
  658. var specialEasing;
  659. var duration;
  660. if (hasOptions) {
  661. easing = options.easing;
  662. start = options.start;
  663. progress = options.progress;
  664. step = options.step;
  665. specialEasing = options.specialEasing;
  666. complete = options.complete;
  667. duration = options.duration;
  668. }
  669. else
  670. duration = options;
  671. specialEasing = specialEasing || {};
  672. duration = duration || 400;
  673. easing = easing || 'swing';
  674. guaranteedNext = guaranteedNext || false;
  675. for (; i < _animations[LEXICON.l]; i++) {
  676. if (_animations[i].el === el) {
  677. animObj = _animations[i];
  678. break;
  679. }
  680. }
  681. if (!animObj) {
  682. animObj = {
  683. el: el,
  684. q: []
  685. };
  686. _animations.push(animObj);
  687. }
  688. for (key in props) {
  689. if (key === _strScrollLeft || key === _strScrollTop)
  690. from[key] = el[key];
  691. else
  692. from[key] = FakejQuery(el).css(key);
  693. }
  694. for (key in from) {
  695. if (from[key] !== props[key] && props[key] !== undefined)
  696. to[key] = props[key];
  697. }
  698. if (!isEmptyObject(to)) {
  699. var timeNow;
  700. var end;
  701. var percent;
  702. var fromVal;
  703. var toVal;
  704. var easedVal;
  705. var timeStart;
  706. var frame;
  707. var elapsed;
  708. var qPos = guaranteedNext ? 0 : inArray(qObj, animObj.q);
  709. var qObj = {
  710. props: to,
  711. duration: hasOptions ? options : duration,
  712. easing: easing,
  713. complete: complete
  714. };
  715. if (qPos === -1) {
  716. qPos = animObj.q[LEXICON.l];
  717. animObj.q.push(qObj);
  718. }
  719. if (qPos === 0) {
  720. if (duration > 0) {
  721. timeStart = COMPATIBILITY.now();
  722. frame = function () {
  723. timeNow = COMPATIBILITY.now();
  724. elapsed = (timeNow - timeStart);
  725. end = qObj.stop || elapsed >= duration;
  726. percent = 1 - ((MATH.max(0, timeStart + duration - timeNow) / duration) || 0);
  727. for (key in to) {
  728. fromVal = parseFloat(from[key]);
  729. toVal = parseFloat(to[key]);
  730. easedVal = (toVal - fromVal) * EASING[specialEasing[key] || easing](percent, percent * duration, 0, 1, duration) + fromVal;
  731. setAnimationValue(el, key, easedVal);
  732. if (isFunction(step)) {
  733. step(easedVal, {
  734. elem: el,
  735. prop: key,
  736. start: fromVal,
  737. now: easedVal,
  738. end: toVal,
  739. pos: percent,
  740. options: {
  741. easing: easing,
  742. speacialEasing: specialEasing,
  743. duration: duration,
  744. complete: complete,
  745. step: step
  746. },
  747. startTime: timeStart
  748. });
  749. }
  750. }
  751. if (isFunction(progress))
  752. progress({}, percent, MATH.max(0, duration - elapsed));
  753. if (end) {
  754. startNextAnimationInQ(animObj);
  755. if (isFunction(complete))
  756. complete();
  757. }
  758. else
  759. qObj.frame = COMPATIBILITY.rAF()(frame);
  760. };
  761. qObj.frame = COMPATIBILITY.rAF()(frame);
  762. }
  763. else {
  764. for (key in to)
  765. setAnimationValue(el, key, to[key]);
  766. startNextAnimationInQ(animObj);
  767. }
  768. }
  769. }
  770. else if (guaranteedNext)
  771. startNextAnimationInQ(animObj);
  772. }
  773. function stop(el, clearQ, jumpToEnd) {
  774. var animObj;
  775. var qObj;
  776. var key;
  777. var i = 0;
  778. for (; i < _animations[LEXICON.l]; i++) {
  779. animObj = _animations[i];
  780. if (animObj.el === el) {
  781. if (animObj.q[LEXICON.l] > 0) {
  782. qObj = animObj.q[0];
  783. qObj.stop = true;
  784. COMPATIBILITY.cAF()(qObj.frame);
  785. animObj.q.splice(0, 1);
  786. if (jumpToEnd)
  787. for (key in qObj.props)
  788. setAnimationValue(el, key, qObj.props[key]);
  789. if (clearQ)
  790. animObj.q = [];
  791. else
  792. startNextAnimationInQ(animObj, false);
  793. }
  794. break;
  795. }
  796. }
  797. }
  798. function elementIsVisible(el) {
  799. return !!(el[LEXICON.oW] || el[LEXICON.oH] || el.getClientRects()[LEXICON.l]);
  800. }
  801. function FakejQuery(selector) {
  802. if (arguments[LEXICON.l] === 0)
  803. return this;
  804. var base = new FakejQuery();
  805. var elements = selector;
  806. var i = 0;
  807. var elms;
  808. var el;
  809. if (_type(selector) == TYPES.s) {
  810. elements = [];
  811. if (selector.charAt(0) === '<') {
  812. el = document.createElement('div');
  813. el.innerHTML = selector;
  814. elms = el.children;
  815. }
  816. else {
  817. elms = document.querySelectorAll(selector);
  818. }
  819. for (; i < elms[LEXICON.l]; i++)
  820. elements.push(elms[i]);
  821. }
  822. if (elements) {
  823. if (_type(elements) != TYPES.s && (!isArrayLike(elements) || elements === window || elements === elements.self))
  824. elements = [elements];
  825. for (i = 0; i < elements[LEXICON.l]; i++)
  826. base[i] = elements[i];
  827. base[LEXICON.l] = elements[LEXICON.l];
  828. }
  829. return base;
  830. };
  831. FakejQuery[LEXICON.p] = {
  832. //EVENTS:
  833. on: function (eventName, handler) {
  834. eventName = (eventName || _strEmpty).match(_rnothtmlwhite) || [_strEmpty];
  835. var eventNameLength = eventName[LEXICON.l];
  836. var i = 0;
  837. var el;
  838. return this.each(function () {
  839. el = this;
  840. try {
  841. if (el.addEventListener) {
  842. for (; i < eventNameLength; i++)
  843. el.addEventListener(eventName[i], handler);
  844. }
  845. else if (el.detachEvent) {
  846. for (; i < eventNameLength; i++)
  847. el.attachEvent('on' + eventName[i], handler);
  848. }
  849. } catch (e) { }
  850. });
  851. },
  852. off: function (eventName, handler) {
  853. eventName = (eventName || _strEmpty).match(_rnothtmlwhite) || [_strEmpty];
  854. var eventNameLength = eventName[LEXICON.l];
  855. var i = 0;
  856. var el;
  857. return this.each(function () {
  858. el = this;
  859. try {
  860. if (el.removeEventListener) {
  861. for (; i < eventNameLength; i++)
  862. el.removeEventListener(eventName[i], handler);
  863. }
  864. else if (el.detachEvent) {
  865. for (; i < eventNameLength; i++)
  866. el.detachEvent('on' + eventName[i], handler);
  867. }
  868. } catch (e) { }
  869. });
  870. },
  871. one: function (eventName, handler) {
  872. eventName = (eventName || _strEmpty).match(_rnothtmlwhite) || [_strEmpty];
  873. return this.each(function () {
  874. var el = FakejQuery(this);
  875. FakejQuery.each(eventName, function (i, oneEventName) {
  876. var oneHandler = function (e) {
  877. handler.call(this, e);
  878. el.off(oneEventName, oneHandler);
  879. };
  880. el.on(oneEventName, oneHandler);
  881. });
  882. });
  883. },
  884. trigger: function (eventName) {
  885. var el;
  886. var event;
  887. return this.each(function () {
  888. el = this;
  889. if (document.createEvent) {
  890. event = document.createEvent('HTMLEvents');
  891. event.initEvent(eventName, true, false);
  892. el.dispatchEvent(event);
  893. }
  894. else {
  895. el.fireEvent('on' + eventName);
  896. }
  897. });
  898. },
  899. //DOM NODE INSERTING / REMOVING:
  900. append: function (child) {
  901. return this.each(function () { insertAdjacentElement(this, 'beforeend', child); });
  902. },
  903. prepend: function (child) {
  904. return this.each(function () { insertAdjacentElement(this, 'afterbegin', child); });
  905. },
  906. before: function (child) {
  907. return this.each(function () { insertAdjacentElement(this, 'beforebegin', child); });
  908. },
  909. after: function (child) {
  910. return this.each(function () { insertAdjacentElement(this, 'afterend', child); });
  911. },
  912. remove: function () {
  913. return this.each(function () {
  914. var el = this;
  915. var parentNode = el.parentNode;
  916. if (parentNode != null)
  917. parentNode.removeChild(el);
  918. });
  919. },
  920. unwrap: function () {
  921. var parents = [];
  922. var i;
  923. var el;
  924. var parent;
  925. this.each(function () {
  926. parent = this.parentNode;
  927. if (inArray(parent, parents) === - 1)
  928. parents.push(parent);
  929. });
  930. for (i = 0; i < parents[LEXICON.l]; i++) {
  931. el = parents[i];
  932. parent = el.parentNode;
  933. while (el.firstChild)
  934. parent.insertBefore(el.firstChild, el);
  935. parent.removeChild(el);
  936. }
  937. return this;
  938. },
  939. wrapAll: function (wrapperHTML) {
  940. var i;
  941. var nodes = this;
  942. var wrapper = FakejQuery(wrapperHTML)[0];
  943. var deepest = wrapper;
  944. var parent = nodes[0].parentNode;
  945. var previousSibling = nodes[0].previousSibling;
  946. while (deepest.childNodes[LEXICON.l] > 0)
  947. deepest = deepest.childNodes[0];
  948. for (i = 0; nodes[LEXICON.l] - i; deepest.firstChild === nodes[0] && i++)
  949. deepest.appendChild(nodes[i]);
  950. var nextSibling = previousSibling ? previousSibling.nextSibling : parent.firstChild;
  951. parent.insertBefore(wrapper, nextSibling);
  952. return this;
  953. },
  954. wrapInner: function (wrapperHTML) {
  955. return this.each(function () {
  956. var el = FakejQuery(this);
  957. var contents = el.contents();
  958. if (contents[LEXICON.l])
  959. contents.wrapAll(wrapperHTML);
  960. else
  961. el.append(wrapperHTML);
  962. });
  963. },
  964. wrap: function (wrapperHTML) {
  965. return this.each(function () { FakejQuery(this).wrapAll(wrapperHTML); });
  966. },
  967. //DOM NODE MANIPULATION / INFORMATION:
  968. css: function (styles, val) {
  969. var el;
  970. var key;
  971. var cptStyle;
  972. var getCptStyle = window.getComputedStyle;
  973. if (_type(styles) == TYPES.s) {
  974. if (val === undefined) {
  975. el = this[0];
  976. cptStyle = getCptStyle ? getCptStyle(el, null) : el.currentStyle[styles];
  977. //https://bugzilla.mozilla.org/show_bug.cgi?id=548397 can be null sometimes if iframe with display: none (firefox only!)
  978. return getCptStyle ? cptStyle != null ? cptStyle.getPropertyValue(styles) : el[LEXICON.s][styles] : cptStyle;
  979. }
  980. else {
  981. return this.each(function () {
  982. setCSSVal(this, styles, val);
  983. });
  984. }
  985. }
  986. else {
  987. return this.each(function () {
  988. for (key in styles)
  989. setCSSVal(this, key, styles[key]);
  990. });
  991. }
  992. },
  993. hasClass: function (className) {
  994. var elem, i = 0;
  995. var classNamePrepared = _strSpace + className + _strSpace;
  996. var classList;
  997. while ((elem = this[i++])) {
  998. classList = elem.classList;
  999. if (classList && classList.contains(className))
  1000. return true;
  1001. else if (elem.nodeType === 1 && (_strSpace + stripAndCollapse(elem.className + _strEmpty) + _strSpace).indexOf(classNamePrepared) > -1)
  1002. return true;
  1003. }
  1004. return false;
  1005. },
  1006. addClass: function (className) {
  1007. var classes;
  1008. var elem;
  1009. var cur;
  1010. var curValue;
  1011. var clazz;
  1012. var finalValue;
  1013. var supportClassList;
  1014. var elmClassList;
  1015. var i = 0;
  1016. var v = 0;
  1017. if (className) {
  1018. classes = className.match(_rnothtmlwhite) || [];
  1019. while ((elem = this[i++])) {
  1020. elmClassList = elem.classList;
  1021. if (supportClassList === undefined)
  1022. supportClassList = elmClassList !== undefined;
  1023. if (supportClassList) {
  1024. while ((clazz = classes[v++]))
  1025. elmClassList.add(clazz);
  1026. }
  1027. else {
  1028. curValue = elem.className + _strEmpty;
  1029. cur = elem.nodeType === 1 && (_strSpace + stripAndCollapse(curValue) + _strSpace);
  1030. if (cur) {
  1031. while ((clazz = classes[v++]))
  1032. if (cur.indexOf(_strSpace + clazz + _strSpace) < 0)
  1033. cur += clazz + _strSpace;
  1034. finalValue = stripAndCollapse(cur);
  1035. if (curValue !== finalValue)
  1036. elem.className = finalValue;
  1037. }
  1038. }
  1039. }
  1040. }
  1041. return this;
  1042. },
  1043. removeClass: function (className) {
  1044. var classes;
  1045. var elem;
  1046. var cur;
  1047. var curValue;
  1048. var clazz;
  1049. var finalValue;
  1050. var supportClassList;
  1051. var elmClassList;
  1052. var i = 0;
  1053. var v = 0;
  1054. if (className) {
  1055. classes = className.match(_rnothtmlwhite) || [];
  1056. while ((elem = this[i++])) {
  1057. elmClassList = elem.classList;
  1058. if (supportClassList === undefined)
  1059. supportClassList = elmClassList !== undefined;
  1060. if (supportClassList) {
  1061. while ((clazz = classes[v++]))
  1062. elmClassList.remove(clazz);
  1063. }
  1064. else {
  1065. curValue = elem.className + _strEmpty;
  1066. cur = elem.nodeType === 1 && (_strSpace + stripAndCollapse(curValue) + _strSpace);
  1067. if (cur) {
  1068. while ((clazz = classes[v++]))
  1069. while (cur.indexOf(_strSpace + clazz + _strSpace) > -1)
  1070. cur = cur.replace(_strSpace + clazz + _strSpace, _strSpace);
  1071. finalValue = stripAndCollapse(cur);
  1072. if (curValue !== finalValue)
  1073. elem.className = finalValue;
  1074. }
  1075. }
  1076. }
  1077. }
  1078. return this;
  1079. },
  1080. hide: function () {
  1081. return this.each(function () { this[LEXICON.s].display = 'none'; });
  1082. },
  1083. show: function () {
  1084. return this.each(function () { this[LEXICON.s].display = 'block'; });
  1085. },
  1086. attr: function (attrName, value) {
  1087. var i = 0;
  1088. var el;
  1089. while (el = this[i++]) {
  1090. if (value === undefined)
  1091. return el.getAttribute(attrName);
  1092. el.setAttribute(attrName, value);
  1093. }
  1094. return this;
  1095. },
  1096. removeAttr: function (attrName) {
  1097. return this.each(function () { this.removeAttribute(attrName); });
  1098. },
  1099. offset: function () {
  1100. var el = this[0];
  1101. var rect = el[LEXICON.bCR]();
  1102. var scrollLeft = window.pageXOffset || document.documentElement[_strScrollLeft];
  1103. var scrollTop = window.pageYOffset || document.documentElement[_strScrollTop];
  1104. return {
  1105. top: rect.top + scrollTop,
  1106. left: rect.left + scrollLeft
  1107. };
  1108. },
  1109. position: function () {
  1110. var el = this[0];
  1111. return {
  1112. top: el.offsetTop,
  1113. left: el.offsetLeft
  1114. };
  1115. },
  1116. scrollLeft: function (value) {
  1117. var i = 0;
  1118. var el;
  1119. while (el = this[i++]) {
  1120. if (value === undefined)
  1121. return el[_strScrollLeft];
  1122. el[_strScrollLeft] = value;
  1123. }
  1124. return this;
  1125. },
  1126. scrollTop: function (value) {
  1127. var i = 0;
  1128. var el;
  1129. while (el = this[i++]) {
  1130. if (value === undefined)
  1131. return el[_strScrollTop];
  1132. el[_strScrollTop] = value;
  1133. }
  1134. return this;
  1135. },
  1136. val: function (value) {
  1137. var el = this[0];
  1138. if (!value)
  1139. return el.value;
  1140. el.value = value;
  1141. return this;
  1142. },
  1143. //DOM TRAVERSAL / FILTERING:
  1144. first: function () {
  1145. return this.eq(0);
  1146. },
  1147. last: function () {
  1148. return this.eq(-1);
  1149. },
  1150. eq: function (index) {
  1151. return FakejQuery(this[index >= 0 ? index : this[LEXICON.l] + index]);
  1152. },
  1153. find: function (selector) {
  1154. var children = [];
  1155. var i;
  1156. this.each(function () {
  1157. var el = this;
  1158. var ch = el.querySelectorAll(selector);
  1159. for (i = 0; i < ch[LEXICON.l]; i++)
  1160. children.push(ch[i]);
  1161. });
  1162. return FakejQuery(children);
  1163. },
  1164. children: function (selector) {
  1165. var children = [];
  1166. var el;
  1167. var ch;
  1168. var i;
  1169. this.each(function () {
  1170. ch = this.children;
  1171. for (i = 0; i < ch[LEXICON.l]; i++) {
  1172. el = ch[i];
  1173. if (selector) {
  1174. if ((el.matches && el.matches(selector)) || matches(el, selector))
  1175. children.push(el);
  1176. }
  1177. else
  1178. children.push(el);
  1179. }
  1180. });
  1181. return FakejQuery(children);
  1182. },
  1183. parent: function (selector) {
  1184. var parents = [];
  1185. var parent;
  1186. this.each(function () {
  1187. parent = this.parentNode;
  1188. if (selector ? FakejQuery(parent).is(selector) : true)
  1189. parents.push(parent);
  1190. });
  1191. return FakejQuery(parents);
  1192. },
  1193. is: function (selector) {
  1194. var el;
  1195. var i;
  1196. for (i = 0; i < this[LEXICON.l]; i++) {
  1197. el = this[i];
  1198. if (selector === ':visible')
  1199. return elementIsVisible(el);
  1200. if (selector === ':hidden')
  1201. return !elementIsVisible(el);
  1202. if ((el.matches && el.matches(selector)) || matches(el, selector))
  1203. return true;
  1204. }
  1205. return false;
  1206. },
  1207. contents: function () {
  1208. var contents = [];
  1209. var childs;
  1210. var i;
  1211. this.each(function () {
  1212. childs = this.childNodes;
  1213. for (i = 0; i < childs[LEXICON.l]; i++)
  1214. contents.push(childs[i]);
  1215. });
  1216. return FakejQuery(contents);
  1217. },
  1218. each: function (callback) {
  1219. return each(this, callback);
  1220. },
  1221. //ANIMATION:
  1222. animate: function (props, duration, easing, complete) {
  1223. return this.each(function () { animate(this, props, duration, easing, complete); });
  1224. },
  1225. stop: function (clearQ, jump) {
  1226. return this.each(function () { stop(this, clearQ, jump); });
  1227. }
  1228. };
  1229. extend(FakejQuery, {
  1230. extend: extend,
  1231. inArray: inArray,
  1232. isEmptyObject: isEmptyObject,
  1233. isPlainObject: isPlainObject,
  1234. each: each
  1235. });
  1236. return FakejQuery;
  1237. })();
  1238. var INSTANCES = (function () {
  1239. var _targets = [];
  1240. var _instancePropertyString = '__overlayScrollbars__';
  1241. /**
  1242. * Register, unregister or get a certain (or all) instances.
  1243. * Register: Pass the target and the instance.
  1244. * Unregister: Pass the target and null.
  1245. * Get Instance: Pass the target from which the instance shall be got.
  1246. * Get Targets: Pass no arguments.
  1247. * @param target The target to which the instance shall be registered / from which the instance shall be unregistered / the instance shall be got
  1248. * @param instance The instance.
  1249. * @returns {*|void} Returns the instance from the given target.
  1250. */
  1251. return function (target, instance) {
  1252. var argLen = arguments[LEXICON.l];
  1253. if (argLen < 1) {
  1254. //return all targets
  1255. return _targets;
  1256. }
  1257. else {
  1258. if (instance) {
  1259. //register instance
  1260. target[_instancePropertyString] = instance;
  1261. _targets.push(target);
  1262. }
  1263. else {
  1264. var index = COMPATIBILITY.inA(target, _targets);
  1265. if (index > -1) {
  1266. if (argLen > 1) {
  1267. //unregister instance
  1268. delete target[_instancePropertyString];
  1269. _targets.splice(index, 1);
  1270. }
  1271. else {
  1272. //get instance from target
  1273. return _targets[index][_instancePropertyString];
  1274. }
  1275. }
  1276. }
  1277. }
  1278. }
  1279. })();
  1280. var PLUGIN = (function () {
  1281. var _plugin;
  1282. var _pluginsGlobals;
  1283. var _pluginsAutoUpdateLoop;
  1284. var _pluginsExtensions = [];
  1285. var _pluginsOptions = (function () {
  1286. var type = COMPATIBILITY.type;
  1287. var possibleTemplateTypes = [
  1288. TYPES.b, //boolean
  1289. TYPES.n, //number
  1290. TYPES.s, //string
  1291. TYPES.a, //array
  1292. TYPES.o, //object
  1293. TYPES.f, //function
  1294. TYPES.z //null
  1295. ];
  1296. var restrictedStringsSplit = ' ';
  1297. var restrictedStringsPossibilitiesSplit = ':';
  1298. var classNameAllowedValues = [TYPES.z, TYPES.s];
  1299. var numberAllowedValues = TYPES.n;
  1300. var booleanNullAllowedValues = [TYPES.z, TYPES.b];
  1301. var booleanTrueTemplate = [true, TYPES.b];
  1302. var booleanFalseTemplate = [false, TYPES.b];
  1303. var callbackTemplate = [null, [TYPES.z, TYPES.f]];
  1304. var inheritedAttrsTemplate = [['style', 'class'], [TYPES.s, TYPES.a, TYPES.z]];
  1305. var resizeAllowedValues = 'n:none b:both h:horizontal v:vertical';
  1306. var overflowBehaviorAllowedValues = 'v-h:visible-hidden v-s:visible-scroll s:scroll h:hidden';
  1307. var scrollbarsVisibilityAllowedValues = 'v:visible h:hidden a:auto';
  1308. var scrollbarsAutoHideAllowedValues = 'n:never s:scroll l:leave m:move';
  1309. var optionsDefaultsAndTemplate = {
  1310. className: ['os-theme-dark', classNameAllowedValues], //null || string
  1311. resize: ['none', resizeAllowedValues], //none || both || horizontal || vertical || n || b || h || v
  1312. sizeAutoCapable: booleanTrueTemplate, //true || false
  1313. clipAlways: booleanTrueTemplate, //true || false
  1314. normalizeRTL: booleanTrueTemplate, //true || false
  1315. paddingAbsolute: booleanFalseTemplate, //true || false
  1316. autoUpdate: [null, booleanNullAllowedValues], //true || false || null
  1317. autoUpdateInterval: [33, numberAllowedValues], //number
  1318. nativeScrollbarsOverlaid: {
  1319. showNativeScrollbars: booleanFalseTemplate, //true || false
  1320. initialize: booleanTrueTemplate //true || false
  1321. },
  1322. overflowBehavior: {
  1323. x: ['scroll', overflowBehaviorAllowedValues], //visible-hidden || visible-scroll || hidden || scroll || v-h || v-s || h || s
  1324. y: ['scroll', overflowBehaviorAllowedValues] //visible-hidden || visible-scroll || hidden || scroll || v-h || v-s || h || s
  1325. },
  1326. scrollbars: {
  1327. visibility: ['auto', scrollbarsVisibilityAllowedValues], //visible || hidden || auto || v || h || a
  1328. autoHide: ['never', scrollbarsAutoHideAllowedValues], //never || scroll || leave || move || n || s || l || m
  1329. autoHideDelay: [800, numberAllowedValues], //number
  1330. dragScrolling: booleanTrueTemplate, //true || false
  1331. clickScrolling: booleanFalseTemplate, //true || false
  1332. touchSupport: booleanTrueTemplate, //true || false
  1333. snapHandle: booleanFalseTemplate //true || false
  1334. },
  1335. textarea: {
  1336. dynWidth: booleanFalseTemplate, //true || false
  1337. dynHeight: booleanFalseTemplate, //true || false
  1338. inheritedAttrs: inheritedAttrsTemplate //string || array || null
  1339. },
  1340. callbacks: {
  1341. onInitialized: callbackTemplate, //null || function
  1342. onInitializationWithdrawn: callbackTemplate, //null || function
  1343. onDestroyed: callbackTemplate, //null || function
  1344. onScrollStart: callbackTemplate, //null || function
  1345. onScroll: callbackTemplate, //null || function
  1346. onScrollStop: callbackTemplate, //null || function
  1347. onOverflowChanged: callbackTemplate, //null || function
  1348. onOverflowAmountChanged: callbackTemplate, //null || function
  1349. onDirectionChanged: callbackTemplate, //null || function
  1350. onContentSizeChanged: callbackTemplate, //null || function
  1351. onHostSizeChanged: callbackTemplate, //null || function
  1352. onUpdated: callbackTemplate //null || function
  1353. }
  1354. };
  1355. var convert = function (template) {
  1356. var recursive = function (obj) {
  1357. var key;
  1358. var val;
  1359. var valType;
  1360. for (key in obj) {
  1361. if (!obj[LEXICON.hOP](key))
  1362. continue;
  1363. val = obj[key];
  1364. valType = type(val);
  1365. if (valType == TYPES.a)
  1366. obj[key] = val[template ? 1 : 0];
  1367. else if (valType == TYPES.o)
  1368. obj[key] = recursive(val);
  1369. }
  1370. return obj;
  1371. };
  1372. return recursive(FRAMEWORK.extend(true, {}, optionsDefaultsAndTemplate));
  1373. };
  1374. return {
  1375. _defaults: convert(),
  1376. _template: convert(true),
  1377. /**
  1378. * Validates the passed object by the passed template.
  1379. * @param obj The object which shall be validated.
  1380. * @param template The template which defines the allowed values and types.
  1381. * @param writeErrors True if errors shall be logged to the console.
  1382. * @param diffObj If a object is passed then only valid differences to this object will be returned.
  1383. * @returns {{}} A object which contains two objects called "default" and "prepared" which contains only the valid properties of the passed original object and discards not different values compared to the passed diffObj.
  1384. */
  1385. _validate: function (obj, template, writeErrors, diffObj) {
  1386. var validatedOptions = {};
  1387. var validatedOptionsPrepared = {};
  1388. var objectCopy = FRAMEWORK.extend(true, {}, obj);
  1389. var inArray = FRAMEWORK.inArray;
  1390. var isEmptyObj = FRAMEWORK.isEmptyObject;
  1391. var checkObjectProps = function (data, template, diffData, validatedOptions, validatedOptionsPrepared, prevPropName) {
  1392. for (var prop in template) {
  1393. if (template[LEXICON.hOP](prop) && data[LEXICON.hOP](prop)) {
  1394. var isValid = false;
  1395. var isDiff = false;
  1396. var templateValue = template[prop];
  1397. var templateValueType = type(templateValue);
  1398. var templateIsComplex = templateValueType == TYPES.o;
  1399. var templateTypes = type(templateValue) != TYPES.a ? [templateValue] : templateValue;
  1400. var dataDiffValue = diffData[prop];
  1401. var dataValue = data[prop];
  1402. var dataValueType = type(dataValue);
  1403. var propPrefix = prevPropName ? prevPropName + '.' : '';
  1404. var error = "The option \"" + propPrefix + prop + "\" wasn't set, because";
  1405. var errorPossibleTypes = [];
  1406. var errorRestrictedStrings = [];
  1407. var restrictedStringValuesSplit;
  1408. var restrictedStringValuesPossibilitiesSplit;
  1409. var isRestrictedValue;
  1410. var mainPossibility;
  1411. var currType;
  1412. var i;
  1413. var v;
  1414. var j;
  1415. dataDiffValue = dataDiffValue === undefined ? {} : dataDiffValue;
  1416. //if the template has a object as value, it means that the options are complex (verschachtelt)
  1417. if (templateIsComplex && dataValueType == TYPES.o) {
  1418. validatedOptions[prop] = {};
  1419. validatedOptionsPrepared[prop] = {};
  1420. checkObjectProps(dataValue, templateValue, dataDiffValue, validatedOptions[prop], validatedOptionsPrepared[prop], propPrefix + prop);
  1421. FRAMEWORK.each([data, validatedOptions, validatedOptionsPrepared], function (index, value) {
  1422. if (isEmptyObj(value[prop])) {
  1423. delete value[prop];
  1424. }
  1425. });
  1426. }
  1427. else if (!templateIsComplex) {
  1428. for (i = 0; i < templateTypes[LEXICON.l]; i++) {
  1429. currType = templateTypes[i];
  1430. templateValueType = type(currType);
  1431. //if currtype is string and starts with restrictedStringPrefix and end with restrictedStringSuffix
  1432. isRestrictedValue = templateValueType == TYPES.s && inArray(currType, possibleTemplateTypes) === -1;
  1433. if (isRestrictedValue) {
  1434. errorPossibleTypes.push(TYPES.s);
  1435. //split it into a array which contains all possible values for example: ["y:yes", "n:no", "m:maybe"]
  1436. restrictedStringValuesSplit = currType.split(restrictedStringsSplit);
  1437. errorRestrictedStrings = errorRestrictedStrings.concat(restrictedStringValuesSplit);
  1438. for (v = 0; v < restrictedStringValuesSplit[LEXICON.l]; v++) {
  1439. //split the possible values into their possibiliteis for example: ["y", "yes"] -> the first is always the mainPossibility
  1440. restrictedStringValuesPossibilitiesSplit = restrictedStringValuesSplit[v].split(restrictedStringsPossibilitiesSplit);
  1441. mainPossibility = restrictedStringValuesPossibilitiesSplit[0];
  1442. for (j = 0; j < restrictedStringValuesPossibilitiesSplit[LEXICON.l]; j++) {
  1443. //if any possibility matches with the dataValue, its valid
  1444. if (dataValue === restrictedStringValuesPossibilitiesSplit[j]) {
  1445. isValid = true;
  1446. break;
  1447. }
  1448. }
  1449. if (isValid)
  1450. break;
  1451. }
  1452. }
  1453. else {
  1454. errorPossibleTypes.push(currType);
  1455. if (dataValueType === currType) {
  1456. isValid = true;
  1457. break;
  1458. }
  1459. }
  1460. }
  1461. if (isValid) {
  1462. isDiff = dataValue !== dataDiffValue;
  1463. if (isDiff)
  1464. validatedOptions[prop] = dataValue;
  1465. if (isRestrictedValue ? inArray(dataDiffValue, restrictedStringValuesPossibilitiesSplit) < 0 : isDiff)
  1466. validatedOptionsPrepared[prop] = isRestrictedValue ? mainPossibility : dataValue;
  1467. }
  1468. else if (writeErrors) {
  1469. console.warn(error + " it doesn't accept the type [ " + dataValueType.toUpperCase() + " ] with the value of \"" + dataValue + "\".\r\n" +
  1470. "Accepted types are: [ " + errorPossibleTypes.join(', ').toUpperCase() + " ]." +
  1471. (errorRestrictedStrings[length] > 0 ? "\r\nValid strings are: [ " + errorRestrictedStrings.join(', ').split(restrictedStringsPossibilitiesSplit).join(', ') + " ]." : ''));
  1472. }
  1473. delete data[prop];
  1474. }
  1475. }
  1476. }
  1477. };
  1478. checkObjectProps(objectCopy, template, diffObj || {}, validatedOptions, validatedOptionsPrepared);
  1479. //add values which aren't specified in the template to the finished validated object to prevent them from being discarded
  1480. /*
  1481. if(keepForeignProps) {
  1482. FRAMEWORK.extend(true, validatedOptions, objectCopy);
  1483. FRAMEWORK.extend(true, validatedOptionsPrepared, objectCopy);
  1484. }
  1485. */
  1486. if (!isEmptyObj(objectCopy) && writeErrors)
  1487. console.warn('The following options are discarded due to invalidity:\r\n' + window.JSON.stringify(objectCopy, null, 2));
  1488. return {
  1489. _default: validatedOptions,
  1490. _prepared: validatedOptionsPrepared
  1491. };
  1492. }
  1493. }
  1494. }());
  1495. /**
  1496. * Initializes the object which contains global information about the plugin and each instance of it.
  1497. */
  1498. function initOverlayScrollbarsStatics() {
  1499. if (!_pluginsGlobals)
  1500. _pluginsGlobals = new OverlayScrollbarsGlobals(_pluginsOptions._defaults);
  1501. if (!_pluginsAutoUpdateLoop)
  1502. _pluginsAutoUpdateLoop = new OverlayScrollbarsAutoUpdateLoop(_pluginsGlobals);
  1503. }
  1504. /**
  1505. * The global object for the OverlayScrollbars objects. It contains resources which every OverlayScrollbars object needs. This object is initialized only once: if the first OverlayScrollbars object gets initialized.
  1506. * @param defaultOptions
  1507. * @constructor
  1508. */
  1509. function OverlayScrollbarsGlobals(defaultOptions) {
  1510. var _base = this;
  1511. var strOverflow = 'overflow';
  1512. var strHidden = 'hidden';
  1513. var strScroll = 'scroll';
  1514. var bodyElement = FRAMEWORK('body');
  1515. var scrollbarDummyElement = FRAMEWORK('<div id="os-dummy-scrollbar-size"><div></div></div>');
  1516. var scrollbarDummyElement0 = scrollbarDummyElement[0];
  1517. var dummyContainerChild = FRAMEWORK(scrollbarDummyElement.children('div').eq(0));
  1518. bodyElement.append(scrollbarDummyElement);
  1519. scrollbarDummyElement.hide().show(); //fix IE8 bug (incorrect measuring)
  1520. var nativeScrollbarSize = calcNativeScrollbarSize(scrollbarDummyElement0);
  1521. var nativeScrollbarIsOverlaid = {
  1522. x: nativeScrollbarSize.x === 0,
  1523. y: nativeScrollbarSize.y === 0
  1524. };
  1525. var msie = (function () {
  1526. var ua = window.navigator.userAgent;
  1527. var strIndexOf = 'indexOf';
  1528. var strSubString = 'substring';
  1529. var msie = ua[strIndexOf]('MSIE ');
  1530. var trident = ua[strIndexOf]('Trident/');
  1531. var edge = ua[strIndexOf]('Edge/');
  1532. var rv = ua[strIndexOf]('rv:');
  1533. var result;
  1534. var parseIntFunc = parseInt;
  1535. // IE 10 or older => return version number
  1536. if (msie > 0)
  1537. result = parseIntFunc(ua[strSubString](msie + 5, ua[strIndexOf]('.', msie)), 10);
  1538. // IE 11 => return version number
  1539. else if (trident > 0)
  1540. result = parseIntFunc(ua[strSubString](rv + 3, ua[strIndexOf]('.', rv)), 10);
  1541. // Edge (IE 12+) => return version number
  1542. else if (edge > 0)
  1543. result = parseIntFunc(ua[strSubString](edge + 5, ua[strIndexOf]('.', edge)), 10);
  1544. // other browser
  1545. return result;
  1546. })();
  1547. FRAMEWORK.extend(_base, {
  1548. defaultOptions: defaultOptions,
  1549. msie: msie,
  1550. autoUpdateLoop: false,
  1551. autoUpdateRecommended: !COMPATIBILITY.mO(),
  1552. nativeScrollbarSize: nativeScrollbarSize,
  1553. nativeScrollbarIsOverlaid: nativeScrollbarIsOverlaid,
  1554. nativeScrollbarStyling: (function () {
  1555. var result = false;
  1556. scrollbarDummyElement.addClass('os-viewport-native-scrollbars-invisible');
  1557. try {
  1558. result = (scrollbarDummyElement.css('scrollbar-width') === 'none' && (msie > 9 || !msie)) || window.getComputedStyle(scrollbarDummyElement0, '::-webkit-scrollbar').getPropertyValue('display') === 'none';
  1559. } catch (ex) { }
  1560. //fix opera bug: scrollbar styles will only appear if overflow value is scroll or auto during the activation of the style.
  1561. //and set overflow to scroll
  1562. //scrollbarDummyElement.css(strOverflow, strHidden).hide().css(strOverflow, strScroll).show();
  1563. //return (scrollbarDummyElement0[LEXICON.oH] - scrollbarDummyElement0[LEXICON.cH]) === 0 && (scrollbarDummyElement0[LEXICON.oW] - scrollbarDummyElement0[LEXICON.cW]) === 0;
  1564. return result;
  1565. })(),
  1566. overlayScrollbarDummySize: { x: 30, y: 30 },
  1567. cssCalc: (function () {
  1568. var dummyStyle = document.createElement('div')[LEXICON.s];
  1569. var strCalc = 'calc';
  1570. var i = -1;
  1571. var prop;
  1572. for (; i < VENDORS._cssPrefixes[LEXICON.l]; i++) {
  1573. prop = i < 0 ? strCalc : VENDORS._cssPrefixes[i] + strCalc;
  1574. dummyStyle.cssText = 'width:' + prop + '(1px);';
  1575. if (dummyStyle[LEXICON.l])
  1576. return prop;
  1577. }
  1578. return null;
  1579. })(),
  1580. restrictedMeasuring: (function () {
  1581. //https://bugzilla.mozilla.org/show_bug.cgi?id=1439305
  1582. scrollbarDummyElement.css(strOverflow, strHidden);
  1583. var scrollSize = {
  1584. w: scrollbarDummyElement0[LEXICON.sW],
  1585. h: scrollbarDummyElement0[LEXICON.sH]
  1586. };
  1587. scrollbarDummyElement.css(strOverflow, 'visible');
  1588. var scrollSize2 = {
  1589. w: scrollbarDummyElement0[LEXICON.sW],
  1590. h: scrollbarDummyElement0[LEXICON.sH]
  1591. };
  1592. return (scrollSize.w - scrollSize2.w) !== 0 || (scrollSize.h - scrollSize2.h) !== 0;
  1593. })(),
  1594. rtlScrollBehavior: (function () {
  1595. scrollbarDummyElement.css({ 'overflow-y': strHidden, 'overflow-x': strScroll, 'direction': 'rtl' }).scrollLeft(0);
  1596. var dummyContainerOffset = scrollbarDummyElement.offset();
  1597. var dummyContainerChildOffset = dummyContainerChild.offset();
  1598. scrollbarDummyElement.scrollLeft(999);
  1599. var dummyContainerScrollOffsetAfterScroll = dummyContainerChild.offset();
  1600. return {
  1601. //origin direction = determines if the zero scroll position is on the left or right side
  1602. //'i' means 'invert' (i === true means that the axis must be inverted to be correct)
  1603. //true = on the left side
  1604. //false = on the right side
  1605. i: dummyContainerOffset.left === dummyContainerChildOffset.left,
  1606. //negative = determines if the maximum scroll is positive or negative
  1607. //'n' means 'negate' (n === true means that the axis must be negated to be correct)
  1608. //true = negative
  1609. //false = positive
  1610. n: dummyContainerChildOffset.left - dummyContainerScrollOffsetAfterScroll.left === 0
  1611. };
  1612. })(),
  1613. supportTransform: VENDORS._cssProperty('transform') !== undefined,
  1614. supportTransition: VENDORS._cssProperty('transition') !== undefined,
  1615. supportPassiveEvents: (function () {
  1616. var supportsPassive = false;
  1617. try {
  1618. window.addEventListener('test', null, Object.defineProperty({}, 'passive', {
  1619. get: function () {
  1620. supportsPassive = true;
  1621. }
  1622. }));
  1623. } catch (e) { }
  1624. return supportsPassive;
  1625. })(),
  1626. supportResizeObserver: !!COMPATIBILITY.rO(),
  1627. supportMutationObserver: !!COMPATIBILITY.mO()
  1628. });
  1629. scrollbarDummyElement.removeAttr(LEXICON.s).remove();
  1630. //Catch zoom event:
  1631. (function () {
  1632. if (nativeScrollbarIsOverlaid.x && nativeScrollbarIsOverlaid.y)
  1633. return;
  1634. var abs = MATH.abs;
  1635. var windowWidth = COMPATIBILITY.wW();
  1636. var windowHeight = COMPATIBILITY.wH();
  1637. var windowDpr = getWindowDPR();
  1638. var onResize = function () {
  1639. if (INSTANCES().length > 0) {
  1640. var newW = COMPATIBILITY.wW();
  1641. var newH = COMPATIBILITY.wH();
  1642. var deltaW = newW - windowWidth;
  1643. var deltaH = newH - windowHeight;
  1644. if (deltaW === 0 && deltaH === 0)
  1645. return;
  1646. var deltaWRatio = MATH.round(newW / (windowWidth / 100.0));
  1647. var deltaHRatio = MATH.round(newH / (windowHeight / 100.0));
  1648. var absDeltaW = abs(deltaW);
  1649. var absDeltaH = abs(deltaH);
  1650. var absDeltaWRatio = abs(deltaWRatio);
  1651. var absDeltaHRatio = abs(deltaHRatio);
  1652. var newDPR = getWindowDPR();
  1653. var deltaIsBigger = absDeltaW > 2 && absDeltaH > 2;
  1654. var difference = !differenceIsBiggerThanOne(absDeltaWRatio, absDeltaHRatio);
  1655. var dprChanged = newDPR !== windowDpr && windowDpr > 0;
  1656. var isZoom = deltaIsBigger && difference && dprChanged;
  1657. var oldScrollbarSize = _base.nativeScrollbarSize;
  1658. var newScrollbarSize;
  1659. if (isZoom) {
  1660. bodyElement.append(scrollbarDummyElement);
  1661. newScrollbarSize = _base.nativeScrollbarSize = calcNativeScrollbarSize(scrollbarDummyElement[0]);
  1662. scrollbarDummyElement.remove();
  1663. if (oldScrollbarSize.x !== newScrollbarSize.x || oldScrollbarSize.y !== newScrollbarSize.y) {
  1664. FRAMEWORK.each(INSTANCES(), function () {
  1665. if (INSTANCES(this))
  1666. INSTANCES(this).update('zoom');
  1667. });
  1668. }
  1669. }
  1670. windowWidth = newW;
  1671. windowHeight = newH;
  1672. windowDpr = newDPR;
  1673. }
  1674. };
  1675. function differenceIsBiggerThanOne(valOne, valTwo) {
  1676. var absValOne = abs(valOne);
  1677. var absValTwo = abs(valTwo);
  1678. return !(absValOne === absValTwo || absValOne + 1 === absValTwo || absValOne - 1 === absValTwo);
  1679. }
  1680. function getWindowDPR() {
  1681. var dDPI = window.screen.deviceXDPI || 0;
  1682. var sDPI = window.screen.logicalXDPI || 1;
  1683. return window.devicePixelRatio || (dDPI / sDPI);
  1684. }
  1685. FRAMEWORK(window).on('resize', onResize);
  1686. })();
  1687. function calcNativeScrollbarSize(measureElement) {
  1688. return {
  1689. x: measureElement[LEXICON.oH] - measureElement[LEXICON.cH],
  1690. y: measureElement[LEXICON.oW] - measureElement[LEXICON.cW]
  1691. };
  1692. }
  1693. }
  1694. /**
  1695. * The object which manages the auto update loop for all OverlayScrollbars objects. This object is initialized only once: if the first OverlayScrollbars object gets initialized.
  1696. * @constructor
  1697. */
  1698. function OverlayScrollbarsAutoUpdateLoop(globals) {
  1699. var _base = this;
  1700. var _inArray = FRAMEWORK.inArray;
  1701. var _getNow = COMPATIBILITY.now;
  1702. var _strAutoUpdate = 'autoUpdate';
  1703. var _strAutoUpdateInterval = _strAutoUpdate + 'Interval';
  1704. var _strLength = LEXICON.l;
  1705. var _loopingInstances = [];
  1706. var _loopingInstancesIntervalCache = [];
  1707. var _loopIsActive = false;
  1708. var _loopIntervalDefault = 33;
  1709. var _loopInterval = _loopIntervalDefault;
  1710. var _loopTimeOld = _getNow();
  1711. var _loopID;
  1712. /**
  1713. * The auto update loop which will run every 50 milliseconds or less if the update interval of a instance is lower than 50 milliseconds.
  1714. */
  1715. var loop = function () {
  1716. if (_loopingInstances[_strLength] > 0 && _loopIsActive) {
  1717. _loopID = COMPATIBILITY.rAF()(function () {
  1718. loop();
  1719. });
  1720. var timeNew = _getNow();
  1721. var timeDelta = timeNew - _loopTimeOld;
  1722. var lowestInterval;
  1723. var instance;
  1724. var instanceOptions;
  1725. var instanceAutoUpdateAllowed;
  1726. var instanceAutoUpdateInterval;
  1727. var now;
  1728. if (timeDelta > _loopInterval) {
  1729. _loopTimeOld = timeNew - (timeDelta % _loopInterval);
  1730. lowestInterval = _loopIntervalDefault;
  1731. for (var i = 0; i < _loopingInstances[_strLength]; i++) {
  1732. instance = _loopingInstances[i];
  1733. if (instance !== undefined) {
  1734. instanceOptions = instance.options();
  1735. instanceAutoUpdateAllowed = instanceOptions[_strAutoUpdate];
  1736. instanceAutoUpdateInterval = MATH.max(1, instanceOptions[_strAutoUpdateInterval]);
  1737. now = _getNow();
  1738. if ((instanceAutoUpdateAllowed === true || instanceAutoUpdateAllowed === null) && (now - _loopingInstancesIntervalCache[i]) > instanceAutoUpdateInterval) {
  1739. instance.update('auto');
  1740. _loopingInstancesIntervalCache[i] = new Date(now += instanceAutoUpdateInterval);
  1741. }
  1742. lowestInterval = MATH.max(1, MATH.min(lowestInterval, instanceAutoUpdateInterval));
  1743. }
  1744. }
  1745. _loopInterval = lowestInterval;
  1746. }
  1747. } else {
  1748. _loopInterval = _loopIntervalDefault;
  1749. }
  1750. };
  1751. /**
  1752. * Add OverlayScrollbars instance to the auto update loop. Only successful if the instance isn't already added.
  1753. * @param instance The instance which shall be updated in a loop automatically.
  1754. */
  1755. _base.add = function (instance) {
  1756. if (_inArray(instance, _loopingInstances) === -1) {
  1757. _loopingInstances.push(instance);
  1758. _loopingInstancesIntervalCache.push(_getNow());
  1759. if (_loopingInstances[_strLength] > 0 && !_loopIsActive) {
  1760. _loopIsActive = true;
  1761. globals.autoUpdateLoop = _loopIsActive;
  1762. loop();
  1763. }
  1764. }
  1765. };
  1766. /**
  1767. * Remove OverlayScrollbars instance from the auto update loop. Only successful if the instance was added before.
  1768. * @param instance The instance which shall be updated in a loop automatically.
  1769. */
  1770. _base.remove = function (instance) {
  1771. var index = _inArray(instance, _loopingInstances);
  1772. if (index > -1) {
  1773. //remove from loopingInstances list
  1774. _loopingInstancesIntervalCache.splice(index, 1);
  1775. _loopingInstances.splice(index, 1);
  1776. //correct update loop behavior
  1777. if (_loopingInstances[_strLength] === 0 && _loopIsActive) {
  1778. _loopIsActive = false;
  1779. globals.autoUpdateLoop = _loopIsActive;
  1780. if (_loopID !== undefined) {
  1781. COMPATIBILITY.cAF()(_loopID);
  1782. _loopID = -1;
  1783. }
  1784. }
  1785. }
  1786. };
  1787. }
  1788. /**
  1789. * A object which manages the scrollbars visibility of the target element.
  1790. * @param pluginTargetElement The element from which the scrollbars shall be hidden.
  1791. * @param options The custom options.
  1792. * @param extensions The custom extensions.
  1793. * @param globals
  1794. * @param autoUpdateLoop
  1795. * @returns {*}
  1796. * @constructor
  1797. */
  1798. function OverlayScrollbarsInstance(pluginTargetElement, options, extensions, globals, autoUpdateLoop) {
  1799. //shortcuts
  1800. var type = COMPATIBILITY.type;
  1801. var inArray = FRAMEWORK.inArray;
  1802. var each = FRAMEWORK.each;
  1803. //make correct instanceof
  1804. var _base = new _plugin();
  1805. var _frameworkProto = FRAMEWORK[LEXICON.p];
  1806. //if passed element is no HTML element: skip and return
  1807. if (!isHTMLElement(pluginTargetElement))
  1808. return;
  1809. //if passed element is already initialized: set passed options if there are any and return its instance
  1810. if (INSTANCES(pluginTargetElement)) {
  1811. var inst = INSTANCES(pluginTargetElement);
  1812. inst.options(options);
  1813. return inst;
  1814. }
  1815. //globals:
  1816. var _nativeScrollbarIsOverlaid;
  1817. var _overlayScrollbarDummySize;
  1818. var _rtlScrollBehavior;
  1819. var _autoUpdateRecommended;
  1820. var _msieVersion;
  1821. var _nativeScrollbarStyling;
  1822. var _cssCalc;
  1823. var _nativeScrollbarSize;
  1824. var _supportTransition;
  1825. var _supportTransform;
  1826. var _supportPassiveEvents;
  1827. var _supportResizeObserver;
  1828. var _supportMutationObserver;
  1829. var _restrictedMeasuring;
  1830. //general readonly:
  1831. var _initialized;
  1832. var _destroyed;
  1833. var _isTextarea;
  1834. var _isBody;
  1835. var _documentMixed;
  1836. var _domExists;
  1837. //general:
  1838. var _isBorderBox;
  1839. var _sizeAutoObserverAdded;
  1840. var _paddingX;
  1841. var _paddingY;
  1842. var _borderX;
  1843. var _borderY;
  1844. var _marginX;
  1845. var _marginY;
  1846. var _isRTL;
  1847. var _sleeping;
  1848. var _contentBorderSize = {};
  1849. var _scrollHorizontalInfo = {};
  1850. var _scrollVerticalInfo = {};
  1851. var _viewportSize = {};
  1852. var _nativeScrollbarMinSize = {};
  1853. //naming:
  1854. var _strMinusHidden = '-hidden';
  1855. var _strMarginMinus = 'margin-';
  1856. var _strPaddingMinus = 'padding-';
  1857. var _strBorderMinus = 'border-';
  1858. var _strTop = 'top';
  1859. var _strRight = 'right';
  1860. var _strBottom = 'bottom';
  1861. var _strLeft = 'left';
  1862. var _strMinMinus = 'min-';
  1863. var _strMaxMinus = 'max-';
  1864. var _strWidth = 'width';
  1865. var _strHeight = 'height';
  1866. var _strFloat = 'float';
  1867. var _strEmpty = '';
  1868. var _strAuto = 'auto';
  1869. var _strSync = 'sync';
  1870. var _strScroll = 'scroll';
  1871. var _strHundredPercent = '100%';
  1872. var _strX = 'x';
  1873. var _strY = 'y';
  1874. var _strDot = '.';
  1875. var _strSpace = ' ';
  1876. var _strScrollbar = 'scrollbar';
  1877. var _strMinusHorizontal = '-horizontal';
  1878. var _strMinusVertical = '-vertical';
  1879. var _strScrollLeft = _strScroll + 'Left';
  1880. var _strScrollTop = _strScroll + 'Top';
  1881. var _strMouseTouchDownEvent = 'mousedown touchstart';
  1882. var _strMouseTouchUpEvent = 'mouseup touchend touchcancel';
  1883. var _strMouseTouchMoveEvent = 'mousemove touchmove';
  1884. var _strMouseTouchEnter = 'mouseenter';
  1885. var _strMouseTouchLeave = 'mouseleave';
  1886. var _strKeyDownEvent = 'keydown';
  1887. var _strKeyUpEvent = 'keyup';
  1888. var _strSelectStartEvent = 'selectstart';
  1889. var _strTransitionEndEvent = 'transitionend webkitTransitionEnd oTransitionEnd';
  1890. var _strResizeObserverProperty = '__overlayScrollbarsRO__';
  1891. //class names:
  1892. var _cassNamesPrefix = 'os-';
  1893. var _classNameHTMLElement = _cassNamesPrefix + 'html';
  1894. var _classNameHostElement = _cassNamesPrefix + 'host';
  1895. var _classNameHostTextareaElement = _classNameHostElement + '-textarea';
  1896. var _classNameHostScrollbarHorizontalHidden = _classNameHostElement + '-' + _strScrollbar + _strMinusHorizontal + _strMinusHidden;
  1897. var _classNameHostScrollbarVerticalHidden = _classNameHostElement + '-' + _strScrollbar + _strMinusVertical + _strMinusHidden;
  1898. var _classNameHostTransition = _classNameHostElement + '-transition';
  1899. var _classNameHostRTL = _classNameHostElement + '-rtl';
  1900. var _classNameHostResizeDisabled = _classNameHostElement + '-resize-disabled';
  1901. var _classNameHostScrolling = _classNameHostElement + '-scrolling';
  1902. var _classNameHostOverflow = _classNameHostElement + '-overflow';
  1903. var _classNameHostOverflowX = _classNameHostOverflow + '-x';
  1904. var _classNameHostOverflowY = _classNameHostOverflow + '-y';
  1905. var _classNameTextareaElement = _cassNamesPrefix + 'textarea';
  1906. var _classNameTextareaCoverElement = _classNameTextareaElement + '-cover';
  1907. var _classNamePaddingElement = _cassNamesPrefix + 'padding';
  1908. var _classNameViewportElement = _cassNamesPrefix + 'viewport';
  1909. var _classNameViewportNativeScrollbarsInvisible = _classNameViewportElement + '-native-scrollbars-invisible';
  1910. var _classNameViewportNativeScrollbarsOverlaid = _classNameViewportElement + '-native-scrollbars-overlaid';
  1911. var _classNameContentElement = _cassNamesPrefix + 'content';
  1912. var _classNameContentArrangeElement = _cassNamesPrefix + 'content-arrange';
  1913. var _classNameContentGlueElement = _cassNamesPrefix + 'content-glue';
  1914. var _classNameSizeAutoObserverElement = _cassNamesPrefix + 'size-auto-observer';
  1915. var _classNameResizeObserverElement = _cassNamesPrefix + 'resize-observer';
  1916. var _classNameResizeObserverItemElement = _cassNamesPrefix + 'resize-observer-item';
  1917. var _classNameResizeObserverItemFinalElement = _classNameResizeObserverItemElement + '-final';
  1918. var _classNameTextInherit = _cassNamesPrefix + 'text-inherit';
  1919. var _classNameScrollbar = _cassNamesPrefix + _strScrollbar;
  1920. var _classNameScrollbarTrack = _classNameScrollbar + '-track';
  1921. var _classNameScrollbarTrackOff = _classNameScrollbarTrack + '-off';
  1922. var _classNameScrollbarHandle = _classNameScrollbar + '-handle';
  1923. var _classNameScrollbarHandleOff = _classNameScrollbarHandle + '-off';
  1924. var _classNameScrollbarUnusable = _classNameScrollbar + '-unusable';
  1925. var _classNameScrollbarAutoHidden = _classNameScrollbar + '-' + _strAuto + _strMinusHidden;
  1926. var _classNameScrollbarCorner = _classNameScrollbar + '-corner';
  1927. var _classNameScrollbarCornerResize = _classNameScrollbarCorner + '-resize';
  1928. var _classNameScrollbarCornerResizeB = _classNameScrollbarCornerResize + '-both';
  1929. var _classNameScrollbarCornerResizeH = _classNameScrollbarCornerResize + _strMinusHorizontal;
  1930. var _classNameScrollbarCornerResizeV = _classNameScrollbarCornerResize + _strMinusVertical;
  1931. var _classNameScrollbarHorizontal = _classNameScrollbar + _strMinusHorizontal;
  1932. var _classNameScrollbarVertical = _classNameScrollbar + _strMinusVertical;
  1933. var _classNameDragging = _cassNamesPrefix + 'dragging';
  1934. var _classNameThemeNone = _cassNamesPrefix + 'theme-none';
  1935. var _classNamesDynamicDestroy = [
  1936. _classNameViewportNativeScrollbarsInvisible,
  1937. _classNameViewportNativeScrollbarsOverlaid,
  1938. _classNameScrollbarTrackOff,
  1939. _classNameScrollbarHandleOff,
  1940. _classNameScrollbarUnusable,
  1941. _classNameScrollbarAutoHidden,
  1942. _classNameScrollbarCornerResize,
  1943. _classNameScrollbarCornerResizeB,
  1944. _classNameScrollbarCornerResizeH,
  1945. _classNameScrollbarCornerResizeV,
  1946. _classNameDragging].join(_strSpace);
  1947. //callbacks:
  1948. var _callbacksInitQeueue = [];
  1949. //options:
  1950. var _defaultOptions;
  1951. var _currentOptions;
  1952. var _currentPreparedOptions;
  1953. //extensions:
  1954. var _extensions = {};
  1955. var _extensionsPrivateMethods = 'added removed on contract';
  1956. //update
  1957. var _lastUpdateTime;
  1958. var _swallowedUpdateHints = {};
  1959. var _swallowedUpdateTimeout;
  1960. var _swallowUpdateLag = 42;
  1961. var _imgs = [];
  1962. //DOM elements:
  1963. var _windowElement;
  1964. var _documentElement;
  1965. var _htmlElement;
  1966. var _bodyElement;
  1967. var _targetElement; //the target element of this OverlayScrollbars object
  1968. var _hostElement; //the host element of this OverlayScrollbars object -> may be the same as targetElement
  1969. var _sizeAutoObserverElement; //observes size auto changes
  1970. var _sizeObserverElement; //observes size and padding changes
  1971. var _paddingElement; //manages the padding
  1972. var _viewportElement; //is the viewport of our scrollbar model
  1973. var _contentElement; //the element which holds the content
  1974. var _contentArrangeElement; //is needed for correct sizing of the content element (only if native scrollbars are overlays)
  1975. var _contentGlueElement; //has always the size of the content element
  1976. var _textareaCoverElement; //only applied if target is a textarea element. Used for correct size calculation and for prevention of uncontrolled scrolling
  1977. var _scrollbarCornerElement;
  1978. var _scrollbarHorizontalElement;
  1979. var _scrollbarHorizontalTrackElement;
  1980. var _scrollbarHorizontalHandleElement;
  1981. var _scrollbarVerticalElement;
  1982. var _scrollbarVerticalTrackElement;
  1983. var _scrollbarVerticalHandleElement;
  1984. var _windowElementNative;
  1985. var _documentElementNative;
  1986. var _targetElementNative;
  1987. var _hostElementNative;
  1988. var _sizeAutoObserverElementNative;
  1989. var _sizeObserverElementNative;
  1990. var _paddingElementNative;
  1991. var _viewportElementNative;
  1992. var _contentElementNative;
  1993. //Cache:
  1994. var _hostSizeCache;
  1995. var _contentScrollSizeCache;
  1996. var _arrangeContentSizeCache;
  1997. var _hasOverflowCache;
  1998. var _hideOverflowCache;
  1999. var _widthAutoCache;
  2000. var _heightAutoCache;
  2001. var _cssMaxValueCache;
  2002. var _cssBoxSizingCache;
  2003. var _cssPaddingCache;
  2004. var _cssBorderCache;
  2005. var _cssMarginCache;
  2006. var _cssDirectionCache;
  2007. var _cssDirectionDetectedCache;
  2008. var _paddingAbsoluteCache;
  2009. var _clipAlwaysCache;
  2010. var _contentGlueSizeCache;
  2011. var _overflowBehaviorCache;
  2012. var _overflowAmountCache;
  2013. var _ignoreOverlayScrollbarHidingCache;
  2014. var _autoUpdateCache;
  2015. var _sizeAutoCapableCache;
  2016. var _contentElementScrollSizeChangeDetectedCache;
  2017. var _hostElementSizeChangeDetectedCache;
  2018. var _scrollbarsVisibilityCache;
  2019. var _scrollbarsAutoHideCache;
  2020. var _scrollbarsClickScrollingCache;
  2021. var _scrollbarsDragScrollingCache;
  2022. var _resizeCache;
  2023. var _normalizeRTLCache;
  2024. var _classNameCache;
  2025. var _oldClassName;
  2026. var _textareaAutoWrappingCache;
  2027. var _textareaInfoCache;
  2028. var _textareaSizeCache;
  2029. var _textareaDynHeightCache;
  2030. var _textareaDynWidthCache;
  2031. var _bodyMinSizeCache;
  2032. var _viewportScrollSizeCache;
  2033. var _displayIsHiddenCache;
  2034. var _updateAutoCache = {};
  2035. //MutationObserver:
  2036. var _mutationObserverHost;
  2037. var _mutationObserverContent;
  2038. var _mutationObserverHostCallback;
  2039. var _mutationObserverContentCallback;
  2040. var _mutationObserversConnected;
  2041. var _mutationObserverAttrsTextarea = ['wrap', 'cols', 'rows'];
  2042. var _mutationObserverAttrsHost = [LEXICON.i, LEXICON.c, LEXICON.s, 'open'];
  2043. //events:
  2044. var _destroyEvents = [];
  2045. //textarea:
  2046. var _textareaHasFocus;
  2047. //scrollbars:
  2048. var _scrollbarsAutoHideTimeoutId;
  2049. var _scrollbarsAutoHideMoveTimeoutId;
  2050. var _scrollbarsAutoHideDelay;
  2051. var _scrollbarsAutoHideNever;
  2052. var _scrollbarsAutoHideScroll;
  2053. var _scrollbarsAutoHideMove;
  2054. var _scrollbarsAutoHideLeave;
  2055. var _scrollbarsHandleHovered;
  2056. var _scrollbarsHandlesDefineScrollPos;
  2057. //resize
  2058. var _resizeNone;
  2059. var _resizeBoth;
  2060. var _resizeHorizontal;
  2061. var _resizeVertical;
  2062. //==== Event Listener ====//
  2063. /**
  2064. * Adds or removes a event listener from the given element.
  2065. * @param element The element to which the event listener shall be applied or removed.
  2066. * @param eventNames The name(s) of the events.
  2067. * @param listener The method which shall be called.
  2068. * @param remove True if the handler shall be removed, false or undefined if the handler shall be added.
  2069. */
  2070. function setupResponsiveEventListener(element, eventNames, listener, remove, passive) {
  2071. var collected = type(eventNames) == TYPES.a && type(listener) == TYPES.a;
  2072. var method = remove ? 'removeEventListener' : 'addEventListener';
  2073. var onOff = remove ? 'off' : 'on';
  2074. var events = collected ? false : eventNames.split(_strSpace)
  2075. var i = 0;
  2076. if (collected) {
  2077. for (; i < eventNames[LEXICON.l]; i++)
  2078. setupResponsiveEventListener(element, eventNames[i], listener[i], remove);
  2079. }
  2080. else {
  2081. for (; i < events[LEXICON.l]; i++) {
  2082. if (_supportPassiveEvents)
  2083. element[0][method](events[i], listener, { passive: passive || false });
  2084. else
  2085. element[onOff](events[i], listener);
  2086. }
  2087. }
  2088. }
  2089. function addDestroyEventListener(element, eventNames, listener, passive) {
  2090. setupResponsiveEventListener(element, eventNames, listener, false, passive);
  2091. _destroyEvents.push(COMPATIBILITY.bind(setupResponsiveEventListener, 0, element, eventNames, listener, true, passive));
  2092. }
  2093. //==== Resize Observer ====//
  2094. /**
  2095. * Adds or removes a resize observer from the given element.
  2096. * @param targetElement The element to which the resize observer shall be added or removed.
  2097. * @param onElementResizedCallback The callback which is fired every time the resize observer registers a size change or false / undefined if the resizeObserver shall be removed.
  2098. */
  2099. function setupResizeObserver(targetElement, onElementResizedCallback) {
  2100. if (targetElement) {
  2101. var resizeObserver = COMPATIBILITY.rO();
  2102. var strAnimationStartEvent = 'animationstart mozAnimationStart webkitAnimationStart MSAnimationStart';
  2103. var strChildNodes = 'childNodes';
  2104. var constScroll = 3333333;
  2105. var callback = function () {
  2106. targetElement[_strScrollTop](constScroll)[_strScrollLeft](_isRTL ? _rtlScrollBehavior.n ? -constScroll : _rtlScrollBehavior.i ? 0 : constScroll : constScroll);
  2107. onElementResizedCallback();
  2108. };
  2109. //add resize observer:
  2110. if (onElementResizedCallback) {
  2111. if (_supportResizeObserver) {
  2112. var element = targetElement.append(generateDiv(_classNameResizeObserverElement + ' observed')).contents()[0];
  2113. var observer = element[_strResizeObserverProperty] = new resizeObserver(callback);
  2114. observer.observe(element);
  2115. }
  2116. else {
  2117. if (_msieVersion > 9 || !_autoUpdateRecommended) {
  2118. targetElement.prepend(
  2119. generateDiv(_classNameResizeObserverElement,
  2120. generateDiv({ c: _classNameResizeObserverItemElement, dir: 'ltr' },
  2121. generateDiv(_classNameResizeObserverItemElement,
  2122. generateDiv(_classNameResizeObserverItemFinalElement)
  2123. ) +
  2124. generateDiv(_classNameResizeObserverItemElement,
  2125. generateDiv({ c: _classNameResizeObserverItemFinalElement, style: 'width: 200%; height: 200%' })
  2126. )
  2127. )
  2128. )
  2129. );
  2130. var observerElement = targetElement[0][strChildNodes][0][strChildNodes][0];
  2131. var shrinkElement = FRAMEWORK(observerElement[strChildNodes][1]);
  2132. var expandElement = FRAMEWORK(observerElement[strChildNodes][0]);
  2133. var expandElementChild = FRAMEWORK(expandElement[0][strChildNodes][0]);
  2134. var widthCache = observerElement[LEXICON.oW];
  2135. var heightCache = observerElement[LEXICON.oH];
  2136. var isDirty;
  2137. var rAFId;
  2138. var currWidth;
  2139. var currHeight;
  2140. var factor = 2;
  2141. var nativeScrollbarSize = globals.nativeScrollbarSize; //care don't make changes to this object!!!
  2142. var reset = function () {
  2143. /*
  2144. var sizeResetWidth = observerElement[LEXICON.oW] + nativeScrollbarSize.x * factor + nativeScrollbarSize.y * factor + _overlayScrollbarDummySize.x + _overlayScrollbarDummySize.y;
  2145. var sizeResetHeight = observerElement[LEXICON.oH] + nativeScrollbarSize.x * factor + nativeScrollbarSize.y * factor + _overlayScrollbarDummySize.x + _overlayScrollbarDummySize.y;
  2146. var expandChildCSS = {};
  2147. expandChildCSS[_strWidth] = sizeResetWidth;
  2148. expandChildCSS[_strHeight] = sizeResetHeight;
  2149. expandElementChild.css(expandChildCSS);
  2150. expandElement[_strScrollLeft](sizeResetWidth)[_strScrollTop](sizeResetHeight);
  2151. shrinkElement[_strScrollLeft](sizeResetWidth)[_strScrollTop](sizeResetHeight);
  2152. */
  2153. expandElement[_strScrollLeft](constScroll)[_strScrollTop](constScroll);
  2154. shrinkElement[_strScrollLeft](constScroll)[_strScrollTop](constScroll);
  2155. };
  2156. var onResized = function () {
  2157. rAFId = 0;
  2158. if (!isDirty)
  2159. return;
  2160. widthCache = currWidth;
  2161. heightCache = currHeight;
  2162. callback();
  2163. };
  2164. var onScroll = function (event) {
  2165. currWidth = observerElement[LEXICON.oW];
  2166. currHeight = observerElement[LEXICON.oH];
  2167. isDirty = currWidth != widthCache || currHeight != heightCache;
  2168. if (event && isDirty && !rAFId) {
  2169. COMPATIBILITY.cAF()(rAFId);
  2170. rAFId = COMPATIBILITY.rAF()(onResized);
  2171. }
  2172. else if (!event)
  2173. onResized();
  2174. reset();
  2175. if (event) {
  2176. COMPATIBILITY.prvD(event);
  2177. COMPATIBILITY.stpP(event);
  2178. }
  2179. return false;
  2180. };
  2181. var expandChildCSS = {};
  2182. var observerElementCSS = {};
  2183. setTopRightBottomLeft(observerElementCSS, _strEmpty, [
  2184. -((nativeScrollbarSize.y + 1) * factor),
  2185. nativeScrollbarSize.x * -factor,
  2186. nativeScrollbarSize.y * -factor,
  2187. -((nativeScrollbarSize.x + 1) * factor)
  2188. ]);
  2189. FRAMEWORK(observerElement).css(observerElementCSS);
  2190. expandElement.on(_strScroll, onScroll);
  2191. shrinkElement.on(_strScroll, onScroll);
  2192. targetElement.on(strAnimationStartEvent, function () {
  2193. onScroll(false);
  2194. });
  2195. //lets assume that the divs will never be that large and a constant value is enough
  2196. expandChildCSS[_strWidth] = constScroll;
  2197. expandChildCSS[_strHeight] = constScroll;
  2198. expandElementChild.css(expandChildCSS);
  2199. reset();
  2200. }
  2201. else {
  2202. var attachEvent = _documentElementNative.attachEvent;
  2203. var isIE = _msieVersion !== undefined;
  2204. if (attachEvent) {
  2205. targetElement.prepend(generateDiv(_classNameResizeObserverElement));
  2206. findFirst(targetElement, _strDot + _classNameResizeObserverElement)[0].attachEvent('onresize', callback);
  2207. }
  2208. else {
  2209. var obj = _documentElementNative.createElement(TYPES.o);
  2210. obj.setAttribute('tabindex', '-1');
  2211. obj.setAttribute(LEXICON.c, _classNameResizeObserverElement);
  2212. obj.onload = function () {
  2213. var wnd = this.contentDocument.defaultView;
  2214. wnd.addEventListener('resize', callback);
  2215. wnd.document.documentElement.style.display = 'none';
  2216. };
  2217. obj.type = 'text/html';
  2218. if (isIE)
  2219. targetElement.prepend(obj);
  2220. obj.data = 'about:blank';
  2221. if (!isIE)
  2222. targetElement.prepend(obj);
  2223. targetElement.on(strAnimationStartEvent, callback);
  2224. }
  2225. }
  2226. }
  2227. if (targetElement[0] === _sizeObserverElementNative) {
  2228. var directionChanged = function () {
  2229. var dir = _hostElement.css('direction');
  2230. var css = {};
  2231. var scrollLeftValue = 0;
  2232. var result = false;
  2233. if (dir !== _cssDirectionDetectedCache) {
  2234. if (dir === 'ltr') {
  2235. css[_strLeft] = 0;
  2236. css[_strRight] = _strAuto;
  2237. scrollLeftValue = constScroll;
  2238. }
  2239. else {
  2240. css[_strLeft] = _strAuto;
  2241. css[_strRight] = 0;
  2242. scrollLeftValue = _rtlScrollBehavior.n ? -constScroll : _rtlScrollBehavior.i ? 0 : constScroll;
  2243. }
  2244. //execution order is important for IE!!!
  2245. _sizeObserverElement.children().eq(0).css(css);
  2246. _sizeObserverElement[_strScrollLeft](scrollLeftValue)[_strScrollTop](constScroll);
  2247. _cssDirectionDetectedCache = dir;
  2248. result = true;
  2249. }
  2250. return result;
  2251. };
  2252. directionChanged();
  2253. addDestroyEventListener(targetElement, _strScroll, function (event) {
  2254. if (directionChanged())
  2255. update();
  2256. COMPATIBILITY.prvD(event);
  2257. COMPATIBILITY.stpP(event);
  2258. return false;
  2259. });
  2260. }
  2261. }
  2262. //remove resize observer:
  2263. else {
  2264. if (_supportResizeObserver) {
  2265. var element = targetElement.contents()[0];
  2266. var resizeObserverObj = element[_strResizeObserverProperty];
  2267. if (resizeObserverObj) {
  2268. resizeObserverObj.disconnect();
  2269. delete element[_strResizeObserverProperty];
  2270. }
  2271. }
  2272. else {
  2273. remove(targetElement.children(_strDot + _classNameResizeObserverElement).eq(0));
  2274. }
  2275. }
  2276. }
  2277. }
  2278. /**
  2279. * Freezes or unfreezes the given resize observer.
  2280. * @param targetElement The element to which the target resize observer is applied.
  2281. * @param freeze True if the resize observer shall be frozen, false otherwise.
  2282. function freezeResizeObserver(targetElement, freeze) {
  2283. if (targetElement !== undefined) {
  2284. if(freeze) {
  2285. if (_supportResizeObserver) {
  2286. var element = targetElement.contents()[0];
  2287. element[_strResizeObserverProperty].unobserve(element);
  2288. }
  2289. else {
  2290. targetElement = targetElement.children(_strDot + _classNameResizeObserverElement).eq(0);
  2291. var w = targetElement.css(_strWidth);
  2292. var h = targetElement.css(_strHeight);
  2293. var css = {};
  2294. css[_strWidth] = w;
  2295. css[_strHeight] = h;
  2296. targetElement.css(css);
  2297. }
  2298. }
  2299. else {
  2300. if (_supportResizeObserver) {
  2301. var element = targetElement.contents()[0];
  2302. element[_strResizeObserverProperty].observe(element);
  2303. }
  2304. else {
  2305. var css = { };
  2306. css[_strHeight] = _strEmpty;
  2307. css[_strWidth] = _strEmpty;
  2308. targetElement.children(_strDot + _classNameResizeObserverElement).eq(0).css(css);
  2309. }
  2310. }
  2311. }
  2312. }
  2313. */
  2314. //==== Mutation Observers ====//
  2315. /**
  2316. * Creates MutationObservers for the host and content Element if they are supported.
  2317. */
  2318. function createMutationObservers() {
  2319. if (_supportMutationObserver) {
  2320. var mutationObserverContentLag = 11;
  2321. var mutationObserver = COMPATIBILITY.mO();
  2322. var contentLastUpdate = COMPATIBILITY.now();
  2323. var mutationTarget;
  2324. var mutationAttrName;
  2325. var contentTimeout;
  2326. var now;
  2327. var sizeAuto;
  2328. var action;
  2329. _mutationObserverHostCallback = function (mutations) {
  2330. var doUpdate = false;
  2331. var mutation;
  2332. if (_initialized && !_sleeping) {
  2333. each(mutations, function () {
  2334. mutation = this;
  2335. mutationTarget = mutation.target;
  2336. mutationAttrName = mutation.attributeName;
  2337. if (mutationAttrName === LEXICON.c)
  2338. doUpdate = hostClassNamesChanged(mutation.oldValue, mutationTarget.className);
  2339. else if (mutationAttrName === LEXICON.s)
  2340. doUpdate = mutation.oldValue !== mutationTarget[LEXICON.s].cssText;
  2341. else
  2342. doUpdate = true;
  2343. if (doUpdate)
  2344. return false;
  2345. });
  2346. if (doUpdate)
  2347. _base.update(_strAuto);
  2348. }
  2349. return doUpdate;
  2350. };
  2351. _mutationObserverContentCallback = function (mutations) {
  2352. var doUpdate = false;
  2353. var mutation;
  2354. if (_initialized && !_sleeping) {
  2355. each(mutations, function () {
  2356. mutation = this;
  2357. doUpdate = isUnknownMutation(mutation);
  2358. return !doUpdate;
  2359. });
  2360. if (doUpdate) {
  2361. now = COMPATIBILITY.now();
  2362. sizeAuto = (_heightAutoCache || _widthAutoCache);
  2363. action = function () {
  2364. if (!_destroyed) {
  2365. contentLastUpdate = now;
  2366. //if cols, rows or wrap attr was changed
  2367. if (_isTextarea)
  2368. textareaUpdate();
  2369. if (sizeAuto)
  2370. update();
  2371. else
  2372. _base.update(_strAuto);
  2373. }
  2374. };
  2375. clearTimeout(contentTimeout);
  2376. if (mutationObserverContentLag <= 0 || now - contentLastUpdate > mutationObserverContentLag || !sizeAuto)
  2377. action();
  2378. else
  2379. contentTimeout = setTimeout(action, mutationObserverContentLag);
  2380. }
  2381. }
  2382. return doUpdate;
  2383. }
  2384. _mutationObserverHost = new mutationObserver(_mutationObserverHostCallback);
  2385. _mutationObserverContent = new mutationObserver(_mutationObserverContentCallback);
  2386. }
  2387. }
  2388. /**
  2389. * Connects the MutationObservers if they are supported.
  2390. */
  2391. function connectMutationObservers() {
  2392. if (_supportMutationObserver && !_mutationObserversConnected) {
  2393. _mutationObserverHost.observe(_hostElementNative, {
  2394. attributes: true,
  2395. attributeOldValue: true,
  2396. attributeFilter: _mutationObserverAttrsHost
  2397. });
  2398. _mutationObserverContent.observe(_isTextarea ? _targetElementNative : _contentElementNative, {
  2399. attributes: true,
  2400. attributeOldValue: true,
  2401. subtree: !_isTextarea,
  2402. childList: !_isTextarea,
  2403. characterData: !_isTextarea,
  2404. attributeFilter: _isTextarea ? _mutationObserverAttrsTextarea : _mutationObserverAttrsHost
  2405. });
  2406. _mutationObserversConnected = true;
  2407. }
  2408. }
  2409. /**
  2410. * Disconnects the MutationObservers if they are supported.
  2411. */
  2412. function disconnectMutationObservers() {
  2413. if (_supportMutationObserver && _mutationObserversConnected) {
  2414. _mutationObserverHost.disconnect();
  2415. _mutationObserverContent.disconnect();
  2416. _mutationObserversConnected = false;
  2417. }
  2418. }
  2419. //==== Events of elements ====//
  2420. /**
  2421. * This method gets called every time the host element gets resized. IMPORTANT: Padding changes are detected too!!
  2422. * It refreshes the hostResizedEventArgs and the hostSizeResizeCache.
  2423. * If there are any size changes, the update method gets called.
  2424. */
  2425. function hostOnResized() {
  2426. if (!_sleeping) {
  2427. var changed;
  2428. var hostSize = {
  2429. w: _sizeObserverElementNative[LEXICON.sW],
  2430. h: _sizeObserverElementNative[LEXICON.sH]
  2431. };
  2432. changed = checkCache(hostSize, _hostElementSizeChangeDetectedCache);
  2433. _hostElementSizeChangeDetectedCache = hostSize;
  2434. if (changed)
  2435. update({ _hostSizeChanged: true });
  2436. }
  2437. }
  2438. /**
  2439. * The mouse enter event of the host element. This event is only needed for the autoHide feature.
  2440. */
  2441. function hostOnMouseEnter() {
  2442. if (_scrollbarsAutoHideLeave)
  2443. refreshScrollbarsAutoHide(true);
  2444. }
  2445. /**
  2446. * The mouse leave event of the host element. This event is only needed for the autoHide feature.
  2447. */
  2448. function hostOnMouseLeave() {
  2449. if (_scrollbarsAutoHideLeave && !_bodyElement.hasClass(_classNameDragging))
  2450. refreshScrollbarsAutoHide(false);
  2451. }
  2452. /**
  2453. * The mouse move event of the host element. This event is only needed for the autoHide "move" feature.
  2454. */
  2455. function hostOnMouseMove() {
  2456. if (_scrollbarsAutoHideMove) {
  2457. refreshScrollbarsAutoHide(true);
  2458. clearTimeout(_scrollbarsAutoHideMoveTimeoutId);
  2459. _scrollbarsAutoHideMoveTimeoutId = setTimeout(function () {
  2460. if (_scrollbarsAutoHideMove && !_destroyed)
  2461. refreshScrollbarsAutoHide(false);
  2462. }, 100);
  2463. }
  2464. }
  2465. /**
  2466. * Prevents text from deselection if attached to the document element on the mousedown event of a DOM element.
  2467. * @param event The select start event.
  2468. */
  2469. function documentOnSelectStart(event) {
  2470. COMPATIBILITY.prvD(event);
  2471. return false;
  2472. }
  2473. /**
  2474. * A callback which will be called after a img element has downloaded its src asynchronous.
  2475. */
  2476. function imgOnLoad() {
  2477. update({ _contentSizeChanged: true });
  2478. }
  2479. /**
  2480. * Adds or removes mouse & touch events of the host element. (for handling auto-hiding of the scrollbars)
  2481. * @param destroy Indicates whether the events shall be added or removed.
  2482. */
  2483. function setupHostMouseTouchEvents(destroy) {
  2484. setupResponsiveEventListener(_hostElement,
  2485. _strMouseTouchMoveEvent,
  2486. hostOnMouseMove,
  2487. (_scrollbarsAutoHideMove ? destroy : true), true);
  2488. setupResponsiveEventListener(_hostElement,
  2489. [_strMouseTouchEnter, _strMouseTouchLeave],
  2490. [hostOnMouseEnter, hostOnMouseLeave],
  2491. (_scrollbarsAutoHideMove ? true : destroy), true);
  2492. //if the plugin is initialized and the mouse is over the host element, make the scrollbars visible
  2493. if (!_initialized && !destroy)
  2494. _hostElement.one('mouseover', hostOnMouseEnter);
  2495. }
  2496. //==== Update Detection ====//
  2497. /**
  2498. * Measures the min width and min height of the body element and refreshes the related cache.
  2499. * @returns {boolean} True if the min width or min height has changed, false otherwise.
  2500. */
  2501. function bodyMinSizeChanged() {
  2502. var bodyMinSize = {};
  2503. if (_isBody && _contentArrangeElement) {
  2504. bodyMinSize.w = parseToZeroOrNumber(_contentArrangeElement.css(_strMinMinus + _strWidth));
  2505. bodyMinSize.h = parseToZeroOrNumber(_contentArrangeElement.css(_strMinMinus + _strHeight));
  2506. bodyMinSize.c = checkCache(bodyMinSize, _bodyMinSizeCache);
  2507. bodyMinSize.f = true; //flag for "measured at least once"
  2508. }
  2509. _bodyMinSizeCache = bodyMinSize;
  2510. return !!bodyMinSize.c;
  2511. }
  2512. /**
  2513. * Returns true if the class names really changed (new class without plugin host prefix)
  2514. * @param oldCassNames The old ClassName string.
  2515. * @param newClassNames The new ClassName string.
  2516. * @returns {boolean} True if the class names has really changed, false otherwise.
  2517. */
  2518. function hostClassNamesChanged(oldCassNames, newClassNames) {
  2519. var currClasses = (newClassNames !== undefined && newClassNames !== null) ? newClassNames.split(_strSpace) : _strEmpty;
  2520. var oldClasses = (oldCassNames !== undefined && oldCassNames !== null) ? oldCassNames.split(_strSpace) : _strEmpty;
  2521. if (currClasses === _strEmpty && oldClasses === _strEmpty)
  2522. return false;
  2523. var diff = getArrayDifferences(oldClasses, currClasses);
  2524. var changed = false;
  2525. var oldClassNames = _oldClassName !== undefined && _oldClassName !== null ? _oldClassName.split(_strSpace) : [_strEmpty];
  2526. var currClassNames = _classNameCache !== undefined && _classNameCache !== null ? _classNameCache.split(_strSpace) : [_strEmpty];
  2527. //remove none theme from diff list to prevent update
  2528. var idx = inArray(_classNameThemeNone, diff);
  2529. var curr;
  2530. var i;
  2531. var v;
  2532. var o;
  2533. var c;
  2534. if (idx > -1)
  2535. diff.splice(idx, 1);
  2536. for (i = 0; i < diff.length; i++) {
  2537. curr = diff[i];
  2538. if (curr.indexOf(_classNameHostElement) !== 0) {
  2539. o = true;
  2540. c = true;
  2541. for (v = 0; v < oldClassNames.length; v++) {
  2542. if (curr === oldClassNames[v]) {
  2543. o = false;
  2544. break;
  2545. }
  2546. }
  2547. for (v = 0; v < currClassNames.length; v++) {
  2548. if (curr === currClassNames[v]) {
  2549. c = false;
  2550. break;
  2551. }
  2552. }
  2553. if (o && c) {
  2554. changed = true;
  2555. break;
  2556. }
  2557. }
  2558. }
  2559. return changed;
  2560. }
  2561. /**
  2562. * Returns true if the given mutation is not from a from the plugin generated element. If the target element is a textarea the mutation is always unknown.
  2563. * @param mutation The mutation which shall be checked.
  2564. * @returns {boolean} True if the mutation is from a unknown element, false otherwise.
  2565. */
  2566. function isUnknownMutation(mutation) {
  2567. var attributeName = mutation.attributeName;
  2568. var mutationTarget = mutation.target;
  2569. var mutationType = mutation.type;
  2570. var strClosest = 'closest';
  2571. if (mutationTarget === _contentElementNative)
  2572. return attributeName === null;
  2573. if (mutationType === 'attributes' && (attributeName === LEXICON.c || attributeName === LEXICON.s) && !_isTextarea) {
  2574. //ignore className changes by the plugin
  2575. if (attributeName === LEXICON.c && FRAMEWORK(mutationTarget).hasClass(_classNameHostElement))
  2576. return hostClassNamesChanged(mutation.oldValue, mutationTarget.getAttribute(LEXICON.c));
  2577. //only do it of browser support it natively
  2578. if (typeof mutationTarget[strClosest] != TYPES.f)
  2579. return true;
  2580. if (mutationTarget[strClosest](_strDot + _classNameResizeObserverElement) !== null ||
  2581. mutationTarget[strClosest](_strDot + _classNameScrollbar) !== null ||
  2582. mutationTarget[strClosest](_strDot + _classNameScrollbarCorner) !== null)
  2583. return false;
  2584. }
  2585. return true;
  2586. }
  2587. /**
  2588. * Returns true if the content size was changed since the last time this method was called.
  2589. * @returns {boolean} True if the content size was changed, false otherwise.
  2590. */
  2591. function updateAutoContentSizeChanged() {
  2592. if (_sleeping)
  2593. return false;
  2594. var contentMeasureElement = getContentMeasureElement();
  2595. var textareaValueLength = _isTextarea && _widthAutoCache && !_textareaAutoWrappingCache ? _targetElement.val().length : 0;
  2596. var setCSS = !_mutationObserversConnected && _widthAutoCache && !_isTextarea;
  2597. var viewportScrollSize = {};
  2598. var css = {};
  2599. var float;
  2600. var bodyMinSizeC;
  2601. var changed;
  2602. var viewportScrollSizeChanged;
  2603. var contentElementScrollSize;
  2604. //fix for https://bugzilla.mozilla.org/show_bug.cgi?id=1439305, it only works with "clipAlways : true"
  2605. //it can work with "clipAlways : false" too, but I had to set the overflow of the viewportElement to hidden every time before measuring
  2606. if (_restrictedMeasuring) {
  2607. viewportScrollSize = {
  2608. x: _viewportElementNative[LEXICON.sW],
  2609. y: _viewportElementNative[LEXICON.sH]
  2610. }
  2611. }
  2612. if (setCSS) {
  2613. float = _contentElement.css(_strFloat);
  2614. css[_strFloat] = _isRTL ? _strRight : _strLeft;
  2615. css[_strWidth] = _strAuto;
  2616. _contentElement.css(css);
  2617. }
  2618. contentElementScrollSize = {
  2619. w: contentMeasureElement[LEXICON.sW] + textareaValueLength,
  2620. h: contentMeasureElement[LEXICON.sH] + textareaValueLength
  2621. };
  2622. if (setCSS) {
  2623. css[_strFloat] = float;
  2624. css[_strWidth] = _strHundredPercent;
  2625. _contentElement.css(css);
  2626. }
  2627. bodyMinSizeC = bodyMinSizeChanged();
  2628. changed = checkCache(contentElementScrollSize, _contentElementScrollSizeChangeDetectedCache);
  2629. viewportScrollSizeChanged = checkCache(viewportScrollSize, _viewportScrollSizeCache);
  2630. _contentElementScrollSizeChangeDetectedCache = contentElementScrollSize;
  2631. _viewportScrollSizeCache = viewportScrollSize;
  2632. return changed || bodyMinSizeC || viewportScrollSizeChanged;
  2633. }
  2634. /**
  2635. * Returns true when a attribute which the MutationObserver would observe has changed.
  2636. * @returns {boolean} True if one of the attributes which a MutationObserver would observe has changed, false or undefined otherwise.
  2637. */
  2638. function meaningfulAttrsChanged() {
  2639. if (_sleeping || _mutationObserversConnected)
  2640. return;
  2641. var changed;
  2642. var elem;
  2643. var curr;
  2644. var cache;
  2645. var checks = [
  2646. {
  2647. _elem: _hostElement,
  2648. _props: _mutationObserverAttrsHost.concat(':visible')
  2649. },
  2650. {
  2651. _elem: _isTextarea ? _targetElement : undefined,
  2652. _props: _mutationObserverAttrsTextarea
  2653. }
  2654. ];
  2655. each(checks, function (index, check) {
  2656. elem = check._elem;
  2657. if (elem) {
  2658. each(check._props, function (index, prop) {
  2659. curr = prop.charAt(0) === ':' ? elem.is(prop) : elem.attr(prop);
  2660. cache = _updateAutoCache[prop];
  2661. changed = changed || checkCache(curr, cache);
  2662. _updateAutoCache[prop] = curr;
  2663. });
  2664. }
  2665. });
  2666. return changed;
  2667. }
  2668. /**
  2669. * Checks is a CSS Property of a child element is affecting the scroll size of the content.
  2670. * @param propertyName The CSS property name.
  2671. * @returns {boolean} True if the property is affecting the content scroll size, false otherwise.
  2672. */
  2673. function isSizeAffectingCSSProperty(propertyName) {
  2674. if (!_initialized)
  2675. return true;
  2676. var flexGrow = 'flex-grow';
  2677. var flexShrink = 'flex-shrink';
  2678. var flexBasis = 'flex-basis';
  2679. var affectingPropsX = [
  2680. _strWidth,
  2681. _strMinMinus + _strWidth,
  2682. _strMaxMinus + _strWidth,
  2683. _strMarginMinus + _strLeft,
  2684. _strMarginMinus + _strRight,
  2685. _strLeft,
  2686. _strRight,
  2687. 'font-weight',
  2688. 'word-spacing',
  2689. flexGrow,
  2690. flexShrink,
  2691. flexBasis
  2692. ];
  2693. var affectingPropsXContentBox = [
  2694. _strPaddingMinus + _strLeft,
  2695. _strPaddingMinus + _strRight,
  2696. _strBorderMinus + _strLeft + _strWidth,
  2697. _strBorderMinus + _strRight + _strWidth
  2698. ];
  2699. var affectingPropsY = [
  2700. _strHeight,
  2701. _strMinMinus + _strHeight,
  2702. _strMaxMinus + _strHeight,
  2703. _strMarginMinus + _strTop,
  2704. _strMarginMinus + _strBottom,
  2705. _strTop,
  2706. _strBottom,
  2707. 'line-height',
  2708. flexGrow,
  2709. flexShrink,
  2710. flexBasis
  2711. ];
  2712. var affectingPropsYContentBox = [
  2713. _strPaddingMinus + _strTop,
  2714. _strPaddingMinus + _strBottom,
  2715. _strBorderMinus + _strTop + _strWidth,
  2716. _strBorderMinus + _strBottom + _strWidth
  2717. ];
  2718. var _strS = 's';
  2719. var _strVS = 'v-s';
  2720. var checkX = _overflowBehaviorCache.x === _strS || _overflowBehaviorCache.x === _strVS;
  2721. var checkY = _overflowBehaviorCache.y === _strS || _overflowBehaviorCache.y === _strVS;
  2722. var sizeIsAffected = false;
  2723. var checkPropertyName = function (arr, name) {
  2724. for (var i = 0; i < arr[LEXICON.l]; i++) {
  2725. if (arr[i] === name)
  2726. return true;
  2727. }
  2728. return false;
  2729. };
  2730. if (checkY) {
  2731. sizeIsAffected = checkPropertyName(affectingPropsY, propertyName);
  2732. if (!sizeIsAffected && !_isBorderBox)
  2733. sizeIsAffected = checkPropertyName(affectingPropsYContentBox, propertyName);
  2734. }
  2735. if (checkX && !sizeIsAffected) {
  2736. sizeIsAffected = checkPropertyName(affectingPropsX, propertyName);
  2737. if (!sizeIsAffected && !_isBorderBox)
  2738. sizeIsAffected = checkPropertyName(affectingPropsXContentBox, propertyName);
  2739. }
  2740. return sizeIsAffected;
  2741. }
  2742. //==== Update ====//
  2743. /**
  2744. * Updates the variables and size of the textarea element, and manages the scroll on new line or new character.
  2745. */
  2746. function textareaUpdate() {
  2747. if (!_sleeping) {
  2748. var wrapAttrOff = !_textareaAutoWrappingCache;
  2749. var minWidth = _viewportSize.w;
  2750. var minHeight = _viewportSize.h;
  2751. var css = {};
  2752. var doMeasure = _widthAutoCache || wrapAttrOff;
  2753. var origWidth;
  2754. var width;
  2755. var origHeight;
  2756. var height;
  2757. //reset min size
  2758. css[_strMinMinus + _strWidth] = _strEmpty;
  2759. css[_strMinMinus + _strHeight] = _strEmpty;
  2760. //set width auto
  2761. css[_strWidth] = _strAuto;
  2762. _targetElement.css(css);
  2763. //measure width
  2764. origWidth = _targetElementNative[LEXICON.oW];
  2765. width = doMeasure ? MATH.max(origWidth, _targetElementNative[LEXICON.sW] - 1) : 1;
  2766. /*width += (_widthAutoCache ? _marginX + (!_isBorderBox ? wrapAttrOff ? 0 : _paddingX + _borderX : 0) : 0);*/
  2767. //set measured width
  2768. css[_strWidth] = _widthAutoCache ? _strAuto /*width*/ : _strHundredPercent;
  2769. css[_strMinMinus + _strWidth] = _strHundredPercent;
  2770. //set height auto
  2771. css[_strHeight] = _strAuto;
  2772. _targetElement.css(css);
  2773. //measure height
  2774. origHeight = _targetElementNative[LEXICON.oH];
  2775. height = MATH.max(origHeight, _targetElementNative[LEXICON.sH] - 1);
  2776. //append correct size values
  2777. css[_strWidth] = width;
  2778. css[_strHeight] = height;
  2779. _textareaCoverElement.css(css);
  2780. //apply min width / min height to prevent textarea collapsing
  2781. css[_strMinMinus + _strWidth] = minWidth /*+ (!_isBorderBox && _widthAutoCache ? _paddingX + _borderX : 0)*/;
  2782. css[_strMinMinus + _strHeight] = minHeight /*+ (!_isBorderBox && _heightAutoCache ? _paddingY + _borderY : 0)*/;
  2783. _targetElement.css(css);
  2784. return {
  2785. _originalWidth: origWidth,
  2786. _originalHeight: origHeight,
  2787. _dynamicWidth: width,
  2788. _dynamicHeight: height
  2789. };
  2790. }
  2791. }
  2792. /**
  2793. * Updates the plugin and DOM to the current options.
  2794. * This method should only be called if a update is 100% required.
  2795. * @param updateHints A objects which contains hints for this update:
  2796. * {
  2797. * _hostSizeChanged : boolean,
  2798. * _contentSizeChanged : boolean,
  2799. * _force : boolean, == preventSwallowing
  2800. * _changedOptions : { }, == preventSwallowing && preventSleep
  2801. * }
  2802. */
  2803. function update(updateHints) {
  2804. clearTimeout(_swallowedUpdateTimeout);
  2805. updateHints = updateHints || {};
  2806. _swallowedUpdateHints._hostSizeChanged |= updateHints._hostSizeChanged;
  2807. _swallowedUpdateHints._contentSizeChanged |= updateHints._contentSizeChanged;
  2808. _swallowedUpdateHints._force |= updateHints._force;
  2809. var now = COMPATIBILITY.now();
  2810. var hostSizeChanged = !!_swallowedUpdateHints._hostSizeChanged;
  2811. var contentSizeChanged = !!_swallowedUpdateHints._contentSizeChanged;
  2812. var force = !!_swallowedUpdateHints._force;
  2813. var changedOptions = updateHints._changedOptions;
  2814. var swallow = _swallowUpdateLag > 0 && _initialized && !_destroyed && !force && !changedOptions && (now - _lastUpdateTime) < _swallowUpdateLag && (!_heightAutoCache && !_widthAutoCache);
  2815. var displayIsHidden;
  2816. if (swallow)
  2817. _swallowedUpdateTimeout = setTimeout(update, _swallowUpdateLag);
  2818. //abort update due to:
  2819. //destroyed
  2820. //swallowing
  2821. //sleeping
  2822. //host is hidden or has false display
  2823. if (_destroyed || swallow || (_sleeping && !changedOptions) || (_initialized && !force && (displayIsHidden = _hostElement.is(':hidden'))) || _hostElement.css('display') === 'inline')
  2824. return;
  2825. _lastUpdateTime = now;
  2826. _swallowedUpdateHints = {};
  2827. //if scrollbar styling is possible and native scrollbars aren't overlaid the scrollbar styling will be applied which hides the native scrollbars completely.
  2828. if (_nativeScrollbarStyling && !(_nativeScrollbarIsOverlaid.x && _nativeScrollbarIsOverlaid.y)) {
  2829. //native scrollbars are hidden, so change the values to zero
  2830. _nativeScrollbarSize.x = 0;
  2831. _nativeScrollbarSize.y = 0;
  2832. }
  2833. else {
  2834. //refresh native scrollbar size (in case of zoom)
  2835. _nativeScrollbarSize = extendDeep({}, globals.nativeScrollbarSize);
  2836. }
  2837. // Scrollbar padding is needed for firefox, because firefox hides scrollbar automatically if the size of the div is too small.
  2838. // The calculation: [scrollbar size +3 *3]
  2839. // (+3 because of possible decoration e.g. borders, margins etc., but only if native scrollbar is NOT a overlaid scrollbar)
  2840. // (*3 because (1)increase / (2)decrease -button and (3)resize handle)
  2841. _nativeScrollbarMinSize = {
  2842. x: (_nativeScrollbarSize.x + (_nativeScrollbarIsOverlaid.x ? 0 : 3)) * 3,
  2843. y: (_nativeScrollbarSize.y + (_nativeScrollbarIsOverlaid.y ? 0 : 3)) * 3
  2844. };
  2845. //changedOptions = changedOptions || { };
  2846. //freezeResizeObserver(_sizeObserverElement, true);
  2847. //freezeResizeObserver(_sizeAutoObserverElement, true);
  2848. var checkCacheAutoForce = function () {
  2849. return checkCache.apply(this, [].slice.call(arguments).concat([force]));
  2850. };
  2851. //save current scroll offset
  2852. var currScroll = {
  2853. x: _viewportElement[_strScrollLeft](),
  2854. y: _viewportElement[_strScrollTop]()
  2855. };
  2856. var currentPreparedOptionsScrollbars = _currentPreparedOptions.scrollbars;
  2857. var currentPreparedOptionsTextarea = _currentPreparedOptions.textarea;
  2858. //scrollbars visibility:
  2859. var scrollbarsVisibility = currentPreparedOptionsScrollbars.visibility;
  2860. var scrollbarsVisibilityChanged = checkCacheAutoForce(scrollbarsVisibility, _scrollbarsVisibilityCache);
  2861. //scrollbars autoHide:
  2862. var scrollbarsAutoHide = currentPreparedOptionsScrollbars.autoHide;
  2863. var scrollbarsAutoHideChanged = checkCacheAutoForce(scrollbarsAutoHide, _scrollbarsAutoHideCache);
  2864. //scrollbars click scrolling
  2865. var scrollbarsClickScrolling = currentPreparedOptionsScrollbars.clickScrolling;
  2866. var scrollbarsClickScrollingChanged = checkCacheAutoForce(scrollbarsClickScrolling, _scrollbarsClickScrollingCache);
  2867. //scrollbars drag scrolling
  2868. var scrollbarsDragScrolling = currentPreparedOptionsScrollbars.dragScrolling;
  2869. var scrollbarsDragScrollingChanged = checkCacheAutoForce(scrollbarsDragScrolling, _scrollbarsDragScrollingCache);
  2870. //className
  2871. var className = _currentPreparedOptions.className;
  2872. var classNameChanged = checkCacheAutoForce(className, _classNameCache);
  2873. //resize
  2874. var resize = _currentPreparedOptions.resize;
  2875. var resizeChanged = checkCacheAutoForce(resize, _resizeCache) && !_isBody; //body can't be resized since the window itself acts as resize possibility.
  2876. //paddingAbsolute
  2877. var paddingAbsolute = _currentPreparedOptions.paddingAbsolute;
  2878. var paddingAbsoluteChanged = checkCacheAutoForce(paddingAbsolute, _paddingAbsoluteCache);
  2879. //clipAlways
  2880. var clipAlways = _currentPreparedOptions.clipAlways;
  2881. var clipAlwaysChanged = checkCacheAutoForce(clipAlways, _clipAlwaysCache);
  2882. //sizeAutoCapable
  2883. var sizeAutoCapable = _currentPreparedOptions.sizeAutoCapable && !_isBody; //body can never be size auto, because it shall be always as big as the viewport.
  2884. var sizeAutoCapableChanged = checkCacheAutoForce(sizeAutoCapable, _sizeAutoCapableCache);
  2885. //showNativeScrollbars
  2886. var ignoreOverlayScrollbarHiding = _currentPreparedOptions.nativeScrollbarsOverlaid.showNativeScrollbars;
  2887. var ignoreOverlayScrollbarHidingChanged = checkCacheAutoForce(ignoreOverlayScrollbarHiding, _ignoreOverlayScrollbarHidingCache);
  2888. //autoUpdate
  2889. var autoUpdate = _currentPreparedOptions.autoUpdate;
  2890. var autoUpdateChanged = checkCacheAutoForce(autoUpdate, _autoUpdateCache);
  2891. //overflowBehavior
  2892. var overflowBehavior = _currentPreparedOptions.overflowBehavior;
  2893. var overflowBehaviorChanged = checkCacheAutoForce(overflowBehavior, _overflowBehaviorCache, force);
  2894. //dynWidth:
  2895. var textareaDynWidth = currentPreparedOptionsTextarea.dynWidth;
  2896. var textareaDynWidthChanged = checkCacheAutoForce(_textareaDynWidthCache, textareaDynWidth);
  2897. //dynHeight:
  2898. var textareaDynHeight = currentPreparedOptionsTextarea.dynHeight;
  2899. var textareaDynHeightChanged = checkCacheAutoForce(_textareaDynHeightCache, textareaDynHeight);
  2900. //scrollbars visibility
  2901. _scrollbarsAutoHideNever = scrollbarsAutoHide === 'n';
  2902. _scrollbarsAutoHideScroll = scrollbarsAutoHide === 's';
  2903. _scrollbarsAutoHideMove = scrollbarsAutoHide === 'm';
  2904. _scrollbarsAutoHideLeave = scrollbarsAutoHide === 'l';
  2905. //scrollbars autoHideDelay
  2906. _scrollbarsAutoHideDelay = currentPreparedOptionsScrollbars.autoHideDelay;
  2907. //old className
  2908. _oldClassName = _classNameCache;
  2909. //resize
  2910. _resizeNone = resize === 'n';
  2911. _resizeBoth = resize === 'b';
  2912. _resizeHorizontal = resize === 'h';
  2913. _resizeVertical = resize === 'v';
  2914. //normalizeRTL
  2915. _normalizeRTLCache = _currentPreparedOptions.normalizeRTL;
  2916. //ignore overlay scrollbar hiding
  2917. ignoreOverlayScrollbarHiding = ignoreOverlayScrollbarHiding && (_nativeScrollbarIsOverlaid.x && _nativeScrollbarIsOverlaid.y);
  2918. //refresh options cache
  2919. _scrollbarsVisibilityCache = scrollbarsVisibility;
  2920. _scrollbarsAutoHideCache = scrollbarsAutoHide;
  2921. _scrollbarsClickScrollingCache = scrollbarsClickScrolling;
  2922. _scrollbarsDragScrollingCache = scrollbarsDragScrolling;
  2923. _classNameCache = className;
  2924. _resizeCache = resize;
  2925. _paddingAbsoluteCache = paddingAbsolute;
  2926. _clipAlwaysCache = clipAlways;
  2927. _sizeAutoCapableCache = sizeAutoCapable;
  2928. _ignoreOverlayScrollbarHidingCache = ignoreOverlayScrollbarHiding;
  2929. _autoUpdateCache = autoUpdate;
  2930. _overflowBehaviorCache = extendDeep({}, overflowBehavior);
  2931. _textareaDynWidthCache = textareaDynWidth;
  2932. _textareaDynHeightCache = textareaDynHeight;
  2933. _hasOverflowCache = _hasOverflowCache || { x: false, y: false };
  2934. //set correct class name to the host element
  2935. if (classNameChanged) {
  2936. removeClass(_hostElement, _oldClassName + _strSpace + _classNameThemeNone);
  2937. addClass(_hostElement, className !== undefined && className !== null && className.length > 0 ? className : _classNameThemeNone);
  2938. }
  2939. //set correct auto Update
  2940. if (autoUpdateChanged) {
  2941. if (autoUpdate === true) {
  2942. disconnectMutationObservers();
  2943. autoUpdateLoop.add(_base);
  2944. }
  2945. else if (autoUpdate === null) {
  2946. if (_autoUpdateRecommended) {
  2947. disconnectMutationObservers();
  2948. autoUpdateLoop.add(_base);
  2949. }
  2950. else {
  2951. autoUpdateLoop.remove(_base);
  2952. connectMutationObservers();
  2953. }
  2954. }
  2955. else {
  2956. autoUpdateLoop.remove(_base);
  2957. connectMutationObservers();
  2958. }
  2959. }
  2960. //activate or deactivate size auto capability
  2961. if (sizeAutoCapableChanged) {
  2962. if (sizeAutoCapable) {
  2963. if (!_contentGlueElement) {
  2964. _contentGlueElement = FRAMEWORK(generateDiv(_classNameContentGlueElement));
  2965. _paddingElement.before(_contentGlueElement);
  2966. }
  2967. else {
  2968. _contentGlueElement.show();
  2969. }
  2970. if (_sizeAutoObserverAdded) {
  2971. _sizeAutoObserverElement.show();
  2972. }
  2973. else {
  2974. _sizeAutoObserverElement = FRAMEWORK(generateDiv(_classNameSizeAutoObserverElement));
  2975. _sizeAutoObserverElementNative = _sizeAutoObserverElement[0];
  2976. _contentGlueElement.before(_sizeAutoObserverElement);
  2977. var oldSize = { w: -1, h: -1 };
  2978. setupResizeObserver(_sizeAutoObserverElement, function () {
  2979. var newSize = {
  2980. w: _sizeAutoObserverElementNative[LEXICON.oW],
  2981. h: _sizeAutoObserverElementNative[LEXICON.oH]
  2982. };
  2983. if (checkCache(newSize, oldSize)) {
  2984. if (_initialized && (_heightAutoCache && newSize.h > 0) || (_widthAutoCache && newSize.w > 0)) {
  2985. update();
  2986. }
  2987. else if (_initialized && (!_heightAutoCache && newSize.h === 0) || (!_widthAutoCache && newSize.w === 0)) {
  2988. update();
  2989. }
  2990. }
  2991. oldSize = newSize;
  2992. });
  2993. _sizeAutoObserverAdded = true;
  2994. //fix heightAuto detector bug if height is fixed but contentHeight is 0.
  2995. //the probability this bug will ever happen is very very low, thats why its ok if we use calc which isn't supported in IE8.
  2996. if (_cssCalc !== null)
  2997. _sizeAutoObserverElement.css(_strHeight, _cssCalc + '(100% + 1px)');
  2998. }
  2999. }
  3000. else {
  3001. if (_sizeAutoObserverAdded)
  3002. _sizeAutoObserverElement.hide();
  3003. if (_contentGlueElement)
  3004. _contentGlueElement.hide();
  3005. }
  3006. }
  3007. //if force, update all resizeObservers too
  3008. if (force) {
  3009. _sizeObserverElement.find('*').trigger(_strScroll);
  3010. if (_sizeAutoObserverAdded)
  3011. _sizeAutoObserverElement.find('*').trigger(_strScroll);
  3012. }
  3013. //display hidden:
  3014. displayIsHidden = displayIsHidden === undefined ? _hostElement.is(':hidden') : displayIsHidden;
  3015. var displayIsHiddenChanged = checkCacheAutoForce(displayIsHidden, _displayIsHiddenCache);
  3016. //textarea AutoWrapping:
  3017. var textareaAutoWrapping = _isTextarea ? _targetElement.attr('wrap') !== 'off' : false;
  3018. var textareaAutoWrappingChanged = checkCacheAutoForce(textareaAutoWrapping, _textareaAutoWrappingCache);
  3019. //detect direction:
  3020. var cssDirection = _hostElement.css('direction');
  3021. var cssDirectionChanged = checkCacheAutoForce(cssDirection, _cssDirectionCache);
  3022. //detect box-sizing:
  3023. var boxSizing = _hostElement.css('box-sizing');
  3024. var boxSizingChanged = checkCacheAutoForce(boxSizing, _cssBoxSizingCache);
  3025. //detect padding:
  3026. var padding = {
  3027. c: force,
  3028. t: parseToZeroOrNumber(_hostElement.css(_strPaddingMinus + _strTop)),
  3029. r: parseToZeroOrNumber(_hostElement.css(_strPaddingMinus + _strRight)),
  3030. b: parseToZeroOrNumber(_hostElement.css(_strPaddingMinus + _strBottom)),
  3031. l: parseToZeroOrNumber(_hostElement.css(_strPaddingMinus + _strLeft))
  3032. };
  3033. //width + height auto detecting var:
  3034. var sizeAutoObserverElementBCRect;
  3035. //exception occurs in IE8 sometimes (unknown exception)
  3036. try {
  3037. sizeAutoObserverElementBCRect = _sizeAutoObserverAdded ? _sizeAutoObserverElementNative[LEXICON.bCR]() : null;
  3038. } catch (ex) {
  3039. return;
  3040. }
  3041. _isRTL = cssDirection === 'rtl';
  3042. _isBorderBox = (boxSizing === 'border-box');
  3043. var isRTLLeft = _isRTL ? _strLeft : _strRight;
  3044. var isRTLRight = _isRTL ? _strRight : _strLeft;
  3045. //detect width auto:
  3046. var widthAutoResizeDetection = false;
  3047. var widthAutoObserverDetection = (_sizeAutoObserverAdded && (_hostElement.css(_strFloat) !== 'none' /*|| _isTextarea */)) ? (MATH.round(sizeAutoObserverElementBCRect.right - sizeAutoObserverElementBCRect.left) === 0) && (!paddingAbsolute ? (_hostElementNative[LEXICON.cW] - _paddingX) > 0 : true) : false;
  3048. if (sizeAutoCapable && !widthAutoObserverDetection) {
  3049. var tmpCurrHostWidth = _hostElementNative[LEXICON.oW];
  3050. var tmpCurrContentGlueWidth = _contentGlueElement.css(_strWidth);
  3051. _contentGlueElement.css(_strWidth, _strAuto);
  3052. var tmpNewHostWidth = _hostElementNative[LEXICON.oW];
  3053. _contentGlueElement.css(_strWidth, tmpCurrContentGlueWidth);
  3054. widthAutoResizeDetection = tmpCurrHostWidth !== tmpNewHostWidth;
  3055. if (!widthAutoResizeDetection) {
  3056. _contentGlueElement.css(_strWidth, tmpCurrHostWidth + 1);
  3057. tmpNewHostWidth = _hostElementNative[LEXICON.oW];
  3058. _contentGlueElement.css(_strWidth, tmpCurrContentGlueWidth);
  3059. widthAutoResizeDetection = tmpCurrHostWidth !== tmpNewHostWidth;
  3060. }
  3061. }
  3062. var widthAuto = (widthAutoObserverDetection || widthAutoResizeDetection) && sizeAutoCapable && !displayIsHidden;
  3063. var widthAutoChanged = checkCacheAutoForce(widthAuto, _widthAutoCache);
  3064. var wasWidthAuto = !widthAuto && _widthAutoCache;
  3065. //detect height auto:
  3066. var heightAuto = _sizeAutoObserverAdded && sizeAutoCapable && !displayIsHidden ? (MATH.round(sizeAutoObserverElementBCRect.bottom - sizeAutoObserverElementBCRect.top) === 0) /* && (!paddingAbsolute && (_msieVersion > 9 || !_msieVersion) ? true : true) */ : false;
  3067. var heightAutoChanged = checkCacheAutoForce(heightAuto, _heightAutoCache);
  3068. var wasHeightAuto = !heightAuto && _heightAutoCache;
  3069. //detect border:
  3070. //we need the border only if border box and auto size
  3071. var strMinusWidth = '-' + _strWidth;
  3072. var updateBorderX = (widthAuto && _isBorderBox) || !_isBorderBox;
  3073. var updateBorderY = (heightAuto && _isBorderBox) || !_isBorderBox;
  3074. var border = {
  3075. c: force,
  3076. t: updateBorderY ? parseToZeroOrNumber(_hostElement.css(_strBorderMinus + _strTop + strMinusWidth), true) : 0,
  3077. r: updateBorderX ? parseToZeroOrNumber(_hostElement.css(_strBorderMinus + _strRight + strMinusWidth), true) : 0,
  3078. b: updateBorderY ? parseToZeroOrNumber(_hostElement.css(_strBorderMinus + _strBottom + strMinusWidth), true) : 0,
  3079. l: updateBorderX ? parseToZeroOrNumber(_hostElement.css(_strBorderMinus + _strLeft + strMinusWidth), true) : 0
  3080. };
  3081. //detect margin:
  3082. var margin = {
  3083. c: force,
  3084. t: parseToZeroOrNumber(_hostElement.css(_strMarginMinus + _strTop)),
  3085. r: parseToZeroOrNumber(_hostElement.css(_strMarginMinus + _strRight)),
  3086. b: parseToZeroOrNumber(_hostElement.css(_strMarginMinus + _strBottom)),
  3087. l: parseToZeroOrNumber(_hostElement.css(_strMarginMinus + _strLeft))
  3088. };
  3089. //detect css max width & height:
  3090. var cssMaxValue = {
  3091. h: String(_hostElement.css(_strMaxMinus + _strHeight)),
  3092. w: String(_hostElement.css(_strMaxMinus + _strWidth))
  3093. };
  3094. //vars to apply correct css
  3095. var contentElementCSS = {};
  3096. var contentGlueElementCSS = {};
  3097. //funcs
  3098. var getHostSize = function () {
  3099. //has to be clientSize because offsetSize respect borders
  3100. return {
  3101. w: _hostElementNative[LEXICON.cW],
  3102. h: _hostElementNative[LEXICON.cH]
  3103. };
  3104. };
  3105. var getViewportSize = function () {
  3106. //viewport size is padding container because it never has padding, margin and a border
  3107. //determine zoom rounding error -> sometimes scrollWidth/Height is smaller than clientWidth/Height
  3108. //if this happens add the difference to the viewportSize to compensate the rounding error
  3109. return {
  3110. w: _paddingElementNative[LEXICON.oW] + MATH.max(0, _contentElementNative[LEXICON.cW] - _contentElementNative[LEXICON.sW]),
  3111. h: _paddingElementNative[LEXICON.oH] + MATH.max(0, _contentElementNative[LEXICON.cH] - _contentElementNative[LEXICON.sH])
  3112. };
  3113. };
  3114. //set info for padding
  3115. var paddingAbsoluteX = _paddingX = padding.l + padding.r;
  3116. var paddingAbsoluteY = _paddingY = padding.t + padding.b;
  3117. paddingAbsoluteX *= paddingAbsolute ? 1 : 0;
  3118. paddingAbsoluteY *= paddingAbsolute ? 1 : 0;
  3119. padding.c = checkCacheAutoForce(padding, _cssPaddingCache);
  3120. //set info for border
  3121. _borderX = border.l + border.r;
  3122. _borderY = border.t + border.b;
  3123. border.c = checkCacheAutoForce(border, _cssBorderCache);
  3124. //set info for margin
  3125. _marginX = margin.l + margin.r;
  3126. _marginY = margin.t + margin.b;
  3127. margin.c = checkCacheAutoForce(margin, _cssMarginCache);
  3128. //set info for css max value
  3129. cssMaxValue.ih = parseToZeroOrNumber(cssMaxValue.h); //ih = integer height
  3130. cssMaxValue.iw = parseToZeroOrNumber(cssMaxValue.w); //iw = integer width
  3131. cssMaxValue.ch = cssMaxValue.h.indexOf('px') > -1; //ch = correct height
  3132. cssMaxValue.cw = cssMaxValue.w.indexOf('px') > -1; //cw = correct width
  3133. cssMaxValue.c = checkCacheAutoForce(cssMaxValue, _cssMaxValueCache);
  3134. //refresh cache
  3135. _displayIsHiddenCache = displayIsHidden;
  3136. _textareaAutoWrappingCache = textareaAutoWrapping;
  3137. _cssDirectionCache = cssDirection;
  3138. _cssBoxSizingCache = boxSizing;
  3139. _widthAutoCache = widthAuto;
  3140. _heightAutoCache = heightAuto;
  3141. _cssPaddingCache = padding;
  3142. _cssBorderCache = border;
  3143. _cssMarginCache = margin;
  3144. _cssMaxValueCache = cssMaxValue;
  3145. //IEFix direction changed
  3146. if (cssDirectionChanged && _sizeAutoObserverAdded)
  3147. _sizeAutoObserverElement.css(_strFloat, isRTLRight);
  3148. //apply padding:
  3149. if (padding.c || cssDirectionChanged || paddingAbsoluteChanged || widthAutoChanged || heightAutoChanged || boxSizingChanged || sizeAutoCapableChanged) {
  3150. var paddingElementCSS = {};
  3151. var textareaCSS = {};
  3152. setTopRightBottomLeft(contentGlueElementCSS, _strMarginMinus, [-padding.t, -padding.r, -padding.b, -padding.l]);
  3153. if (paddingAbsolute) {
  3154. setTopRightBottomLeft(paddingElementCSS, _strEmpty, [padding.t, padding.r, padding.b, padding.l]);
  3155. if (_isTextarea)
  3156. setTopRightBottomLeft(textareaCSS, _strPaddingMinus);
  3157. else
  3158. setTopRightBottomLeft(contentElementCSS, _strPaddingMinus);
  3159. }
  3160. else {
  3161. setTopRightBottomLeft(paddingElementCSS, _strEmpty);
  3162. if (_isTextarea)
  3163. setTopRightBottomLeft(textareaCSS, _strPaddingMinus, [padding.t, padding.r, padding.b, padding.l]);
  3164. else
  3165. setTopRightBottomLeft(contentElementCSS, _strPaddingMinus, [padding.t, padding.r, padding.b, padding.l]);
  3166. }
  3167. _paddingElement.css(paddingElementCSS);
  3168. _targetElement.css(textareaCSS);
  3169. }
  3170. //viewport size is padding container because it never has padding, margin and a border.
  3171. _viewportSize = getViewportSize();
  3172. //update Textarea
  3173. var textareaSize = _isTextarea ? textareaUpdate() : false;
  3174. var textareaSizeChanged = _isTextarea && checkCacheAutoForce(textareaSize, _textareaSizeCache);
  3175. var textareaDynOrigSize = _isTextarea && textareaSize ? {
  3176. w: textareaDynWidth ? textareaSize._dynamicWidth : textareaSize._originalWidth,
  3177. h: textareaDynHeight ? textareaSize._dynamicHeight : textareaSize._originalHeight
  3178. } : {};
  3179. _textareaSizeCache = textareaSize;
  3180. //fix height auto / width auto in cooperation with current padding & boxSizing behavior:
  3181. if (heightAuto && (heightAutoChanged || paddingAbsoluteChanged || boxSizingChanged || cssMaxValue.c || padding.c || border.c)) {
  3182. /*
  3183. if (cssMaxValue.ch)
  3184. contentElementCSS[_strMaxMinus + _strHeight] =
  3185. (cssMaxValue.ch ? (cssMaxValue.ih - paddingAbsoluteY + (_isBorderBox ? -_borderY : _paddingY))
  3186. : _strEmpty);
  3187. */
  3188. contentElementCSS[_strHeight] = _strAuto;
  3189. }
  3190. else if (heightAutoChanged || paddingAbsoluteChanged) {
  3191. contentElementCSS[_strMaxMinus + _strHeight] = _strEmpty;
  3192. contentElementCSS[_strHeight] = _strHundredPercent;
  3193. }
  3194. if (widthAuto && (widthAutoChanged || paddingAbsoluteChanged || boxSizingChanged || cssMaxValue.c || padding.c || border.c || cssDirectionChanged)) {
  3195. /*
  3196. if (cssMaxValue.cw)
  3197. contentElementCSS[_strMaxMinus + _strWidth] =
  3198. (cssMaxValue.cw ? (cssMaxValue.iw - paddingAbsoluteX + (_isBorderBox ? -_borderX : _paddingX)) +
  3199. (_nativeScrollbarIsOverlaid.y ? _overlayScrollbarDummySize.y : 0)
  3200. : _strEmpty);
  3201. */
  3202. contentElementCSS[_strWidth] = _strAuto;
  3203. contentGlueElementCSS[_strMaxMinus + _strWidth] = _strHundredPercent; //IE Fix
  3204. }
  3205. else if (widthAutoChanged || paddingAbsoluteChanged) {
  3206. contentElementCSS[_strMaxMinus + _strWidth] = _strEmpty;
  3207. contentElementCSS[_strWidth] = _strHundredPercent;
  3208. contentElementCSS[_strFloat] = _strEmpty;
  3209. contentGlueElementCSS[_strMaxMinus + _strWidth] = _strEmpty; //IE Fix
  3210. }
  3211. if (widthAuto) {
  3212. if (!cssMaxValue.cw)
  3213. contentElementCSS[_strMaxMinus + _strWidth] = _strEmpty;
  3214. //textareaDynOrigSize.w || _strAuto :: doesnt works because applied margin will shift width
  3215. contentGlueElementCSS[_strWidth] = _strAuto;
  3216. contentElementCSS[_strWidth] = _strAuto;
  3217. contentElementCSS[_strFloat] = isRTLRight;
  3218. }
  3219. else {
  3220. contentGlueElementCSS[_strWidth] = _strEmpty;
  3221. }
  3222. if (heightAuto) {
  3223. if (!cssMaxValue.ch)
  3224. contentElementCSS[_strMaxMinus + _strHeight] = _strEmpty;
  3225. //textareaDynOrigSize.h || _contentElementNative[LEXICON.cH] :: use for anti scroll jumping
  3226. contentGlueElementCSS[_strHeight] = textareaDynOrigSize.h || _contentElementNative[LEXICON.cH];
  3227. }
  3228. else {
  3229. contentGlueElementCSS[_strHeight] = _strEmpty;
  3230. }
  3231. if (sizeAutoCapable)
  3232. _contentGlueElement.css(contentGlueElementCSS);
  3233. _contentElement.css(contentElementCSS);
  3234. //CHECKPOINT HERE ~
  3235. contentElementCSS = {};
  3236. contentGlueElementCSS = {};
  3237. //if [content(host) client / scroll size, or target element direction, or content(host) max-sizes] changed, or force is true
  3238. if (hostSizeChanged || contentSizeChanged || textareaSizeChanged || cssDirectionChanged || boxSizingChanged || paddingAbsoluteChanged || widthAutoChanged || widthAuto || heightAutoChanged || heightAuto || cssMaxValue.c || ignoreOverlayScrollbarHidingChanged || overflowBehaviorChanged || clipAlwaysChanged || resizeChanged || scrollbarsVisibilityChanged || scrollbarsAutoHideChanged || scrollbarsDragScrollingChanged || scrollbarsClickScrollingChanged || textareaDynWidthChanged || textareaDynHeightChanged || textareaAutoWrappingChanged) {
  3239. var strOverflow = 'overflow';
  3240. var strOverflowX = strOverflow + '-x';
  3241. var strOverflowY = strOverflow + '-y';
  3242. var strHidden = 'hidden';
  3243. var strVisible = 'visible';
  3244. //decide whether the content overflow must get hidden for correct overflow measuring, it !MUST! be always hidden if the height is auto
  3245. var hideOverflow4CorrectMeasuring = _restrictedMeasuring ?
  3246. (_nativeScrollbarIsOverlaid.x || _nativeScrollbarIsOverlaid.y) || //it must be hidden if native scrollbars are overlaid
  3247. (_viewportSize.w < _nativeScrollbarMinSize.y || _viewportSize.h < _nativeScrollbarMinSize.x) || //it must be hidden if host-element is too small
  3248. heightAuto || displayIsHiddenChanged //it must be hidden if height is auto or display was changed
  3249. : heightAuto; //if there is not the restricted Measuring bug, it must be hidden if the height is auto
  3250. //Reset the viewport (very important for natively overlaid scrollbars and zoom change
  3251. //don't change the overflow prop as it is very expensive and affects performance !A LOT!
  3252. var viewportElementResetCSS = {};
  3253. var resetXTmp = _hasOverflowCache.y && _hideOverflowCache.ys && !ignoreOverlayScrollbarHiding && !_nativeScrollbarStyling ? (_nativeScrollbarIsOverlaid.y ? _viewportElement.css(isRTLLeft) : -_nativeScrollbarSize.y) : 0;
  3254. var resetBottomTmp = _hasOverflowCache.x && _hideOverflowCache.xs && !ignoreOverlayScrollbarHiding && !_nativeScrollbarStyling ? (_nativeScrollbarIsOverlaid.x ? _viewportElement.css(_strBottom) : -_nativeScrollbarSize.x) : 0;
  3255. setTopRightBottomLeft(viewportElementResetCSS, _strEmpty);
  3256. _viewportElement.css(viewportElementResetCSS);
  3257. if (hideOverflow4CorrectMeasuring)
  3258. _contentElement.css(strOverflow, strHidden);
  3259. //measure several sizes:
  3260. var contentMeasureElement = getContentMeasureElement();
  3261. //in Firefox content element has to have overflow hidden, else element margins aren't calculated properly, this element prevents this bug, but only if scrollbars aren't overlaid
  3262. var contentMeasureElementGuaranty = _restrictedMeasuring && !hideOverflow4CorrectMeasuring ? _viewportElementNative : contentMeasureElement;
  3263. var contentSize = {
  3264. //use clientSize because natively overlaidScrollbars add borders
  3265. w: textareaDynOrigSize.w || contentMeasureElement[LEXICON.cW],
  3266. h: textareaDynOrigSize.h || contentMeasureElement[LEXICON.cH]
  3267. };
  3268. var scrollSize = {
  3269. w: MATH.max(contentMeasureElement[LEXICON.sW], contentMeasureElementGuaranty[LEXICON.sW]),
  3270. h: MATH.max(contentMeasureElement[LEXICON.sH], contentMeasureElementGuaranty[LEXICON.sH])
  3271. };
  3272. //apply the correct viewport style and measure viewport size
  3273. viewportElementResetCSS[_strBottom] = wasHeightAuto ? _strEmpty : resetBottomTmp;
  3274. viewportElementResetCSS[isRTLLeft] = wasWidthAuto ? _strEmpty : resetXTmp;
  3275. _viewportElement.css(viewportElementResetCSS);
  3276. _viewportSize = getViewportSize();
  3277. //measure and correct several sizes
  3278. var hostSize = getHostSize();
  3279. var contentGlueSize = {
  3280. //client/scrollSize + AbsolutePadding -> because padding is only applied to the paddingElement if its absolute, so you have to add it manually
  3281. //hostSize is clientSize -> so padding should be added manually, right? FALSE! Because content glue is inside hostElement, so we don't have to worry about padding
  3282. w: MATH.max((widthAuto ? contentSize.w : scrollSize.w) + paddingAbsoluteX, hostSize.w),
  3283. h: MATH.max((heightAuto ? contentSize.h : scrollSize.h) + paddingAbsoluteY, hostSize.h)
  3284. };
  3285. contentGlueSize.c = checkCacheAutoForce(contentGlueSize, _contentGlueSizeCache);
  3286. _contentGlueSizeCache = contentGlueSize;
  3287. //apply correct contentGlue size
  3288. if (sizeAutoCapable) {
  3289. //size contentGlue correctly to make sure the element has correct size if the sizing switches to auto
  3290. if (contentGlueSize.c || (heightAuto || widthAuto)) {
  3291. contentGlueElementCSS[_strWidth] = contentGlueSize.w;
  3292. contentGlueElementCSS[_strHeight] = contentGlueSize.h;
  3293. //textarea-sizes are already calculated correctly at this point
  3294. if (!_isTextarea) {
  3295. contentSize = {
  3296. //use clientSize because natively overlaidScrollbars add borders
  3297. w: contentMeasureElement[LEXICON.cW],
  3298. h: contentMeasureElement[LEXICON.cH]
  3299. };
  3300. }
  3301. }
  3302. var textareaCoverCSS = {};
  3303. var setContentGlueElementCSSfunction = function (horizontal) {
  3304. var scrollbarVars = getScrollbarVars(horizontal);
  3305. var wh = scrollbarVars._w_h;
  3306. var strWH = scrollbarVars._width_height;
  3307. var autoSize = horizontal ? widthAuto : heightAuto;
  3308. var borderSize = horizontal ? _borderX : _borderY;
  3309. var paddingSize = horizontal ? _paddingX : _paddingY;
  3310. var marginSize = horizontal ? _marginX : _marginY;
  3311. var maxSize = contentGlueElementCSS[strWH] + (_isBorderBox ? borderSize : -paddingSize);
  3312. //make contentGlue size -1 if element is not auto sized, to make sure that a resize event happens when the element shrinks
  3313. if (!autoSize || (!autoSize && border.c))
  3314. contentGlueElementCSS[strWH] = hostSize[wh] - (_isBorderBox ? 0 : paddingSize + borderSize) - 1 - marginSize;
  3315. //if size is auto and host is same size as max size, make content glue size +1 to make sure size changes will be detected
  3316. if (autoSize && cssMaxValue['c' + wh] && cssMaxValue['i' + wh] === maxSize)
  3317. contentGlueElementCSS[strWH] = maxSize + (_isBorderBox ? 0 : paddingSize) + 1;
  3318. //if size is auto and host is smaller than size as min size, make content glue size -1 to make sure size changes will be detected (this is only needed if padding is 0)
  3319. if (autoSize && (contentSize[wh] < _viewportSize[wh]) && (horizontal && _isTextarea ? !textareaAutoWrapping : true)) {
  3320. if (_isTextarea)
  3321. textareaCoverCSS[strWH] = parseToZeroOrNumber(_textareaCoverElement.css(strWH)) - 1;
  3322. contentGlueElementCSS[strWH] -= 1;
  3323. }
  3324. //make sure content glue size is at least 1
  3325. if (contentSize[wh] > 0)
  3326. contentGlueElementCSS[strWH] = MATH.max(1, contentGlueElementCSS[strWH]);
  3327. };
  3328. setContentGlueElementCSSfunction(true);
  3329. setContentGlueElementCSSfunction(false);
  3330. if (_isTextarea)
  3331. _textareaCoverElement.css(textareaCoverCSS);
  3332. _contentGlueElement.css(contentGlueElementCSS);
  3333. }
  3334. if (widthAuto)
  3335. contentElementCSS[_strWidth] = _strHundredPercent;
  3336. if (widthAuto && !_isBorderBox && !_mutationObserversConnected)
  3337. contentElementCSS[_strFloat] = 'none';
  3338. //apply and reset content style
  3339. _contentElement.css(contentElementCSS);
  3340. contentElementCSS = {};
  3341. //measure again, but this time all correct sizes:
  3342. var contentScrollSize = {
  3343. w: MATH.max(contentMeasureElement[LEXICON.sW], contentMeasureElementGuaranty[LEXICON.sW]),
  3344. h: MATH.max(contentMeasureElement[LEXICON.sH], contentMeasureElementGuaranty[LEXICON.sH])
  3345. };
  3346. contentScrollSize.c = contentSizeChanged = checkCacheAutoForce(contentScrollSize, _contentScrollSizeCache);
  3347. _contentScrollSizeCache = contentScrollSize;
  3348. //remove overflow hidden to restore overflow
  3349. if (hideOverflow4CorrectMeasuring)
  3350. _contentElement.css(strOverflow, _strEmpty);
  3351. //refresh viewport size after correct measuring
  3352. _viewportSize = getViewportSize();
  3353. hostSize = getHostSize();
  3354. hostSizeChanged = checkCacheAutoForce(hostSize, _hostSizeCache);
  3355. _hostSizeCache = hostSize;
  3356. var hideOverflowForceTextarea = _isTextarea && (_viewportSize.w === 0 || _viewportSize.h === 0);
  3357. var previousOverflowAmount = _overflowAmountCache;
  3358. var overflowBehaviorIsVS = {};
  3359. var overflowBehaviorIsVH = {};
  3360. var overflowBehaviorIsS = {};
  3361. var overflowAmount = {};
  3362. var hasOverflow = {};
  3363. var hideOverflow = {};
  3364. var canScroll = {};
  3365. var viewportRect = _paddingElementNative[LEXICON.bCR]();
  3366. var setOverflowVariables = function (horizontal) {
  3367. var scrollbarVars = getScrollbarVars(horizontal);
  3368. var scrollbarVarsInverted = getScrollbarVars(!horizontal);
  3369. var xyI = scrollbarVarsInverted._x_y;
  3370. var xy = scrollbarVars._x_y;
  3371. var wh = scrollbarVars._w_h;
  3372. var widthHeight = scrollbarVars._width_height;
  3373. var scrollMax = _strScroll + scrollbarVars._Left_Top + 'Max';
  3374. var fractionalOverflowAmount = viewportRect[widthHeight] ? MATH.abs(viewportRect[widthHeight] - _viewportSize[wh]) : 0;
  3375. var checkFractionalOverflowAmount = previousOverflowAmount && previousOverflowAmount[xy] > 0 && _viewportElementNative[scrollMax] === 0;
  3376. overflowBehaviorIsVS[xy] = overflowBehavior[xy] === 'v-s';
  3377. overflowBehaviorIsVH[xy] = overflowBehavior[xy] === 'v-h';
  3378. overflowBehaviorIsS[xy] = overflowBehavior[xy] === 's';
  3379. overflowAmount[xy] = MATH.max(0, MATH.round((contentScrollSize[wh] - _viewportSize[wh]) * 100) / 100);
  3380. overflowAmount[xy] *= (hideOverflowForceTextarea || (checkFractionalOverflowAmount && fractionalOverflowAmount > 0 && fractionalOverflowAmount < 1)) ? 0 : 1;
  3381. hasOverflow[xy] = overflowAmount[xy] > 0;
  3382. //hideOverflow:
  3383. //x || y : true === overflow is hidden by "overflow: scroll" OR "overflow: hidden"
  3384. //xs || ys : true === overflow is hidden by "overflow: scroll"
  3385. hideOverflow[xy] = overflowBehaviorIsVS[xy] || overflowBehaviorIsVH[xy] ? (hasOverflow[xyI] && !overflowBehaviorIsVS[xyI] && !overflowBehaviorIsVH[xyI]) : hasOverflow[xy];
  3386. hideOverflow[xy + 's'] = hideOverflow[xy] ? (overflowBehaviorIsS[xy] || overflowBehaviorIsVS[xy]) : false;
  3387. canScroll[xy] = hasOverflow[xy] && hideOverflow[xy + 's'];
  3388. };
  3389. setOverflowVariables(true);
  3390. setOverflowVariables(false);
  3391. overflowAmount.c = checkCacheAutoForce(overflowAmount, _overflowAmountCache);
  3392. _overflowAmountCache = overflowAmount;
  3393. hasOverflow.c = checkCacheAutoForce(hasOverflow, _hasOverflowCache);
  3394. _hasOverflowCache = hasOverflow;
  3395. hideOverflow.c = checkCacheAutoForce(hideOverflow, _hideOverflowCache);
  3396. _hideOverflowCache = hideOverflow;
  3397. //if native scrollbar is overlay at x OR y axis, prepare DOM
  3398. if (_nativeScrollbarIsOverlaid.x || _nativeScrollbarIsOverlaid.y) {
  3399. var borderDesign = 'px solid transparent';
  3400. var contentArrangeElementCSS = {};
  3401. var arrangeContent = {};
  3402. var arrangeChanged = force;
  3403. var setContentElementCSS;
  3404. if (hasOverflow.x || hasOverflow.y) {
  3405. arrangeContent.w = _nativeScrollbarIsOverlaid.y && hasOverflow.y ? contentScrollSize.w + _overlayScrollbarDummySize.y : _strEmpty;
  3406. arrangeContent.h = _nativeScrollbarIsOverlaid.x && hasOverflow.x ? contentScrollSize.h + _overlayScrollbarDummySize.x : _strEmpty;
  3407. arrangeChanged = checkCacheAutoForce(arrangeContent, _arrangeContentSizeCache);
  3408. _arrangeContentSizeCache = arrangeContent;
  3409. }
  3410. if (hasOverflow.c || hideOverflow.c || contentScrollSize.c || cssDirectionChanged || widthAutoChanged || heightAutoChanged || widthAuto || heightAuto || ignoreOverlayScrollbarHidingChanged) {
  3411. contentElementCSS[_strMarginMinus + isRTLRight] = contentElementCSS[_strBorderMinus + isRTLRight] = _strEmpty;
  3412. setContentElementCSS = function (horizontal) {
  3413. var scrollbarVars = getScrollbarVars(horizontal);
  3414. var scrollbarVarsInverted = getScrollbarVars(!horizontal);
  3415. var xy = scrollbarVars._x_y;
  3416. var strDirection = horizontal ? _strBottom : isRTLLeft;
  3417. var invertedAutoSize = horizontal ? heightAuto : widthAuto;
  3418. if (_nativeScrollbarIsOverlaid[xy] && hasOverflow[xy] && hideOverflow[xy + 's']) {
  3419. contentElementCSS[_strMarginMinus + strDirection] = invertedAutoSize ? (ignoreOverlayScrollbarHiding ? _strEmpty : _overlayScrollbarDummySize[xy]) : _strEmpty;
  3420. contentElementCSS[_strBorderMinus + strDirection] = ((horizontal ? !invertedAutoSize : true) && !ignoreOverlayScrollbarHiding) ? (_overlayScrollbarDummySize[xy] + borderDesign) : _strEmpty;
  3421. }
  3422. else {
  3423. arrangeContent[scrollbarVarsInverted._w_h] =
  3424. contentElementCSS[_strMarginMinus + strDirection] =
  3425. contentElementCSS[_strBorderMinus + strDirection] = _strEmpty;
  3426. arrangeChanged = true;
  3427. }
  3428. };
  3429. if (_nativeScrollbarStyling) {
  3430. if (ignoreOverlayScrollbarHiding)
  3431. removeClass(_viewportElement, _classNameViewportNativeScrollbarsInvisible);
  3432. else
  3433. addClass(_viewportElement, _classNameViewportNativeScrollbarsInvisible);
  3434. }
  3435. else {
  3436. setContentElementCSS(true);
  3437. setContentElementCSS(false);
  3438. }
  3439. }
  3440. if (ignoreOverlayScrollbarHiding) {
  3441. arrangeContent.w = arrangeContent.h = _strEmpty;
  3442. arrangeChanged = true;
  3443. }
  3444. if (arrangeChanged && !_nativeScrollbarStyling) {
  3445. contentArrangeElementCSS[_strWidth] = hideOverflow.y ? arrangeContent.w : _strEmpty;
  3446. contentArrangeElementCSS[_strHeight] = hideOverflow.x ? arrangeContent.h : _strEmpty;
  3447. if (!_contentArrangeElement) {
  3448. _contentArrangeElement = FRAMEWORK(generateDiv(_classNameContentArrangeElement));
  3449. _viewportElement.prepend(_contentArrangeElement);
  3450. }
  3451. _contentArrangeElement.css(contentArrangeElementCSS);
  3452. }
  3453. _contentElement.css(contentElementCSS);
  3454. }
  3455. var viewportElementCSS = {};
  3456. var paddingElementCSS = {};
  3457. var setViewportCSS;
  3458. if (hostSizeChanged || hasOverflow.c || hideOverflow.c || contentScrollSize.c || overflowBehaviorChanged || boxSizingChanged || ignoreOverlayScrollbarHidingChanged || cssDirectionChanged || clipAlwaysChanged || heightAutoChanged) {
  3459. viewportElementCSS[isRTLRight] = _strEmpty;
  3460. setViewportCSS = function (horizontal) {
  3461. var scrollbarVars = getScrollbarVars(horizontal);
  3462. var scrollbarVarsInverted = getScrollbarVars(!horizontal);
  3463. var xy = scrollbarVars._x_y;
  3464. var XY = scrollbarVars._X_Y;
  3465. var strDirection = horizontal ? _strBottom : isRTLLeft;
  3466. var reset = function () {
  3467. viewportElementCSS[strDirection] = _strEmpty;
  3468. _contentBorderSize[scrollbarVarsInverted._w_h] = 0;
  3469. };
  3470. if (hasOverflow[xy] && hideOverflow[xy + 's']) {
  3471. viewportElementCSS[strOverflow + XY] = _strScroll;
  3472. if (ignoreOverlayScrollbarHiding || _nativeScrollbarStyling) {
  3473. reset();
  3474. }
  3475. else {
  3476. viewportElementCSS[strDirection] = -(_nativeScrollbarIsOverlaid[xy] ? _overlayScrollbarDummySize[xy] : _nativeScrollbarSize[xy]);
  3477. _contentBorderSize[scrollbarVarsInverted._w_h] = _nativeScrollbarIsOverlaid[xy] ? _overlayScrollbarDummySize[scrollbarVarsInverted._x_y] : 0;
  3478. }
  3479. } else {
  3480. viewportElementCSS[strOverflow + XY] = _strEmpty;
  3481. reset();
  3482. }
  3483. };
  3484. setViewportCSS(true);
  3485. setViewportCSS(false);
  3486. // if the scroll container is too small and if there is any overflow with no overlay scrollbar (and scrollbar styling isn't possible),
  3487. // make viewport element greater in size (Firefox hide Scrollbars fix)
  3488. // because firefox starts hiding scrollbars on too small elements
  3489. // with this behavior the overflow calculation may be incorrect or the scrollbars would appear suddenly
  3490. // https://bugzilla.mozilla.org/show_bug.cgi?id=292284
  3491. if (!_nativeScrollbarStyling
  3492. && (_viewportSize.h < _nativeScrollbarMinSize.x || _viewportSize.w < _nativeScrollbarMinSize.y)
  3493. && ((hasOverflow.x && hideOverflow.x && !_nativeScrollbarIsOverlaid.x) || (hasOverflow.y && hideOverflow.y && !_nativeScrollbarIsOverlaid.y))) {
  3494. viewportElementCSS[_strPaddingMinus + _strTop] = _nativeScrollbarMinSize.x;
  3495. viewportElementCSS[_strMarginMinus + _strTop] = -_nativeScrollbarMinSize.x;
  3496. viewportElementCSS[_strPaddingMinus + isRTLRight] = _nativeScrollbarMinSize.y;
  3497. viewportElementCSS[_strMarginMinus + isRTLRight] = -_nativeScrollbarMinSize.y;
  3498. }
  3499. else {
  3500. viewportElementCSS[_strPaddingMinus + _strTop] =
  3501. viewportElementCSS[_strMarginMinus + _strTop] =
  3502. viewportElementCSS[_strPaddingMinus + isRTLRight] =
  3503. viewportElementCSS[_strMarginMinus + isRTLRight] = _strEmpty;
  3504. }
  3505. viewportElementCSS[_strPaddingMinus + isRTLLeft] =
  3506. viewportElementCSS[_strMarginMinus + isRTLLeft] = _strEmpty;
  3507. //if there is any overflow (x OR y axis) and this overflow shall be hidden, make overflow hidden, else overflow visible
  3508. if ((hasOverflow.x && hideOverflow.x) || (hasOverflow.y && hideOverflow.y) || hideOverflowForceTextarea) {
  3509. //only hide if is Textarea
  3510. if (_isTextarea && hideOverflowForceTextarea) {
  3511. paddingElementCSS[strOverflowX] =
  3512. paddingElementCSS[strOverflowY] = strHidden;
  3513. }
  3514. }
  3515. else {
  3516. if (!clipAlways || (overflowBehaviorIsVH.x || overflowBehaviorIsVS.x || overflowBehaviorIsVH.y || overflowBehaviorIsVS.y)) {
  3517. //only un-hide if Textarea
  3518. if (_isTextarea) {
  3519. paddingElementCSS[strOverflowX] =
  3520. paddingElementCSS[strOverflowY] = _strEmpty;
  3521. }
  3522. viewportElementCSS[strOverflowX] =
  3523. viewportElementCSS[strOverflowY] = strVisible;
  3524. }
  3525. }
  3526. _paddingElement.css(paddingElementCSS);
  3527. _viewportElement.css(viewportElementCSS);
  3528. viewportElementCSS = {};
  3529. //force soft redraw in webkit because without the scrollbars will may appear because DOM wont be redrawn under special conditions
  3530. if ((hasOverflow.c || boxSizingChanged || widthAutoChanged || heightAutoChanged) && !(_nativeScrollbarIsOverlaid.x && _nativeScrollbarIsOverlaid.y)) {
  3531. var elementStyle = _contentElementNative[LEXICON.s];
  3532. var dump;
  3533. elementStyle.webkitTransform = 'scale(1)';
  3534. elementStyle.display = 'run-in';
  3535. dump = _contentElementNative[LEXICON.oH];
  3536. elementStyle.display = _strEmpty; //|| dump; //use dump to prevent it from deletion if minify
  3537. elementStyle.webkitTransform = _strEmpty;
  3538. }
  3539. /*
  3540. //force hard redraw in webkit if native overlaid scrollbars shall appear
  3541. if (ignoreOverlayScrollbarHidingChanged && ignoreOverlayScrollbarHiding) {
  3542. _hostElement.hide();
  3543. var dump = _hostElementNative[LEXICON.oH];
  3544. _hostElement.show();
  3545. }
  3546. */
  3547. }
  3548. //change to direction RTL and width auto Bugfix in Webkit
  3549. //without this fix, the DOM still thinks the scrollbar is LTR and thus the content is shifted to the left
  3550. contentElementCSS = {};
  3551. if (cssDirectionChanged || widthAutoChanged || heightAutoChanged) {
  3552. if (_isRTL && widthAuto) {
  3553. var floatTmp = _contentElement.css(_strFloat);
  3554. var posLeftWithoutFloat = MATH.round(_contentElement.css(_strFloat, _strEmpty).css(_strLeft, _strEmpty).position().left);
  3555. _contentElement.css(_strFloat, floatTmp);
  3556. var posLeftWithFloat = MATH.round(_contentElement.position().left);
  3557. if (posLeftWithoutFloat !== posLeftWithFloat)
  3558. contentElementCSS[_strLeft] = posLeftWithoutFloat;
  3559. }
  3560. else {
  3561. contentElementCSS[_strLeft] = _strEmpty;
  3562. }
  3563. }
  3564. _contentElement.css(contentElementCSS);
  3565. //handle scroll position
  3566. if (_isTextarea && contentSizeChanged) {
  3567. var textareaInfo = getTextareaInfo();
  3568. if (textareaInfo) {
  3569. var textareaRowsChanged = _textareaInfoCache === undefined ? true : textareaInfo._rows !== _textareaInfoCache._rows;
  3570. var cursorRow = textareaInfo._cursorRow;
  3571. var cursorCol = textareaInfo._cursorColumn;
  3572. var widestRow = textareaInfo._widestRow;
  3573. var lastRow = textareaInfo._rows;
  3574. var lastCol = textareaInfo._columns;
  3575. var cursorPos = textareaInfo._cursorPosition;
  3576. var cursorMax = textareaInfo._cursorMax;
  3577. var cursorIsLastPosition = (cursorPos >= cursorMax && _textareaHasFocus);
  3578. var textareaScrollAmount = {
  3579. x: (!textareaAutoWrapping && (cursorCol === lastCol && cursorRow === widestRow)) ? _overflowAmountCache.x : -1,
  3580. y: (textareaAutoWrapping ? cursorIsLastPosition || textareaRowsChanged && (previousOverflowAmount ? (currScroll.y === previousOverflowAmount.y) : false) : (cursorIsLastPosition || textareaRowsChanged) && cursorRow === lastRow) ? _overflowAmountCache.y : -1
  3581. };
  3582. currScroll.x = textareaScrollAmount.x > -1 ? (_isRTL && _normalizeRTLCache && _rtlScrollBehavior.i ? 0 : textareaScrollAmount.x) : currScroll.x; //if inverted, scroll to 0 -> normalized this means to max scroll offset.
  3583. currScroll.y = textareaScrollAmount.y > -1 ? textareaScrollAmount.y : currScroll.y;
  3584. }
  3585. _textareaInfoCache = textareaInfo;
  3586. }
  3587. if (_isRTL && _rtlScrollBehavior.i && _nativeScrollbarIsOverlaid.y && hasOverflow.x && _normalizeRTLCache)
  3588. currScroll.x += _contentBorderSize.w || 0;
  3589. if (widthAuto)
  3590. _hostElement[_strScrollLeft](0);
  3591. if (heightAuto)
  3592. _hostElement[_strScrollTop](0);
  3593. _viewportElement[_strScrollLeft](currScroll.x)[_strScrollTop](currScroll.y);
  3594. //scrollbars management:
  3595. var scrollbarsVisibilityVisible = scrollbarsVisibility === 'v';
  3596. var scrollbarsVisibilityHidden = scrollbarsVisibility === 'h';
  3597. var scrollbarsVisibilityAuto = scrollbarsVisibility === 'a';
  3598. var showScrollbarH = COMPATIBILITY.bind(refreshScrollbarAppearance, 0, true, true, canScroll.x);
  3599. var showScrollbarV = COMPATIBILITY.bind(refreshScrollbarAppearance, 0, false, true, canScroll.y);
  3600. var hideScrollbarH = COMPATIBILITY.bind(refreshScrollbarAppearance, 0, true, false, canScroll.x);
  3601. var hideScrollbarV = COMPATIBILITY.bind(refreshScrollbarAppearance, 0, false, false, canScroll.y);
  3602. //manage class name which indicates scrollable overflow
  3603. if (hideOverflow.x || hideOverflow.y)
  3604. addClass(_hostElement, _classNameHostOverflow);
  3605. else
  3606. removeClass(_hostElement, _classNameHostOverflow);
  3607. if (hideOverflow.x)
  3608. addClass(_hostElement, _classNameHostOverflowX);
  3609. else
  3610. removeClass(_hostElement, _classNameHostOverflowX);
  3611. if (hideOverflow.y)
  3612. addClass(_hostElement, _classNameHostOverflowY);
  3613. else
  3614. removeClass(_hostElement, _classNameHostOverflowY);
  3615. //add or remove rtl class name for styling purposes
  3616. if (cssDirectionChanged) {
  3617. if (_isRTL)
  3618. addClass(_hostElement, _classNameHostRTL);
  3619. else
  3620. removeClass(_hostElement, _classNameHostRTL);
  3621. }
  3622. //manage the resize feature (CSS3 resize "polyfill" for this plugin)
  3623. if (_isBody)
  3624. addClass(_hostElement, _classNameHostResizeDisabled);
  3625. if (resizeChanged) {
  3626. removeClass(_scrollbarCornerElement, [
  3627. _classNameScrollbarCornerResize,
  3628. _classNameScrollbarCornerResizeB,
  3629. _classNameScrollbarCornerResizeH,
  3630. _classNameScrollbarCornerResizeV].join(_strSpace));
  3631. if (_resizeNone) {
  3632. addClass(_hostElement, _classNameHostResizeDisabled);
  3633. }
  3634. else {
  3635. removeClass(_hostElement, _classNameHostResizeDisabled);
  3636. addClass(_scrollbarCornerElement, _classNameScrollbarCornerResize);
  3637. if (_resizeBoth)
  3638. addClass(_scrollbarCornerElement, _classNameScrollbarCornerResizeB);
  3639. else if (_resizeHorizontal)
  3640. addClass(_scrollbarCornerElement, _classNameScrollbarCornerResizeH);
  3641. else if (_resizeVertical)
  3642. addClass(_scrollbarCornerElement, _classNameScrollbarCornerResizeV);
  3643. }
  3644. }
  3645. //manage the scrollbars general visibility + the scrollbar interactivity (unusable class name)
  3646. if (scrollbarsVisibilityChanged || overflowBehaviorChanged || hideOverflow.c || hasOverflow.c || ignoreOverlayScrollbarHidingChanged) {
  3647. if (ignoreOverlayScrollbarHiding) {
  3648. if (ignoreOverlayScrollbarHidingChanged) {
  3649. removeClass(_hostElement, _classNameHostScrolling);
  3650. if (ignoreOverlayScrollbarHiding) {
  3651. hideScrollbarH();
  3652. hideScrollbarV();
  3653. }
  3654. }
  3655. }
  3656. else if (scrollbarsVisibilityAuto) {
  3657. if (canScroll.x)
  3658. showScrollbarH();
  3659. else
  3660. hideScrollbarH();
  3661. if (canScroll.y)
  3662. showScrollbarV();
  3663. else
  3664. hideScrollbarV();
  3665. }
  3666. else if (scrollbarsVisibilityVisible) {
  3667. showScrollbarH();
  3668. showScrollbarV();
  3669. }
  3670. else if (scrollbarsVisibilityHidden) {
  3671. hideScrollbarH();
  3672. hideScrollbarV();
  3673. }
  3674. }
  3675. //manage the scrollbars auto hide feature (auto hide them after specific actions)
  3676. if (scrollbarsAutoHideChanged || ignoreOverlayScrollbarHidingChanged) {
  3677. if (_scrollbarsAutoHideLeave || _scrollbarsAutoHideMove) {
  3678. setupHostMouseTouchEvents(true);
  3679. setupHostMouseTouchEvents();
  3680. }
  3681. else {
  3682. setupHostMouseTouchEvents(true);
  3683. }
  3684. if (_scrollbarsAutoHideNever)
  3685. refreshScrollbarsAutoHide(true);
  3686. else
  3687. refreshScrollbarsAutoHide(false, true);
  3688. }
  3689. //manage scrollbars handle length & offset - don't remove!
  3690. if (hostSizeChanged || overflowAmount.c || heightAutoChanged || widthAutoChanged || resizeChanged || boxSizingChanged || paddingAbsoluteChanged || ignoreOverlayScrollbarHidingChanged || cssDirectionChanged) {
  3691. refreshScrollbarHandleLength(true);
  3692. refreshScrollbarHandleOffset(true);
  3693. refreshScrollbarHandleLength(false);
  3694. refreshScrollbarHandleOffset(false);
  3695. }
  3696. //manage interactivity
  3697. if (scrollbarsClickScrollingChanged)
  3698. refreshScrollbarsInteractive(true, scrollbarsClickScrolling);
  3699. if (scrollbarsDragScrollingChanged)
  3700. refreshScrollbarsInteractive(false, scrollbarsDragScrolling);
  3701. //callbacks:
  3702. if (cssDirectionChanged) {
  3703. dispatchCallback('onDirectionChanged', {
  3704. isRTL: _isRTL,
  3705. dir: cssDirection
  3706. });
  3707. }
  3708. if (hostSizeChanged) {
  3709. dispatchCallback('onHostSizeChanged', {
  3710. width: _hostSizeCache.w,
  3711. height: _hostSizeCache.h
  3712. });
  3713. }
  3714. if (contentSizeChanged) {
  3715. dispatchCallback('onContentSizeChanged', {
  3716. width: _contentScrollSizeCache.w,
  3717. height: _contentScrollSizeCache.h
  3718. });
  3719. }
  3720. if (hasOverflow.c || hideOverflow.c) {
  3721. dispatchCallback('onOverflowChanged', {
  3722. x: hasOverflow.x,
  3723. y: hasOverflow.y,
  3724. xScrollable: hideOverflow.xs,
  3725. yScrollable: hideOverflow.ys,
  3726. clipped: hideOverflow.x || hideOverflow.y
  3727. });
  3728. }
  3729. if (overflowAmount.c) {
  3730. dispatchCallback('onOverflowAmountChanged', {
  3731. x: overflowAmount.x,
  3732. y: overflowAmount.y
  3733. });
  3734. }
  3735. }
  3736. //fix body min size
  3737. if (_isBody && _bodyMinSizeCache && (_hasOverflowCache.c || _bodyMinSizeCache.c)) {
  3738. //its possible that no min size was measured until now, because the content arrange element was just added now, in this case, measure now the min size.
  3739. if (!_bodyMinSizeCache.f)
  3740. bodyMinSizeChanged();
  3741. if (_nativeScrollbarIsOverlaid.y && _hasOverflowCache.x)
  3742. _contentElement.css(_strMinMinus + _strWidth, _bodyMinSizeCache.w + _overlayScrollbarDummySize.y);
  3743. if (_nativeScrollbarIsOverlaid.x && _hasOverflowCache.y)
  3744. _contentElement.css(_strMinMinus + _strHeight, _bodyMinSizeCache.h + _overlayScrollbarDummySize.x);
  3745. _bodyMinSizeCache.c = false;
  3746. }
  3747. //freezeResizeObserver(_sizeObserverElement, false);
  3748. //freezeResizeObserver(_sizeAutoObserverElement, false);
  3749. dispatchCallback('onUpdated', { forced: force });
  3750. }
  3751. //==== Options ====//
  3752. /**
  3753. * Sets new options but doesn't call the update method.
  3754. * @param newOptions The object which contains the new options.
  3755. * @returns {*} A object which contains the changed options.
  3756. */
  3757. function setOptions(newOptions) {
  3758. var validatedOpts = _pluginsOptions._validate(newOptions, _pluginsOptions._template, true, _currentOptions)
  3759. _currentOptions = extendDeep({}, _currentOptions, validatedOpts._default);
  3760. _currentPreparedOptions = extendDeep({}, _currentPreparedOptions, validatedOpts._prepared);
  3761. return validatedOpts._prepared;
  3762. }
  3763. //==== Structure ====//
  3764. /**
  3765. * Builds or destroys the wrapper and helper DOM elements.
  3766. * @param destroy Indicates whether the DOM shall be build or destroyed.
  3767. */
  3768. function setupStructureDOM(destroy) {
  3769. var strParent = 'parent';
  3770. var classNameResizeObserverHost = 'os-resize-observer-host';
  3771. var classNameTextareaElementFull = _classNameTextareaElement + _strSpace + _classNameTextInherit;
  3772. var textareaClass = _isTextarea ? _strSpace + _classNameTextInherit : _strEmpty;
  3773. var adoptAttrs = _currentPreparedOptions.textarea.inheritedAttrs;
  3774. var adoptAttrsMap = {};
  3775. var applyAdoptedAttrs = function () {
  3776. var applyAdoptedAttrsElm = destroy ? _targetElement : _hostElement;
  3777. each(adoptAttrsMap, function (key, value) {
  3778. if (type(value) == TYPES.s) {
  3779. if (key == LEXICON.c)
  3780. applyAdoptedAttrsElm.addClass(value);
  3781. else
  3782. applyAdoptedAttrsElm.attr(key, value);
  3783. }
  3784. });
  3785. };
  3786. var hostElementClassNames = [
  3787. _classNameHostElement,
  3788. _classNameHostTextareaElement,
  3789. _classNameHostResizeDisabled,
  3790. _classNameHostRTL,
  3791. _classNameHostScrollbarHorizontalHidden,
  3792. _classNameHostScrollbarVerticalHidden,
  3793. _classNameHostTransition,
  3794. _classNameHostScrolling,
  3795. _classNameHostOverflow,
  3796. _classNameHostOverflowX,
  3797. _classNameHostOverflowY,
  3798. _classNameThemeNone,
  3799. _classNameTextareaElement,
  3800. _classNameTextInherit,
  3801. _classNameCache].join(_strSpace);
  3802. var hostElementCSS = {};
  3803. //get host element as first element, because that's the most upper element and required for the other elements
  3804. _hostElement = _hostElement || (_isTextarea ? (_domExists ? _targetElement[strParent]()[strParent]()[strParent]()[strParent]() : FRAMEWORK(generateDiv(_classNameHostTextareaElement))) : _targetElement);
  3805. _contentElement = _contentElement || selectOrGenerateDivByClass(_classNameContentElement + textareaClass);
  3806. _viewportElement = _viewportElement || selectOrGenerateDivByClass(_classNameViewportElement + textareaClass);
  3807. _paddingElement = _paddingElement || selectOrGenerateDivByClass(_classNamePaddingElement + textareaClass);
  3808. _sizeObserverElement = _sizeObserverElement || selectOrGenerateDivByClass(classNameResizeObserverHost);
  3809. _textareaCoverElement = _textareaCoverElement || (_isTextarea ? selectOrGenerateDivByClass(_classNameTextareaCoverElement) : undefined);
  3810. //on destroy, remove all generated class names from the host element before collecting the adopted attributes
  3811. //to prevent adopting generated class names
  3812. if (destroy)
  3813. removeClass(_hostElement, hostElementClassNames);
  3814. //collect all adopted attributes
  3815. adoptAttrs = type(adoptAttrs) == TYPES.s ? adoptAttrs.split(_strSpace) : adoptAttrs;
  3816. if (type(adoptAttrs) == TYPES.a && _isTextarea) {
  3817. each(adoptAttrs, function (i, v) {
  3818. if (type(v) == TYPES.s) {
  3819. adoptAttrsMap[v] = destroy ? _hostElement.attr(v) : _targetElement.attr(v);
  3820. }
  3821. });
  3822. }
  3823. if (!destroy) {
  3824. if (_isTextarea) {
  3825. if (!_currentPreparedOptions.sizeAutoCapable) {
  3826. hostElementCSS[_strWidth] = _targetElement.css(_strWidth);
  3827. hostElementCSS[_strHeight] = _targetElement.css(_strHeight);
  3828. }
  3829. if (!_domExists)
  3830. _targetElement.addClass(_classNameTextInherit).wrap(_hostElement);
  3831. //jQuery clones elements in wrap functions, so we have to select them again
  3832. _hostElement = _targetElement[strParent]().css(hostElementCSS);
  3833. }
  3834. if (!_domExists) {
  3835. //add the correct class to the target element
  3836. addClass(_targetElement, _isTextarea ? classNameTextareaElementFull : _classNameHostElement);
  3837. //wrap the content into the generated elements to create the required DOM
  3838. _hostElement.wrapInner(_contentElement)
  3839. .wrapInner(_viewportElement)
  3840. .wrapInner(_paddingElement)
  3841. .prepend(_sizeObserverElement);
  3842. //jQuery clones elements in wrap functions, so we have to select them again
  3843. _contentElement = findFirst(_hostElement, _strDot + _classNameContentElement);
  3844. _viewportElement = findFirst(_hostElement, _strDot + _classNameViewportElement);
  3845. _paddingElement = findFirst(_hostElement, _strDot + _classNamePaddingElement);
  3846. if (_isTextarea) {
  3847. _contentElement.prepend(_textareaCoverElement);
  3848. applyAdoptedAttrs();
  3849. }
  3850. }
  3851. if (_nativeScrollbarStyling)
  3852. addClass(_viewportElement, _classNameViewportNativeScrollbarsInvisible);
  3853. if (_nativeScrollbarIsOverlaid.x && _nativeScrollbarIsOverlaid.y)
  3854. addClass(_viewportElement, _classNameViewportNativeScrollbarsOverlaid);
  3855. if (_isBody)
  3856. addClass(_htmlElement, _classNameHTMLElement);
  3857. _sizeObserverElementNative = _sizeObserverElement[0];
  3858. _hostElementNative = _hostElement[0];
  3859. _paddingElementNative = _paddingElement[0];
  3860. _viewportElementNative = _viewportElement[0];
  3861. _contentElementNative = _contentElement[0];
  3862. }
  3863. else {
  3864. if (_domExists && _initialized) {
  3865. //clear size observer
  3866. _sizeObserverElement.children().remove();
  3867. //remove the style property and classes from already generated elements
  3868. each([_paddingElement, _viewportElement, _contentElement, _textareaCoverElement], function (i, elm) {
  3869. if (elm) {
  3870. removeClass(elm.removeAttr(LEXICON.s), _classNamesDynamicDestroy);
  3871. }
  3872. });
  3873. //add classes to the host element which was removed previously to match the expected DOM
  3874. addClass(_hostElement, _isTextarea ? _classNameHostTextareaElement : _classNameHostElement);
  3875. }
  3876. else {
  3877. //remove size observer
  3878. remove(_sizeObserverElement);
  3879. //unwrap the content to restore DOM
  3880. _contentElement.contents()
  3881. .unwrap()
  3882. .unwrap()
  3883. .unwrap();
  3884. if (_isTextarea) {
  3885. _targetElement.unwrap();
  3886. remove(_hostElement);
  3887. remove(_textareaCoverElement);
  3888. applyAdoptedAttrs();
  3889. }
  3890. }
  3891. if (_isTextarea)
  3892. _targetElement.removeAttr(LEXICON.s);
  3893. if (_isBody)
  3894. removeClass(_htmlElement, _classNameHTMLElement);
  3895. }
  3896. }
  3897. /**
  3898. * Adds or removes all wrapper elements interactivity events.
  3899. * @param destroy Indicates whether the Events shall be added or removed.
  3900. */
  3901. function setupStructureEvents() {
  3902. var textareaKeyDownRestrictedKeyCodes = [
  3903. 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 123, //F1 to F12
  3904. 33, 34, //page up, page down
  3905. 37, 38, 39, 40, //left, up, right, down arrows
  3906. 16, 17, 18, 19, 20, 144 //Shift, Ctrl, Alt, Pause, CapsLock, NumLock
  3907. ];
  3908. var textareaKeyDownKeyCodesList = [];
  3909. var textareaUpdateIntervalID;
  3910. var scrollStopTimeoutId;
  3911. var scrollStopDelay = 175;
  3912. var strFocus = 'focus';
  3913. function updateTextarea(doClearInterval) {
  3914. textareaUpdate();
  3915. _base.update(_strAuto);
  3916. if (doClearInterval && _autoUpdateRecommended)
  3917. clearInterval(textareaUpdateIntervalID);
  3918. }
  3919. function textareaOnScroll(event) {
  3920. _targetElement[_strScrollLeft](_rtlScrollBehavior.i && _normalizeRTLCache ? 9999999 : 0);
  3921. _targetElement[_strScrollTop](0);
  3922. COMPATIBILITY.prvD(event);
  3923. COMPATIBILITY.stpP(event);
  3924. return false;
  3925. }
  3926. function textareaOnDrop(event) {
  3927. setTimeout(function () {
  3928. if (!_destroyed)
  3929. updateTextarea();
  3930. }, 50);
  3931. }
  3932. function textareaOnFocus() {
  3933. _textareaHasFocus = true;
  3934. addClass(_hostElement, strFocus);
  3935. }
  3936. function textareaOnFocusout() {
  3937. _textareaHasFocus = false;
  3938. textareaKeyDownKeyCodesList = [];
  3939. removeClass(_hostElement, strFocus);
  3940. updateTextarea(true);
  3941. }
  3942. function textareaOnKeyDown(event) {
  3943. var keyCode = event.keyCode;
  3944. if (inArray(keyCode, textareaKeyDownRestrictedKeyCodes) < 0) {
  3945. if (!textareaKeyDownKeyCodesList[LEXICON.l]) {
  3946. updateTextarea();
  3947. textareaUpdateIntervalID = setInterval(updateTextarea, 1000 / 60);
  3948. }
  3949. if (inArray(keyCode, textareaKeyDownKeyCodesList) < 0)
  3950. textareaKeyDownKeyCodesList.push(keyCode);
  3951. }
  3952. }
  3953. function textareaOnKeyUp(event) {
  3954. var keyCode = event.keyCode;
  3955. var index = inArray(keyCode, textareaKeyDownKeyCodesList);
  3956. if (inArray(keyCode, textareaKeyDownRestrictedKeyCodes) < 0) {
  3957. if (index > -1)
  3958. textareaKeyDownKeyCodesList.splice(index, 1);
  3959. if (!textareaKeyDownKeyCodesList[LEXICON.l])
  3960. updateTextarea(true);
  3961. }
  3962. }
  3963. function contentOnTransitionEnd(event) {
  3964. if (_autoUpdateCache === true)
  3965. return;
  3966. event = event.originalEvent || event;
  3967. if (isSizeAffectingCSSProperty(event.propertyName))
  3968. _base.update(_strAuto);
  3969. }
  3970. function viewportOnScroll(event) {
  3971. if (!_sleeping) {
  3972. if (scrollStopTimeoutId !== undefined)
  3973. clearTimeout(scrollStopTimeoutId);
  3974. else {
  3975. if (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove)
  3976. refreshScrollbarsAutoHide(true);
  3977. if (!nativeOverlayScrollbarsAreActive())
  3978. addClass(_hostElement, _classNameHostScrolling);
  3979. dispatchCallback('onScrollStart', event);
  3980. }
  3981. //if a scrollbars handle gets dragged, the mousemove event is responsible for refreshing the handle offset
  3982. //because if CSS scroll-snap is used, the handle offset gets only refreshed on every snap point
  3983. //this looks laggy & clunky, it looks much better if the offset refreshes with the mousemove
  3984. if (!_scrollbarsHandlesDefineScrollPos) {
  3985. refreshScrollbarHandleOffset(true);
  3986. refreshScrollbarHandleOffset(false);
  3987. }
  3988. dispatchCallback('onScroll', event);
  3989. scrollStopTimeoutId = setTimeout(function () {
  3990. if (!_destroyed) {
  3991. //OnScrollStop:
  3992. clearTimeout(scrollStopTimeoutId);
  3993. scrollStopTimeoutId = undefined;
  3994. if (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove)
  3995. refreshScrollbarsAutoHide(false);
  3996. if (!nativeOverlayScrollbarsAreActive())
  3997. removeClass(_hostElement, _classNameHostScrolling);
  3998. dispatchCallback('onScrollStop', event);
  3999. }
  4000. }, scrollStopDelay);
  4001. }
  4002. }
  4003. if (_isTextarea) {
  4004. if (_msieVersion > 9 || !_autoUpdateRecommended) {
  4005. addDestroyEventListener(_targetElement, 'input', updateTextarea);
  4006. }
  4007. else {
  4008. addDestroyEventListener(_targetElement,
  4009. [_strKeyDownEvent, _strKeyUpEvent],
  4010. [textareaOnKeyDown, textareaOnKeyUp]);
  4011. }
  4012. addDestroyEventListener(_targetElement,
  4013. [_strScroll, 'drop', strFocus, strFocus + 'out'],
  4014. [textareaOnScroll, textareaOnDrop, textareaOnFocus, textareaOnFocusout]);
  4015. }
  4016. else {
  4017. addDestroyEventListener(_contentElement, _strTransitionEndEvent, contentOnTransitionEnd);
  4018. }
  4019. addDestroyEventListener(_viewportElement, _strScroll, viewportOnScroll, true);
  4020. }
  4021. //==== Scrollbars ====//
  4022. /**
  4023. * Builds or destroys all scrollbar DOM elements (scrollbar, track, handle)
  4024. * @param destroy Indicates whether the DOM shall be build or destroyed.
  4025. */
  4026. function setupScrollbarsDOM(destroy) {
  4027. var selectOrGenerateScrollbarDOM = function (isHorizontal) {
  4028. var scrollbarClassName = isHorizontal ? _classNameScrollbarHorizontal : _classNameScrollbarVertical;
  4029. var scrollbar = selectOrGenerateDivByClass(_classNameScrollbar + _strSpace + scrollbarClassName, true);
  4030. var track = selectOrGenerateDivByClass(_classNameScrollbarTrack, scrollbar);
  4031. var handle = selectOrGenerateDivByClass(_classNameScrollbarHandle, scrollbar);
  4032. if (!_domExists && !destroy) {
  4033. scrollbar.append(track);
  4034. track.append(handle);
  4035. }
  4036. return {
  4037. _scrollbar: scrollbar,
  4038. _track: track,
  4039. _handle: handle
  4040. };
  4041. };
  4042. function resetScrollbarDOM(isHorizontal) {
  4043. var scrollbarVars = getScrollbarVars(isHorizontal);
  4044. var scrollbar = scrollbarVars._scrollbar;
  4045. var track = scrollbarVars._track;
  4046. var handle = scrollbarVars._handle;
  4047. if (_domExists && _initialized) {
  4048. each([scrollbar, track, handle], function (i, elm) {
  4049. removeClass(elm.removeAttr(LEXICON.s), _classNamesDynamicDestroy);
  4050. });
  4051. }
  4052. else {
  4053. remove(scrollbar || selectOrGenerateScrollbarDOM(isHorizontal)._scrollbar);
  4054. }
  4055. }
  4056. var horizontalElements;
  4057. var verticalElements;
  4058. if (!destroy) {
  4059. horizontalElements = selectOrGenerateScrollbarDOM(true);
  4060. verticalElements = selectOrGenerateScrollbarDOM();
  4061. _scrollbarHorizontalElement = horizontalElements._scrollbar;
  4062. _scrollbarHorizontalTrackElement = horizontalElements._track;
  4063. _scrollbarHorizontalHandleElement = horizontalElements._handle;
  4064. _scrollbarVerticalElement = verticalElements._scrollbar;
  4065. _scrollbarVerticalTrackElement = verticalElements._track;
  4066. _scrollbarVerticalHandleElement = verticalElements._handle;
  4067. if (!_domExists) {
  4068. _paddingElement.after(_scrollbarVerticalElement);
  4069. _paddingElement.after(_scrollbarHorizontalElement);
  4070. }
  4071. }
  4072. else {
  4073. resetScrollbarDOM(true);
  4074. resetScrollbarDOM();
  4075. }
  4076. }
  4077. /**
  4078. * Initializes all scrollbar interactivity events. (track and handle dragging, clicking, scrolling)
  4079. * @param isHorizontal True if the target scrollbar is the horizontal scrollbar, false if the target scrollbar is the vertical scrollbar.
  4080. */
  4081. function setupScrollbarEvents(isHorizontal) {
  4082. var scrollbarVars = getScrollbarVars(isHorizontal);
  4083. var scrollbarVarsInfo = scrollbarVars._info;
  4084. var insideIFrame = _windowElementNative.top !== _windowElementNative;
  4085. var xy = scrollbarVars._x_y;
  4086. var XY = scrollbarVars._X_Y;
  4087. var scroll = _strScroll + scrollbarVars._Left_Top;
  4088. var strActive = 'active';
  4089. var strSnapHandle = 'snapHandle';
  4090. var scrollDurationFactor = 1;
  4091. var increaseDecreaseScrollAmountKeyCodes = [16, 17]; //shift, ctrl
  4092. var trackTimeout;
  4093. var mouseDownScroll;
  4094. var mouseDownOffset;
  4095. var mouseDownInvertedScale;
  4096. function getPointerPosition(event) {
  4097. return _msieVersion && insideIFrame ? event['screen' + XY] : COMPATIBILITY.page(event)[xy]; //use screen coordinates in EDGE & IE because the page values are incorrect in frames.
  4098. }
  4099. function getPreparedScrollbarsOption(name) {
  4100. return _currentPreparedOptions.scrollbars[name];
  4101. }
  4102. function increaseTrackScrollAmount() {
  4103. scrollDurationFactor = 0.5;
  4104. }
  4105. function decreaseTrackScrollAmount() {
  4106. scrollDurationFactor = 1;
  4107. }
  4108. function documentKeyDown(event) {
  4109. if (inArray(event.keyCode, increaseDecreaseScrollAmountKeyCodes) > -1)
  4110. increaseTrackScrollAmount();
  4111. }
  4112. function documentKeyUp(event) {
  4113. if (inArray(event.keyCode, increaseDecreaseScrollAmountKeyCodes) > -1)
  4114. decreaseTrackScrollAmount();
  4115. }
  4116. function onMouseTouchDownContinue(event) {
  4117. var originalEvent = event.originalEvent || event;
  4118. var isTouchEvent = originalEvent.touches !== undefined;
  4119. return _sleeping || _destroyed || nativeOverlayScrollbarsAreActive() || !_scrollbarsDragScrollingCache || (isTouchEvent && !getPreparedScrollbarsOption('touchSupport')) ? false : COMPATIBILITY.mBtn(event) === 1 || isTouchEvent;
  4120. }
  4121. function documentDragMove(event) {
  4122. if (onMouseTouchDownContinue(event)) {
  4123. var trackLength = scrollbarVarsInfo._trackLength;
  4124. var handleLength = scrollbarVarsInfo._handleLength;
  4125. var scrollRange = scrollbarVarsInfo._maxScroll;
  4126. var scrollRaw = (getPointerPosition(event) - mouseDownOffset) * mouseDownInvertedScale;
  4127. var scrollDeltaPercent = scrollRaw / (trackLength - handleLength);
  4128. var scrollDelta = (scrollRange * scrollDeltaPercent);
  4129. scrollDelta = isFinite(scrollDelta) ? scrollDelta : 0;
  4130. if (_isRTL && isHorizontal && !_rtlScrollBehavior.i)
  4131. scrollDelta *= -1;
  4132. _viewportElement[scroll](MATH.round(mouseDownScroll + scrollDelta));
  4133. if (_scrollbarsHandlesDefineScrollPos)
  4134. refreshScrollbarHandleOffset(isHorizontal, mouseDownScroll + scrollDelta);
  4135. if (!_supportPassiveEvents)
  4136. COMPATIBILITY.prvD(event);
  4137. }
  4138. else
  4139. documentMouseTouchUp(event);
  4140. }
  4141. function documentMouseTouchUp(event) {
  4142. event = event || event.originalEvent;
  4143. setupResponsiveEventListener(_documentElement,
  4144. [_strMouseTouchMoveEvent, _strMouseTouchUpEvent, _strKeyDownEvent, _strKeyUpEvent, _strSelectStartEvent],
  4145. [documentDragMove, documentMouseTouchUp, documentKeyDown, documentKeyUp, documentOnSelectStart],
  4146. true);
  4147. if (_scrollbarsHandlesDefineScrollPos)
  4148. refreshScrollbarHandleOffset(isHorizontal, true);
  4149. _scrollbarsHandlesDefineScrollPos = false;
  4150. removeClass(_bodyElement, _classNameDragging);
  4151. removeClass(scrollbarVars._handle, strActive);
  4152. removeClass(scrollbarVars._track, strActive);
  4153. removeClass(scrollbarVars._scrollbar, strActive);
  4154. mouseDownScroll = undefined;
  4155. mouseDownOffset = undefined;
  4156. mouseDownInvertedScale = 1;
  4157. decreaseTrackScrollAmount();
  4158. if (trackTimeout !== undefined) {
  4159. _base.scrollStop();
  4160. clearTimeout(trackTimeout);
  4161. trackTimeout = undefined;
  4162. }
  4163. if (event) {
  4164. var rect = _hostElementNative[LEXICON.bCR]();
  4165. var mouseInsideHost = event.clientX >= rect.left && event.clientX <= rect.right && event.clientY >= rect.top && event.clientY <= rect.bottom;
  4166. //if mouse is outside host element
  4167. if (!mouseInsideHost)
  4168. hostOnMouseLeave();
  4169. if (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove)
  4170. refreshScrollbarsAutoHide(false);
  4171. }
  4172. }
  4173. function onHandleMouseTouchDown(event) {
  4174. if (onMouseTouchDownContinue(event))
  4175. onHandleMouseTouchDownAction(event);
  4176. }
  4177. function onHandleMouseTouchDownAction(event) {
  4178. mouseDownScroll = _viewportElement[scroll]();
  4179. mouseDownScroll = isNaN(mouseDownScroll) ? 0 : mouseDownScroll;
  4180. if (_isRTL && isHorizontal && !_rtlScrollBehavior.n || !_isRTL)
  4181. mouseDownScroll = mouseDownScroll < 0 ? 0 : mouseDownScroll;
  4182. mouseDownInvertedScale = getHostElementInvertedScale()[xy];
  4183. mouseDownOffset = getPointerPosition(event);
  4184. _scrollbarsHandlesDefineScrollPos = !getPreparedScrollbarsOption(strSnapHandle);
  4185. addClass(_bodyElement, _classNameDragging);
  4186. addClass(scrollbarVars._handle, strActive);
  4187. addClass(scrollbarVars._scrollbar, strActive);
  4188. setupResponsiveEventListener(_documentElement,
  4189. [_strMouseTouchMoveEvent, _strMouseTouchUpEvent, _strSelectStartEvent],
  4190. [documentDragMove, documentMouseTouchUp, documentOnSelectStart]);
  4191. if (_msieVersion || !_documentMixed)
  4192. COMPATIBILITY.prvD(event);
  4193. COMPATIBILITY.stpP(event);
  4194. }
  4195. function onTrackMouseTouchDown(event) {
  4196. if (onMouseTouchDownContinue(event)) {
  4197. var scrollDistance = MATH.round(_viewportSize[scrollbarVars._w_h]);
  4198. var trackOffset = scrollbarVars._track.offset()[scrollbarVars._left_top];
  4199. var ctrlKey = event.ctrlKey;
  4200. var instantScroll = event.shiftKey;
  4201. var instantScrollTransition = instantScroll && ctrlKey;
  4202. var isFirstIteration = true;
  4203. var easing = 'linear';
  4204. var decreaseScroll;
  4205. var finishedCondition;
  4206. var scrollActionFinsished = function (transition) {
  4207. if (_scrollbarsHandlesDefineScrollPos)
  4208. refreshScrollbarHandleOffset(isHorizontal, transition);
  4209. };
  4210. var scrollActionInstantFinished = function () {
  4211. scrollActionFinsished();
  4212. onHandleMouseTouchDownAction(event);
  4213. };
  4214. var scrollAction = function () {
  4215. if (!_destroyed) {
  4216. var mouseOffset = (mouseDownOffset - trackOffset) * mouseDownInvertedScale;
  4217. var handleOffset = scrollbarVarsInfo._handleOffset;
  4218. var trackLength = scrollbarVarsInfo._trackLength;
  4219. var handleLength = scrollbarVarsInfo._handleLength;
  4220. var scrollRange = scrollbarVarsInfo._maxScroll;
  4221. var currScroll = scrollbarVarsInfo._currentScroll;
  4222. var scrollDuration = 270 * scrollDurationFactor;
  4223. var timeoutDelay = isFirstIteration ? MATH.max(400, scrollDuration) : scrollDuration;
  4224. var instantScrollPosition = scrollRange * ((mouseOffset - (handleLength / 2)) / (trackLength - handleLength)); // 100% * positionPercent
  4225. var rtlIsNormal = _isRTL && isHorizontal && ((!_rtlScrollBehavior.i && !_rtlScrollBehavior.n) || _normalizeRTLCache);
  4226. var decreaseScrollCondition = rtlIsNormal ? handleOffset < mouseOffset : handleOffset > mouseOffset;
  4227. var scrollObj = {};
  4228. var animationObj = {
  4229. easing: easing,
  4230. step: function (now) {
  4231. if (_scrollbarsHandlesDefineScrollPos) {
  4232. _viewportElement[scroll](now); //https://github.com/jquery/jquery/issues/4340
  4233. refreshScrollbarHandleOffset(isHorizontal, now);
  4234. }
  4235. }
  4236. };
  4237. instantScrollPosition = isFinite(instantScrollPosition) ? instantScrollPosition : 0;
  4238. instantScrollPosition = _isRTL && isHorizontal && !_rtlScrollBehavior.i ? (scrollRange - instantScrollPosition) : instantScrollPosition;
  4239. //_base.scrollStop();
  4240. if (instantScroll) {
  4241. _viewportElement[scroll](instantScrollPosition); //scroll instantly to new position
  4242. if (instantScrollTransition) {
  4243. //get the scroll position after instant scroll (in case CSS Snap Points are used) to get the correct snapped scroll position
  4244. //and the animation stops at the correct point
  4245. instantScrollPosition = _viewportElement[scroll]();
  4246. //scroll back to the position before instant scrolling so animation can be performed
  4247. _viewportElement[scroll](currScroll);
  4248. instantScrollPosition = rtlIsNormal && _rtlScrollBehavior.i ? (scrollRange - instantScrollPosition) : instantScrollPosition;
  4249. instantScrollPosition = rtlIsNormal && _rtlScrollBehavior.n ? -instantScrollPosition : instantScrollPosition;
  4250. scrollObj[xy] = instantScrollPosition;
  4251. _base.scroll(scrollObj, extendDeep(animationObj, {
  4252. duration: 130,
  4253. complete: scrollActionInstantFinished
  4254. }));
  4255. }
  4256. else
  4257. scrollActionInstantFinished();
  4258. }
  4259. else {
  4260. decreaseScroll = isFirstIteration ? decreaseScrollCondition : decreaseScroll;
  4261. finishedCondition = rtlIsNormal
  4262. ? (decreaseScroll ? handleOffset + handleLength >= mouseOffset : handleOffset <= mouseOffset)
  4263. : (decreaseScroll ? handleOffset <= mouseOffset : handleOffset + handleLength >= mouseOffset);
  4264. if (finishedCondition) {
  4265. clearTimeout(trackTimeout);
  4266. _base.scrollStop();
  4267. trackTimeout = undefined;
  4268. scrollActionFinsished(true);
  4269. }
  4270. else {
  4271. trackTimeout = setTimeout(scrollAction, timeoutDelay);
  4272. scrollObj[xy] = (decreaseScroll ? '-=' : '+=') + scrollDistance;
  4273. _base.scroll(scrollObj, extendDeep(animationObj, {
  4274. duration: scrollDuration
  4275. }));
  4276. }
  4277. isFirstIteration = false;
  4278. }
  4279. }
  4280. };
  4281. if (ctrlKey)
  4282. increaseTrackScrollAmount();
  4283. mouseDownInvertedScale = getHostElementInvertedScale()[xy];
  4284. mouseDownOffset = COMPATIBILITY.page(event)[xy];
  4285. _scrollbarsHandlesDefineScrollPos = !getPreparedScrollbarsOption(strSnapHandle);
  4286. addClass(_bodyElement, _classNameDragging);
  4287. addClass(scrollbarVars._track, strActive);
  4288. addClass(scrollbarVars._scrollbar, strActive);
  4289. setupResponsiveEventListener(_documentElement,
  4290. [_strMouseTouchUpEvent, _strKeyDownEvent, _strKeyUpEvent, _strSelectStartEvent],
  4291. [documentMouseTouchUp, documentKeyDown, documentKeyUp, documentOnSelectStart]);
  4292. scrollAction();
  4293. COMPATIBILITY.prvD(event);
  4294. COMPATIBILITY.stpP(event);
  4295. }
  4296. }
  4297. function onTrackMouseTouchEnter(event) {
  4298. //make sure both scrollbars will stay visible if one scrollbar is hovered if autoHide is "scroll" or "move".
  4299. _scrollbarsHandleHovered = true;
  4300. if (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove)
  4301. refreshScrollbarsAutoHide(true);
  4302. }
  4303. function onTrackMouseTouchLeave(event) {
  4304. _scrollbarsHandleHovered = false;
  4305. if (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove)
  4306. refreshScrollbarsAutoHide(false);
  4307. }
  4308. function onScrollbarMouseTouchDown(event) {
  4309. COMPATIBILITY.stpP(event);
  4310. }
  4311. addDestroyEventListener(scrollbarVars._handle,
  4312. _strMouseTouchDownEvent,
  4313. onHandleMouseTouchDown);
  4314. addDestroyEventListener(scrollbarVars._track,
  4315. [_strMouseTouchDownEvent, _strMouseTouchEnter, _strMouseTouchLeave],
  4316. [onTrackMouseTouchDown, onTrackMouseTouchEnter, onTrackMouseTouchLeave]);
  4317. addDestroyEventListener(scrollbarVars._scrollbar,
  4318. _strMouseTouchDownEvent,
  4319. onScrollbarMouseTouchDown);
  4320. if (_supportTransition) {
  4321. addDestroyEventListener(scrollbarVars._scrollbar, _strTransitionEndEvent, function (event) {
  4322. if (event.target !== scrollbarVars._scrollbar[0])
  4323. return;
  4324. refreshScrollbarHandleLength(isHorizontal);
  4325. refreshScrollbarHandleOffset(isHorizontal);
  4326. });
  4327. }
  4328. }
  4329. /**
  4330. * Shows or hides the given scrollbar and applied a class name which indicates if the scrollbar is scrollable or not.
  4331. * @param isHorizontal True if the horizontal scrollbar is the target, false if the vertical scrollbar is the target.
  4332. * @param shallBeVisible True if the scrollbar shall be shown, false if hidden.
  4333. * @param canScroll True if the scrollbar is scrollable, false otherwise.
  4334. */
  4335. function refreshScrollbarAppearance(isHorizontal, shallBeVisible, canScroll) {
  4336. var scrollbarClassName = isHorizontal ? _classNameHostScrollbarHorizontalHidden : _classNameHostScrollbarVerticalHidden;
  4337. var scrollbarElement = isHorizontal ? _scrollbarHorizontalElement : _scrollbarVerticalElement;
  4338. if (shallBeVisible)
  4339. removeClass(_hostElement, scrollbarClassName);
  4340. else
  4341. addClass(_hostElement, scrollbarClassName);
  4342. if (canScroll)
  4343. removeClass(scrollbarElement, _classNameScrollbarUnusable);
  4344. else
  4345. addClass(scrollbarElement, _classNameScrollbarUnusable);
  4346. }
  4347. /**
  4348. * Autoshows / autohides both scrollbars with.
  4349. * @param shallBeVisible True if the scrollbars shall be autoshown (only the case if they are hidden by a autohide), false if the shall be auto hidden.
  4350. * @param delayfree True if the scrollbars shall be hidden without a delay, false or undefined otherwise.
  4351. */
  4352. function refreshScrollbarsAutoHide(shallBeVisible, delayfree) {
  4353. clearTimeout(_scrollbarsAutoHideTimeoutId);
  4354. if (shallBeVisible) {
  4355. //if(_hasOverflowCache.x && _hideOverflowCache.xs)
  4356. removeClass(_scrollbarHorizontalElement, _classNameScrollbarAutoHidden);
  4357. //if(_hasOverflowCache.y && _hideOverflowCache.ys)
  4358. removeClass(_scrollbarVerticalElement, _classNameScrollbarAutoHidden);
  4359. }
  4360. else {
  4361. var anyActive;
  4362. var strActive = 'active';
  4363. var hide = function () {
  4364. if (!_scrollbarsHandleHovered && !_destroyed) {
  4365. anyActive = _scrollbarHorizontalHandleElement.hasClass(strActive) || _scrollbarVerticalHandleElement.hasClass(strActive);
  4366. if (!anyActive && (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove || _scrollbarsAutoHideLeave))
  4367. addClass(_scrollbarHorizontalElement, _classNameScrollbarAutoHidden);
  4368. if (!anyActive && (_scrollbarsAutoHideScroll || _scrollbarsAutoHideMove || _scrollbarsAutoHideLeave))
  4369. addClass(_scrollbarVerticalElement, _classNameScrollbarAutoHidden);
  4370. }
  4371. };
  4372. if (_scrollbarsAutoHideDelay > 0 && delayfree !== true)
  4373. _scrollbarsAutoHideTimeoutId = setTimeout(hide, _scrollbarsAutoHideDelay);
  4374. else
  4375. hide();
  4376. }
  4377. }
  4378. /**
  4379. * Refreshes the handle length of the given scrollbar.
  4380. * @param isHorizontal True if the horizontal scrollbar handle shall be refreshed, false if the vertical one shall be refreshed.
  4381. */
  4382. function refreshScrollbarHandleLength(isHorizontal) {
  4383. var handleCSS = {};
  4384. var scrollbarVars = getScrollbarVars(isHorizontal);
  4385. var scrollbarVarsInfo = scrollbarVars._info;
  4386. var digit = 1000000;
  4387. //get and apply intended handle length
  4388. var handleRatio = MATH.min(1, (_hostSizeCache[scrollbarVars._w_h] - (_paddingAbsoluteCache ? (isHorizontal ? _paddingX : _paddingY) : 0)) / _contentScrollSizeCache[scrollbarVars._w_h]);
  4389. handleCSS[scrollbarVars._width_height] = (MATH.floor(handleRatio * 100 * digit) / digit) + '%'; //the last * digit / digit is for flooring to the 4th digit
  4390. if (!nativeOverlayScrollbarsAreActive())
  4391. scrollbarVars._handle.css(handleCSS);
  4392. //measure the handle length to respect min & max length
  4393. scrollbarVarsInfo._handleLength = scrollbarVars._handle[0]['offset' + scrollbarVars._Width_Height];
  4394. scrollbarVarsInfo._handleLengthRatio = handleRatio;
  4395. }
  4396. /**
  4397. * Refreshes the handle offset of the given scrollbar.
  4398. * @param isHorizontal True if the horizontal scrollbar handle shall be refreshed, false if the vertical one shall be refreshed.
  4399. * @param scrollOrTransition The scroll position of the given scrollbar axis to which the handle shall be moved or a boolean which indicates whether a transition shall be applied. If undefined or boolean if the current scroll-offset is taken. (if isHorizontal ? scrollLeft : scrollTop)
  4400. */
  4401. function refreshScrollbarHandleOffset(isHorizontal, scrollOrTransition) {
  4402. var transition = type(scrollOrTransition) == TYPES.b;
  4403. var transitionDuration = 250;
  4404. var isRTLisHorizontal = _isRTL && isHorizontal;
  4405. var scrollbarVars = getScrollbarVars(isHorizontal);
  4406. var scrollbarVarsInfo = scrollbarVars._info;
  4407. var strTranslateBrace = 'translate(';
  4408. var strTransform = VENDORS._cssProperty('transform');
  4409. var strTransition = VENDORS._cssProperty('transition');
  4410. var nativeScroll = isHorizontal ? _viewportElement[_strScrollLeft]() : _viewportElement[_strScrollTop]();
  4411. var currentScroll = scrollOrTransition === undefined || transition ? nativeScroll : scrollOrTransition;
  4412. //measure the handle length to respect min & max length
  4413. var handleLength = scrollbarVarsInfo._handleLength;
  4414. var trackLength = scrollbarVars._track[0]['offset' + scrollbarVars._Width_Height];
  4415. var handleTrackDiff = trackLength - handleLength;
  4416. var handleCSS = {};
  4417. var transformOffset;
  4418. var translateValue;
  4419. //DONT use the variable '_contentScrollSizeCache[scrollbarVars._w_h]' instead of '_viewportElement[0]['scroll' + scrollbarVars._Width_Height]'
  4420. // because its a bit behind during the small delay when content size updates
  4421. //(delay = mutationObserverContentLag, if its 0 then this var could be used)
  4422. var maxScroll = (_viewportElementNative[_strScroll + scrollbarVars._Width_Height] - _viewportElementNative['client' + scrollbarVars._Width_Height]) * (_rtlScrollBehavior.n && isRTLisHorizontal ? -1 : 1); //* -1 if rtl scroll max is negative
  4423. var getScrollRatio = function (base) {
  4424. return isNaN(base / maxScroll) ? 0 : MATH.max(0, MATH.min(1, base / maxScroll));
  4425. };
  4426. var getHandleOffset = function (scrollRatio) {
  4427. var offset = handleTrackDiff * scrollRatio;
  4428. offset = isNaN(offset) ? 0 : offset;
  4429. offset = (isRTLisHorizontal && !_rtlScrollBehavior.i) ? (trackLength - handleLength - offset) : offset;
  4430. offset = MATH.max(0, offset);
  4431. return offset;
  4432. };
  4433. var scrollRatio = getScrollRatio(nativeScroll);
  4434. var unsnappedScrollRatio = getScrollRatio(currentScroll);
  4435. var handleOffset = getHandleOffset(unsnappedScrollRatio);
  4436. var snappedHandleOffset = getHandleOffset(scrollRatio);
  4437. scrollbarVarsInfo._maxScroll = maxScroll;
  4438. scrollbarVarsInfo._currentScroll = nativeScroll;
  4439. scrollbarVarsInfo._currentScrollRatio = scrollRatio;
  4440. if (_supportTransform) {
  4441. transformOffset = isRTLisHorizontal ? -(trackLength - handleLength - handleOffset) : handleOffset; //in px
  4442. //transformOffset = (transformOffset / trackLength * 100) * (trackLength / handleLength); //in %
  4443. translateValue = isHorizontal ? strTranslateBrace + transformOffset + 'px, 0)' : strTranslateBrace + '0, ' + transformOffset + 'px)';
  4444. handleCSS[strTransform] = translateValue;
  4445. //apply or clear up transition
  4446. if (_supportTransition)
  4447. handleCSS[strTransition] = transition && MATH.abs(handleOffset - scrollbarVarsInfo._handleOffset) > 1 ? getCSSTransitionString(scrollbarVars._handle) + ', ' + (strTransform + _strSpace + transitionDuration + 'ms') : _strEmpty;
  4448. }
  4449. else
  4450. handleCSS[scrollbarVars._left_top] = handleOffset;
  4451. //only apply css if offset has changed and overflow exists.
  4452. if (!nativeOverlayScrollbarsAreActive()) {
  4453. scrollbarVars._handle.css(handleCSS);
  4454. //clear up transition
  4455. if (_supportTransform && _supportTransition && transition) {
  4456. scrollbarVars._handle.one(_strTransitionEndEvent, function () {
  4457. if (!_destroyed)
  4458. scrollbarVars._handle.css(strTransition, _strEmpty);
  4459. });
  4460. }
  4461. }
  4462. scrollbarVarsInfo._handleOffset = handleOffset;
  4463. scrollbarVarsInfo._snappedHandleOffset = snappedHandleOffset;
  4464. scrollbarVarsInfo._trackLength = trackLength;
  4465. }
  4466. /**
  4467. * Refreshes the interactivity of the given scrollbar element.
  4468. * @param isTrack True if the track element is the target, false if the handle element is the target.
  4469. * @param value True for interactivity false for no interactivity.
  4470. */
  4471. function refreshScrollbarsInteractive(isTrack, value) {
  4472. var action = value ? 'removeClass' : 'addClass';
  4473. var element1 = isTrack ? _scrollbarHorizontalTrackElement : _scrollbarHorizontalHandleElement;
  4474. var element2 = isTrack ? _scrollbarVerticalTrackElement : _scrollbarVerticalHandleElement;
  4475. var className = isTrack ? _classNameScrollbarTrackOff : _classNameScrollbarHandleOff;
  4476. element1[action](className);
  4477. element2[action](className);
  4478. }
  4479. /**
  4480. * Returns a object which is used for fast access for specific variables.
  4481. * @param isHorizontal True if the horizontal scrollbar vars shall be accessed, false if the vertical scrollbar vars shall be accessed.
  4482. * @returns {{wh: string, WH: string, lt: string, _wh: string, _lt: string, t: *, h: *, c: {}, s: *}}
  4483. */
  4484. function getScrollbarVars(isHorizontal) {
  4485. return {
  4486. _width_height: isHorizontal ? _strWidth : _strHeight,
  4487. _Width_Height: isHorizontal ? 'Width' : 'Height',
  4488. _left_top: isHorizontal ? _strLeft : _strTop,
  4489. _Left_Top: isHorizontal ? 'Left' : 'Top',
  4490. _x_y: isHorizontal ? _strX : _strY,
  4491. _X_Y: isHorizontal ? 'X' : 'Y',
  4492. _w_h: isHorizontal ? 'w' : 'h',
  4493. _l_t: isHorizontal ? 'l' : 't',
  4494. _track: isHorizontal ? _scrollbarHorizontalTrackElement : _scrollbarVerticalTrackElement,
  4495. _handle: isHorizontal ? _scrollbarHorizontalHandleElement : _scrollbarVerticalHandleElement,
  4496. _scrollbar: isHorizontal ? _scrollbarHorizontalElement : _scrollbarVerticalElement,
  4497. _info: isHorizontal ? _scrollHorizontalInfo : _scrollVerticalInfo
  4498. };
  4499. }
  4500. //==== Scrollbar Corner ====//
  4501. /**
  4502. * Builds or destroys the scrollbar corner DOM element.
  4503. * @param destroy Indicates whether the DOM shall be build or destroyed.
  4504. */
  4505. function setupScrollbarCornerDOM(destroy) {
  4506. _scrollbarCornerElement = _scrollbarCornerElement || selectOrGenerateDivByClass(_classNameScrollbarCorner, true);
  4507. if (!destroy) {
  4508. if (!_domExists) {
  4509. _hostElement.append(_scrollbarCornerElement);
  4510. }
  4511. }
  4512. else {
  4513. if (_domExists && _initialized) {
  4514. removeClass(_scrollbarCornerElement.removeAttr(LEXICON.s), _classNamesDynamicDestroy);
  4515. }
  4516. else {
  4517. remove(_scrollbarCornerElement);
  4518. }
  4519. }
  4520. }
  4521. /**
  4522. * Initializes all scrollbar corner interactivity events.
  4523. */
  4524. function setupScrollbarCornerEvents() {
  4525. var insideIFrame = _windowElementNative.top !== _windowElementNative;
  4526. var mouseDownPosition = {};
  4527. var mouseDownSize = {};
  4528. var mouseDownInvertedScale = {};
  4529. var reconnectMutationObserver;
  4530. function documentDragMove(event) {
  4531. if (onMouseTouchDownContinue(event)) {
  4532. var pageOffset = getCoordinates(event);
  4533. var hostElementCSS = {};
  4534. if (_resizeHorizontal || _resizeBoth)
  4535. hostElementCSS[_strWidth] = (mouseDownSize.w + (pageOffset.x - mouseDownPosition.x) * mouseDownInvertedScale.x);
  4536. if (_resizeVertical || _resizeBoth)
  4537. hostElementCSS[_strHeight] = (mouseDownSize.h + (pageOffset.y - mouseDownPosition.y) * mouseDownInvertedScale.y);
  4538. _hostElement.css(hostElementCSS);
  4539. COMPATIBILITY.stpP(event);
  4540. }
  4541. else {
  4542. documentMouseTouchUp(event);
  4543. }
  4544. }
  4545. function documentMouseTouchUp(event) {
  4546. var eventIsTrusted = event !== undefined;
  4547. setupResponsiveEventListener(_documentElement,
  4548. [_strSelectStartEvent, _strMouseTouchMoveEvent, _strMouseTouchUpEvent],
  4549. [documentOnSelectStart, documentDragMove, documentMouseTouchUp],
  4550. true);
  4551. removeClass(_bodyElement, _classNameDragging);
  4552. if (_scrollbarCornerElement.releaseCapture)
  4553. _scrollbarCornerElement.releaseCapture();
  4554. if (eventIsTrusted) {
  4555. if (reconnectMutationObserver)
  4556. connectMutationObservers();
  4557. _base.update(_strAuto);
  4558. }
  4559. reconnectMutationObserver = false;
  4560. }
  4561. function onMouseTouchDownContinue(event) {
  4562. var originalEvent = event.originalEvent || event;
  4563. var isTouchEvent = originalEvent.touches !== undefined;
  4564. return _sleeping || _destroyed ? false : COMPATIBILITY.mBtn(event) === 1 || isTouchEvent;
  4565. }
  4566. function getCoordinates(event) {
  4567. return _msieVersion && insideIFrame ? { x: event.screenX, y: event.screenY } : COMPATIBILITY.page(event);
  4568. }
  4569. addDestroyEventListener(_scrollbarCornerElement, _strMouseTouchDownEvent, function (event) {
  4570. if (onMouseTouchDownContinue(event) && !_resizeNone) {
  4571. if (_mutationObserversConnected) {
  4572. reconnectMutationObserver = true;
  4573. disconnectMutationObservers();
  4574. }
  4575. mouseDownPosition = getCoordinates(event);
  4576. mouseDownSize.w = _hostElementNative[LEXICON.oW] - (!_isBorderBox ? _paddingX : 0);
  4577. mouseDownSize.h = _hostElementNative[LEXICON.oH] - (!_isBorderBox ? _paddingY : 0);
  4578. mouseDownInvertedScale = getHostElementInvertedScale();
  4579. setupResponsiveEventListener(_documentElement,
  4580. [_strSelectStartEvent, _strMouseTouchMoveEvent, _strMouseTouchUpEvent],
  4581. [documentOnSelectStart, documentDragMove, documentMouseTouchUp]);
  4582. addClass(_bodyElement, _classNameDragging);
  4583. if (_scrollbarCornerElement.setCapture)
  4584. _scrollbarCornerElement.setCapture();
  4585. COMPATIBILITY.prvD(event);
  4586. COMPATIBILITY.stpP(event);
  4587. }
  4588. });
  4589. }
  4590. //==== Utils ====//
  4591. /**
  4592. * Calls the callback with the given name. The Context of this callback is always _base (this).
  4593. * @param name The name of the target which shall be called.
  4594. * @param args The args with which the callback shall be called.
  4595. */
  4596. function dispatchCallback(name, args) {
  4597. if (_initialized) {
  4598. var callback = _currentPreparedOptions.callbacks[name];
  4599. var extensionOnName = name;
  4600. var ext;
  4601. if (extensionOnName.substr(0, 2) === 'on')
  4602. extensionOnName = extensionOnName.substr(2, 1).toLowerCase() + extensionOnName.substr(3);
  4603. if (type(callback) == TYPES.f)
  4604. callback.call(_base, args);
  4605. each(_extensions, function () {
  4606. ext = this;
  4607. if (type(ext.on) == TYPES.f)
  4608. ext.on(extensionOnName, args);
  4609. });
  4610. }
  4611. else if (!_destroyed)
  4612. _callbacksInitQeueue.push({ n: name, a: args });
  4613. }
  4614. /**
  4615. * Sets the "top, right, bottom, left" properties, with a given prefix, of the given css object.
  4616. * @param targetCSSObject The css object to which the values shall be applied.
  4617. * @param prefix The prefix of the "top, right, bottom, left" css properties. (example: 'padding-' is a valid prefix)
  4618. * @param values A array of values which shall be applied to the "top, right, bottom, left" -properties. The array order is [top, right, bottom, left].
  4619. * If this argument is undefined the value '' (empty string) will be applied to all properties.
  4620. */
  4621. function setTopRightBottomLeft(targetCSSObject, prefix, values) {
  4622. if (values === undefined)
  4623. values = [_strEmpty, _strEmpty, _strEmpty, _strEmpty];
  4624. targetCSSObject[prefix + _strTop] = values[0];
  4625. targetCSSObject[prefix + _strRight] = values[1];
  4626. targetCSSObject[prefix + _strBottom] = values[2];
  4627. targetCSSObject[prefix + _strLeft] = values[3];
  4628. }
  4629. /**
  4630. * Returns the computed CSS transition string from the given element.
  4631. * @param element The element from which the transition string shall be returned.
  4632. * @returns {string} The CSS transition string from the given element.
  4633. */
  4634. function getCSSTransitionString(element) {
  4635. var transitionStr = VENDORS._cssProperty('transition');
  4636. var assembledValue = element.css(transitionStr);
  4637. if (assembledValue)
  4638. return assembledValue;
  4639. var regExpString = '\\s*(' + '([^,(]+(\\(.+?\\))?)+' + ')[\\s,]*';
  4640. var regExpMain = new RegExp(regExpString);
  4641. var regExpValidate = new RegExp('^(' + regExpString + ')+$');
  4642. var properties = 'property duration timing-function delay'.split(' ');
  4643. var result = [];
  4644. var strResult;
  4645. var valueArray;
  4646. var i = 0;
  4647. var j;
  4648. var splitCssStyleByComma = function (str) {
  4649. strResult = [];
  4650. if (!str.match(regExpValidate))
  4651. return str;
  4652. while (str.match(regExpMain)) {
  4653. strResult.push(RegExp.$1);
  4654. str = str.replace(regExpMain, _strEmpty);
  4655. }
  4656. return strResult;
  4657. };
  4658. for (; i < properties[LEXICON.l]; i++) {
  4659. valueArray = splitCssStyleByComma(element.css(transitionStr + '-' + properties[i]));
  4660. for (j = 0; j < valueArray[LEXICON.l]; j++)
  4661. result[j] = (result[j] ? result[j] + _strSpace : _strEmpty) + valueArray[j];
  4662. }
  4663. return result.join(', ');
  4664. }
  4665. /**
  4666. * Calculates the host-elements inverted scale. (invertedScale = 1 / scale)
  4667. * @returns {{x: number, y: number}} The scale of the host-element.
  4668. */
  4669. function getHostElementInvertedScale() {
  4670. var rect = _paddingElementNative[LEXICON.bCR]();
  4671. return {
  4672. x: _supportTransform ? 1 / (MATH.round(rect.width) / _paddingElementNative[LEXICON.oW]) || 1 : 1,
  4673. y: _supportTransform ? 1 / (MATH.round(rect.height) / _paddingElementNative[LEXICON.oH]) || 1 : 1
  4674. };
  4675. }
  4676. /**
  4677. * Checks whether the given object is a HTMLElement.
  4678. * @param o The object which shall be checked.
  4679. * @returns {boolean} True the given object is a HTMLElement, false otherwise.
  4680. */
  4681. function isHTMLElement(o) {
  4682. var strOwnerDocument = 'ownerDocument';
  4683. var strHTMLElement = 'HTMLElement';
  4684. var wnd = o && o[strOwnerDocument] ? (o[strOwnerDocument].parentWindow || window) : window;
  4685. return (
  4686. typeof wnd[strHTMLElement] == TYPES.o ? o instanceof wnd[strHTMLElement] : //DOM2
  4687. o && typeof o == TYPES.o && o !== null && o.nodeType === 1 && typeof o.nodeName == TYPES.s
  4688. );
  4689. }
  4690. /**
  4691. * Compares 2 arrays and returns the differences between them as a array.
  4692. * @param a1 The first array which shall be compared.
  4693. * @param a2 The second array which shall be compared.
  4694. * @returns {Array} The differences between the two arrays.
  4695. */
  4696. function getArrayDifferences(a1, a2) {
  4697. var a = [];
  4698. var diff = [];
  4699. var i;
  4700. var k;
  4701. for (i = 0; i < a1.length; i++)
  4702. a[a1[i]] = true;
  4703. for (i = 0; i < a2.length; i++) {
  4704. if (a[a2[i]])
  4705. delete a[a2[i]];
  4706. else
  4707. a[a2[i]] = true;
  4708. }
  4709. for (k in a)
  4710. diff.push(k);
  4711. return diff;
  4712. }
  4713. /**
  4714. * Returns Zero or the number to which the value can be parsed.
  4715. * @param value The value which shall be parsed.
  4716. * @param toFloat Indicates whether the number shall be parsed to a float.
  4717. */
  4718. function parseToZeroOrNumber(value, toFloat) {
  4719. var num = toFloat ? parseFloat(value) : parseInt(value, 10);
  4720. return isNaN(num) ? 0 : num;
  4721. }
  4722. /**
  4723. * Gets several information of the textarea and returns them as a object or undefined if the browser doesn't support it.
  4724. * @returns {{cursorRow: Number, cursorCol, rows: Number, cols: number, wRow: number, pos: number, max : number}} or undefined if not supported.
  4725. */
  4726. function getTextareaInfo() {
  4727. //read needed values
  4728. var textareaCursorPosition = _targetElementNative.selectionStart;
  4729. if (textareaCursorPosition === undefined)
  4730. return;
  4731. var textareaValue = _targetElement.val();
  4732. var textareaLength = textareaValue[LEXICON.l];
  4733. var textareaRowSplit = textareaValue.split('\n');
  4734. var textareaLastRow = textareaRowSplit[LEXICON.l];
  4735. var textareaCurrentCursorRowSplit = textareaValue.substr(0, textareaCursorPosition).split('\n');
  4736. var widestRow = 0;
  4737. var textareaLastCol = 0;
  4738. var cursorRow = textareaCurrentCursorRowSplit[LEXICON.l];
  4739. var cursorCol = textareaCurrentCursorRowSplit[textareaCurrentCursorRowSplit[LEXICON.l] - 1][LEXICON.l];
  4740. var rowCols;
  4741. var i;
  4742. //get widest Row and the last column of the textarea
  4743. for (i = 0; i < textareaRowSplit[LEXICON.l]; i++) {
  4744. rowCols = textareaRowSplit[i][LEXICON.l];
  4745. if (rowCols > textareaLastCol) {
  4746. widestRow = i + 1;
  4747. textareaLastCol = rowCols;
  4748. }
  4749. }
  4750. return {
  4751. _cursorRow: cursorRow, //cursorRow
  4752. _cursorColumn: cursorCol, //cursorCol
  4753. _rows: textareaLastRow, //rows
  4754. _columns: textareaLastCol, //cols
  4755. _widestRow: widestRow, //wRow
  4756. _cursorPosition: textareaCursorPosition, //pos
  4757. _cursorMax: textareaLength //max
  4758. };
  4759. }
  4760. /**
  4761. * Determines whether native overlay scrollbars are active.
  4762. * @returns {boolean} True if native overlay scrollbars are active, false otherwise.
  4763. */
  4764. function nativeOverlayScrollbarsAreActive() {
  4765. return (_ignoreOverlayScrollbarHidingCache && (_nativeScrollbarIsOverlaid.x && _nativeScrollbarIsOverlaid.y));
  4766. }
  4767. /**
  4768. * Gets the element which is used to measure the content size.
  4769. * @returns {*} TextareaCover if target element is textarea else the ContentElement.
  4770. */
  4771. function getContentMeasureElement() {
  4772. return _isTextarea ? _textareaCoverElement[0] : _contentElementNative;
  4773. }
  4774. /**
  4775. * Generates a string which represents a HTML div with the given classes or attributes.
  4776. * @param classesOrAttrs The class of the div as string or a object which represents the attributes of the div. (The class attribute can also be written as "className".)
  4777. * @param content The content of the div as string.
  4778. * @returns {string} The concated string which represents a HTML div and its content.
  4779. */
  4780. function generateDiv(classesOrAttrs, content) {
  4781. return '<div ' + (classesOrAttrs ? type(classesOrAttrs) == TYPES.s ?
  4782. 'class="' + classesOrAttrs + '"' :
  4783. (function () {
  4784. var key;
  4785. var attrs = _strEmpty;
  4786. if (FRAMEWORK.isPlainObject(classesOrAttrs)) {
  4787. for (key in classesOrAttrs)
  4788. attrs += (key === 'c' ? 'class' : key) + '="' + classesOrAttrs[key] + '" ';
  4789. }
  4790. return attrs;
  4791. })() :
  4792. _strEmpty) +
  4793. '>' +
  4794. (content || _strEmpty) +
  4795. '</div>';
  4796. }
  4797. /**
  4798. * Selects or generates a div with the given class attribute.
  4799. * @param className The class names (divided by spaces) of the div which shall be selected or generated.
  4800. * @param selectParentOrOnlyChildren The parent element from which of the element shall be selected. (if undefined or boolean its hostElement)
  4801. * If its a boolean it decides whether only the children of the host element shall be selected.
  4802. * @returns {*} The generated or selected element.
  4803. */
  4804. function selectOrGenerateDivByClass(className, selectParentOrOnlyChildren) {
  4805. var onlyChildren = type(selectParentOrOnlyChildren) == TYPES.b;
  4806. var selectParent = onlyChildren ? _hostElement : (selectParentOrOnlyChildren || _hostElement);
  4807. return (_domExists && !selectParent[LEXICON.l])
  4808. ? null
  4809. : _domExists
  4810. ? selectParent[onlyChildren ? 'children' : 'find'](_strDot + className.replace(/\s/g, _strDot)).eq(0)
  4811. : FRAMEWORK(generateDiv(className))
  4812. }
  4813. /**
  4814. * Gets the value of the given property from the given object.
  4815. * @param obj The object from which the property value shall be got.
  4816. * @param path The property of which the value shall be got.
  4817. * @returns {*} Returns the value of the searched property or undefined of the property wasn't found.
  4818. */
  4819. function getObjectPropVal(obj, path) {
  4820. var splits = path.split(_strDot);
  4821. var i = 0;
  4822. var val;
  4823. for (; i < splits.length; i++) {
  4824. if (!obj[LEXICON.hOP](splits[i]))
  4825. return;
  4826. val = obj[splits[i]];
  4827. if (i < splits.length && type(val) == TYPES.o)
  4828. obj = val;
  4829. }
  4830. return val;
  4831. }
  4832. /**
  4833. * Sets the value of the given property from the given object.
  4834. * @param obj The object from which the property value shall be set.
  4835. * @param path The property of which the value shall be set.
  4836. * @param val The value of the property which shall be set.
  4837. */
  4838. function setObjectPropVal(obj, path, val) {
  4839. var splits = path.split(_strDot);
  4840. var splitsLength = splits.length;
  4841. var i = 0;
  4842. var extendObj = {};
  4843. var extendObjRoot = extendObj;
  4844. for (; i < splitsLength; i++)
  4845. extendObj = extendObj[splits[i]] = i + 1 < splitsLength ? {} : val;
  4846. FRAMEWORK.extend(obj, extendObjRoot, true);
  4847. }
  4848. //==== Utils Cache ====//
  4849. /**
  4850. * Compares two values or objects and returns true if they aren't equal.
  4851. * @param current The first value or object which shall be compared.
  4852. * @param cache The second value or object which shall be compared.
  4853. * @param force If true the returned value is always true.
  4854. * @returns {boolean} True if both values or objects aren't equal or force is true, false otherwise.
  4855. */
  4856. function checkCache(current, cache, force) {
  4857. if (force)
  4858. return force;
  4859. if (type(current) == TYPES.o && type(cache) == TYPES.o) {
  4860. for (var prop in current) {
  4861. if (prop !== 'c') {
  4862. if (current[LEXICON.hOP](prop) && cache[LEXICON.hOP](prop)) {
  4863. if (checkCache(current[prop], cache[prop]))
  4864. return true;
  4865. }
  4866. else {
  4867. return true;
  4868. }
  4869. }
  4870. }
  4871. }
  4872. else {
  4873. return current !== cache;
  4874. }
  4875. return false;
  4876. }
  4877. //==== Shortcuts ====//
  4878. /**
  4879. * jQuery extend method shortcut with a appended "true" as first argument.
  4880. */
  4881. function extendDeep() {
  4882. return FRAMEWORK.extend.apply(this, [true].concat([].slice.call(arguments)));
  4883. }
  4884. /**
  4885. * jQuery addClass method shortcut.
  4886. */
  4887. function addClass(el, classes) {
  4888. return _frameworkProto.addClass.call(el, classes);
  4889. }
  4890. /**
  4891. * jQuery removeClass method shortcut.
  4892. */
  4893. function removeClass(el, classes) {
  4894. return _frameworkProto.removeClass.call(el, classes);
  4895. }
  4896. /**
  4897. * jQuery remove method shortcut.
  4898. */
  4899. function remove(el) {
  4900. return _frameworkProto.remove.call(el);
  4901. }
  4902. /**
  4903. * Finds the first child element with the given selector of the given element.
  4904. * @param el The root element from which the selector shall be valid.
  4905. * @param selector The selector of the searched element.
  4906. * @returns {*} The first element which is a child of the given element and matches the givens selector.
  4907. */
  4908. function findFirst(el, selector) {
  4909. return _frameworkProto.find.call(el, selector).eq(0);
  4910. }
  4911. //==== API ====//
  4912. /**
  4913. * Puts the instance to sleep. It wont respond to any changes in the DOM and won't update. Scrollbar Interactivity is also disabled as well as the resize handle.
  4914. * This behavior can be reset by calling the update method.
  4915. */
  4916. _base.sleep = function () {
  4917. _sleeping = true;
  4918. };
  4919. /**
  4920. * Updates the plugin and DOM to the current options.
  4921. * This method should only be called if a update is 100% required.
  4922. * @param force True if every property shall be updated and the cache shall be ignored.
  4923. * !INTERNAL USAGE! : force can be a string "auto", "sync" or "zoom" too
  4924. * if "auto" then before a real update the content size and host element attributes gets checked, and if they changed only then the update method will be called.
  4925. * if "sync" then the async update process (MutationObserver or UpdateLoop) gets synchronized and a corresponding update takes place if one was needed due to pending changes.
  4926. * if "zoom" then a update takes place where it's assumed that content and host size changed
  4927. * @returns {boolean|undefined}
  4928. * If force is "sync" then a boolean is returned which indicates whether a update was needed due to pending changes.
  4929. * If force is "auto" then a boolean is returned whether a update was needed due to attribute or size changes.
  4930. * undefined otherwise.
  4931. */
  4932. _base.update = function (force) {
  4933. if (_destroyed)
  4934. return;
  4935. var attrsChanged;
  4936. var contentSizeC;
  4937. var isString = type(force) == TYPES.s;
  4938. var imgElementSelector = 'img';
  4939. var imgElementLoadEvent = 'load';
  4940. var doUpdateAuto;
  4941. var mutHost;
  4942. var mutContent;
  4943. if (isString) {
  4944. if (force === _strAuto) {
  4945. attrsChanged = meaningfulAttrsChanged();
  4946. contentSizeC = updateAutoContentSizeChanged();
  4947. doUpdateAuto = attrsChanged || contentSizeC;
  4948. if (doUpdateAuto) {
  4949. update({
  4950. _contentSizeChanged: contentSizeC,
  4951. _changedOptions: _initialized ? undefined : _currentPreparedOptions
  4952. });
  4953. }
  4954. }
  4955. else if (force === _strSync) {
  4956. if (_mutationObserversConnected) {
  4957. mutHost = _mutationObserverHostCallback(_mutationObserverHost.takeRecords());
  4958. mutContent = _mutationObserverContentCallback(_mutationObserverContent.takeRecords());
  4959. }
  4960. else {
  4961. mutHost = _base.update(_strAuto);
  4962. }
  4963. }
  4964. else if (force === 'zoom') {
  4965. update({
  4966. _hostSizeChanged: true,
  4967. _contentSizeChanged: true
  4968. });
  4969. }
  4970. }
  4971. else {
  4972. force = _sleeping || force;
  4973. _sleeping = false;
  4974. if (!_base.update(_strSync) || force)
  4975. update({ _force: force });
  4976. }
  4977. if (!_isTextarea) {
  4978. _contentElement.find(imgElementSelector).each(function (i, el) {
  4979. var index = COMPATIBILITY.inA(el, _imgs);
  4980. if (index === -1)
  4981. FRAMEWORK(el).off(imgElementLoadEvent, imgOnLoad).on(imgElementLoadEvent, imgOnLoad);
  4982. });
  4983. }
  4984. return doUpdateAuto || mutHost || mutContent;
  4985. };
  4986. /**
  4987. Gets or sets the current options. The update method will be called automatically if new options were set.
  4988. * @param newOptions If new options are given, then the new options will be set, if new options aren't given (undefined or a not a plain object) then the current options will be returned.
  4989. * @param value If new options is a property path string, then this value will be used to set the option to which the property path string leads.
  4990. * @returns {*}
  4991. */
  4992. _base.options = function (newOptions, value) {
  4993. var option = {};
  4994. var changedOps;
  4995. //return current options if newOptions are undefined or empty
  4996. if (FRAMEWORK.isEmptyObject(newOptions) || !FRAMEWORK.isPlainObject(newOptions)) {
  4997. if (type(newOptions) == TYPES.s) {
  4998. if (arguments.length > 1) {
  4999. setObjectPropVal(option, newOptions, value);
  5000. changedOps = setOptions(option);
  5001. }
  5002. else
  5003. return getObjectPropVal(_currentOptions, newOptions);
  5004. }
  5005. else
  5006. return _currentOptions;
  5007. }
  5008. else {
  5009. changedOps = setOptions(newOptions);
  5010. }
  5011. if (!FRAMEWORK.isEmptyObject(changedOps)) {
  5012. update({ _changedOptions: changedOps });
  5013. }
  5014. };
  5015. /**
  5016. * Restore the DOM, disconnects all observers, remove all resize observers and put the instance to sleep.
  5017. */
  5018. _base.destroy = function () {
  5019. if (_destroyed)
  5020. return;
  5021. //remove this instance from auto update loop
  5022. autoUpdateLoop.remove(_base);
  5023. //disconnect all mutation observers
  5024. disconnectMutationObservers();
  5025. //remove all resize observers
  5026. setupResizeObserver(_sizeObserverElement);
  5027. setupResizeObserver(_sizeAutoObserverElement);
  5028. //remove all extensions
  5029. for (var extName in _extensions)
  5030. _base.removeExt(extName);
  5031. //remove all 'destroy' events
  5032. while (_destroyEvents[LEXICON.l] > 0)
  5033. _destroyEvents.pop()();
  5034. //remove all events from host element
  5035. setupHostMouseTouchEvents(true);
  5036. //remove all helper / detection elements
  5037. if (_contentGlueElement)
  5038. remove(_contentGlueElement);
  5039. if (_contentArrangeElement)
  5040. remove(_contentArrangeElement);
  5041. if (_sizeAutoObserverAdded)
  5042. remove(_sizeAutoObserverElement);
  5043. //remove all generated DOM
  5044. setupScrollbarsDOM(true);
  5045. setupScrollbarCornerDOM(true);
  5046. setupStructureDOM(true);
  5047. //remove all generated image load events
  5048. for (var i = 0; i < _imgs[LEXICON.l]; i++)
  5049. FRAMEWORK(_imgs[i]).off('load', imgOnLoad);
  5050. _imgs = undefined;
  5051. _destroyed = true;
  5052. _sleeping = true;
  5053. //remove this instance from the instances list
  5054. INSTANCES(pluginTargetElement, 0);
  5055. dispatchCallback('onDestroyed');
  5056. //remove all properties and methods
  5057. //for (var property in _base)
  5058. // delete _base[property];
  5059. //_base = undefined;
  5060. };
  5061. /**
  5062. * Scrolls to a given position or element.
  5063. * @param coordinates
  5064. * 1. Can be "coordinates" which looks like:
  5065. * { x : ?, y : ? } OR Object with x and y properties
  5066. * { left : ?, top : ? } OR Object with left and top properties
  5067. * { l : ?, t : ? } OR Object with l and t properties
  5068. * [ ?, ? ] OR Array where the first two element are the coordinates (first is x, second is y)
  5069. * ? A single value which stays for both axis
  5070. * A value can be a number, a string or a calculation.
  5071. *
  5072. * Operators:
  5073. * [NONE] The current scroll will be overwritten by the value.
  5074. * '+=' The value will be added to the current scroll offset
  5075. * '-=' The value will be subtracted from the current scroll offset
  5076. * '*=' The current scroll wil be multiplicated by the value.
  5077. * '/=' The current scroll wil be divided by the value.
  5078. *
  5079. * Units:
  5080. * [NONE] The value is the final scroll amount. final = (value * 1)
  5081. * 'px' Same as none
  5082. * '%' The value is dependent on the current scroll value. final = ((currentScrollValue / 100) * value)
  5083. * 'vw' The value is multiplicated by the viewport width. final = (value * viewportWidth)
  5084. * 'vh' The value is multiplicated by the viewport height. final = (value * viewportHeight)
  5085. *
  5086. * example final values:
  5087. * 200, '200px', '50%', '1vw', '1vh', '+=200', '/=1vw', '*=2px', '-=5vh', '+=33%', '+= 50% - 2px', '-= 1vw - 50%'
  5088. *
  5089. * 2. Can be a HTML or jQuery element:
  5090. * The final scroll offset is the offset (without margin) of the given HTML / jQuery element.
  5091. *
  5092. * 3. Can be a object with a HTML or jQuery element with additional settings:
  5093. * {
  5094. * el : [HTMLElement, jQuery element], MUST be specified, else this object isn't valid.
  5095. * scroll : [string, array, object], Default value is 'always'.
  5096. * block : [string, array, object], Default value is 'begin'.
  5097. * margin : [number, boolean, array, object] Default value is false.
  5098. * }
  5099. *
  5100. * Possible scroll settings are:
  5101. * 'always' Scrolls always.
  5102. * 'ifneeded' Scrolls only if the element isnt fully in view.
  5103. * 'never' Scrolls never.
  5104. *
  5105. * Possible block settings are:
  5106. * 'begin' Both axis shall be docked to the "begin" edge. - The element will be docked to the top and left edge of the viewport.
  5107. * 'end' Both axis shall be docked to the "end" edge. - The element will be docked to the bottom and right edge of the viewport. (If direction is RTL to the bottom and left edge.)
  5108. * 'center' Both axis shall be docked to "center". - The element will be centered in the viewport.
  5109. * 'nearest' The element will be docked to the nearest edge(s).
  5110. *
  5111. * Possible margin settings are: -- The actual margin of the element wont be affect, this option affects only the final scroll offset.
  5112. * [BOOLEAN] If true the css margin of the element will be used, if false no margin will be used.
  5113. * [NUMBER] The margin will be used for all edges.
  5114. *
  5115. * @param duration The duration of the scroll animation, OR a jQuery animation configuration object.
  5116. * @param easing The animation easing.
  5117. * @param complete The animation complete callback.
  5118. * @returns {{
  5119. * position: {x: number, y: number},
  5120. * ratio: {x: number, y: number},
  5121. * max: {x: number, y: number},
  5122. * handleOffset: {x: number, y: number},
  5123. * handleLength: {x: number, y: number},
  5124. * handleLengthRatio: {x: number, y: number}, t
  5125. * rackLength: {x: number, y: number},
  5126. * isRTL: boolean,
  5127. * isRTLNormalized: boolean
  5128. * }}
  5129. */
  5130. _base.scroll = function (coordinates, duration, easing, complete) {
  5131. if (arguments.length === 0 || coordinates === undefined) {
  5132. var infoX = _scrollHorizontalInfo;
  5133. var infoY = _scrollVerticalInfo;
  5134. var normalizeInvert = _normalizeRTLCache && _isRTL && _rtlScrollBehavior.i;
  5135. var normalizeNegate = _normalizeRTLCache && _isRTL && _rtlScrollBehavior.n;
  5136. var scrollX = infoX._currentScroll;
  5137. var scrollXRatio = infoX._currentScrollRatio;
  5138. var maxScrollX = infoX._maxScroll;
  5139. scrollXRatio = normalizeInvert ? 1 - scrollXRatio : scrollXRatio;
  5140. scrollX = normalizeInvert ? maxScrollX - scrollX : scrollX;
  5141. scrollX *= normalizeNegate ? -1 : 1;
  5142. maxScrollX *= normalizeNegate ? -1 : 1;
  5143. return {
  5144. position: {
  5145. x: scrollX,
  5146. y: infoY._currentScroll
  5147. },
  5148. ratio: {
  5149. x: scrollXRatio,
  5150. y: infoY._currentScrollRatio
  5151. },
  5152. max: {
  5153. x: maxScrollX,
  5154. y: infoY._maxScroll
  5155. },
  5156. handleOffset: {
  5157. x: infoX._handleOffset,
  5158. y: infoY._handleOffset
  5159. },
  5160. handleLength: {
  5161. x: infoX._handleLength,
  5162. y: infoY._handleLength
  5163. },
  5164. handleLengthRatio: {
  5165. x: infoX._handleLengthRatio,
  5166. y: infoY._handleLengthRatio
  5167. },
  5168. trackLength: {
  5169. x: infoX._trackLength,
  5170. y: infoY._trackLength
  5171. },
  5172. snappedHandleOffset: {
  5173. x: infoX._snappedHandleOffset,
  5174. y: infoY._snappedHandleOffset
  5175. },
  5176. isRTL: _isRTL,
  5177. isRTLNormalized: _normalizeRTLCache
  5178. };
  5179. }
  5180. _base.update(_strSync);
  5181. var normalizeRTL = _normalizeRTLCache;
  5182. var coordinatesXAxisProps = [_strX, _strLeft, 'l'];
  5183. var coordinatesYAxisProps = [_strY, _strTop, 't'];
  5184. var coordinatesOperators = ['+=', '-=', '*=', '/='];
  5185. var durationIsObject = type(duration) == TYPES.o;
  5186. var completeCallback = durationIsObject ? duration.complete : complete;
  5187. var i;
  5188. var finalScroll = {};
  5189. var specialEasing = {};
  5190. var doScrollLeft;
  5191. var doScrollTop;
  5192. var animationOptions;
  5193. var strEnd = 'end';
  5194. var strBegin = 'begin';
  5195. var strCenter = 'center';
  5196. var strNearest = 'nearest';
  5197. var strAlways = 'always';
  5198. var strNever = 'never';
  5199. var strIfNeeded = 'ifneeded';
  5200. var strLength = LEXICON.l;
  5201. var settingsAxis;
  5202. var settingsScroll;
  5203. var settingsBlock;
  5204. var settingsMargin;
  5205. var finalElement;
  5206. var elementObjSettingsAxisValues = [_strX, _strY, 'xy', 'yx'];
  5207. var elementObjSettingsBlockValues = [strBegin, strEnd, strCenter, strNearest];
  5208. var elementObjSettingsScrollValues = [strAlways, strNever, strIfNeeded];
  5209. var coordinatesIsElementObj = coordinates[LEXICON.hOP]('el');
  5210. var possibleElement = coordinatesIsElementObj ? coordinates.el : coordinates;
  5211. var possibleElementIsJQuery = possibleElement instanceof FRAMEWORK || JQUERY ? possibleElement instanceof JQUERY : false;
  5212. var possibleElementIsHTMLElement = possibleElementIsJQuery ? false : isHTMLElement(possibleElement);
  5213. var updateScrollbarInfos = function () {
  5214. if (doScrollLeft)
  5215. refreshScrollbarHandleOffset(true);
  5216. if (doScrollTop)
  5217. refreshScrollbarHandleOffset(false);
  5218. };
  5219. var proxyCompleteCallback = type(completeCallback) != TYPES.f ? undefined : function () {
  5220. updateScrollbarInfos();
  5221. completeCallback();
  5222. };
  5223. function checkSettingsStringValue(currValue, allowedValues) {
  5224. for (i = 0; i < allowedValues[strLength]; i++) {
  5225. if (currValue === allowedValues[i])
  5226. return true;
  5227. }
  5228. return false;
  5229. }
  5230. function getRawScroll(isX, coordinates) {
  5231. var coordinateProps = isX ? coordinatesXAxisProps : coordinatesYAxisProps;
  5232. coordinates = type(coordinates) == TYPES.s || type(coordinates) == TYPES.n ? [coordinates, coordinates] : coordinates;
  5233. if (type(coordinates) == TYPES.a)
  5234. return isX ? coordinates[0] : coordinates[1];
  5235. else if (type(coordinates) == TYPES.o) {
  5236. //decides RTL normalization "hack" with .n
  5237. //normalizeRTL = type(coordinates.n) == TYPES.b ? coordinates.n : normalizeRTL;
  5238. for (i = 0; i < coordinateProps[strLength]; i++)
  5239. if (coordinateProps[i] in coordinates)
  5240. return coordinates[coordinateProps[i]];
  5241. }
  5242. }
  5243. function getFinalScroll(isX, rawScroll) {
  5244. var isString = type(rawScroll) == TYPES.s;
  5245. var operator;
  5246. var amount;
  5247. var scrollInfo = isX ? _scrollHorizontalInfo : _scrollVerticalInfo;
  5248. var currScroll = scrollInfo._currentScroll;
  5249. var maxScroll = scrollInfo._maxScroll;
  5250. var mult = ' * ';
  5251. var finalValue;
  5252. var isRTLisX = _isRTL && isX;
  5253. var normalizeShortcuts = isRTLisX && _rtlScrollBehavior.n && !normalizeRTL;
  5254. var strReplace = 'replace';
  5255. var evalFunc = eval;
  5256. var possibleOperator;
  5257. if (isString) {
  5258. //check operator
  5259. if (rawScroll[strLength] > 2) {
  5260. possibleOperator = rawScroll.substr(0, 2);
  5261. if (inArray(possibleOperator, coordinatesOperators) > -1)
  5262. operator = possibleOperator;
  5263. }
  5264. //calculate units and shortcuts
  5265. rawScroll = operator ? rawScroll.substr(2) : rawScroll;
  5266. rawScroll = rawScroll
  5267. [strReplace](/min/g, 0) //'min' = 0%
  5268. [strReplace](/</g, 0) //'<' = 0%
  5269. [strReplace](/max/g, (normalizeShortcuts ? '-' : _strEmpty) + _strHundredPercent) //'max' = 100%
  5270. [strReplace](/>/g, (normalizeShortcuts ? '-' : _strEmpty) + _strHundredPercent) //'>' = 100%
  5271. [strReplace](/px/g, _strEmpty)
  5272. [strReplace](/%/g, mult + (maxScroll * (isRTLisX && _rtlScrollBehavior.n ? -1 : 1) / 100.0))
  5273. [strReplace](/vw/g, mult + _viewportSize.w)
  5274. [strReplace](/vh/g, mult + _viewportSize.h);
  5275. amount = parseToZeroOrNumber(isNaN(rawScroll) ? parseToZeroOrNumber(evalFunc(rawScroll), true).toFixed() : rawScroll);
  5276. }
  5277. else {
  5278. amount = rawScroll;
  5279. }
  5280. if (amount !== undefined && !isNaN(amount) && type(amount) == TYPES.n) {
  5281. var normalizeIsRTLisX = normalizeRTL && isRTLisX;
  5282. var operatorCurrScroll = currScroll * (normalizeIsRTLisX && _rtlScrollBehavior.n ? -1 : 1);
  5283. var invert = normalizeIsRTLisX && _rtlScrollBehavior.i;
  5284. var negate = normalizeIsRTLisX && _rtlScrollBehavior.n;
  5285. operatorCurrScroll = invert ? (maxScroll - operatorCurrScroll) : operatorCurrScroll;
  5286. switch (operator) {
  5287. case '+=':
  5288. finalValue = operatorCurrScroll + amount;
  5289. break;
  5290. case '-=':
  5291. finalValue = operatorCurrScroll - amount;
  5292. break;
  5293. case '*=':
  5294. finalValue = operatorCurrScroll * amount;
  5295. break;
  5296. case '/=':
  5297. finalValue = operatorCurrScroll / amount;
  5298. break;
  5299. default:
  5300. finalValue = amount;
  5301. break;
  5302. }
  5303. finalValue = invert ? maxScroll - finalValue : finalValue;
  5304. finalValue *= negate ? -1 : 1;
  5305. finalValue = isRTLisX && _rtlScrollBehavior.n ? MATH.min(0, MATH.max(maxScroll, finalValue)) : MATH.max(0, MATH.min(maxScroll, finalValue));
  5306. }
  5307. return finalValue === currScroll ? undefined : finalValue;
  5308. }
  5309. function getPerAxisValue(value, valueInternalType, defaultValue, allowedValues) {
  5310. var resultDefault = [defaultValue, defaultValue];
  5311. var valueType = type(value);
  5312. var valueArrLength;
  5313. var valueArrItem;
  5314. //value can be [ string, or array of two strings ]
  5315. if (valueType == valueInternalType) {
  5316. value = [value, value];
  5317. }
  5318. else if (valueType == TYPES.a) {
  5319. valueArrLength = value[strLength];
  5320. if (valueArrLength > 2 || valueArrLength < 1)
  5321. value = resultDefault;
  5322. else {
  5323. if (valueArrLength === 1)
  5324. value[1] = defaultValue;
  5325. for (i = 0; i < valueArrLength; i++) {
  5326. valueArrItem = value[i];
  5327. if (type(valueArrItem) != valueInternalType || !checkSettingsStringValue(valueArrItem, allowedValues)) {
  5328. value = resultDefault;
  5329. break;
  5330. }
  5331. }
  5332. }
  5333. }
  5334. else if (valueType == TYPES.o)
  5335. value = [value[_strX] || defaultValue, value[_strY] || defaultValue];
  5336. else
  5337. value = resultDefault;
  5338. return { x: value[0], y: value[1] };
  5339. }
  5340. function generateMargin(marginTopRightBottomLeftArray) {
  5341. var result = [];
  5342. var currValue;
  5343. var currValueType;
  5344. var valueDirections = [_strTop, _strRight, _strBottom, _strLeft];
  5345. for (i = 0; i < marginTopRightBottomLeftArray[strLength]; i++) {
  5346. if (i === valueDirections[strLength])
  5347. break;
  5348. currValue = marginTopRightBottomLeftArray[i];
  5349. currValueType = type(currValue);
  5350. if (currValueType == TYPES.b)
  5351. result.push(currValue ? parseToZeroOrNumber(finalElement.css(_strMarginMinus + valueDirections[i])) : 0);
  5352. else
  5353. result.push(currValueType == TYPES.n ? currValue : 0);
  5354. }
  5355. return result;
  5356. }
  5357. if (possibleElementIsJQuery || possibleElementIsHTMLElement) {
  5358. //get settings
  5359. var margin = coordinatesIsElementObj ? coordinates.margin : 0;
  5360. var axis = coordinatesIsElementObj ? coordinates.axis : 0;
  5361. var scroll = coordinatesIsElementObj ? coordinates.scroll : 0;
  5362. var block = coordinatesIsElementObj ? coordinates.block : 0;
  5363. var marginDefault = [0, 0, 0, 0];
  5364. var marginType = type(margin);
  5365. var marginLength;
  5366. finalElement = possibleElementIsJQuery ? possibleElement : FRAMEWORK(possibleElement);
  5367. if (finalElement[strLength] > 0) {
  5368. //margin can be [ boolean, number, array of 2, array of 4, object ]
  5369. if (marginType == TYPES.n || marginType == TYPES.b)
  5370. margin = generateMargin([margin, margin, margin, margin]);
  5371. else if (marginType == TYPES.a) {
  5372. marginLength = margin[strLength];
  5373. if (marginLength === 2)
  5374. margin = generateMargin([margin[0], margin[1], margin[0], margin[1]]);
  5375. else if (marginLength >= 4)
  5376. margin = generateMargin(margin);
  5377. else
  5378. margin = marginDefault;
  5379. }
  5380. else if (marginType == TYPES.o)
  5381. margin = generateMargin([margin[_strTop], margin[_strRight], margin[_strBottom], margin[_strLeft]]);
  5382. else
  5383. margin = marginDefault;
  5384. //block = type(block) === TYPES.b ? block ? [ strNearest, strBegin ] : [ strNearest, strEnd ] : block;
  5385. settingsAxis = checkSettingsStringValue(axis, elementObjSettingsAxisValues) ? axis : 'xy';
  5386. settingsScroll = getPerAxisValue(scroll, TYPES.s, strAlways, elementObjSettingsScrollValues);
  5387. settingsBlock = getPerAxisValue(block, TYPES.s, strBegin, elementObjSettingsBlockValues);
  5388. settingsMargin = margin;
  5389. var viewportScroll = {
  5390. l: _scrollHorizontalInfo._currentScroll,
  5391. t: _scrollVerticalInfo._currentScroll
  5392. };
  5393. // use padding element instead of viewport element because padding element has never padding, margin or position applied.
  5394. var viewportOffset = _paddingElement.offset();
  5395. //get coordinates
  5396. var elementOffset = finalElement.offset();
  5397. var doNotScroll = {
  5398. x: settingsScroll.x == strNever || settingsAxis == _strY,
  5399. y: settingsScroll.y == strNever || settingsAxis == _strX
  5400. };
  5401. elementOffset[_strTop] -= settingsMargin[0];
  5402. elementOffset[_strLeft] -= settingsMargin[3];
  5403. var elementScrollCoordinates = {
  5404. x: MATH.round(elementOffset[_strLeft] - viewportOffset[_strLeft] + viewportScroll.l),
  5405. y: MATH.round(elementOffset[_strTop] - viewportOffset[_strTop] + viewportScroll.t)
  5406. };
  5407. if (_isRTL) {
  5408. if (!_rtlScrollBehavior.n && !_rtlScrollBehavior.i)
  5409. elementScrollCoordinates.x = MATH.round(viewportOffset[_strLeft] - elementOffset[_strLeft] + viewportScroll.l);
  5410. if (_rtlScrollBehavior.n && normalizeRTL)
  5411. elementScrollCoordinates.x *= -1;
  5412. if (_rtlScrollBehavior.i && normalizeRTL)
  5413. elementScrollCoordinates.x = MATH.round(viewportOffset[_strLeft] - elementOffset[_strLeft] + (_scrollHorizontalInfo._maxScroll - viewportScroll.l));
  5414. }
  5415. //measuring is required
  5416. if (settingsBlock.x != strBegin || settingsBlock.y != strBegin || settingsScroll.x == strIfNeeded || settingsScroll.y == strIfNeeded || _isRTL) {
  5417. var measuringElm = finalElement[0];
  5418. var rawElementSize = _supportTransform ? measuringElm[LEXICON.bCR]() : {
  5419. width: measuringElm[LEXICON.oW],
  5420. height: measuringElm[LEXICON.oH]
  5421. };
  5422. var elementSize = {
  5423. w: rawElementSize[_strWidth] + settingsMargin[3] + settingsMargin[1],
  5424. h: rawElementSize[_strHeight] + settingsMargin[0] + settingsMargin[2]
  5425. };
  5426. var finalizeBlock = function (isX) {
  5427. var vars = getScrollbarVars(isX);
  5428. var wh = vars._w_h;
  5429. var lt = vars._left_top;
  5430. var xy = vars._x_y;
  5431. var blockIsEnd = settingsBlock[xy] == (isX ? _isRTL ? strBegin : strEnd : strEnd);
  5432. var blockIsCenter = settingsBlock[xy] == strCenter;
  5433. var blockIsNearest = settingsBlock[xy] == strNearest;
  5434. var scrollNever = settingsScroll[xy] == strNever;
  5435. var scrollIfNeeded = settingsScroll[xy] == strIfNeeded;
  5436. var vpSize = _viewportSize[wh];
  5437. var vpOffset = viewportOffset[lt];
  5438. var elSize = elementSize[wh];
  5439. var elOffset = elementOffset[lt];
  5440. var divide = blockIsCenter ? 2 : 1;
  5441. var elementCenterOffset = elOffset + (elSize / 2);
  5442. var viewportCenterOffset = vpOffset + (vpSize / 2);
  5443. var isInView =
  5444. elSize <= vpSize
  5445. && elOffset >= vpOffset
  5446. && elOffset + elSize <= vpOffset + vpSize;
  5447. if (scrollNever)
  5448. doNotScroll[xy] = true;
  5449. else if (!doNotScroll[xy]) {
  5450. if (blockIsNearest || scrollIfNeeded) {
  5451. doNotScroll[xy] = scrollIfNeeded ? isInView : false;
  5452. blockIsEnd = elSize < vpSize ? elementCenterOffset > viewportCenterOffset : elementCenterOffset < viewportCenterOffset;
  5453. }
  5454. elementScrollCoordinates[xy] -= blockIsEnd || blockIsCenter ? ((vpSize / divide) - (elSize / divide)) * (isX && _isRTL && normalizeRTL ? -1 : 1) : 0;
  5455. }
  5456. };
  5457. finalizeBlock(true);
  5458. finalizeBlock(false);
  5459. }
  5460. if (doNotScroll.y)
  5461. delete elementScrollCoordinates.y;
  5462. if (doNotScroll.x)
  5463. delete elementScrollCoordinates.x;
  5464. coordinates = elementScrollCoordinates;
  5465. }
  5466. }
  5467. finalScroll[_strScrollLeft] = getFinalScroll(true, getRawScroll(true, coordinates));
  5468. finalScroll[_strScrollTop] = getFinalScroll(false, getRawScroll(false, coordinates));
  5469. doScrollLeft = finalScroll[_strScrollLeft] !== undefined;
  5470. doScrollTop = finalScroll[_strScrollTop] !== undefined;
  5471. if ((doScrollLeft || doScrollTop) && (duration > 0 || durationIsObject)) {
  5472. if (durationIsObject) {
  5473. duration.complete = proxyCompleteCallback;
  5474. _viewportElement.animate(finalScroll, duration);
  5475. }
  5476. else {
  5477. animationOptions = {
  5478. duration: duration,
  5479. complete: proxyCompleteCallback
  5480. };
  5481. if (type(easing) == TYPES.a || FRAMEWORK.isPlainObject(easing)) {
  5482. specialEasing[_strScrollLeft] = easing[0] || easing.x;
  5483. specialEasing[_strScrollTop] = easing[1] || easing.y;
  5484. animationOptions.specialEasing = specialEasing;
  5485. }
  5486. else {
  5487. animationOptions.easing = easing;
  5488. }
  5489. _viewportElement.animate(finalScroll, animationOptions);
  5490. }
  5491. }
  5492. else {
  5493. if (doScrollLeft)
  5494. _viewportElement[_strScrollLeft](finalScroll[_strScrollLeft]);
  5495. if (doScrollTop)
  5496. _viewportElement[_strScrollTop](finalScroll[_strScrollTop]);
  5497. updateScrollbarInfos();
  5498. }
  5499. };
  5500. /**
  5501. * Stops all scroll animations.
  5502. * @returns {*} The current OverlayScrollbars instance (for chaining).
  5503. */
  5504. _base.scrollStop = function (param1, param2, param3) {
  5505. _viewportElement.stop(param1, param2, param3);
  5506. return _base;
  5507. };
  5508. /**
  5509. * Returns all relevant elements.
  5510. * @param elementName The name of the element which shall be returned.
  5511. * @returns {{target: *, host: *, padding: *, viewport: *, content: *, scrollbarHorizontal: {scrollbar: *, track: *, handle: *}, scrollbarVertical: {scrollbar: *, track: *, handle: *}, scrollbarCorner: *} | *}
  5512. */
  5513. _base.getElements = function (elementName) {
  5514. var obj = {
  5515. target: _targetElementNative,
  5516. host: _hostElementNative,
  5517. padding: _paddingElementNative,
  5518. viewport: _viewportElementNative,
  5519. content: _contentElementNative,
  5520. scrollbarHorizontal: {
  5521. scrollbar: _scrollbarHorizontalElement[0],
  5522. track: _scrollbarHorizontalTrackElement[0],
  5523. handle: _scrollbarHorizontalHandleElement[0]
  5524. },
  5525. scrollbarVertical: {
  5526. scrollbar: _scrollbarVerticalElement[0],
  5527. track: _scrollbarVerticalTrackElement[0],
  5528. handle: _scrollbarVerticalHandleElement[0]
  5529. },
  5530. scrollbarCorner: _scrollbarCornerElement[0]
  5531. };
  5532. return type(elementName) == TYPES.s ? getObjectPropVal(obj, elementName) : obj;
  5533. };
  5534. /**
  5535. * Returns a object which describes the current state of this instance.
  5536. * @param stateProperty A specific property from the state object which shall be returned.
  5537. * @returns {{widthAuto, heightAuto, overflowAmount, hideOverflow, hasOverflow, contentScrollSize, viewportSize, hostSize, autoUpdate} | *}
  5538. */
  5539. _base.getState = function (stateProperty) {
  5540. function prepare(obj) {
  5541. if (!FRAMEWORK.isPlainObject(obj))
  5542. return obj;
  5543. var extended = extendDeep({}, obj);
  5544. var changePropertyName = function (from, to) {
  5545. if (extended[LEXICON.hOP](from)) {
  5546. extended[to] = extended[from];
  5547. delete extended[from];
  5548. }
  5549. };
  5550. changePropertyName('w', _strWidth); //change w to width
  5551. changePropertyName('h', _strHeight); //change h to height
  5552. delete extended.c; //delete c (the 'changed' prop)
  5553. return extended;
  5554. };
  5555. var obj = {
  5556. destroyed: !!prepare(_destroyed),
  5557. sleeping: !!prepare(_sleeping),
  5558. autoUpdate: prepare(!_mutationObserversConnected),
  5559. widthAuto: prepare(_widthAutoCache),
  5560. heightAuto: prepare(_heightAutoCache),
  5561. padding: prepare(_cssPaddingCache),
  5562. overflowAmount: prepare(_overflowAmountCache),
  5563. hideOverflow: prepare(_hideOverflowCache),
  5564. hasOverflow: prepare(_hasOverflowCache),
  5565. contentScrollSize: prepare(_contentScrollSizeCache),
  5566. viewportSize: prepare(_viewportSize),
  5567. hostSize: prepare(_hostSizeCache),
  5568. documentMixed: prepare(_documentMixed)
  5569. };
  5570. return type(stateProperty) == TYPES.s ? getObjectPropVal(obj, stateProperty) : obj;
  5571. };
  5572. /**
  5573. * Gets all or specific extension instance.
  5574. * @param extName The name of the extension from which the instance shall be got.
  5575. * @returns {{}} The instance of the extension with the given name or undefined if the instance couldn't be found.
  5576. */
  5577. _base.ext = function (extName) {
  5578. var result;
  5579. var privateMethods = _extensionsPrivateMethods.split(' ');
  5580. var i = 0;
  5581. if (type(extName) == TYPES.s) {
  5582. if (_extensions[LEXICON.hOP](extName)) {
  5583. result = extendDeep({}, _extensions[extName]);
  5584. for (; i < privateMethods.length; i++)
  5585. delete result[privateMethods[i]];
  5586. }
  5587. }
  5588. else {
  5589. result = {};
  5590. for (i in _extensions)
  5591. result[i] = extendDeep({}, _base.ext(i));
  5592. }
  5593. return result;
  5594. };
  5595. /**
  5596. * Adds a extension to this instance.
  5597. * @param extName The name of the extension which shall be added.
  5598. * @param extensionOptions The extension options which shall be used.
  5599. * @returns {{}} The instance of the added extension or undefined if the extension couldn't be added properly.
  5600. */
  5601. _base.addExt = function (extName, extensionOptions) {
  5602. var registeredExtensionObj = _plugin.extension(extName);
  5603. var instance;
  5604. var instanceAdded;
  5605. var instanceContract;
  5606. var contractResult;
  5607. var contractFulfilled = true;
  5608. if (registeredExtensionObj) {
  5609. if (!_extensions[LEXICON.hOP](extName)) {
  5610. instance = registeredExtensionObj.extensionFactory.call(_base,
  5611. extendDeep({}, registeredExtensionObj.defaultOptions),
  5612. FRAMEWORK,
  5613. COMPATIBILITY);
  5614. if (instance) {
  5615. instanceContract = instance.contract;
  5616. if (type(instanceContract) == TYPES.f) {
  5617. contractResult = instanceContract(window);
  5618. contractFulfilled = type(contractResult) == TYPES.b ? contractResult : contractFulfilled;
  5619. }
  5620. if (contractFulfilled) {
  5621. _extensions[extName] = instance;
  5622. instanceAdded = instance.added;
  5623. if (type(instanceAdded) == TYPES.f)
  5624. instanceAdded(extensionOptions);
  5625. return _base.ext(extName);
  5626. }
  5627. }
  5628. }
  5629. else
  5630. return _base.ext(extName);
  5631. }
  5632. else
  5633. console.warn("A extension with the name \"" + extName + "\" isn't registered.");
  5634. };
  5635. /**
  5636. * Removes a extension from this instance.
  5637. * @param extName The name of the extension which shall be removed.
  5638. * @returns {boolean} True if the extension was removed, false otherwise e.g. if the extension wasn't added before.
  5639. */
  5640. _base.removeExt = function (extName) {
  5641. var instance = _extensions[extName];
  5642. var instanceRemoved;
  5643. if (instance) {
  5644. delete _extensions[extName];
  5645. instanceRemoved = instance.removed;
  5646. if (type(instanceRemoved) == TYPES.f)
  5647. instanceRemoved();
  5648. return true;
  5649. }
  5650. return false;
  5651. };
  5652. /**
  5653. * Constructs the plugin.
  5654. * @param targetElement The element to which the plugin shall be applied.
  5655. * @param options The initial options of the plugin.
  5656. * @param extensions The extension(s) which shall be added right after the initialization.
  5657. * @returns {boolean} True if the plugin was successfully initialized, false otherwise.
  5658. */
  5659. function construct(targetElement, options, extensions) {
  5660. _defaultOptions = globals.defaultOptions;
  5661. _nativeScrollbarStyling = globals.nativeScrollbarStyling;
  5662. _nativeScrollbarSize = extendDeep({}, globals.nativeScrollbarSize);
  5663. _nativeScrollbarIsOverlaid = extendDeep({}, globals.nativeScrollbarIsOverlaid);
  5664. _overlayScrollbarDummySize = extendDeep({}, globals.overlayScrollbarDummySize);
  5665. _rtlScrollBehavior = extendDeep({}, globals.rtlScrollBehavior);
  5666. //parse & set options but don't update
  5667. setOptions(extendDeep({}, _defaultOptions, options));
  5668. _cssCalc = globals.cssCalc;
  5669. _msieVersion = globals.msie;
  5670. _autoUpdateRecommended = globals.autoUpdateRecommended;
  5671. _supportTransition = globals.supportTransition;
  5672. _supportTransform = globals.supportTransform;
  5673. _supportPassiveEvents = globals.supportPassiveEvents;
  5674. _supportResizeObserver = globals.supportResizeObserver;
  5675. _supportMutationObserver = globals.supportMutationObserver;
  5676. _restrictedMeasuring = globals.restrictedMeasuring;
  5677. _documentElement = FRAMEWORK(targetElement.ownerDocument);
  5678. _documentElementNative = _documentElement[0];
  5679. _windowElement = FRAMEWORK(_documentElementNative.defaultView || _documentElementNative.parentWindow);
  5680. _windowElementNative = _windowElement[0];
  5681. _htmlElement = findFirst(_documentElement, 'html');
  5682. _bodyElement = findFirst(_htmlElement, 'body');
  5683. _targetElement = FRAMEWORK(targetElement);
  5684. _targetElementNative = _targetElement[0];
  5685. _isTextarea = _targetElement.is('textarea');
  5686. _isBody = _targetElement.is('body');
  5687. _documentMixed = _documentElementNative !== document;
  5688. /* On a div Element The if checks only whether:
  5689. * - the targetElement has the class "os-host"
  5690. * - the targetElement has a a child with the class "os-padding"
  5691. *
  5692. * If that's the case, its assumed the DOM has already the following structure:
  5693. * (The ".os-host" element is the targetElement)
  5694. *
  5695. * <div class="os-host">
  5696. * <div class="os-resize-observer-host"></div>
  5697. * <div class="os-padding">
  5698. * <div class="os-viewport">
  5699. * <div class="os-content"></div>
  5700. * </div>
  5701. * </div>
  5702. * <div class="os-scrollbar os-scrollbar-horizontal ">
  5703. * <div class="os-scrollbar-track">
  5704. * <div class="os-scrollbar-handle"></div>
  5705. * </div>
  5706. * </div>
  5707. * <div class="os-scrollbar os-scrollbar-vertical">
  5708. * <div class="os-scrollbar-track">
  5709. * <div class="os-scrollbar-handle"></div>
  5710. * </div>
  5711. * </div>
  5712. * <div class="os-scrollbar-corner"></div>
  5713. * </div>
  5714. *
  5715. * =====================================================================================
  5716. *
  5717. * On a Textarea Element The if checks only whether:
  5718. * - the targetElement has the class "os-textarea"
  5719. * - the targetElement is inside a element with the class "os-content"
  5720. *
  5721. * If that's the case, its assumed the DOM has already the following structure:
  5722. * (The ".os-textarea" (textarea) element is the targetElement)
  5723. *
  5724. * <div class="os-host-textarea">
  5725. * <div class="os-resize-observer-host"></div>
  5726. * <div class="os-padding os-text-inherit">
  5727. * <div class="os-viewport os-text-inherit">
  5728. * <div class="os-content os-text-inherit">
  5729. * <div class="os-textarea-cover"></div>
  5730. * <textarea class="os-textarea os-text-inherit"></textarea>
  5731. * </div>
  5732. * </div>
  5733. * </div>
  5734. * <div class="os-scrollbar os-scrollbar-horizontal ">
  5735. * <div class="os-scrollbar-track">
  5736. * <div class="os-scrollbar-handle"></div>
  5737. * </div>
  5738. * </div>
  5739. * <div class="os-scrollbar os-scrollbar-vertical">
  5740. * <div class="os-scrollbar-track">
  5741. * <div class="os-scrollbar-handle"></div>
  5742. * </div>
  5743. * </div>
  5744. * <div class="os-scrollbar-corner"></div>
  5745. * </div>
  5746. */
  5747. _domExists = _isTextarea
  5748. ? _targetElement.hasClass(_classNameTextareaElement) && _targetElement.parent().hasClass(_classNameContentElement)
  5749. : _targetElement.hasClass(_classNameHostElement) && _targetElement.children(_strDot + _classNamePaddingElement)[LEXICON.l];
  5750. var initBodyScroll;
  5751. var bodyMouseTouchDownListener;
  5752. //check if the plugin hasn't to be initialized
  5753. if (_nativeScrollbarIsOverlaid.x && _nativeScrollbarIsOverlaid.y && !_currentPreparedOptions.nativeScrollbarsOverlaid.initialize) {
  5754. dispatchCallback('onInitializationWithdrawn');
  5755. if (_domExists) {
  5756. setupStructureDOM(true);
  5757. setupScrollbarsDOM(true);
  5758. setupScrollbarCornerDOM(true);
  5759. }
  5760. _destroyed = true;
  5761. _sleeping = true;
  5762. return _base;
  5763. }
  5764. if (_isBody) {
  5765. initBodyScroll = {};
  5766. initBodyScroll.l = MATH.max(_targetElement[_strScrollLeft](), _htmlElement[_strScrollLeft](), _windowElement[_strScrollLeft]());
  5767. initBodyScroll.t = MATH.max(_targetElement[_strScrollTop](), _htmlElement[_strScrollTop](), _windowElement[_strScrollTop]());
  5768. bodyMouseTouchDownListener = function () {
  5769. _viewportElement.removeAttr('tabindex');
  5770. setupResponsiveEventListener(_viewportElement, _strMouseTouchDownEvent, bodyMouseTouchDownListener, true, true);
  5771. }
  5772. }
  5773. //build OverlayScrollbars DOM
  5774. setupStructureDOM();
  5775. setupScrollbarsDOM();
  5776. setupScrollbarCornerDOM();
  5777. //create OverlayScrollbars events
  5778. setupStructureEvents();
  5779. setupScrollbarEvents(true);
  5780. setupScrollbarEvents(false);
  5781. setupScrollbarCornerEvents();
  5782. //create mutation observers
  5783. createMutationObservers();
  5784. //build resize observer for the host element
  5785. setupResizeObserver(_sizeObserverElement, hostOnResized);
  5786. if (_isBody) {
  5787. //apply the body scroll to handle it right in the update method
  5788. _viewportElement[_strScrollLeft](initBodyScroll.l)[_strScrollTop](initBodyScroll.t);
  5789. //set the focus on the viewport element so you dont have to click on the page to use keyboard keys (up / down / space) for scrolling
  5790. if (document.activeElement == targetElement && _viewportElementNative.focus) {
  5791. //set a tabindex to make the viewportElement focusable
  5792. _viewportElement.attr('tabindex', '-1');
  5793. _viewportElementNative.focus();
  5794. /* the tabindex has to be removed due to;
  5795. * If you set the tabindex attribute on an <div>, then its child content cannot be scrolled with the arrow keys unless you set tabindex on the content, too
  5796. * https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/tabindex
  5797. */
  5798. setupResponsiveEventListener(_viewportElement, _strMouseTouchDownEvent, bodyMouseTouchDownListener, false, true);
  5799. }
  5800. }
  5801. //update for the first time & initialize cache
  5802. _base.update(_strAuto);
  5803. //the plugin is initialized now!
  5804. _initialized = true;
  5805. dispatchCallback('onInitialized');
  5806. //call all callbacks which would fire before the initialized was complete
  5807. each(_callbacksInitQeueue, function (index, value) { dispatchCallback(value.n, value.a); });
  5808. _callbacksInitQeueue = [];
  5809. //add extensions
  5810. if (type(extensions) == TYPES.s)
  5811. extensions = [extensions];
  5812. if (COMPATIBILITY.isA(extensions))
  5813. each(extensions, function (index, value) { _base.addExt(value); });
  5814. else if (FRAMEWORK.isPlainObject(extensions))
  5815. each(extensions, function (key, value) { _base.addExt(key, value); });
  5816. //add the transition class for transitions AFTER the first update & AFTER the applied extensions (for preventing unwanted transitions)
  5817. setTimeout(function () {
  5818. if (_supportTransition && !_destroyed)
  5819. addClass(_hostElement, _classNameHostTransition);
  5820. }, 333);
  5821. return _base;
  5822. }
  5823. if (_plugin.valid(construct(pluginTargetElement, options, extensions))) {
  5824. INSTANCES(pluginTargetElement, _base);
  5825. }
  5826. return _base;
  5827. }
  5828. /**
  5829. * Initializes a new OverlayScrollbarsInstance object or changes options if already initialized or returns the current instance.
  5830. * @param pluginTargetElements The elements to which the Plugin shall be initialized.
  5831. * @param options The custom options with which the plugin shall be initialized.
  5832. * @param extensions The extension(s) which shall be added right after initialization.
  5833. * @returns {*}
  5834. */
  5835. _plugin = window[PLUGINNAME] = function (pluginTargetElements, options, extensions) {
  5836. if (arguments[LEXICON.l] === 0)
  5837. return this;
  5838. var arr = [];
  5839. var optsIsPlainObj = FRAMEWORK.isPlainObject(options);
  5840. var inst;
  5841. var result;
  5842. //pluginTargetElements is null or undefined
  5843. if (!pluginTargetElements)
  5844. return optsIsPlainObj || !options ? result : arr;
  5845. /*
  5846. pluginTargetElements will be converted to:
  5847. 1. A jQueryElement Array
  5848. 2. A HTMLElement Array
  5849. 3. A Array with a single HTML Element
  5850. so pluginTargetElements is always a array.
  5851. */
  5852. pluginTargetElements = pluginTargetElements[LEXICON.l] != undefined ? pluginTargetElements : [pluginTargetElements[0] || pluginTargetElements];
  5853. initOverlayScrollbarsStatics();
  5854. if (pluginTargetElements[LEXICON.l] > 0) {
  5855. if (optsIsPlainObj) {
  5856. FRAMEWORK.each(pluginTargetElements, function (i, v) {
  5857. inst = v;
  5858. if (inst !== undefined)
  5859. arr.push(OverlayScrollbarsInstance(inst, options, extensions, _pluginsGlobals, _pluginsAutoUpdateLoop));
  5860. });
  5861. }
  5862. else {
  5863. FRAMEWORK.each(pluginTargetElements, function (i, v) {
  5864. inst = INSTANCES(v);
  5865. if ((options === '!' && _plugin.valid(inst)) || (COMPATIBILITY.type(options) == TYPES.f && options(v, inst)))
  5866. arr.push(inst);
  5867. else if (options === undefined)
  5868. arr.push(inst);
  5869. });
  5870. }
  5871. result = arr[LEXICON.l] === 1 ? arr[0] : arr;
  5872. }
  5873. return result;
  5874. };
  5875. /**
  5876. * Returns a object which contains global information about the plugin and each instance of it.
  5877. * The returned object is just a copy, that means that changes to the returned object won't have any effect to the original object.
  5878. */
  5879. _plugin.globals = function () {
  5880. initOverlayScrollbarsStatics();
  5881. var globals = FRAMEWORK.extend(true, {}, _pluginsGlobals);
  5882. delete globals['msie'];
  5883. return globals;
  5884. };
  5885. /**
  5886. * Gets or Sets the default options for each new plugin initialization.
  5887. * @param newDefaultOptions The object with which the default options shall be extended.
  5888. */
  5889. _plugin.defaultOptions = function (newDefaultOptions) {
  5890. initOverlayScrollbarsStatics();
  5891. var currDefaultOptions = _pluginsGlobals.defaultOptions;
  5892. if (newDefaultOptions === undefined)
  5893. return FRAMEWORK.extend(true, {}, currDefaultOptions);
  5894. //set the new default options
  5895. _pluginsGlobals.defaultOptions = FRAMEWORK.extend(true, {}, currDefaultOptions, _pluginsOptions._validate(newDefaultOptions, _pluginsOptions._template, true, currDefaultOptions)._default);
  5896. };
  5897. /**
  5898. * Checks whether the passed instance is a non-destroyed OverlayScrollbars instance.
  5899. * @param osInstance The potential OverlayScrollbars instance which shall be checked.
  5900. * @returns {boolean} True if the passed value is a non-destroyed OverlayScrollbars instance, false otherwise.
  5901. */
  5902. _plugin.valid = function (osInstance) {
  5903. return osInstance instanceof _plugin && !osInstance.getState().destroyed;
  5904. };
  5905. /**
  5906. * Registers, Unregisters or returns a extension.
  5907. * Register: Pass the name and the extension. (defaultOptions is optional)
  5908. * Unregister: Pass the name and anything except a function as extension parameter.
  5909. * Get extension: Pass the name of the extension which shall be got.
  5910. * Get all extensions: Pass no arguments.
  5911. * @param extensionName The name of the extension which shall be registered, unregistered or returned.
  5912. * @param extension A function which generates the instance of the extension or anything other to remove a already registered extension.
  5913. * @param defaultOptions The default options which shall be used for the registered extension.
  5914. */
  5915. _plugin.extension = function (extensionName, extension, defaultOptions) {
  5916. var extNameTypeString = COMPATIBILITY.type(extensionName) == TYPES.s;
  5917. var argLen = arguments[LEXICON.l];
  5918. var i = 0;
  5919. if (argLen < 1 || !extNameTypeString) {
  5920. //return a copy of all extension objects
  5921. return FRAMEWORK.extend(true, { length: _pluginsExtensions[LEXICON.l] }, _pluginsExtensions);
  5922. }
  5923. else if (extNameTypeString) {
  5924. if (COMPATIBILITY.type(extension) == TYPES.f) {
  5925. //register extension
  5926. _pluginsExtensions.push({
  5927. name: extensionName,
  5928. extensionFactory: extension,
  5929. defaultOptions: defaultOptions
  5930. });
  5931. }
  5932. else {
  5933. for (; i < _pluginsExtensions[LEXICON.l]; i++) {
  5934. if (_pluginsExtensions[i].name === extensionName) {
  5935. if (argLen > 1)
  5936. _pluginsExtensions.splice(i, 1); //remove extension
  5937. else
  5938. return FRAMEWORK.extend(true, {}, _pluginsExtensions[i]); //return extension with the given name
  5939. }
  5940. }
  5941. }
  5942. }
  5943. };
  5944. return _plugin;
  5945. })();
  5946. if (JQUERY && JQUERY.fn) {
  5947. /**
  5948. * The jQuery initialization interface.
  5949. * @param options The initial options for the construction of the plugin. To initialize the plugin, this option has to be a object! If it isn't a object, the instance(s) are returned and the plugin wont be initialized.
  5950. * @param extensions The extension(s) which shall be added right after initialization.
  5951. * @returns {*} After initialization it returns the jQuery element array, else it returns the instance(s) of the elements which are selected.
  5952. */
  5953. JQUERY.fn.overlayScrollbars = function (options, extensions) {
  5954. var _elements = this;
  5955. if (JQUERY.isPlainObject(options)) {
  5956. JQUERY.each(_elements, function () { PLUGIN(this, options, extensions); });
  5957. return _elements;
  5958. }
  5959. else
  5960. return PLUGIN(_elements, options);
  5961. };
  5962. }
  5963. return PLUGIN;
  5964. }
  5965. ));