123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465 |
- /**
- * jQuery WeUI V1.2.1
- * By 言川
- * http://lihongxun945.github.io/jquery-weui/
- */
- /* global $:true */
- /* global WebKitCSSMatrix:true */
- (function($) {
- "use strict";
- $.fn.transitionEnd = function(callback) {
- var events = ['webkitTransitionEnd', 'transitionend', 'oTransitionEnd', 'MSTransitionEnd', 'msTransitionEnd'],
- i, dom = this;
- function fireCallBack(e) {
- /*jshint validthis:true */
- if (e.target !== this) return;
- callback.call(this, e);
- for (i = 0; i < events.length; i++) {
- dom.off(events[i], fireCallBack);
- }
- }
- if (callback) {
- for (i = 0; i < events.length; i++) {
- dom.on(events[i], fireCallBack);
- }
- }
- return this;
- };
- $.support = (function() {
- var support = {
- touch: !!(('ontouchstart' in window) || window.DocumentTouch && document instanceof window.DocumentTouch)
- };
- return support;
- })();
- $.touchEvents = {
- start: $.support.touch ? 'touchstart' : 'mousedown',
- move: $.support.touch ? 'touchmove' : 'mousemove',
- end: $.support.touch ? 'touchend' : 'mouseup'
- };
- $.getTouchPosition = function(e) {
- e = e.originalEvent || e; //jquery wrap the originevent
- if(e.type === 'touchstart' || e.type === 'touchmove' || e.type === 'touchend') {
- return {
- x: e.targetTouches[0].pageX,
- y: e.targetTouches[0].pageY
- };
- } else {
- return {
- x: e.pageX,
- y: e.pageY
- };
- }
- };
- $.fn.scrollHeight = function() {
- return this[0].scrollHeight;
- };
- $.fn.transform = function(transform) {
- for (var i = 0; i < this.length; i++) {
- var elStyle = this[i].style;
- elStyle.webkitTransform = elStyle.MsTransform = elStyle.msTransform = elStyle.MozTransform = elStyle.OTransform = elStyle.transform = transform;
- }
- return this;
- };
- $.fn.transition = function(duration) {
- if (typeof duration !== 'string') {
- duration = duration + 'ms';
- }
- for (var i = 0; i < this.length; i++) {
- var elStyle = this[i].style;
- elStyle.webkitTransitionDuration = elStyle.MsTransitionDuration = elStyle.msTransitionDuration = elStyle.MozTransitionDuration = elStyle.OTransitionDuration = elStyle.transitionDuration = duration;
- }
- return this;
- };
- $.getTranslate = function (el, axis) {
- var matrix, curTransform, curStyle, transformMatrix;
- // automatic axis detection
- if (typeof axis === 'undefined') {
- axis = 'x';
- }
- curStyle = window.getComputedStyle(el, null);
- if (window.WebKitCSSMatrix) {
- // Some old versions of Webkit choke when 'none' is passed; pass
- // empty string instead in this case
- transformMatrix = new WebKitCSSMatrix(curStyle.webkitTransform === 'none' ? '' : curStyle.webkitTransform);
- }
- else {
- transformMatrix = curStyle.MozTransform || curStyle.OTransform || curStyle.MsTransform || curStyle.msTransform || curStyle.transform || curStyle.getPropertyValue('transform').replace('translate(', 'matrix(1, 0, 0, 1,');
- matrix = transformMatrix.toString().split(',');
- }
- if (axis === 'x') {
- //Latest Chrome and webkits Fix
- if (window.WebKitCSSMatrix)
- curTransform = transformMatrix.m41;
- //Crazy IE10 Matrix
- else if (matrix.length === 16)
- curTransform = parseFloat(matrix[12]);
- //Normal Browsers
- else
- curTransform = parseFloat(matrix[4]);
- }
- if (axis === 'y') {
- //Latest Chrome and webkits Fix
- if (window.WebKitCSSMatrix)
- curTransform = transformMatrix.m42;
- //Crazy IE10 Matrix
- else if (matrix.length === 16)
- curTransform = parseFloat(matrix[13]);
- //Normal Browsers
- else
- curTransform = parseFloat(matrix[5]);
- }
- return curTransform || 0;
- };
- $.requestAnimationFrame = function (callback) {
- if (window.requestAnimationFrame) return window.requestAnimationFrame(callback);
- else if (window.webkitRequestAnimationFrame) return window.webkitRequestAnimationFrame(callback);
- else if (window.mozRequestAnimationFrame) return window.mozRequestAnimationFrame(callback);
- else {
- return window.setTimeout(callback, 1000 / 60);
- }
- };
- $.cancelAnimationFrame = function (id) {
- if (window.cancelAnimationFrame) return window.cancelAnimationFrame(id);
- else if (window.webkitCancelAnimationFrame) return window.webkitCancelAnimationFrame(id);
- else if (window.mozCancelAnimationFrame) return window.mozCancelAnimationFrame(id);
- else {
- return window.clearTimeout(id);
- }
- };
- $.fn.join = function(arg) {
- return this.toArray().join(arg);
- }
- })($);
- /*===========================
- Template7 Template engine
- ===========================*/
- /* global $:true */
- /* jshint unused:false */
- /* jshint forin:false */
- +function ($) {
- "use strict";
- $.Template7 = $.t7 = (function () {
- function isArray(arr) {
- return Object.prototype.toString.apply(arr) === '[object Array]';
- }
- function isObject(obj) {
- return obj instanceof Object;
- }
- function isFunction(func) {
- return typeof func === 'function';
- }
- var cache = {};
- function helperToSlices(string) {
- var helperParts = string.replace(/[{}#}]/g, '').split(' ');
- var slices = [];
- var shiftIndex, i, j;
- for (i = 0; i < helperParts.length; i++) {
- var part = helperParts[i];
- if (i === 0) slices.push(part);
- else {
- if (part.indexOf('"') === 0) {
- // Plain String
- if (part.match(/"/g).length === 2) {
- // One word string
- slices.push(part);
- }
- else {
- // Find closed Index
- shiftIndex = 0;
- for (j = i + 1; j < helperParts.length; j++) {
- part += ' ' + helperParts[j];
- if (helperParts[j].indexOf('"') >= 0) {
- shiftIndex = j;
- slices.push(part);
- break;
- }
- }
- if (shiftIndex) i = shiftIndex;
- }
- }
- else {
- if (part.indexOf('=') > 0) {
- // Hash
- var hashParts = part.split('=');
- var hashName = hashParts[0];
- var hashContent = hashParts[1];
- if (hashContent.match(/"/g).length !== 2) {
- shiftIndex = 0;
- for (j = i + 1; j < helperParts.length; j++) {
- hashContent += ' ' + helperParts[j];
- if (helperParts[j].indexOf('"') >= 0) {
- shiftIndex = j;
- break;
- }
- }
- if (shiftIndex) i = shiftIndex;
- }
- var hash = [hashName, hashContent.replace(/"/g,'')];
- slices.push(hash);
- }
- else {
- // Plain variable
- slices.push(part);
- }
- }
- }
- }
- return slices;
- }
- function stringToBlocks(string) {
- var blocks = [], i, j, k;
- if (!string) return [];
- var _blocks = string.split(/({{[^{^}]*}})/);
- for (i = 0; i < _blocks.length; i++) {
- var block = _blocks[i];
- if (block === '') continue;
- if (block.indexOf('{{') < 0) {
- blocks.push({
- type: 'plain',
- content: block
- });
- }
- else {
- if (block.indexOf('{/') >= 0) {
- continue;
- }
- if (block.indexOf('{#') < 0 && block.indexOf(' ') < 0 && block.indexOf('else') < 0) {
- // Simple variable
- blocks.push({
- type: 'variable',
- contextName: block.replace(/[{}]/g, '')
- });
- continue;
- }
- // Helpers
- var helperSlices = helperToSlices(block);
- var helperName = helperSlices[0];
- var helperContext = [];
- var helperHash = {};
- for (j = 1; j < helperSlices.length; j++) {
- var slice = helperSlices[j];
- if (isArray(slice)) {
- // Hash
- helperHash[slice[0]] = slice[1] === 'false' ? false : slice[1];
- }
- else {
- helperContext.push(slice);
- }
- }
- if (block.indexOf('{#') >= 0) {
- // Condition/Helper
- var helperStartIndex = i;
- var helperContent = '';
- var elseContent = '';
- var toSkip = 0;
- var shiftIndex;
- var foundClosed = false, foundElse = false, foundClosedElse = false, depth = 0;
- for (j = i + 1; j < _blocks.length; j++) {
- if (_blocks[j].indexOf('{{#') >= 0) {
- depth ++;
- }
- if (_blocks[j].indexOf('{{/') >= 0) {
- depth --;
- }
- if (_blocks[j].indexOf('{{#' + helperName) >= 0) {
- helperContent += _blocks[j];
- if (foundElse) elseContent += _blocks[j];
- toSkip ++;
- }
- else if (_blocks[j].indexOf('{{/' + helperName) >= 0) {
- if (toSkip > 0) {
- toSkip--;
- helperContent += _blocks[j];
- if (foundElse) elseContent += _blocks[j];
- }
- else {
- shiftIndex = j;
- foundClosed = true;
- break;
- }
- }
- else if (_blocks[j].indexOf('else') >= 0 && depth === 0) {
- foundElse = true;
- }
- else {
- if (!foundElse) helperContent += _blocks[j];
- if (foundElse) elseContent += _blocks[j];
- }
- }
- if (foundClosed) {
- if (shiftIndex) i = shiftIndex;
- blocks.push({
- type: 'helper',
- helperName: helperName,
- contextName: helperContext,
- content: helperContent,
- inverseContent: elseContent,
- hash: helperHash
- });
- }
- }
- else if (block.indexOf(' ') > 0) {
- blocks.push({
- type: 'helper',
- helperName: helperName,
- contextName: helperContext,
- hash: helperHash
- });
- }
- }
- }
- return blocks;
- }
- var Template7 = function (template) {
- var t = this;
- t.template = template;
- function getCompileFn(block, depth) {
- if (block.content) return compile(block.content, depth);
- else return function () {return ''; };
- }
- function getCompileInverse(block, depth) {
- if (block.inverseContent) return compile(block.inverseContent, depth);
- else return function () {return ''; };
- }
- function getCompileVar(name, ctx) {
- var variable, parts, levelsUp = 0, initialCtx = ctx;
- if (name.indexOf('../') === 0) {
- levelsUp = name.split('../').length - 1;
- var newDepth = ctx.split('_')[1] - levelsUp;
- ctx = 'ctx_' + (newDepth >= 1 ? newDepth : 1);
- parts = name.split('../')[levelsUp].split('.');
- }
- else if (name.indexOf('@global') === 0) {
- ctx = '$.Template7.global';
- parts = name.split('@global.')[1].split('.');
- }
- else if (name.indexOf('@root') === 0) {
- ctx = 'ctx_1';
- parts = name.split('@root.')[1].split('.');
- }
- else {
- parts = name.split('.');
- }
- variable = ctx;
- for (var i = 0; i < parts.length; i++) {
- var part = parts[i];
- if (part.indexOf('@') === 0) {
- if (i > 0) {
- variable += '[(data && data.' + part.replace('@', '') + ')]';
- }
- else {
- variable = '(data && data.' + name.replace('@', '') + ')';
- }
- }
- else {
- if (isFinite(part)) {
- variable += '[' + part + ']';
- }
- else {
- if (part.indexOf('this') === 0) {
- variable = part.replace('this', ctx);
- }
- else {
- variable += '.' + part;
- }
- }
- }
- }
- return variable;
- }
- function getCompiledArguments(contextArray, ctx) {
- var arr = [];
- for (var i = 0; i < contextArray.length; i++) {
- if (contextArray[i].indexOf('"') === 0) arr.push(contextArray[i]);
- else {
- arr.push(getCompileVar(contextArray[i], ctx));
- }
- }
- return arr.join(', ');
- }
- function compile(template, depth) {
- depth = depth || 1;
- template = template || t.template;
- if (typeof template !== 'string') {
- throw new Error('Template7: Template must be a string');
- }
- var blocks = stringToBlocks(template);
- if (blocks.length === 0) {
- return function () { return ''; };
- }
- var ctx = 'ctx_' + depth;
- var resultString = '(function (' + ctx + ', data) {\n';
- if (depth === 1) {
- resultString += 'function isArray(arr){return Object.prototype.toString.apply(arr) === \'[object Array]\';}\n';
- resultString += 'function isFunction(func){return (typeof func === \'function\');}\n';
- resultString += 'function c(val, ctx) {if (typeof val !== "undefined") {if (isFunction(val)) {return val.call(ctx);} else return val;} else return "";}\n';
- }
- resultString += 'var r = \'\';\n';
- var i, j, context;
- for (i = 0; i < blocks.length; i++) {
- var block = blocks[i];
- // Plain block
- if (block.type === 'plain') {
- resultString += 'r +=\'' + (block.content).replace(/\r/g, '\\r').replace(/\n/g, '\\n').replace(/'/g, '\\' + '\'') + '\';';
- continue;
- }
- var variable, compiledArguments;
- // Variable block
- if (block.type === 'variable') {
- variable = getCompileVar(block.contextName, ctx);
- resultString += 'r += c(' + variable + ', ' + ctx + ');';
- }
- // Helpers block
- if (block.type === 'helper') {
- if (block.helperName in t.helpers) {
- compiledArguments = getCompiledArguments(block.contextName, ctx);
- resultString += 'r += ($.Template7.helpers.' + block.helperName + ').call(' + ctx + ', ' + (compiledArguments && (compiledArguments + ', ')) +'{hash:' + JSON.stringify(block.hash) + ', data: data || {}, fn: ' + getCompileFn(block, depth+1) + ', inverse: ' + getCompileInverse(block, depth+1) + ', root: ctx_1});';
- }
- else {
- if (block.contextName.length > 0) {
- throw new Error('Template7: Missing helper: "' + block.helperName + '"');
- }
- else {
- variable = getCompileVar(block.helperName, ctx);
- resultString += 'if (' + variable + ') {';
- resultString += 'if (isArray(' + variable + ')) {';
- resultString += 'r += ($.Template7.helpers.each).call(' + ctx + ', ' + variable + ', {hash:' + JSON.stringify(block.hash) + ', data: data || {}, fn: ' + getCompileFn(block, depth+1) + ', inverse: ' + getCompileInverse(block, depth+1) + ', root: ctx_1});';
- resultString += '}else {';
- resultString += 'r += ($.Template7.helpers.with).call(' + ctx + ', ' + variable + ', {hash:' + JSON.stringify(block.hash) + ', data: data || {}, fn: ' + getCompileFn(block, depth+1) + ', inverse: ' + getCompileInverse(block, depth+1) + ', root: ctx_1});';
- resultString += '}}';
- }
- }
- }
- }
- resultString += '\nreturn r;})';
- return eval.call(window, resultString);
- }
- t.compile = function (template) {
- if (!t.compiled) {
- t.compiled = compile(template);
- }
- return t.compiled;
- };
- };
- Template7.prototype = {
- options: {},
- helpers: {
- 'if': function (context, options) {
- if (isFunction(context)) { context = context.call(this); }
- if (context) {
- return options.fn(this, options.data);
- }
- else {
- return options.inverse(this, options.data);
- }
- },
- 'unless': function (context, options) {
- if (isFunction(context)) { context = context.call(this); }
- if (!context) {
- return options.fn(this, options.data);
- }
- else {
- return options.inverse(this, options.data);
- }
- },
- 'each': function (context, options) {
- var ret = '', i = 0;
- if (isFunction(context)) { context = context.call(this); }
- if (isArray(context)) {
- if (options.hash.reverse) {
- context = context.reverse();
- }
- for (i = 0; i < context.length; i++) {
- ret += options.fn(context[i], {first: i === 0, last: i === context.length - 1, index: i});
- }
- if (options.hash.reverse) {
- context = context.reverse();
- }
- }
- else {
- for (var key in context) {
- i++;
- ret += options.fn(context[key], {key: key});
- }
- }
- if (i > 0) return ret;
- else return options.inverse(this);
- },
- 'with': function (context, options) {
- if (isFunction(context)) { context = context.call(this); }
- return options.fn(context);
- },
- 'join': function (context, options) {
- if (isFunction(context)) { context = context.call(this); }
- return context.join(options.hash.delimiter || options.hash.delimeter);
- },
- 'js': function (expression, options) {
- var func;
- if (expression.indexOf('return')>=0) {
- func = '(function(){'+expression+'})';
- }
- else {
- func = '(function(){return ('+expression+')})';
- }
- return eval.call(this, func).call(this);
- },
- 'js_compare': function (expression, options) {
- var func;
- if (expression.indexOf('return')>=0) {
- func = '(function(){'+expression+'})';
- }
- else {
- func = '(function(){return ('+expression+')})';
- }
- var condition = eval.call(this, func).call(this);
- if (condition) {
- return options.fn(this, options.data);
- }
- else {
- return options.inverse(this, options.data);
- }
- }
- }
- };
- var t7 = function (template, data) {
- if (arguments.length === 2) {
- var instance = new Template7(template);
- var rendered = instance.compile()(data);
- instance = null;
- return (rendered);
- }
- else return new Template7(template);
- };
- t7.registerHelper = function (name, fn) {
- Template7.prototype.helpers[name] = fn;
- };
- t7.unregisterHelper = function (name) {
- Template7.prototype.helpers[name] = undefined;
- delete Template7.prototype.helpers[name];
- };
- t7.compile = function (template, options) {
- var instance = new Template7(template, options);
- return instance.compile();
- };
- t7.options = Template7.prototype.options;
- t7.helpers = Template7.prototype.helpers;
- return t7;
- })();
- }($);
- /*! Hammer.JS - v2.0.8 - 2016-04-23
- * http://hammerjs.github.io/
- *
- * Copyright (c) 2016 Jorik Tangelder;
- * Licensed under the MIT license */
- (function(window, document, exportName, undefined) {
- 'use strict';
- var VENDOR_PREFIXES = ['', 'webkit', 'Moz', 'MS', 'ms', 'o'];
- var TEST_ELEMENT = document.createElement('div');
- var TYPE_FUNCTION = 'function';
- var round = Math.round;
- var abs = Math.abs;
- var now = Date.now;
- /**
- * set a timeout with a given scope
- * @param {Function} fn
- * @param {Number} timeout
- * @param {Object} context
- * @returns {number}
- */
- function setTimeoutContext(fn, timeout, context) {
- return setTimeout(bindFn(fn, context), timeout);
- }
- /**
- * if the argument is an array, we want to execute the fn on each entry
- * if it aint an array we don't want to do a thing.
- * this is used by all the methods that accept a single and array argument.
- * @param {*|Array} arg
- * @param {String} fn
- * @param {Object} [context]
- * @returns {Boolean}
- */
- function invokeArrayArg(arg, fn, context) {
- if (Array.isArray(arg)) {
- each(arg, context[fn], context);
- return true;
- }
- return false;
- }
- /**
- * walk objects and arrays
- * @param {Object} obj
- * @param {Function} iterator
- * @param {Object} context
- */
- function each(obj, iterator, context) {
- var i;
- if (!obj) {
- return;
- }
- if (obj.forEach) {
- obj.forEach(iterator, context);
- } else if (obj.length !== undefined) {
- i = 0;
- while (i < obj.length) {
- iterator.call(context, obj[i], i, obj);
- i++;
- }
- } else {
- for (i in obj) {
- obj.hasOwnProperty(i) && iterator.call(context, obj[i], i, obj);
- }
- }
- }
- /**
- * wrap a method with a deprecation warning and stack trace
- * @param {Function} method
- * @param {String} name
- * @param {String} message
- * @returns {Function} A new function wrapping the supplied method.
- */
- function deprecate(method, name, message) {
- var deprecationMessage = 'DEPRECATED METHOD: ' + name + '\n' + message + ' AT \n';
- return function() {
- var e = new Error('get-stack-trace');
- var stack = e && e.stack ? e.stack.replace(/^[^\(]+?[\n$]/gm, '')
- .replace(/^\s+at\s+/gm, '')
- .replace(/^Object.<anonymous>\s*\(/gm, '{anonymous}()@') : 'Unknown Stack Trace';
- var log = window.console && (window.console.warn || window.console.log);
- if (log) {
- log.call(window.console, deprecationMessage, stack);
- }
- return method.apply(this, arguments);
- };
- }
- /**
- * extend object.
- * means that properties in dest will be overwritten by the ones in src.
- * @param {Object} target
- * @param {...Object} objects_to_assign
- * @returns {Object} target
- */
- var assign;
- if (typeof Object.assign !== 'function') {
- assign = function assign(target) {
- if (target === undefined || target === null) {
- throw new TypeError('Cannot convert undefined or null to object');
- }
- var output = Object(target);
- for (var index = 1; index < arguments.length; index++) {
- var source = arguments[index];
- if (source !== undefined && source !== null) {
- for (var nextKey in source) {
- if (source.hasOwnProperty(nextKey)) {
- output[nextKey] = source[nextKey];
- }
- }
- }
- }
- return output;
- };
- } else {
- assign = Object.assign;
- }
- /**
- * extend object.
- * means that properties in dest will be overwritten by the ones in src.
- * @param {Object} dest
- * @param {Object} src
- * @param {Boolean} [merge=false]
- * @returns {Object} dest
- */
- var extend = deprecate(function extend(dest, src, merge) {
- var keys = Object.keys(src);
- var i = 0;
- while (i < keys.length) {
- if (!merge || (merge && dest[keys[i]] === undefined)) {
- dest[keys[i]] = src[keys[i]];
- }
- i++;
- }
- return dest;
- }, 'extend', 'Use `assign`.');
- /**
- * merge the values from src in the dest.
- * means that properties that exist in dest will not be overwritten by src
- * @param {Object} dest
- * @param {Object} src
- * @returns {Object} dest
- */
- var merge = deprecate(function merge(dest, src) {
- return extend(dest, src, true);
- }, 'merge', 'Use `assign`.');
- /**
- * simple class inheritance
- * @param {Function} child
- * @param {Function} base
- * @param {Object} [properties]
- */
- function inherit(child, base, properties) {
- var baseP = base.prototype,
- childP;
- childP = child.prototype = Object.create(baseP);
- childP.constructor = child;
- childP._super = baseP;
- if (properties) {
- assign(childP, properties);
- }
- }
- /**
- * simple function bind
- * @param {Function} fn
- * @param {Object} context
- * @returns {Function}
- */
- function bindFn(fn, context) {
- return function boundFn() {
- return fn.apply(context, arguments);
- };
- }
- /**
- * let a boolean value also be a function that must return a boolean
- * this first item in args will be used as the context
- * @param {Boolean|Function} val
- * @param {Array} [args]
- * @returns {Boolean}
- */
- function boolOrFn(val, args) {
- if (typeof val == TYPE_FUNCTION) {
- return val.apply(args ? args[0] || undefined : undefined, args);
- }
- return val;
- }
- /**
- * use the val2 when val1 is undefined
- * @param {*} val1
- * @param {*} val2
- * @returns {*}
- */
- function ifUndefined(val1, val2) {
- return (val1 === undefined) ? val2 : val1;
- }
- /**
- * addEventListener with multiple events at once
- * @param {EventTarget} target
- * @param {String} types
- * @param {Function} handler
- */
- function addEventListeners(target, types, handler) {
- each(splitStr(types), function(type) {
- target.addEventListener(type, handler, false);
- });
- }
- /**
- * removeEventListener with multiple events at once
- * @param {EventTarget} target
- * @param {String} types
- * @param {Function} handler
- */
- function removeEventListeners(target, types, handler) {
- each(splitStr(types), function(type) {
- target.removeEventListener(type, handler, false);
- });
- }
- /**
- * find if a node is in the given parent
- * @method hasParent
- * @param {HTMLElement} node
- * @param {HTMLElement} parent
- * @return {Boolean} found
- */
- function hasParent(node, parent) {
- while (node) {
- if (node == parent) {
- return true;
- }
- node = node.parentNode;
- }
- return false;
- }
- /**
- * small indexOf wrapper
- * @param {String} str
- * @param {String} find
- * @returns {Boolean} found
- */
- function inStr(str, find) {
- return str.indexOf(find) > -1;
- }
- /**
- * split string on whitespace
- * @param {String} str
- * @returns {Array} words
- */
- function splitStr(str) {
- return str.trim().split(/\s+/g);
- }
- /**
- * find if a array contains the object using indexOf or a simple polyFill
- * @param {Array} src
- * @param {String} find
- * @param {String} [findByKey]
- * @return {Boolean|Number} false when not found, or the index
- */
- function inArray(src, find, findByKey) {
- if (src.indexOf && !findByKey) {
- return src.indexOf(find);
- } else {
- var i = 0;
- while (i < src.length) {
- if ((findByKey && src[i][findByKey] == find) || (!findByKey && src[i] === find)) {
- return i;
- }
- i++;
- }
- return -1;
- }
- }
- /**
- * convert array-like objects to real arrays
- * @param {Object} obj
- * @returns {Array}
- */
- function toArray(obj) {
- return Array.prototype.slice.call(obj, 0);
- }
- /**
- * unique array with objects based on a key (like 'id') or just by the array's value
- * @param {Array} src [{id:1},{id:2},{id:1}]
- * @param {String} [key]
- * @param {Boolean} [sort=False]
- * @returns {Array} [{id:1},{id:2}]
- */
- function uniqueArray(src, key, sort) {
- var results = [];
- var values = [];
- var i = 0;
- while (i < src.length) {
- var val = key ? src[i][key] : src[i];
- if (inArray(values, val) < 0) {
- results.push(src[i]);
- }
- values[i] = val;
- i++;
- }
- if (sort) {
- if (!key) {
- results = results.sort();
- } else {
- results = results.sort(function sortUniqueArray(a, b) {
- return a[key] > b[key];
- });
- }
- }
- return results;
- }
- /**
- * get the prefixed property
- * @param {Object} obj
- * @param {String} property
- * @returns {String|Undefined} prefixed
- */
- function prefixed(obj, property) {
- var prefix, prop;
- var camelProp = property[0].toUpperCase() + property.slice(1);
- var i = 0;
- while (i < VENDOR_PREFIXES.length) {
- prefix = VENDOR_PREFIXES[i];
- prop = (prefix) ? prefix + camelProp : property;
- if (prop in obj) {
- return prop;
- }
- i++;
- }
- return undefined;
- }
- /**
- * get a unique id
- * @returns {number} uniqueId
- */
- var _uniqueId = 1;
- function uniqueId() {
- return _uniqueId++;
- }
- /**
- * get the window object of an element
- * @param {HTMLElement} element
- * @returns {DocumentView|Window}
- */
- function getWindowForElement(element) {
- var doc = element.ownerDocument || element;
- return (doc.defaultView || doc.parentWindow || window);
- }
- var MOBILE_REGEX = /mobile|tablet|ip(ad|hone|od)|android/i;
- var SUPPORT_TOUCH = ('ontouchstart' in window);
- var SUPPORT_POINTER_EVENTS = prefixed(window, 'PointerEvent') !== undefined;
- var SUPPORT_ONLY_TOUCH = SUPPORT_TOUCH && MOBILE_REGEX.test(navigator.userAgent);
- var INPUT_TYPE_TOUCH = 'touch';
- var INPUT_TYPE_PEN = 'pen';
- var INPUT_TYPE_MOUSE = 'mouse';
- var INPUT_TYPE_KINECT = 'kinect';
- var COMPUTE_INTERVAL = 25;
- var INPUT_START = 1;
- var INPUT_MOVE = 2;
- var INPUT_END = 4;
- var INPUT_CANCEL = 8;
- var DIRECTION_NONE = 1;
- var DIRECTION_LEFT = 2;
- var DIRECTION_RIGHT = 4;
- var DIRECTION_UP = 8;
- var DIRECTION_DOWN = 16;
- var DIRECTION_HORIZONTAL = DIRECTION_LEFT | DIRECTION_RIGHT;
- var DIRECTION_VERTICAL = DIRECTION_UP | DIRECTION_DOWN;
- var DIRECTION_ALL = DIRECTION_HORIZONTAL | DIRECTION_VERTICAL;
- var PROPS_XY = ['x', 'y'];
- var PROPS_CLIENT_XY = ['clientX', 'clientY'];
- /**
- * create new input type manager
- * @param {Manager} manager
- * @param {Function} callback
- * @returns {Input}
- * @constructor
- */
- function Input(manager, callback) {
- var self = this;
- this.manager = manager;
- this.callback = callback;
- this.element = manager.element;
- this.target = manager.options.inputTarget;
- // smaller wrapper around the handler, for the scope and the enabled state of the manager,
- // so when disabled the input events are completely bypassed.
- this.domHandler = function(ev) {
- if (boolOrFn(manager.options.enable, [manager])) {
- self.handler(ev);
- }
- };
- this.init();
- }
- Input.prototype = {
- /**
- * should handle the inputEvent data and trigger the callback
- * @virtual
- */
- handler: function() { },
- /**
- * bind the events
- */
- init: function() {
- this.evEl && addEventListeners(this.element, this.evEl, this.domHandler);
- this.evTarget && addEventListeners(this.target, this.evTarget, this.domHandler);
- this.evWin && addEventListeners(getWindowForElement(this.element), this.evWin, this.domHandler);
- },
- /**
- * unbind the events
- */
- destroy: function() {
- this.evEl && removeEventListeners(this.element, this.evEl, this.domHandler);
- this.evTarget && removeEventListeners(this.target, this.evTarget, this.domHandler);
- this.evWin && removeEventListeners(getWindowForElement(this.element), this.evWin, this.domHandler);
- }
- };
- /**
- * create new input type manager
- * called by the Manager constructor
- * @param {Hammer} manager
- * @returns {Input}
- */
- function createInputInstance(manager) {
- var Type;
- var inputClass = manager.options.inputClass;
- if (inputClass) {
- Type = inputClass;
- } else if (SUPPORT_POINTER_EVENTS) {
- Type = PointerEventInput;
- } else if (SUPPORT_ONLY_TOUCH) {
- Type = TouchInput;
- } else if (!SUPPORT_TOUCH) {
- Type = MouseInput;
- } else {
- Type = TouchMouseInput;
- }
- return new (Type)(manager, inputHandler);
- }
- /**
- * handle input events
- * @param {Manager} manager
- * @param {String} eventType
- * @param {Object} input
- */
- function inputHandler(manager, eventType, input) {
- var pointersLen = input.pointers.length;
- var changedPointersLen = input.changedPointers.length;
- var isFirst = (eventType & INPUT_START && (pointersLen - changedPointersLen === 0));
- var isFinal = (eventType & (INPUT_END | INPUT_CANCEL) && (pointersLen - changedPointersLen === 0));
- input.isFirst = !!isFirst;
- input.isFinal = !!isFinal;
- if (isFirst) {
- manager.session = {};
- }
- // source event is the normalized value of the domEvents
- // like 'touchstart, mouseup, pointerdown'
- input.eventType = eventType;
- // compute scale, rotation etc
- computeInputData(manager, input);
- // emit secret event
- manager.emit('hammer.input', input);
- manager.recognize(input);
- manager.session.prevInput = input;
- }
- /**
- * extend the data with some usable properties like scale, rotate, velocity etc
- * @param {Object} manager
- * @param {Object} input
- */
- function computeInputData(manager, input) {
- var session = manager.session;
- var pointers = input.pointers;
- var pointersLength = pointers.length;
- // store the first input to calculate the distance and direction
- if (!session.firstInput) {
- session.firstInput = simpleCloneInputData(input);
- }
- // to compute scale and rotation we need to store the multiple touches
- if (pointersLength > 1 && !session.firstMultiple) {
- session.firstMultiple = simpleCloneInputData(input);
- } else if (pointersLength === 1) {
- session.firstMultiple = false;
- }
- var firstInput = session.firstInput;
- var firstMultiple = session.firstMultiple;
- var offsetCenter = firstMultiple ? firstMultiple.center : firstInput.center;
- var center = input.center = getCenter(pointers);
- input.timeStamp = now();
- input.deltaTime = input.timeStamp - firstInput.timeStamp;
- input.angle = getAngle(offsetCenter, center);
- input.distance = getDistance(offsetCenter, center);
- computeDeltaXY(session, input);
- input.offsetDirection = getDirection(input.deltaX, input.deltaY);
- var overallVelocity = getVelocity(input.deltaTime, input.deltaX, input.deltaY);
- input.overallVelocityX = overallVelocity.x;
- input.overallVelocityY = overallVelocity.y;
- input.overallVelocity = (abs(overallVelocity.x) > abs(overallVelocity.y)) ? overallVelocity.x : overallVelocity.y;
- input.scale = firstMultiple ? getScale(firstMultiple.pointers, pointers) : 1;
- input.rotation = firstMultiple ? getRotation(firstMultiple.pointers, pointers) : 0;
- input.maxPointers = !session.prevInput ? input.pointers.length : ((input.pointers.length >
- session.prevInput.maxPointers) ? input.pointers.length : session.prevInput.maxPointers);
- computeIntervalInputData(session, input);
- // find the correct target
- var target = manager.element;
- if (hasParent(input.srcEvent.target, target)) {
- target = input.srcEvent.target;
- }
- input.target = target;
- }
- function computeDeltaXY(session, input) {
- var center = input.center;
- var offset = session.offsetDelta || {};
- var prevDelta = session.prevDelta || {};
- var prevInput = session.prevInput || {};
- if (input.eventType === INPUT_START || prevInput.eventType === INPUT_END) {
- prevDelta = session.prevDelta = {
- x: prevInput.deltaX || 0,
- y: prevInput.deltaY || 0
- };
- offset = session.offsetDelta = {
- x: center.x,
- y: center.y
- };
- }
- input.deltaX = prevDelta.x + (center.x - offset.x);
- input.deltaY = prevDelta.y + (center.y - offset.y);
- }
- /**
- * velocity is calculated every x ms
- * @param {Object} session
- * @param {Object} input
- */
- function computeIntervalInputData(session, input) {
- var last = session.lastInterval || input,
- deltaTime = input.timeStamp - last.timeStamp,
- velocity, velocityX, velocityY, direction;
- if (input.eventType != INPUT_CANCEL && (deltaTime > COMPUTE_INTERVAL || last.velocity === undefined)) {
- var deltaX = input.deltaX - last.deltaX;
- var deltaY = input.deltaY - last.deltaY;
- var v = getVelocity(deltaTime, deltaX, deltaY);
- velocityX = v.x;
- velocityY = v.y;
- velocity = (abs(v.x) > abs(v.y)) ? v.x : v.y;
- direction = getDirection(deltaX, deltaY);
- session.lastInterval = input;
- } else {
- // use latest velocity info if it doesn't overtake a minimum period
- velocity = last.velocity;
- velocityX = last.velocityX;
- velocityY = last.velocityY;
- direction = last.direction;
- }
- input.velocity = velocity;
- input.velocityX = velocityX;
- input.velocityY = velocityY;
- input.direction = direction;
- }
- /**
- * create a simple clone from the input used for storage of firstInput and firstMultiple
- * @param {Object} input
- * @returns {Object} clonedInputData
- */
- function simpleCloneInputData(input) {
- // make a simple copy of the pointers because we will get a reference if we don't
- // we only need clientXY for the calculations
- var pointers = [];
- var i = 0;
- while (i < input.pointers.length) {
- pointers[i] = {
- clientX: round(input.pointers[i].clientX),
- clientY: round(input.pointers[i].clientY)
- };
- i++;
- }
- return {
- timeStamp: now(),
- pointers: pointers,
- center: getCenter(pointers),
- deltaX: input.deltaX,
- deltaY: input.deltaY
- };
- }
- /**
- * get the center of all the pointers
- * @param {Array} pointers
- * @return {Object} center contains `x` and `y` properties
- */
- function getCenter(pointers) {
- var pointersLength = pointers.length;
- // no need to loop when only one touch
- if (pointersLength === 1) {
- return {
- x: round(pointers[0].clientX),
- y: round(pointers[0].clientY)
- };
- }
- var x = 0, y = 0, i = 0;
- while (i < pointersLength) {
- x += pointers[i].clientX;
- y += pointers[i].clientY;
- i++;
- }
- return {
- x: round(x / pointersLength),
- y: round(y / pointersLength)
- };
- }
- /**
- * calculate the velocity between two points. unit is in px per ms.
- * @param {Number} deltaTime
- * @param {Number} x
- * @param {Number} y
- * @return {Object} velocity `x` and `y`
- */
- function getVelocity(deltaTime, x, y) {
- return {
- x: x / deltaTime || 0,
- y: y / deltaTime || 0
- };
- }
- /**
- * get the direction between two points
- * @param {Number} x
- * @param {Number} y
- * @return {Number} direction
- */
- function getDirection(x, y) {
- if (x === y) {
- return DIRECTION_NONE;
- }
- if (abs(x) >= abs(y)) {
- return x < 0 ? DIRECTION_LEFT : DIRECTION_RIGHT;
- }
- return y < 0 ? DIRECTION_UP : DIRECTION_DOWN;
- }
- /**
- * calculate the absolute distance between two points
- * @param {Object} p1 {x, y}
- * @param {Object} p2 {x, y}
- * @param {Array} [props] containing x and y keys
- * @return {Number} distance
- */
- function getDistance(p1, p2, props) {
- if (!props) {
- props = PROPS_XY;
- }
- var x = p2[props[0]] - p1[props[0]],
- y = p2[props[1]] - p1[props[1]];
- return Math.sqrt((x * x) + (y * y));
- }
- /**
- * calculate the angle between two coordinates
- * @param {Object} p1
- * @param {Object} p2
- * @param {Array} [props] containing x and y keys
- * @return {Number} angle
- */
- function getAngle(p1, p2, props) {
- if (!props) {
- props = PROPS_XY;
- }
- var x = p2[props[0]] - p1[props[0]],
- y = p2[props[1]] - p1[props[1]];
- return Math.atan2(y, x) * 180 / Math.PI;
- }
- /**
- * calculate the rotation degrees between two pointersets
- * @param {Array} start array of pointers
- * @param {Array} end array of pointers
- * @return {Number} rotation
- */
- function getRotation(start, end) {
- return getAngle(end[1], end[0], PROPS_CLIENT_XY) + getAngle(start[1], start[0], PROPS_CLIENT_XY);
- }
- /**
- * calculate the scale factor between two pointersets
- * no scale is 1, and goes down to 0 when pinched together, and bigger when pinched out
- * @param {Array} start array of pointers
- * @param {Array} end array of pointers
- * @return {Number} scale
- */
- function getScale(start, end) {
- return getDistance(end[0], end[1], PROPS_CLIENT_XY) / getDistance(start[0], start[1], PROPS_CLIENT_XY);
- }
- var MOUSE_INPUT_MAP = {
- mousedown: INPUT_START,
- mousemove: INPUT_MOVE,
- mouseup: INPUT_END
- };
- var MOUSE_ELEMENT_EVENTS = 'mousedown';
- var MOUSE_WINDOW_EVENTS = 'mousemove mouseup';
- /**
- * Mouse events input
- * @constructor
- * @extends Input
- */
- function MouseInput() {
- this.evEl = MOUSE_ELEMENT_EVENTS;
- this.evWin = MOUSE_WINDOW_EVENTS;
- this.pressed = false; // mousedown state
- Input.apply(this, arguments);
- }
- inherit(MouseInput, Input, {
- /**
- * handle mouse events
- * @param {Object} ev
- */
- handler: function MEhandler(ev) {
- var eventType = MOUSE_INPUT_MAP[ev.type];
- // on start we want to have the left mouse button down
- if (eventType & INPUT_START && ev.button === 0) {
- this.pressed = true;
- }
- if (eventType & INPUT_MOVE && ev.which !== 1) {
- eventType = INPUT_END;
- }
- // mouse must be down
- if (!this.pressed) {
- return;
- }
- if (eventType & INPUT_END) {
- this.pressed = false;
- }
- this.callback(this.manager, eventType, {
- pointers: [ev],
- changedPointers: [ev],
- pointerType: INPUT_TYPE_MOUSE,
- srcEvent: ev
- });
- }
- });
- var POINTER_INPUT_MAP = {
- pointerdown: INPUT_START,
- pointermove: INPUT_MOVE,
- pointerup: INPUT_END,
- pointercancel: INPUT_CANCEL,
- pointerout: INPUT_CANCEL
- };
- // in IE10 the pointer types is defined as an enum
- var IE10_POINTER_TYPE_ENUM = {
- 2: INPUT_TYPE_TOUCH,
- 3: INPUT_TYPE_PEN,
- 4: INPUT_TYPE_MOUSE,
- 5: INPUT_TYPE_KINECT // see https://twitter.com/jacobrossi/status/480596438489890816
- };
- var POINTER_ELEMENT_EVENTS = 'pointerdown';
- var POINTER_WINDOW_EVENTS = 'pointermove pointerup pointercancel';
- // IE10 has prefixed support, and case-sensitive
- if (window.MSPointerEvent && !window.PointerEvent) {
- POINTER_ELEMENT_EVENTS = 'MSPointerDown';
- POINTER_WINDOW_EVENTS = 'MSPointerMove MSPointerUp MSPointerCancel';
- }
- /**
- * Pointer events input
- * @constructor
- * @extends Input
- */
- function PointerEventInput() {
- this.evEl = POINTER_ELEMENT_EVENTS;
- this.evWin = POINTER_WINDOW_EVENTS;
- Input.apply(this, arguments);
- this.store = (this.manager.session.pointerEvents = []);
- }
- inherit(PointerEventInput, Input, {
- /**
- * handle mouse events
- * @param {Object} ev
- */
- handler: function PEhandler(ev) {
- var store = this.store;
- var removePointer = false;
- var eventTypeNormalized = ev.type.toLowerCase().replace('ms', '');
- var eventType = POINTER_INPUT_MAP[eventTypeNormalized];
- var pointerType = IE10_POINTER_TYPE_ENUM[ev.pointerType] || ev.pointerType;
- var isTouch = (pointerType == INPUT_TYPE_TOUCH);
- // get index of the event in the store
- var storeIndex = inArray(store, ev.pointerId, 'pointerId');
- // start and mouse must be down
- if (eventType & INPUT_START && (ev.button === 0 || isTouch)) {
- if (storeIndex < 0) {
- store.push(ev);
- storeIndex = store.length - 1;
- }
- } else if (eventType & (INPUT_END | INPUT_CANCEL)) {
- removePointer = true;
- }
- // it not found, so the pointer hasn't been down (so it's probably a hover)
- if (storeIndex < 0) {
- return;
- }
- // update the event in the store
- store[storeIndex] = ev;
- this.callback(this.manager, eventType, {
- pointers: store,
- changedPointers: [ev],
- pointerType: pointerType,
- srcEvent: ev
- });
- if (removePointer) {
- // remove from the store
- store.splice(storeIndex, 1);
- }
- }
- });
- var SINGLE_TOUCH_INPUT_MAP = {
- touchstart: INPUT_START,
- touchmove: INPUT_MOVE,
- touchend: INPUT_END,
- touchcancel: INPUT_CANCEL
- };
- var SINGLE_TOUCH_TARGET_EVENTS = 'touchstart';
- var SINGLE_TOUCH_WINDOW_EVENTS = 'touchstart touchmove touchend touchcancel';
- /**
- * Touch events input
- * @constructor
- * @extends Input
- */
- function SingleTouchInput() {
- this.evTarget = SINGLE_TOUCH_TARGET_EVENTS;
- this.evWin = SINGLE_TOUCH_WINDOW_EVENTS;
- this.started = false;
- Input.apply(this, arguments);
- }
- inherit(SingleTouchInput, Input, {
- handler: function TEhandler(ev) {
- var type = SINGLE_TOUCH_INPUT_MAP[ev.type];
- // should we handle the touch events?
- if (type === INPUT_START) {
- this.started = true;
- }
- if (!this.started) {
- return;
- }
- var touches = normalizeSingleTouches.call(this, ev, type);
- // when done, reset the started state
- if (type & (INPUT_END | INPUT_CANCEL) && touches[0].length - touches[1].length === 0) {
- this.started = false;
- }
- this.callback(this.manager, type, {
- pointers: touches[0],
- changedPointers: touches[1],
- pointerType: INPUT_TYPE_TOUCH,
- srcEvent: ev
- });
- }
- });
- /**
- * @this {TouchInput}
- * @param {Object} ev
- * @param {Number} type flag
- * @returns {undefined|Array} [all, changed]
- */
- function normalizeSingleTouches(ev, type) {
- var all = toArray(ev.touches);
- var changed = toArray(ev.changedTouches);
- if (type & (INPUT_END | INPUT_CANCEL)) {
- all = uniqueArray(all.concat(changed), 'identifier', true);
- }
- return [all, changed];
- }
- var TOUCH_INPUT_MAP = {
- touchstart: INPUT_START,
- touchmove: INPUT_MOVE,
- touchend: INPUT_END,
- touchcancel: INPUT_CANCEL
- };
- var TOUCH_TARGET_EVENTS = 'touchstart touchmove touchend touchcancel';
- /**
- * Multi-user touch events input
- * @constructor
- * @extends Input
- */
- function TouchInput() {
- this.evTarget = TOUCH_TARGET_EVENTS;
- this.targetIds = {};
- Input.apply(this, arguments);
- }
- inherit(TouchInput, Input, {
- handler: function MTEhandler(ev) {
- var type = TOUCH_INPUT_MAP[ev.type];
- var touches = getTouches.call(this, ev, type);
- if (!touches) {
- return;
- }
- this.callback(this.manager, type, {
- pointers: touches[0],
- changedPointers: touches[1],
- pointerType: INPUT_TYPE_TOUCH,
- srcEvent: ev
- });
- }
- });
- /**
- * @this {TouchInput}
- * @param {Object} ev
- * @param {Number} type flag
- * @returns {undefined|Array} [all, changed]
- */
- function getTouches(ev, type) {
- var allTouches = toArray(ev.touches);
- var targetIds = this.targetIds;
- // when there is only one touch, the process can be simplified
- if (type & (INPUT_START | INPUT_MOVE) && allTouches.length === 1) {
- targetIds[allTouches[0].identifier] = true;
- return [allTouches, allTouches];
- }
- var i,
- targetTouches,
- changedTouches = toArray(ev.changedTouches),
- changedTargetTouches = [],
- target = this.target;
- // get target touches from touches
- targetTouches = allTouches.filter(function(touch) {
- return hasParent(touch.target, target);
- });
- // collect touches
- if (type === INPUT_START) {
- i = 0;
- while (i < targetTouches.length) {
- targetIds[targetTouches[i].identifier] = true;
- i++;
- }
- }
- // filter changed touches to only contain touches that exist in the collected target ids
- i = 0;
- while (i < changedTouches.length) {
- if (targetIds[changedTouches[i].identifier]) {
- changedTargetTouches.push(changedTouches[i]);
- }
- // cleanup removed touches
- if (type & (INPUT_END | INPUT_CANCEL)) {
- delete targetIds[changedTouches[i].identifier];
- }
- i++;
- }
- if (!changedTargetTouches.length) {
- return;
- }
- return [
- // merge targetTouches with changedTargetTouches so it contains ALL touches, including 'end' and 'cancel'
- uniqueArray(targetTouches.concat(changedTargetTouches), 'identifier', true),
- changedTargetTouches
- ];
- }
- /**
- * Combined touch and mouse input
- *
- * Touch has a higher priority then mouse, and while touching no mouse events are allowed.
- * This because touch devices also emit mouse events while doing a touch.
- *
- * @constructor
- * @extends Input
- */
- var DEDUP_TIMEOUT = 2500;
- var DEDUP_DISTANCE = 25;
- function TouchMouseInput() {
- Input.apply(this, arguments);
- var handler = bindFn(this.handler, this);
- this.touch = new TouchInput(this.manager, handler);
- this.mouse = new MouseInput(this.manager, handler);
- this.primaryTouch = null;
- this.lastTouches = [];
- }
- inherit(TouchMouseInput, Input, {
- /**
- * handle mouse and touch events
- * @param {Hammer} manager
- * @param {String} inputEvent
- * @param {Object} inputData
- */
- handler: function TMEhandler(manager, inputEvent, inputData) {
- var isTouch = (inputData.pointerType == INPUT_TYPE_TOUCH),
- isMouse = (inputData.pointerType == INPUT_TYPE_MOUSE);
- if (isMouse && inputData.sourceCapabilities && inputData.sourceCapabilities.firesTouchEvents) {
- return;
- }
- // when we're in a touch event, record touches to de-dupe synthetic mouse event
- if (isTouch) {
- recordTouches.call(this, inputEvent, inputData);
- } else if (isMouse && isSyntheticEvent.call(this, inputData)) {
- return;
- }
- this.callback(manager, inputEvent, inputData);
- },
- /**
- * remove the event listeners
- */
- destroy: function destroy() {
- this.touch.destroy();
- this.mouse.destroy();
- }
- });
- function recordTouches(eventType, eventData) {
- if (eventType & INPUT_START) {
- this.primaryTouch = eventData.changedPointers[0].identifier;
- setLastTouch.call(this, eventData);
- } else if (eventType & (INPUT_END | INPUT_CANCEL)) {
- setLastTouch.call(this, eventData);
- }
- }
- function setLastTouch(eventData) {
- var touch = eventData.changedPointers[0];
- if (touch.identifier === this.primaryTouch) {
- var lastTouch = {x: touch.clientX, y: touch.clientY};
- this.lastTouches.push(lastTouch);
- var lts = this.lastTouches;
- var removeLastTouch = function() {
- var i = lts.indexOf(lastTouch);
- if (i > -1) {
- lts.splice(i, 1);
- }
- };
- setTimeout(removeLastTouch, DEDUP_TIMEOUT);
- }
- }
- function isSyntheticEvent(eventData) {
- var x = eventData.srcEvent.clientX, y = eventData.srcEvent.clientY;
- for (var i = 0; i < this.lastTouches.length; i++) {
- var t = this.lastTouches[i];
- var dx = Math.abs(x - t.x), dy = Math.abs(y - t.y);
- if (dx <= DEDUP_DISTANCE && dy <= DEDUP_DISTANCE) {
- return true;
- }
- }
- return false;
- }
- var PREFIXED_TOUCH_ACTION = prefixed(TEST_ELEMENT.style, 'touchAction');
- var NATIVE_TOUCH_ACTION = PREFIXED_TOUCH_ACTION !== undefined;
- // magical touchAction value
- var TOUCH_ACTION_COMPUTE = 'compute';
- var TOUCH_ACTION_AUTO = 'auto';
- var TOUCH_ACTION_MANIPULATION = 'manipulation'; // not implemented
- var TOUCH_ACTION_NONE = 'none';
- var TOUCH_ACTION_PAN_X = 'pan-x';
- var TOUCH_ACTION_PAN_Y = 'pan-y';
- var TOUCH_ACTION_MAP = getTouchActionProps();
- /**
- * Touch Action
- * sets the touchAction property or uses the js alternative
- * @param {Manager} manager
- * @param {String} value
- * @constructor
- */
- function TouchAction(manager, value) {
- this.manager = manager;
- this.set(value);
- }
- TouchAction.prototype = {
- /**
- * set the touchAction value on the element or enable the polyfill
- * @param {String} value
- */
- set: function(value) {
- // find out the touch-action by the event handlers
- if (value == TOUCH_ACTION_COMPUTE) {
- value = this.compute();
- }
- if (NATIVE_TOUCH_ACTION && this.manager.element.style && TOUCH_ACTION_MAP[value]) {
- this.manager.element.style[PREFIXED_TOUCH_ACTION] = value;
- }
- this.actions = value.toLowerCase().trim();
- },
- /**
- * just re-set the touchAction value
- */
- update: function() {
- this.set(this.manager.options.touchAction);
- },
- /**
- * compute the value for the touchAction property based on the recognizer's settings
- * @returns {String} value
- */
- compute: function() {
- var actions = [];
- each(this.manager.recognizers, function(recognizer) {
- if (boolOrFn(recognizer.options.enable, [recognizer])) {
- actions = actions.concat(recognizer.getTouchAction());
- }
- });
- return cleanTouchActions(actions.join(' '));
- },
- /**
- * this method is called on each input cycle and provides the preventing of the browser behavior
- * @param {Object} input
- */
- preventDefaults: function(input) {
- var srcEvent = input.srcEvent;
- var direction = input.offsetDirection;
- // if the touch action did prevented once this session
- if (this.manager.session.prevented) {
- srcEvent.preventDefault();
- return;
- }
- var actions = this.actions;
- var hasNone = inStr(actions, TOUCH_ACTION_NONE) && !TOUCH_ACTION_MAP[TOUCH_ACTION_NONE];
- var hasPanY = inStr(actions, TOUCH_ACTION_PAN_Y) && !TOUCH_ACTION_MAP[TOUCH_ACTION_PAN_Y];
- var hasPanX = inStr(actions, TOUCH_ACTION_PAN_X) && !TOUCH_ACTION_MAP[TOUCH_ACTION_PAN_X];
- if (hasNone) {
- //do not prevent defaults if this is a tap gesture
- var isTapPointer = input.pointers.length === 1;
- var isTapMovement = input.distance < 2;
- var isTapTouchTime = input.deltaTime < 250;
- if (isTapPointer && isTapMovement && isTapTouchTime) {
- return;
- }
- }
- if (hasPanX && hasPanY) {
- // `pan-x pan-y` means browser handles all scrolling/panning, do not prevent
- return;
- }
- if (hasNone ||
- (hasPanY && direction & DIRECTION_HORIZONTAL) ||
- (hasPanX && direction & DIRECTION_VERTICAL)) {
- return this.preventSrc(srcEvent);
- }
- },
- /**
- * call preventDefault to prevent the browser's default behavior (scrolling in most cases)
- * @param {Object} srcEvent
- */
- preventSrc: function(srcEvent) {
- this.manager.session.prevented = true;
- srcEvent.preventDefault();
- }
- };
- /**
- * when the touchActions are collected they are not a valid value, so we need to clean things up. *
- * @param {String} actions
- * @returns {*}
- */
- function cleanTouchActions(actions) {
- // none
- if (inStr(actions, TOUCH_ACTION_NONE)) {
- return TOUCH_ACTION_NONE;
- }
- var hasPanX = inStr(actions, TOUCH_ACTION_PAN_X);
- var hasPanY = inStr(actions, TOUCH_ACTION_PAN_Y);
- // if both pan-x and pan-y are set (different recognizers
- // for different directions, e.g. horizontal pan but vertical swipe?)
- // we need none (as otherwise with pan-x pan-y combined none of these
- // recognizers will work, since the browser would handle all panning
- if (hasPanX && hasPanY) {
- return TOUCH_ACTION_NONE;
- }
- // pan-x OR pan-y
- if (hasPanX || hasPanY) {
- return hasPanX ? TOUCH_ACTION_PAN_X : TOUCH_ACTION_PAN_Y;
- }
- // manipulation
- if (inStr(actions, TOUCH_ACTION_MANIPULATION)) {
- return TOUCH_ACTION_MANIPULATION;
- }
- return TOUCH_ACTION_AUTO;
- }
- function getTouchActionProps() {
- if (!NATIVE_TOUCH_ACTION) {
- return false;
- }
- var touchMap = {};
- var cssSupports = window.CSS && window.CSS.supports;
- ['auto', 'manipulation', 'pan-y', 'pan-x', 'pan-x pan-y', 'none'].forEach(function(val) {
- // If css.supports is not supported but there is native touch-action assume it supports
- // all values. This is the case for IE 10 and 11.
- touchMap[val] = cssSupports ? window.CSS.supports('touch-action', val) : true;
- });
- return touchMap;
- }
- /**
- * Recognizer flow explained; *
- * All recognizers have the initial state of POSSIBLE when a input session starts.
- * The definition of a input session is from the first input until the last input, with all it's movement in it. *
- * Example session for mouse-input: mousedown -> mousemove -> mouseup
- *
- * On each recognizing cycle (see Manager.recognize) the .recognize() method is executed
- * which determines with state it should be.
- *
- * If the recognizer has the state FAILED, CANCELLED or RECOGNIZED (equals ENDED), it is reset to
- * POSSIBLE to give it another change on the next cycle.
- *
- * Possible
- * |
- * +-----+---------------+
- * | |
- * +-----+-----+ |
- * | | |
- * Failed Cancelled |
- * +-------+------+
- * | |
- * Recognized Began
- * |
- * Changed
- * |
- * Ended/Recognized
- */
- var STATE_POSSIBLE = 1;
- var STATE_BEGAN = 2;
- var STATE_CHANGED = 4;
- var STATE_ENDED = 8;
- var STATE_RECOGNIZED = STATE_ENDED;
- var STATE_CANCELLED = 16;
- var STATE_FAILED = 32;
- /**
- * Recognizer
- * Every recognizer needs to extend from this class.
- * @constructor
- * @param {Object} options
- */
- function Recognizer(options) {
- this.options = assign({}, this.defaults, options || {});
- this.id = uniqueId();
- this.manager = null;
- // default is enable true
- this.options.enable = ifUndefined(this.options.enable, true);
- this.state = STATE_POSSIBLE;
- this.simultaneous = {};
- this.requireFail = [];
- }
- Recognizer.prototype = {
- /**
- * @virtual
- * @type {Object}
- */
- defaults: {},
- /**
- * set options
- * @param {Object} options
- * @return {Recognizer}
- */
- set: function(options) {
- assign(this.options, options);
- // also update the touchAction, in case something changed about the directions/enabled state
- this.manager && this.manager.touchAction.update();
- return this;
- },
- /**
- * recognize simultaneous with an other recognizer.
- * @param {Recognizer} otherRecognizer
- * @returns {Recognizer} this
- */
- recognizeWith: function(otherRecognizer) {
- if (invokeArrayArg(otherRecognizer, 'recognizeWith', this)) {
- return this;
- }
- var simultaneous = this.simultaneous;
- otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
- if (!simultaneous[otherRecognizer.id]) {
- simultaneous[otherRecognizer.id] = otherRecognizer;
- otherRecognizer.recognizeWith(this);
- }
- return this;
- },
- /**
- * drop the simultaneous link. it doesnt remove the link on the other recognizer.
- * @param {Recognizer} otherRecognizer
- * @returns {Recognizer} this
- */
- dropRecognizeWith: function(otherRecognizer) {
- if (invokeArrayArg(otherRecognizer, 'dropRecognizeWith', this)) {
- return this;
- }
- otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
- delete this.simultaneous[otherRecognizer.id];
- return this;
- },
- /**
- * recognizer can only run when an other is failing
- * @param {Recognizer} otherRecognizer
- * @returns {Recognizer} this
- */
- requireFailure: function(otherRecognizer) {
- if (invokeArrayArg(otherRecognizer, 'requireFailure', this)) {
- return this;
- }
- var requireFail = this.requireFail;
- otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
- if (inArray(requireFail, otherRecognizer) === -1) {
- requireFail.push(otherRecognizer);
- otherRecognizer.requireFailure(this);
- }
- return this;
- },
- /**
- * drop the requireFailure link. it does not remove the link on the other recognizer.
- * @param {Recognizer} otherRecognizer
- * @returns {Recognizer} this
- */
- dropRequireFailure: function(otherRecognizer) {
- if (invokeArrayArg(otherRecognizer, 'dropRequireFailure', this)) {
- return this;
- }
- otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
- var index = inArray(this.requireFail, otherRecognizer);
- if (index > -1) {
- this.requireFail.splice(index, 1);
- }
- return this;
- },
- /**
- * has require failures boolean
- * @returns {boolean}
- */
- hasRequireFailures: function() {
- return this.requireFail.length > 0;
- },
- /**
- * if the recognizer can recognize simultaneous with an other recognizer
- * @param {Recognizer} otherRecognizer
- * @returns {Boolean}
- */
- canRecognizeWith: function(otherRecognizer) {
- return !!this.simultaneous[otherRecognizer.id];
- },
- /**
- * You should use `tryEmit` instead of `emit` directly to check
- * that all the needed recognizers has failed before emitting.
- * @param {Object} input
- */
- emit: function(input) {
- var self = this;
- var state = this.state;
- function emit(event) {
- self.manager.emit(event, input);
- }
- // 'panstart' and 'panmove'
- if (state < STATE_ENDED) {
- emit(self.options.event + stateStr(state));
- }
- emit(self.options.event); // simple 'eventName' events
- if (input.additionalEvent) { // additional event(panleft, panright, pinchin, pinchout...)
- emit(input.additionalEvent);
- }
- // panend and pancancel
- if (state >= STATE_ENDED) {
- emit(self.options.event + stateStr(state));
- }
- },
- /**
- * Check that all the require failure recognizers has failed,
- * if true, it emits a gesture event,
- * otherwise, setup the state to FAILED.
- * @param {Object} input
- */
- tryEmit: function(input) {
- if (this.canEmit()) {
- return this.emit(input);
- }
- // it's failing anyway
- this.state = STATE_FAILED;
- },
- /**
- * can we emit?
- * @returns {boolean}
- */
- canEmit: function() {
- var i = 0;
- while (i < this.requireFail.length) {
- if (!(this.requireFail[i].state & (STATE_FAILED | STATE_POSSIBLE))) {
- return false;
- }
- i++;
- }
- return true;
- },
- /**
- * update the recognizer
- * @param {Object} inputData
- */
- recognize: function(inputData) {
- // make a new copy of the inputData
- // so we can change the inputData without messing up the other recognizers
- var inputDataClone = assign({}, inputData);
- // is is enabled and allow recognizing?
- if (!boolOrFn(this.options.enable, [this, inputDataClone])) {
- this.reset();
- this.state = STATE_FAILED;
- return;
- }
- // reset when we've reached the end
- if (this.state & (STATE_RECOGNIZED | STATE_CANCELLED | STATE_FAILED)) {
- this.state = STATE_POSSIBLE;
- }
- this.state = this.process(inputDataClone);
- // the recognizer has recognized a gesture
- // so trigger an event
- if (this.state & (STATE_BEGAN | STATE_CHANGED | STATE_ENDED | STATE_CANCELLED)) {
- this.tryEmit(inputDataClone);
- }
- },
- /**
- * return the state of the recognizer
- * the actual recognizing happens in this method
- * @virtual
- * @param {Object} inputData
- * @returns {Const} STATE
- */
- process: function(inputData) { }, // jshint ignore:line
- /**
- * return the preferred touch-action
- * @virtual
- * @returns {Array}
- */
- getTouchAction: function() { },
- /**
- * called when the gesture isn't allowed to recognize
- * like when another is being recognized or it is disabled
- * @virtual
- */
- reset: function() { }
- };
- /**
- * get a usable string, used as event postfix
- * @param {Const} state
- * @returns {String} state
- */
- function stateStr(state) {
- if (state & STATE_CANCELLED) {
- return 'cancel';
- } else if (state & STATE_ENDED) {
- return 'end';
- } else if (state & STATE_CHANGED) {
- return 'move';
- } else if (state & STATE_BEGAN) {
- return 'start';
- }
- return '';
- }
- /**
- * direction cons to string
- * @param {Const} direction
- * @returns {String}
- */
- function directionStr(direction) {
- if (direction == DIRECTION_DOWN) {
- return 'down';
- } else if (direction == DIRECTION_UP) {
- return 'up';
- } else if (direction == DIRECTION_LEFT) {
- return 'left';
- } else if (direction == DIRECTION_RIGHT) {
- return 'right';
- }
- return '';
- }
- /**
- * get a recognizer by name if it is bound to a manager
- * @param {Recognizer|String} otherRecognizer
- * @param {Recognizer} recognizer
- * @returns {Recognizer}
- */
- function getRecognizerByNameIfManager(otherRecognizer, recognizer) {
- var manager = recognizer.manager;
- if (manager) {
- return manager.get(otherRecognizer);
- }
- return otherRecognizer;
- }
- /**
- * This recognizer is just used as a base for the simple attribute recognizers.
- * @constructor
- * @extends Recognizer
- */
- function AttrRecognizer() {
- Recognizer.apply(this, arguments);
- }
- inherit(AttrRecognizer, Recognizer, {
- /**
- * @namespace
- * @memberof AttrRecognizer
- */
- defaults: {
- /**
- * @type {Number}
- * @default 1
- */
- pointers: 1
- },
- /**
- * Used to check if it the recognizer receives valid input, like input.distance > 10.
- * @memberof AttrRecognizer
- * @param {Object} input
- * @returns {Boolean} recognized
- */
- attrTest: function(input) {
- var optionPointers = this.options.pointers;
- return optionPointers === 0 || input.pointers.length === optionPointers;
- },
- /**
- * Process the input and return the state for the recognizer
- * @memberof AttrRecognizer
- * @param {Object} input
- * @returns {*} State
- */
- process: function(input) {
- var state = this.state;
- var eventType = input.eventType;
- var isRecognized = state & (STATE_BEGAN | STATE_CHANGED);
- var isValid = this.attrTest(input);
- // on cancel input and we've recognized before, return STATE_CANCELLED
- if (isRecognized && (eventType & INPUT_CANCEL || !isValid)) {
- return state | STATE_CANCELLED;
- } else if (isRecognized || isValid) {
- if (eventType & INPUT_END) {
- return state | STATE_ENDED;
- } else if (!(state & STATE_BEGAN)) {
- return STATE_BEGAN;
- }
- return state | STATE_CHANGED;
- }
- return STATE_FAILED;
- }
- });
- /**
- * Pan
- * Recognized when the pointer is down and moved in the allowed direction.
- * @constructor
- * @extends AttrRecognizer
- */
- function PanRecognizer() {
- AttrRecognizer.apply(this, arguments);
- this.pX = null;
- this.pY = null;
- }
- inherit(PanRecognizer, AttrRecognizer, {
- /**
- * @namespace
- * @memberof PanRecognizer
- */
- defaults: {
- event: 'pan',
- threshold: 10,
- pointers: 1,
- direction: DIRECTION_ALL
- },
- getTouchAction: function() {
- var direction = this.options.direction;
- var actions = [];
- if (direction & DIRECTION_HORIZONTAL) {
- actions.push(TOUCH_ACTION_PAN_Y);
- }
- if (direction & DIRECTION_VERTICAL) {
- actions.push(TOUCH_ACTION_PAN_X);
- }
- return actions;
- },
- directionTest: function(input) {
- var options = this.options;
- var hasMoved = true;
- var distance = input.distance;
- var direction = input.direction;
- var x = input.deltaX;
- var y = input.deltaY;
- // lock to axis?
- if (!(direction & options.direction)) {
- if (options.direction & DIRECTION_HORIZONTAL) {
- direction = (x === 0) ? DIRECTION_NONE : (x < 0) ? DIRECTION_LEFT : DIRECTION_RIGHT;
- hasMoved = x != this.pX;
- distance = Math.abs(input.deltaX);
- } else {
- direction = (y === 0) ? DIRECTION_NONE : (y < 0) ? DIRECTION_UP : DIRECTION_DOWN;
- hasMoved = y != this.pY;
- distance = Math.abs(input.deltaY);
- }
- }
- input.direction = direction;
- return hasMoved && distance > options.threshold && direction & options.direction;
- },
- attrTest: function(input) {
- return AttrRecognizer.prototype.attrTest.call(this, input) &&
- (this.state & STATE_BEGAN || (!(this.state & STATE_BEGAN) && this.directionTest(input)));
- },
- emit: function(input) {
- this.pX = input.deltaX;
- this.pY = input.deltaY;
- var direction = directionStr(input.direction);
- if (direction) {
- input.additionalEvent = this.options.event + direction;
- }
- this._super.emit.call(this, input);
- }
- });
- /**
- * Pinch
- * Recognized when two or more pointers are moving toward (zoom-in) or away from each other (zoom-out).
- * @constructor
- * @extends AttrRecognizer
- */
- function PinchRecognizer() {
- AttrRecognizer.apply(this, arguments);
- }
- inherit(PinchRecognizer, AttrRecognizer, {
- /**
- * @namespace
- * @memberof PinchRecognizer
- */
- defaults: {
- event: 'pinch',
- threshold: 0,
- pointers: 2
- },
- getTouchAction: function() {
- return [TOUCH_ACTION_NONE];
- },
- attrTest: function(input) {
- return this._super.attrTest.call(this, input) &&
- (Math.abs(input.scale - 1) > this.options.threshold || this.state & STATE_BEGAN);
- },
- emit: function(input) {
- if (input.scale !== 1) {
- var inOut = input.scale < 1 ? 'in' : 'out';
- input.additionalEvent = this.options.event + inOut;
- }
- this._super.emit.call(this, input);
- }
- });
- /**
- * Press
- * Recognized when the pointer is down for x ms without any movement.
- * @constructor
- * @extends Recognizer
- */
- function PressRecognizer() {
- Recognizer.apply(this, arguments);
- this._timer = null;
- this._input = null;
- }
- inherit(PressRecognizer, Recognizer, {
- /**
- * @namespace
- * @memberof PressRecognizer
- */
- defaults: {
- event: 'press',
- pointers: 1,
- time: 251, // minimal time of the pointer to be pressed
- threshold: 9 // a minimal movement is ok, but keep it low
- },
- getTouchAction: function() {
- return [TOUCH_ACTION_AUTO];
- },
- process: function(input) {
- var options = this.options;
- var validPointers = input.pointers.length === options.pointers;
- var validMovement = input.distance < options.threshold;
- var validTime = input.deltaTime > options.time;
- this._input = input;
- // we only allow little movement
- // and we've reached an end event, so a tap is possible
- if (!validMovement || !validPointers || (input.eventType & (INPUT_END | INPUT_CANCEL) && !validTime)) {
- this.reset();
- } else if (input.eventType & INPUT_START) {
- this.reset();
- this._timer = setTimeoutContext(function() {
- this.state = STATE_RECOGNIZED;
- this.tryEmit();
- }, options.time, this);
- } else if (input.eventType & INPUT_END) {
- return STATE_RECOGNIZED;
- }
- return STATE_FAILED;
- },
- reset: function() {
- clearTimeout(this._timer);
- },
- emit: function(input) {
- if (this.state !== STATE_RECOGNIZED) {
- return;
- }
- if (input && (input.eventType & INPUT_END)) {
- this.manager.emit(this.options.event + 'up', input);
- } else {
- this._input.timeStamp = now();
- this.manager.emit(this.options.event, this._input);
- }
- }
- });
- /**
- * Rotate
- * Recognized when two or more pointer are moving in a circular motion.
- * @constructor
- * @extends AttrRecognizer
- */
- function RotateRecognizer() {
- AttrRecognizer.apply(this, arguments);
- }
- inherit(RotateRecognizer, AttrRecognizer, {
- /**
- * @namespace
- * @memberof RotateRecognizer
- */
- defaults: {
- event: 'rotate',
- threshold: 0,
- pointers: 2
- },
- getTouchAction: function() {
- return [TOUCH_ACTION_NONE];
- },
- attrTest: function(input) {
- return this._super.attrTest.call(this, input) &&
- (Math.abs(input.rotation) > this.options.threshold || this.state & STATE_BEGAN);
- }
- });
- /**
- * Swipe
- * Recognized when the pointer is moving fast (velocity), with enough distance in the allowed direction.
- * @constructor
- * @extends AttrRecognizer
- */
- function SwipeRecognizer() {
- AttrRecognizer.apply(this, arguments);
- }
- inherit(SwipeRecognizer, AttrRecognizer, {
- /**
- * @namespace
- * @memberof SwipeRecognizer
- */
- defaults: {
- event: 'swipe',
- threshold: 10,
- velocity: 0.3,
- direction: DIRECTION_HORIZONTAL | DIRECTION_VERTICAL,
- pointers: 1
- },
- getTouchAction: function() {
- return PanRecognizer.prototype.getTouchAction.call(this);
- },
- attrTest: function(input) {
- var direction = this.options.direction;
- var velocity;
- if (direction & (DIRECTION_HORIZONTAL | DIRECTION_VERTICAL)) {
- velocity = input.overallVelocity;
- } else if (direction & DIRECTION_HORIZONTAL) {
- velocity = input.overallVelocityX;
- } else if (direction & DIRECTION_VERTICAL) {
- velocity = input.overallVelocityY;
- }
- return this._super.attrTest.call(this, input) &&
- direction & input.offsetDirection &&
- input.distance > this.options.threshold &&
- input.maxPointers == this.options.pointers &&
- abs(velocity) > this.options.velocity && input.eventType & INPUT_END;
- },
- emit: function(input) {
- var direction = directionStr(input.offsetDirection);
- if (direction) {
- this.manager.emit(this.options.event + direction, input);
- }
- this.manager.emit(this.options.event, input);
- }
- });
- /**
- * A tap is ecognized when the pointer is doing a small tap/click. Multiple taps are recognized if they occur
- * between the given interval and position. The delay option can be used to recognize multi-taps without firing
- * a single tap.
- *
- * The eventData from the emitted event contains the property `tapCount`, which contains the amount of
- * multi-taps being recognized.
- * @constructor
- * @extends Recognizer
- */
- function TapRecognizer() {
- Recognizer.apply(this, arguments);
- // previous time and center,
- // used for tap counting
- this.pTime = false;
- this.pCenter = false;
- this._timer = null;
- this._input = null;
- this.count = 0;
- }
- inherit(TapRecognizer, Recognizer, {
- /**
- * @namespace
- * @memberof PinchRecognizer
- */
- defaults: {
- event: 'tap',
- pointers: 1,
- taps: 1,
- interval: 300, // max time between the multi-tap taps
- time: 250, // max time of the pointer to be down (like finger on the screen)
- threshold: 9, // a minimal movement is ok, but keep it low
- posThreshold: 10 // a multi-tap can be a bit off the initial position
- },
- getTouchAction: function() {
- return [TOUCH_ACTION_MANIPULATION];
- },
- process: function(input) {
- var options = this.options;
- var validPointers = input.pointers.length === options.pointers;
- var validMovement = input.distance < options.threshold;
- var validTouchTime = input.deltaTime < options.time;
- this.reset();
- if ((input.eventType & INPUT_START) && (this.count === 0)) {
- return this.failTimeout();
- }
- // we only allow little movement
- // and we've reached an end event, so a tap is possible
- if (validMovement && validTouchTime && validPointers) {
- if (input.eventType != INPUT_END) {
- return this.failTimeout();
- }
- var validInterval = this.pTime ? (input.timeStamp - this.pTime < options.interval) : true;
- var validMultiTap = !this.pCenter || getDistance(this.pCenter, input.center) < options.posThreshold;
- this.pTime = input.timeStamp;
- this.pCenter = input.center;
- if (!validMultiTap || !validInterval) {
- this.count = 1;
- } else {
- this.count += 1;
- }
- this._input = input;
- // if tap count matches we have recognized it,
- // else it has began recognizing...
- var tapCount = this.count % options.taps;
- if (tapCount === 0) {
- // no failing requirements, immediately trigger the tap event
- // or wait as long as the multitap interval to trigger
- if (!this.hasRequireFailures()) {
- return STATE_RECOGNIZED;
- } else {
- this._timer = setTimeoutContext(function() {
- this.state = STATE_RECOGNIZED;
- this.tryEmit();
- }, options.interval, this);
- return STATE_BEGAN;
- }
- }
- }
- return STATE_FAILED;
- },
- failTimeout: function() {
- this._timer = setTimeoutContext(function() {
- this.state = STATE_FAILED;
- }, this.options.interval, this);
- return STATE_FAILED;
- },
- reset: function() {
- clearTimeout(this._timer);
- },
- emit: function() {
- if (this.state == STATE_RECOGNIZED) {
- this._input.tapCount = this.count;
- this.manager.emit(this.options.event, this._input);
- }
- }
- });
- /**
- * Simple way to create a manager with a default set of recognizers.
- * @param {HTMLElement} element
- * @param {Object} [options]
- * @constructor
- */
- function Hammer(element, options) {
- options = options || {};
- options.recognizers = ifUndefined(options.recognizers, Hammer.defaults.preset);
- return new Manager(element, options);
- }
- /**
- * @const {string}
- */
- Hammer.VERSION = '2.0.8';
- /**
- * default settings
- * @namespace
- */
- Hammer.defaults = {
- /**
- * set if DOM events are being triggered.
- * But this is slower and unused by simple implementations, so disabled by default.
- * @type {Boolean}
- * @default false
- */
- domEvents: false,
- /**
- * The value for the touchAction property/fallback.
- * When set to `compute` it will magically set the correct value based on the added recognizers.
- * @type {String}
- * @default compute
- */
- touchAction: TOUCH_ACTION_COMPUTE,
- /**
- * @type {Boolean}
- * @default true
- */
- enable: true,
- /**
- * EXPERIMENTAL FEATURE -- can be removed/changed
- * Change the parent input target element.
- * If Null, then it is being set the to main element.
- * @type {Null|EventTarget}
- * @default null
- */
- inputTarget: null,
- /**
- * force an input class
- * @type {Null|Function}
- * @default null
- */
- inputClass: null,
- /**
- * Default recognizer setup when calling `Hammer()`
- * When creating a new Manager these will be skipped.
- * @type {Array}
- */
- preset: [
- // RecognizerClass, options, [recognizeWith, ...], [requireFailure, ...]
- [RotateRecognizer, {enable: false}],
- [PinchRecognizer, {enable: false}, ['rotate']],
- [SwipeRecognizer, {direction: DIRECTION_HORIZONTAL}],
- [PanRecognizer, {direction: DIRECTION_HORIZONTAL}, ['swipe']],
- [TapRecognizer],
- [TapRecognizer, {event: 'doubletap', taps: 2}, ['tap']],
- [PressRecognizer]
- ],
- /**
- * Some CSS properties can be used to improve the working of Hammer.
- * Add them to this method and they will be set when creating a new Manager.
- * @namespace
- */
- cssProps: {
- /**
- * Disables text selection to improve the dragging gesture. Mainly for desktop browsers.
- * @type {String}
- * @default 'none'
- */
- userSelect: 'none',
- /**
- * Disable the Windows Phone grippers when pressing an element.
- * @type {String}
- * @default 'none'
- */
- touchSelect: 'none',
- /**
- * Disables the default callout shown when you touch and hold a touch target.
- * On iOS, when you touch and hold a touch target such as a link, Safari displays
- * a callout containing information about the link. This property allows you to disable that callout.
- * @type {String}
- * @default 'none'
- */
- touchCallout: 'none',
- /**
- * Specifies whether zooming is enabled. Used by IE10>
- * @type {String}
- * @default 'none'
- */
- contentZooming: 'none',
- /**
- * Specifies that an entire element should be draggable instead of its contents. Mainly for desktop browsers.
- * @type {String}
- * @default 'none'
- */
- userDrag: 'none',
- /**
- * Overrides the highlight color shown when the user taps a link or a JavaScript
- * clickable element in iOS. This property obeys the alpha value, if specified.
- * @type {String}
- * @default 'rgba(0,0,0,0)'
- */
- tapHighlightColor: 'rgba(0,0,0,0)'
- }
- };
- var STOP = 1;
- var FORCED_STOP = 2;
- /**
- * Manager
- * @param {HTMLElement} element
- * @param {Object} [options]
- * @constructor
- */
- function Manager(element, options) {
- this.options = assign({}, Hammer.defaults, options || {});
- this.options.inputTarget = this.options.inputTarget || element;
- this.handlers = {};
- this.session = {};
- this.recognizers = [];
- this.oldCssProps = {};
- this.element = element;
- this.input = createInputInstance(this);
- this.touchAction = new TouchAction(this, this.options.touchAction);
- toggleCssProps(this, true);
- each(this.options.recognizers, function(item) {
- var recognizer = this.add(new (item[0])(item[1]));
- item[2] && recognizer.recognizeWith(item[2]);
- item[3] && recognizer.requireFailure(item[3]);
- }, this);
- }
- Manager.prototype = {
- /**
- * set options
- * @param {Object} options
- * @returns {Manager}
- */
- set: function(options) {
- assign(this.options, options);
- // Options that need a little more setup
- if (options.touchAction) {
- this.touchAction.update();
- }
- if (options.inputTarget) {
- // Clean up existing event listeners and reinitialize
- this.input.destroy();
- this.input.target = options.inputTarget;
- this.input.init();
- }
- return this;
- },
- /**
- * stop recognizing for this session.
- * This session will be discarded, when a new [input]start event is fired.
- * When forced, the recognizer cycle is stopped immediately.
- * @param {Boolean} [force]
- */
- stop: function(force) {
- this.session.stopped = force ? FORCED_STOP : STOP;
- },
- /**
- * run the recognizers!
- * called by the inputHandler function on every movement of the pointers (touches)
- * it walks through all the recognizers and tries to detect the gesture that is being made
- * @param {Object} inputData
- */
- recognize: function(inputData) {
- var session = this.session;
- if (session.stopped) {
- return;
- }
- // run the touch-action polyfill
- this.touchAction.preventDefaults(inputData);
- var recognizer;
- var recognizers = this.recognizers;
- // this holds the recognizer that is being recognized.
- // so the recognizer's state needs to be BEGAN, CHANGED, ENDED or RECOGNIZED
- // if no recognizer is detecting a thing, it is set to `null`
- var curRecognizer = session.curRecognizer;
- // reset when the last recognizer is recognized
- // or when we're in a new session
- if (!curRecognizer || (curRecognizer && curRecognizer.state & STATE_RECOGNIZED)) {
- curRecognizer = session.curRecognizer = null;
- }
- var i = 0;
- while (i < recognizers.length) {
- recognizer = recognizers[i];
- // find out if we are allowed try to recognize the input for this one.
- // 1. allow if the session is NOT forced stopped (see the .stop() method)
- // 2. allow if we still haven't recognized a gesture in this session, or the this recognizer is the one
- // that is being recognized.
- // 3. allow if the recognizer is allowed to run simultaneous with the current recognized recognizer.
- // this can be setup with the `recognizeWith()` method on the recognizer.
- if (session.stopped !== FORCED_STOP && ( // 1
- !curRecognizer || recognizer == curRecognizer || // 2
- recognizer.canRecognizeWith(curRecognizer))) { // 3
- recognizer.recognize(inputData);
- } else {
- recognizer.reset();
- }
- // if the recognizer has been recognizing the input as a valid gesture, we want to store this one as the
- // current active recognizer. but only if we don't already have an active recognizer
- if (!curRecognizer && recognizer.state & (STATE_BEGAN | STATE_CHANGED | STATE_ENDED)) {
- curRecognizer = session.curRecognizer = recognizer;
- }
- i++;
- }
- },
- /**
- * get a recognizer by its event name.
- * @param {Recognizer|String} recognizer
- * @returns {Recognizer|Null}
- */
- get: function(recognizer) {
- if (recognizer instanceof Recognizer) {
- return recognizer;
- }
- var recognizers = this.recognizers;
- for (var i = 0; i < recognizers.length; i++) {
- if (recognizers[i].options.event == recognizer) {
- return recognizers[i];
- }
- }
- return null;
- },
- /**
- * add a recognizer to the manager
- * existing recognizers with the same event name will be removed
- * @param {Recognizer} recognizer
- * @returns {Recognizer|Manager}
- */
- add: function(recognizer) {
- if (invokeArrayArg(recognizer, 'add', this)) {
- return this;
- }
- // remove existing
- var existing = this.get(recognizer.options.event);
- if (existing) {
- this.remove(existing);
- }
- this.recognizers.push(recognizer);
- recognizer.manager = this;
- this.touchAction.update();
- return recognizer;
- },
- /**
- * remove a recognizer by name or instance
- * @param {Recognizer|String} recognizer
- * @returns {Manager}
- */
- remove: function(recognizer) {
- if (invokeArrayArg(recognizer, 'remove', this)) {
- return this;
- }
- recognizer = this.get(recognizer);
- // let's make sure this recognizer exists
- if (recognizer) {
- var recognizers = this.recognizers;
- var index = inArray(recognizers, recognizer);
- if (index !== -1) {
- recognizers.splice(index, 1);
- this.touchAction.update();
- }
- }
- return this;
- },
- /**
- * bind event
- * @param {String} events
- * @param {Function} handler
- * @returns {EventEmitter} this
- */
- on: function(events, handler) {
- if (events === undefined) {
- return;
- }
- if (handler === undefined) {
- return;
- }
- var handlers = this.handlers;
- each(splitStr(events), function(event) {
- handlers[event] = handlers[event] || [];
- handlers[event].push(handler);
- });
- return this;
- },
- /**
- * unbind event, leave emit blank to remove all handlers
- * @param {String} events
- * @param {Function} [handler]
- * @returns {EventEmitter} this
- */
- off: function(events, handler) {
- if (events === undefined) {
- return;
- }
- var handlers = this.handlers;
- each(splitStr(events), function(event) {
- if (!handler) {
- delete handlers[event];
- } else {
- handlers[event] && handlers[event].splice(inArray(handlers[event], handler), 1);
- }
- });
- return this;
- },
- /**
- * emit event to the listeners
- * @param {String} event
- * @param {Object} data
- */
- emit: function(event, data) {
- // we also want to trigger dom events
- if (this.options.domEvents) {
- triggerDomEvent(event, data);
- }
- // no handlers, so skip it all
- var handlers = this.handlers[event] && this.handlers[event].slice();
- if (!handlers || !handlers.length) {
- return;
- }
- data.type = event;
- data.preventDefault = function() {
- data.srcEvent.preventDefault();
- };
- var i = 0;
- while (i < handlers.length) {
- handlers[i](data);
- i++;
- }
- },
- /**
- * destroy the manager and unbinds all events
- * it doesn't unbind dom events, that is the user own responsibility
- */
- destroy: function() {
- this.element && toggleCssProps(this, false);
- this.handlers = {};
- this.session = {};
- this.input.destroy();
- this.element = null;
- }
- };
- /**
- * add/remove the css properties as defined in manager.options.cssProps
- * @param {Manager} manager
- * @param {Boolean} add
- */
- function toggleCssProps(manager, add) {
- var element = manager.element;
- if (!element.style) {
- return;
- }
- var prop;
- each(manager.options.cssProps, function(value, name) {
- prop = prefixed(element.style, name);
- if (add) {
- manager.oldCssProps[prop] = element.style[prop];
- element.style[prop] = value;
- } else {
- element.style[prop] = manager.oldCssProps[prop] || '';
- }
- });
- if (!add) {
- manager.oldCssProps = {};
- }
- }
- /**
- * trigger dom event
- * @param {String} event
- * @param {Object} data
- */
- function triggerDomEvent(event, data) {
- var gestureEvent = document.createEvent('Event');
- gestureEvent.initEvent(event, true, true);
- gestureEvent.gesture = data;
- data.target.dispatchEvent(gestureEvent);
- }
- assign(Hammer, {
- INPUT_START: INPUT_START,
- INPUT_MOVE: INPUT_MOVE,
- INPUT_END: INPUT_END,
- INPUT_CANCEL: INPUT_CANCEL,
- STATE_POSSIBLE: STATE_POSSIBLE,
- STATE_BEGAN: STATE_BEGAN,
- STATE_CHANGED: STATE_CHANGED,
- STATE_ENDED: STATE_ENDED,
- STATE_RECOGNIZED: STATE_RECOGNIZED,
- STATE_CANCELLED: STATE_CANCELLED,
- STATE_FAILED: STATE_FAILED,
- DIRECTION_NONE: DIRECTION_NONE,
- DIRECTION_LEFT: DIRECTION_LEFT,
- DIRECTION_RIGHT: DIRECTION_RIGHT,
- DIRECTION_UP: DIRECTION_UP,
- DIRECTION_DOWN: DIRECTION_DOWN,
- DIRECTION_HORIZONTAL: DIRECTION_HORIZONTAL,
- DIRECTION_VERTICAL: DIRECTION_VERTICAL,
- DIRECTION_ALL: DIRECTION_ALL,
- Manager: Manager,
- Input: Input,
- TouchAction: TouchAction,
- TouchInput: TouchInput,
- MouseInput: MouseInput,
- PointerEventInput: PointerEventInput,
- TouchMouseInput: TouchMouseInput,
- SingleTouchInput: SingleTouchInput,
- Recognizer: Recognizer,
- AttrRecognizer: AttrRecognizer,
- Tap: TapRecognizer,
- Pan: PanRecognizer,
- Swipe: SwipeRecognizer,
- Pinch: PinchRecognizer,
- Rotate: RotateRecognizer,
- Press: PressRecognizer,
- on: addEventListeners,
- off: removeEventListeners,
- each: each,
- merge: merge,
- extend: extend,
- assign: assign,
- inherit: inherit,
- bindFn: bindFn,
- prefixed: prefixed
- });
- // this prevents errors when Hammer is loaded in the presence of an AMD
- // style loader but by script tag, not by the loader.
- var freeGlobal = (typeof window !== 'undefined' ? window : (typeof self !== 'undefined' ? self : {})); // jshint ignore:line
- freeGlobal.Hammer = Hammer;
- if (typeof define === 'function' && define.amd) {
- define(function() {
- return Hammer;
- });
- } else if (typeof module != 'undefined' && module.exports) {
- module.exports = Hammer;
- } else {
- window[exportName] = Hammer;
- }
- })(window, document, 'Hammer');
- + function($) {
- "use strict";
- var defaults;
-
- $.modal = function(params, onOpen) {
- params = $.extend({}, defaults, params);
- var buttons = params.buttons;
- var buttonsHtml = buttons.map(function(d, i) {
- return '<a href="javascript:;" class="weui-dialog__btn ' + (d.className || "") + '">' + d.text + '</a>';
- }).join("");
- var tpl = '<div class="weui-dialog">' +
- '<div class="weui-dialog__hd"><strong class="weui-dialog__title">' + params.title + '</strong></div>' +
- ( params.text ? '<div class="weui-dialog__bd">'+params.text+'</div>' : '')+
- '<div class="weui-dialog__ft">' + buttonsHtml + '</div>' +
- '</div>';
-
- var dialog = $.openModal(tpl, onOpen);
- dialog.find(".weui-dialog__btn").each(function(i, e) {
- var el = $(e);
- el.click(function() {
- //先关闭对话框,再调用回调函数
- if(params.autoClose) $.closeModal();
- if(buttons[i].onClick) {
- buttons[i].onClick.call(dialog);
- }
- });
- });
- return dialog;
- };
- $.openModal = function(tpl, onOpen) {
- var mask = $("<div class='weui-mask'></div>").appendTo(document.body);
- mask.show();
- var dialog = $(tpl).appendTo(document.body);
-
- if (onOpen) {
- dialog.transitionEnd(function () {
- onOpen.call(dialog);
- });
- }
- dialog.show();
- mask.addClass("weui-mask--visible");
- dialog.addClass("weui-dialog--visible");
- return dialog;
- }
- $.closeModal = function() {
- $(".weui-mask--visible").removeClass("weui-mask--visible").transitionEnd(function() {
- $(this).remove();
- });
- $(".weui-dialog--visible").removeClass("weui-dialog--visible").transitionEnd(function() {
- $(this).remove();
- });
- };
- $.alert = function(text, title, onOK) {
- var config;
- if (typeof text === 'object') {
- config = text;
- } else {
- if (typeof title === 'function') {
- onOK = arguments[1];
- title = undefined;
- }
- config = {
- text: text,
- title: title,
- onOK: onOK
- }
- }
- return $.modal({
- text: config.text,
- title: config.title,
- buttons: [{
- text: defaults.buttonOK,
- className: "primary",
- onClick: config.onOK
- }]
- });
- }
- $.confirm = function(text, title, onOK, onCancel) {
- var config;
- if (typeof text === 'object') {
- config = text
- } else {
- if (typeof title === 'function') {
- onCancel = arguments[2];
- onOK = arguments[1];
- title = undefined;
- }
- config = {
- text: text,
- title: title,
- onOK: onOK,
- onCancel: onCancel
- }
- }
- return $.modal({
- text: config.text,
- title: config.title,
- buttons: [
- {
- text: defaults.buttonCancel,
- className: "default",
- onClick: config.onCancel
- },
- {
- text: defaults.buttonOK,
- className: "primary",
- onClick: config.onOK
- }]
- });
- };
- //如果参数过多,建议通过 config 对象进行配置,而不是传入多个参数。
- $.prompt = function(text, title, onOK, onCancel, input) {
- var config;
- if (typeof text === 'object') {
- config = text;
- } else {
- if (typeof title === 'function') {
- input = arguments[3];
- onCancel = arguments[2];
- onOK = arguments[1];
- title = undefined;
- }
- config = {
- text: text,
- title: title,
- input: input,
- onOK: onOK,
- onCancel: onCancel,
- empty: false //allow empty
- }
- }
- var modal = $.modal({
- text: '<p class="weui-prompt-text">'+(config.text || '')+'</p><input type="text" class="weui-input weui-prompt-input" id="weui-prompt-input" value="' + (config.input || '') + '" />',
- title: config.title,
- autoClose: false,
- buttons: [
- {
- text: defaults.buttonCancel,
- className: "default",
- onClick: function () {
- $.closeModal();
- config.onCancel && config.onCancel.call(modal);
- }
- },
- {
- text: defaults.buttonOK,
- className: "primary",
- onClick: function() {
- var input = $("#weui-prompt-input").val();
- if (!config.empty && (input === "" || input === null)) {
- modal.find('.weui-prompt-input').focus()[0].select();
- return false;
- }
- $.closeModal();
- config.onOK && config.onOK.call(modal, input);
- }
- }]
- }, function () {
- this.find('.weui-prompt-input').focus()[0].select();
- });
- return modal;
- };
- //如果参数过多,建议通过 config 对象进行配置,而不是传入多个参数。
- $.login = function(text, title, onOK, onCancel, username, password) {
- var config;
- if (typeof text === 'object') {
- config = text;
- } else {
- if (typeof title === 'function') {
- password = arguments[4];
- username = arguments[3];
- onCancel = arguments[2];
- onOK = arguments[1];
- title = undefined;
- }
- config = {
- text: text,
- title: title,
- username: username,
- password: password,
- onOK: onOK,
- onCancel: onCancel
- }
- }
- var modal = $.modal({
- text: '<p class="weui-prompt-text">'+(config.text || '')+'</p>' +
- '<input type="text" class="weui-input weui-prompt-input" id="weui-prompt-username" value="' + (config.username || '') + '" placeholder="输入用户名" />' +
- '<input type="password" class="weui-input weui-prompt-input" id="weui-prompt-password" value="' + (config.password || '') + '" placeholder="输入密码" />',
- title: config.title,
- autoClose: false,
- buttons: [
- {
- text: defaults.buttonCancel,
- className: "default",
- onClick: function () {
- $.closeModal();
- config.onCancel && config.onCancel.call(modal);
- }
- }, {
- text: defaults.buttonOK,
- className: "primary",
- onClick: function() {
- var username = $("#weui-prompt-username").val();
- var password = $("#weui-prompt-password").val();
- if (!config.empty && (username === "" || username === null)) {
- modal.find('#weui-prompt-username').focus()[0].select();
- return false;
- }
- if (!config.empty && (password === "" || password === null)) {
- modal.find('#weui-prompt-password').focus()[0].select();
- return false;
- }
- $.closeModal();
- config.onOK && config.onOK.call(modal, username, password);
- }
- }]
- }, function () {
- this.find('#weui-prompt-username').focus()[0].select();
- });
- return modal;
- };
- defaults = $.modal.prototype.defaults = {
- title: "提示",
- text: undefined,
- buttonOK: "确定",
- buttonCancel: "取消",
- buttons: [{
- text: "确定",
- className: "primary"
- }],
- autoClose: true //点击按钮自动关闭对话框,如果你不希望点击按钮就关闭对话框,可以把这个设置为false
- };
- }($);
- + function($) {
- "use strict";
- var defaults;
-
- var show = function(html, className) {
- className = className || "";
- var mask = $("<div class='weui-mask_transparent'></div>").appendTo(document.body);
- var tpl = '<div class="weui-toast ' + className + '">' + html + '</div>';
- var dialog = $(tpl).appendTo(document.body);
- dialog.addClass("weui-toast--visible");
- dialog.show();
- };
- var hide = function(callback) {
- $(".weui-mask_transparent").remove();
- var done = false;
- var $el = $(".weui-toast--visible").removeClass("weui-toast--visible").transitionEnd(function() {
- var $this = $(this);
- $this.remove();
- callback && callback();
- done = true
- });
- setTimeout(function () {
- if (!done) {
- $el.remove()
- callback && callback();
- }
- }, 1000)
- }
- $.toast = function(text, style, callback) {
- if(typeof style === "function") {
- callback = style;
- }
- var className, iconClassName = 'weui-icon-success-no-circle';
- var duration = toastDefaults.duration;
- if(style == "cancel") {
- className = "weui-toast_cancel";
- iconClassName = 'weui-icon-cancel'
- } else if(style == "forbidden") {
- className = "weui-toast--forbidden";
- iconClassName = 'weui-icon-warn'
- } else if(style == "text") {
- className = "weui-toast--text";
- } else if(typeof style === typeof 1) {
- duration = style
- }
- show('<i class="' + iconClassName + ' weui-icon_toast"></i><p class="weui-toast_content">' + (text || "已经完成") + '</p>', className);
- setTimeout(function() {
- hide(callback);
- }, duration);
- }
- $.showLoading = function(text) {
- var html = '<div class="weui_loading">';
- html += '<i class="weui-loading weui-icon_toast"></i>';
- html += '</div>';
- html += '<p class="weui-toast_content">' + (text || "数据加载中") + '</p>';
- show(html, 'weui_loading_toast');
- }
- $.hideLoading = function() {
- hide();
- }
- //自己添加的,用于上传图片
- $.showUpLoading = function(text) {
- var html = '<div class="weui_loading">';
- html += '<i class="weui-loading weui-icon_toast"></i>';
- html += '</div>';
- html += '<p class="weui-toast_content">' + (text || "文件上传中") + '</p>';
- show(html, 'weui_loading_toast');
- }
- //自己添加的,用于上传图片
- $.hideUpLoading = function() {
- hide();
- }
- var toastDefaults = $.toast.prototype.defaults = {
- duration: 2500
- }
- }($);
- + function($) {
- "use strict";
- var defaults;
-
- var show = function(params) {
- var mask = $("<div class='weui-mask weui-actions_mask'></div>").appendTo(document.body);
- var actions = params.actions || [];
- var actionsHtml = actions.map(function(d, i) {
- return '<div class="weui-actionsheet__cell ' + (d.className || "") + '">' + d.text + '</div>';
- }).join("");
- var titleHtml = "";
-
- if (params.title) {
- titleHtml = '<div class="weui-actionsheet__title"><p class="weui-actionsheet__title-text">' + params.title + '</p></div>';
- }
- var tpl = '<div class="weui-actionsheet " id="weui-actionsheet">'+
- titleHtml +
- '<div class="weui-actionsheet__menu">'+
- actionsHtml +
- '</div>'+
- '<div class="weui-actionsheet__action">'+
- '<div class="weui-actionsheet__cell weui-actionsheet_cancel">取消</div>'+
- '</div>'+
- '</div>';
- var dialog = $(tpl).appendTo(document.body);
- dialog.find(".weui-actionsheet__menu .weui-actionsheet__cell, .weui-actionsheet__action .weui-actionsheet__cell").each(function(i, e) {
- $(e).click(function() {
- $.closeActions();
- params.onClose && params.onClose();
- if(actions[i] && actions[i].onClick) {
- actions[i].onClick();
- }
- })
- });
- mask.show();
- dialog.show();
- mask.addClass("weui-mask--visible");
- dialog.addClass("weui-actionsheet_toggle");
- };
- var hide = function() {
- $(".weui-mask").removeClass("weui-mask--visible").transitionEnd(function() {
- $(this).remove();
- });
- $(".weui-actionsheet").removeClass("weui-actionsheet_toggle").transitionEnd(function() {
- $(this).remove();
- });
- }
- $.actions = function(params) {
- params = $.extend({}, defaults, params);
- show(params);
- }
- $.closeActions = function() {
- hide();
- }
- $(document).on("click", ".weui-actions_mask", function() {
- $.closeActions();
- });
- var defaults = $.actions.prototype.defaults = {
- title: undefined,
- onClose: undefined,
- /*actions: [{
- text: "菜单",
- className: "color-danger",
- onClick: function() {
- console.log(1);
- }
- },{
- text: "菜单2",
- className: "color-success",
- onClick: function() {
- console.log(2);
- }
- }]*/
- }
- }($);
- /* ===============================================================================
- ************ Pull to refreh ************
- =============================================================================== */
- /* global $:true */
- +function ($) {
- "use strict";
- var PTR = function(el, opt) {
- if (typeof opt === typeof function () {}) {
- opt = {
- onRefresh: opt
- }
- }
- if (typeof opt === typeof 'a') {
- opt = undefined
- }
- this.opt = $.extend(PTR.defaults, opt || {});
- this.container = $(el);
- this.attachEvents();
- }
- PTR.defaults = {
- distance: 50,
- onRefresh: undefined,
- onPull: undefined
- }
- PTR.prototype.touchStart = function(e) {
- if(this.container.hasClass("refreshing")) return;
- var p = $.getTouchPosition(e);
- this.start = p;
- this.diffX = this.diffY = 0;
- };
- PTR.prototype.touchMove= function(e) {
- if(this.container.hasClass("refreshing")) return;
- if(!this.start) return false;
- if(this.container.scrollTop() > 0) return;
- var p = $.getTouchPosition(e);
- this.diffX = p.x - this.start.x;
- this.diffY = p.y - this.start.y;
- if (Math.abs(this.diffX) > Math.abs(this.diffY)) return true; // 说明是左右方向的拖动
- if(this.diffY < 0) return;
- this.container.addClass("touching");
- e.preventDefault();
- e.stopPropagation();
- this.diffY = Math.pow(this.diffY, 0.75);
- this.container.css("transform", "translate3d(0, "+this.diffY+"px, 0)");
- this.triggerPull(this.diffY)
- };
- PTR.prototype.touchEnd = function() {
- this.start = false;
- if(this.diffY <= 0 || this.container.hasClass("refreshing")) return;
- this.container.removeClass("touching");
- this.container.removeClass("pull-down pull-up");
- this.container.css("transform", "");
- if(Math.abs(this.diffY) <= this.opt.distance) {
- } else {
- this.triggerPullToRefresh();
- }
- };
- PTR.prototype.triggerPullToRefresh = function() {
- this.triggerPull(this.opt.distance)
- this.container.removeClass('pull-up').addClass("refreshing");
- if (this.opt.onRefresh) {
- this.opt.onRefresh.call(this)
- }
- this.container.trigger("pull-to-refresh");
- }
- PTR.prototype.triggerPull = function(diffY) {
- if(diffY < this.opt.distance) {
- this.container.removeClass("pull-up").addClass("pull-down");
- } else {
- this.container.removeClass("pull-down").addClass("pull-up");
- }
- if (this.opt.onPull) {
- this.opt.onPull.call(this, Math.floor(diffY / this.opt.distance * 100))
- }
- this.container.trigger("pull");
- }
- PTR.prototype.pullToRefreshDone = function() {
- this.container.removeClass("refreshing");
- }
- PTR.prototype.attachEvents = function() {
- var el = this.container;
- el.addClass("weui-pull-to-refresh");
- el.on($.touchEvents.start, $.proxy(this.touchStart, this));
- el.on($.touchEvents.move, $.proxy(this.touchMove, this));
- el.on($.touchEvents.end, $.proxy(this.touchEnd, this));
- };
- var pullToRefreshDone = function(el) {
- $(el).removeClass("refreshing");
- }
- $.fn.pullToRefresh = function(opt) {
- return this.each(function() {
- var $this = $(this)
- var ptr = $this.data('ptr')
- if (!ptr) $this.data('ptr', ptr = new PTR(this, opt))
- if (typeof opt === typeof 'a') {
- ptr[opt].call(ptr)
- }
- });
- }
- $.fn.pullToRefreshDone = function() {
- return this.each(function() {
- pullToRefreshDone(this);
- });
- }
- }($);
- /* ===============================================================================
- ************ Infinite ************
- =============================================================================== */
- /* global $:true */
- +function ($) {
- "use strict";
- // fix https://github.com/lihongxun945/jquery-weui/issues/442
- // chrome will always return 0, when use document.body.scrollTop
- // https://stackoverflow.com/questions/43717316/google-chrome-document-body-scrolltop-always-returns-0
- var getOffset = function (container) {
- var tagName = container[0].tagName.toUpperCase()
- var scrollTop
- if (tagName === 'BODY' || tagName === 'HTML') {
- scrollTop = container.scrollTop() || $(window).scrollTop()
- } else {
- scrollTop = container.scrollTop()
- }
- var offset = container.scrollHeight() - ($(window).height() + scrollTop)
- return offset
- }
- var Infinite = function(el, distance) {
- this.container = $(el);
- this.container.data("infinite", this);
- this.distance = distance || 50;
- this.attachEvents();
- }
- Infinite.prototype.scroll = function() {
- var container = this.container;
- this._check();
- }
- Infinite.prototype.attachEvents = function(off) {
- var el = this.container;
- var scrollContainer = (el[0].tagName.toUpperCase() === "BODY" ? $(document) : el);
- scrollContainer[off ? "off" : "on"]("scroll", $.proxy(this.scroll, this));
- };
- Infinite.prototype.detachEvents = function(off) {
- this.attachEvents(true);
- }
- Infinite.prototype._check = function() {
- var offset = getOffset(this.container);
- if(Math.abs(offset) <= this.distance) {
- this.container.trigger("infinite");
- }
- }
- var infinite = function(el) {
- attachEvents(el);
- }
- $.fn.infinite = function(distance) {
- return this.each(function() {
- new Infinite(this, distance);
- });
- }
- $.fn.destroyInfinite = function() {
- return this.each(function() {
- var infinite = $(this).data("infinite");
- if(infinite && infinite.detachEvents) infinite.detachEvents();
- });
- }
- }($);
- /* global $:true */
- +function ($) {
- "use strict";
- var ITEM_ON = "weui-bar__item--on";
- var showTab = function(a) {
- var $a = $(a);
- if($a.hasClass(ITEM_ON)) return;
- var href = $a.attr("href");
- if(!/^#/.test(href)) return ;
- $a.parent().find("."+ITEM_ON).removeClass(ITEM_ON);
- $a.addClass(ITEM_ON);
- var bd = $a.parents(".weui-tab").find(".weui-tab__bd");
- bd.find(".weui-tab__bd-item--active").removeClass("weui-tab__bd-item--active");
- $(href).addClass("weui-tab__bd-item--active");
- }
- $.showTab = showTab;
- $(document).on("click", ".weui-navbar__item, .weui-tabbar__item", function(e) {
- var $a = $(e.currentTarget);
- var href = $a.attr("href");
- if($a.hasClass(ITEM_ON)) return;
- if(!/^#/.test(href)) return;
- e.preventDefault();
- showTab($a);
- });
- }($);
- /* global $:true */
- + function($) {
- "use strict";
- $(document).on("click touchstart", ".weui-search-bar__label", function(e) {
- $(e.target).parents(".weui-search-bar").addClass("weui-search-bar_focusing").find('input').focus();
- })
- /*
- .on("blur", ".weui-search-bar__input", function(e) {
- var $input = $(e.target);
- if(!$input.val()) $input.parents(".weui-search-bar").removeClass("weui-search-bar_focusing");
- })
- */
- .on("click", ".weui-search-bar__cancel-btn", function(e) {
- var $input = $(e.target).parents(".weui-search-bar").removeClass("weui-search-bar_focusing").find(".weui-search-bar__input").val("").blur();
- })
- .on("click", ".weui-icon-clear", function(e) {
- var $input = $(e.target).parents(".weui-search-bar").find(".weui-search-bar__input").val("").focus();
- });
- }($);
- /*===========================
- Device/OS Detection
- ===========================*/
- /* global $:true */
- ;(function ($) {
- "use strict";
- var device = {};
- var ua = navigator.userAgent;
- var android = ua.match(/(Android);?[\s\/]+([\d.]+)?/);
- var ipad = ua.match(/(iPad).*OS\s([\d_]+)/);
- var ipod = ua.match(/(iPod)(.*OS\s([\d_]+))?/);
- var iphone = !ipad && ua.match(/(iPhone\sOS)\s([\d_]+)/);
- device.ios = device.android = device.iphone = device.ipad = device.androidChrome = false;
-
- // Android
- if (android) {
- device.os = 'android';
- device.osVersion = android[2];
- device.android = true;
- device.androidChrome = ua.toLowerCase().indexOf('chrome') >= 0;
- }
- if (ipad || iphone || ipod) {
- device.os = 'ios';
- device.ios = true;
- }
- // iOS
- if (iphone && !ipod) {
- device.osVersion = iphone[2].replace(/_/g, '.');
- device.iphone = true;
- }
- if (ipad) {
- device.osVersion = ipad[2].replace(/_/g, '.');
- device.ipad = true;
- }
- if (ipod) {
- device.osVersion = ipod[3] ? ipod[3].replace(/_/g, '.') : null;
- device.iphone = true;
- }
- // iOS 8+ changed UA
- if (device.ios && device.osVersion && ua.indexOf('Version/') >= 0) {
- if (device.osVersion.split('.')[0] === '10') {
- device.osVersion = ua.toLowerCase().split('version/')[1].split(' ')[0];
- }
- }
- // Webview
- device.webView = (iphone || ipad || ipod) && ua.match(/.*AppleWebKit(?!.*Safari)/i);
-
- // Minimal UI
- if (device.os && device.os === 'ios') {
- var osVersionArr = device.osVersion.split('.');
- device.minimalUi = !device.webView &&
- (ipod || iphone) &&
- (osVersionArr[0] * 1 === 7 ? osVersionArr[1] * 1 >= 1 : osVersionArr[0] * 1 > 7) &&
- $('meta[name="viewport"]').length > 0 && $('meta[name="viewport"]').attr('content').indexOf('minimal-ui') >= 0;
- }
- // Check for status bar and fullscreen app mode
- var windowWidth = $(window).width();
- var windowHeight = $(window).height();
- device.statusBar = false;
- if (device.webView && (windowWidth * windowHeight === screen.width * screen.height)) {
- device.statusBar = true;
- }
- else {
- device.statusBar = false;
- }
- // Classes
- var classNames = [];
- // Pixel Ratio
- device.pixelRatio = window.devicePixelRatio || 1;
- classNames.push('pixel-ratio-' + Math.floor(device.pixelRatio));
- if (device.pixelRatio >= 2) {
- classNames.push('retina');
- }
- // OS classes
- if (device.os) {
- classNames.push(device.os, device.os + '-' + device.osVersion.split('.')[0], device.os + '-' + device.osVersion.replace(/\./g, '-'));
- if (device.os === 'ios') {
- var major = parseInt(device.osVersion.split('.')[0], 10);
- for (var i = major - 1; i >= 6; i--) {
- classNames.push('ios-gt-' + i);
- }
- }
-
- }
- // Status bar classes
- if (device.statusBar) {
- classNames.push('with-statusbar-overlay');
- }
- else {
- $('html').removeClass('with-statusbar-overlay');
- }
- // Add html classes
- if (classNames.length > 0) $('html').addClass(classNames.join(' '));
- $.device = device;
- })($);
- /*======================================================
- ************ Picker ************
- ======================================================*/
- /* global $:true */
- /* jshint unused:false */
- /* jshint multistr:true */
- + function($) {
- "use strict";
- var Picker = function (params) {
- var p = this;
- var defaults = {
- updateValuesOnMomentum: false,
- updateValuesOnTouchmove: true,
- rotateEffect: false,
- momentumRatio: 7,
- freeMode: false,
- // Common settings
- scrollToInput: true,
- inputReadOnly: true,
- toolbar: true,
- toolbarCloseText: '完成',
- title: '请选择',
- toolbarTemplate: '<div class="toolbar">\
- <div class="toolbar-inner">\
- <a href="javascript:;" class="picker-button close-picker">{{closeText}}</a>\
- <h1 class="title">{{title}}</h1>\
- </div>\
- </div>',
- };
- params = params || {};
- for (var def in defaults) {
- if (typeof params[def] === 'undefined') {
- params[def] = defaults[def];
- }
- }
- p.params = params;
- p.cols = [];
- p.initialized = false;
-
- // Inline flag
- p.inline = p.params.container ? true : false;
- // 3D Transforms origin bug, only on safari
- var originBug = $.device.ios || (navigator.userAgent.toLowerCase().indexOf('safari') >= 0 && navigator.userAgent.toLowerCase().indexOf('chrome') < 0) && !$.device.android;
- // Should be converted to popover
- function isPopover() {
- var toPopover = false;
- if (!p.params.convertToPopover && !p.params.onlyInPopover) return toPopover;
- if (!p.inline && p.params.input) {
- if (p.params.onlyInPopover) toPopover = true;
- else {
- if ($.device.ios) {
- toPopover = $.device.ipad ? true : false;
- }
- else {
- if ($(window).width() >= 768) toPopover = true;
- }
- }
- }
- return toPopover;
- }
- function inPopover() {
- if (p.opened && p.container && p.container.length > 0 && p.container.parents('.popover').length > 0) return true;
- else return false;
- }
- // Value
- p.setValue = function (arrValues, transition) {
- var valueIndex = 0;
- for (var i = 0; i < p.cols.length; i++) {
- if (p.cols[i] && !p.cols[i].divider) {
- p.cols[i].setValue(arrValues[valueIndex], transition);
- valueIndex++;
- }
- }
- };
- p.updateValue = function () {
- var newValue = [];
- var newDisplayValue = [];
- for (var i = 0; i < p.cols.length; i++) {
- if (!p.cols[i].divider) {
- newValue.push(p.cols[i].value);
- newDisplayValue.push(p.cols[i].displayValue);
- }
- }
- if (newValue.indexOf(undefined) >= 0) {
- return;
- }
- p.value = newValue;
- p.displayValue = newDisplayValue;
- if (p.params.onChange) {
- p.params.onChange(p, p.value, p.displayValue);
- }
- if (p.input && p.input.length > 0) {
- $(p.input).val(p.params.formatValue ? p.params.formatValue(p, p.value, p.displayValue) : p.value.join(' '));
- $(p.input).trigger('change');
- }
- };
- // Columns Handlers
- p.initPickerCol = function (colElement, updateItems) {
- var colContainer = $(colElement);
- var colIndex = colContainer.index();
- var col = p.cols[colIndex];
- if (col.divider) return;
- col.container = colContainer;
- col.wrapper = col.container.find('.picker-items-col-wrapper');
- col.items = col.wrapper.find('.picker-item');
-
- var i, j;
- var wrapperHeight, itemHeight, itemsHeight, minTranslate, maxTranslate;
- col.replaceValues = function (values, displayValues) {
- col.destroyEvents();
- col.values = values;
- col.displayValues = displayValues;
- var newItemsHTML = p.columnHTML(col, true);
- col.wrapper.html(newItemsHTML);
- col.items = col.wrapper.find('.picker-item');
- col.calcSize();
- col.setValue(col.values[0] || '', 0, true);
- col.initEvents();
- };
- col.calcSize = function () {
- if (!col.values.length) return;
- if (p.params.rotateEffect) {
- col.container.removeClass('picker-items-col-absolute');
- if (!col.width) col.container.css({width:''});
- }
- var colWidth, colHeight;
- colWidth = 0;
- colHeight = col.container[0].offsetHeight;
- wrapperHeight = col.wrapper[0].offsetHeight;
- itemHeight = col.items[0].offsetHeight;
- itemsHeight = itemHeight * col.items.length;
- minTranslate = colHeight / 2 - itemsHeight + itemHeight / 2;
- maxTranslate = colHeight / 2 - itemHeight / 2;
- if (col.width) {
- colWidth = col.width;
- if (parseInt(colWidth, 10) === colWidth) colWidth = colWidth + 'px';
- col.container.css({width: colWidth});
- }
- if (p.params.rotateEffect) {
- if (!col.width) {
- col.items.each(function () {
- var item = $(this);
- item.css({width:'auto'});
- colWidth = Math.max(colWidth, item[0].offsetWidth);
- item.css({width:''});
- });
- col.container.css({width: (colWidth + 2) + 'px'});
- }
- col.container.addClass('picker-items-col-absolute');
- }
- };
- col.calcSize();
-
- col.wrapper.transform('translate3d(0,' + maxTranslate + 'px,0)').transition(0);
- var activeIndex = 0;
- var animationFrameId;
- // Set Value Function
- col.setValue = function (newValue, transition, valueCallbacks) {
- if (typeof transition === 'undefined') transition = '';
- var newActiveIndex = col.wrapper.find('.picker-item[data-picker-value="' + newValue + '"]').index();
- if(typeof newActiveIndex === 'undefined' || newActiveIndex === -1) {
- col.value = col.displayValue = newValue;
- return;
- }
- var newTranslate = -newActiveIndex * itemHeight + maxTranslate;
- // Update wrapper
- col.wrapper.transition(transition);
- col.wrapper.transform('translate3d(0,' + (newTranslate) + 'px,0)');
-
- // Watch items
- if (p.params.updateValuesOnMomentum && col.activeIndex && col.activeIndex !== newActiveIndex ) {
- $.cancelAnimationFrame(animationFrameId);
- col.wrapper.transitionEnd(function(){
- $.cancelAnimationFrame(animationFrameId);
- });
- updateDuringScroll();
- }
- // Update items
- col.updateItems(newActiveIndex, newTranslate, transition, valueCallbacks);
- };
- col.updateItems = function (activeIndex, translate, transition, valueCallbacks) {
- if (typeof translate === 'undefined') {
- translate = $.getTranslate(col.wrapper[0], 'y');
- }
- if(typeof activeIndex === 'undefined') activeIndex = -Math.round((translate - maxTranslate)/itemHeight);
- if (activeIndex < 0) activeIndex = 0;
- if (activeIndex >= col.items.length) activeIndex = col.items.length - 1;
- var previousActiveIndex = col.activeIndex;
- col.activeIndex = activeIndex;
- /*
- col.wrapper.find('.picker-selected, .picker-after-selected, .picker-before-selected').removeClass('picker-selected picker-after-selected picker-before-selected');
- col.items.transition(transition);
- var selectedItem = col.items.eq(activeIndex).addClass('picker-selected').transform('');
- var prevItems = selectedItem.prevAll().addClass('picker-before-selected');
- var nextItems = selectedItem.nextAll().addClass('picker-after-selected');
- */
- //去掉 .picker-after-selected, .picker-before-selected 以提高性能
- col.wrapper.find('.picker-selected').removeClass('picker-selected');
- if (p.params.rotateEffect) {
- col.items.transition(transition);
- }
- var selectedItem = col.items.eq(activeIndex).addClass('picker-selected').transform('');
- if (valueCallbacks || typeof valueCallbacks === 'undefined') {
- // Update values
- col.value = selectedItem.attr('data-picker-value');
- col.displayValue = col.displayValues ? col.displayValues[activeIndex] : col.value;
- // On change callback
- if (previousActiveIndex !== activeIndex) {
- if (col.onChange) {
- col.onChange(p, col.value, col.displayValue);
- }
- p.updateValue();
- }
- }
-
- // Set 3D rotate effect
- if (!p.params.rotateEffect) {
- return;
- }
- var percentage = (translate - (Math.floor((translate - maxTranslate)/itemHeight) * itemHeight + maxTranslate)) / itemHeight;
-
- col.items.each(function () {
- var item = $(this);
- var itemOffsetTop = item.index() * itemHeight;
- var translateOffset = maxTranslate - translate;
- var itemOffset = itemOffsetTop - translateOffset;
- var percentage = itemOffset / itemHeight;
- var itemsFit = Math.ceil(col.height / itemHeight / 2) + 1;
-
- var angle = (-18*percentage);
- if (angle > 180) angle = 180;
- if (angle < -180) angle = -180;
- // Far class
- if (Math.abs(percentage) > itemsFit) item.addClass('picker-item-far');
- else item.removeClass('picker-item-far');
- // Set transform
- item.transform('translate3d(0, ' + (-translate + maxTranslate) + 'px, ' + (originBug ? -110 : 0) + 'px) rotateX(' + angle + 'deg)');
- });
- };
- function updateDuringScroll() {
- animationFrameId = $.requestAnimationFrame(function () {
- col.updateItems(undefined, undefined, 0);
- updateDuringScroll();
- });
- }
- // Update items on init
- if (updateItems) col.updateItems(0, maxTranslate, 0);
- var allowItemClick = true;
- var isTouched, isMoved, touchStartY, touchCurrentY, touchStartTime, touchEndTime, startTranslate, returnTo, currentTranslate, prevTranslate, velocityTranslate, velocityTime;
- function handleTouchStart (e) {
- if (isMoved || isTouched) return;
- e.preventDefault();
- isTouched = true;
- var position = $.getTouchPosition(e);
- touchStartY = touchCurrentY = position.y;
- touchStartTime = (new Date()).getTime();
-
- allowItemClick = true;
- startTranslate = currentTranslate = $.getTranslate(col.wrapper[0], 'y');
- }
- function handleTouchMove (e) {
- if (!isTouched) return;
- e.preventDefault();
- allowItemClick = false;
- var position = $.getTouchPosition(e);
- touchCurrentY = position.y;
- if (!isMoved) {
- // First move
- $.cancelAnimationFrame(animationFrameId);
- isMoved = true;
- startTranslate = currentTranslate = $.getTranslate(col.wrapper[0], 'y');
- col.wrapper.transition(0);
- }
- e.preventDefault();
- var diff = touchCurrentY - touchStartY;
- currentTranslate = startTranslate + diff;
- returnTo = undefined;
- // Normalize translate
- if (currentTranslate < minTranslate) {
- currentTranslate = minTranslate - Math.pow(minTranslate - currentTranslate, 0.8);
- returnTo = 'min';
- }
- if (currentTranslate > maxTranslate) {
- currentTranslate = maxTranslate + Math.pow(currentTranslate - maxTranslate, 0.8);
- returnTo = 'max';
- }
- // Transform wrapper
- col.wrapper.transform('translate3d(0,' + currentTranslate + 'px,0)');
- // Update items
- col.updateItems(undefined, currentTranslate, 0, p.params.updateValuesOnTouchmove);
-
- // Calc velocity
- velocityTranslate = currentTranslate - prevTranslate || currentTranslate;
- velocityTime = (new Date()).getTime();
- prevTranslate = currentTranslate;
- }
- function handleTouchEnd (e) {
- if (!isTouched || !isMoved) {
- isTouched = isMoved = false;
- return;
- }
- isTouched = isMoved = false;
- col.wrapper.transition('');
- if (returnTo) {
- if (returnTo === 'min') {
- col.wrapper.transform('translate3d(0,' + minTranslate + 'px,0)');
- }
- else col.wrapper.transform('translate3d(0,' + maxTranslate + 'px,0)');
- }
- touchEndTime = new Date().getTime();
- var velocity, newTranslate;
- if (touchEndTime - touchStartTime > 300) {
- newTranslate = currentTranslate;
- }
- else {
- velocity = Math.abs(velocityTranslate / (touchEndTime - velocityTime));
- newTranslate = currentTranslate + velocityTranslate * p.params.momentumRatio;
- }
- newTranslate = Math.max(Math.min(newTranslate, maxTranslate), minTranslate);
- // Active Index
- var activeIndex = -Math.floor((newTranslate - maxTranslate)/itemHeight);
- // Normalize translate
- if (!p.params.freeMode) newTranslate = -activeIndex * itemHeight + maxTranslate;
- // Transform wrapper
- col.wrapper.transform('translate3d(0,' + (parseInt(newTranslate,10)) + 'px,0)');
- // Update items
- col.updateItems(activeIndex, newTranslate, '', true);
- // Watch items
- if (p.params.updateValuesOnMomentum) {
- updateDuringScroll();
- col.wrapper.transitionEnd(function(){
- $.cancelAnimationFrame(animationFrameId);
- });
- }
- // Allow click
- setTimeout(function () {
- allowItemClick = true;
- }, 100);
- }
- function handleClick(e) {
- if (!allowItemClick) return;
- $.cancelAnimationFrame(animationFrameId);
- /*jshint validthis:true */
- var value = $(this).attr('data-picker-value');
- col.setValue(value);
- }
- col.initEvents = function (detach) {
- var method = detach ? 'off' : 'on';
- col.container[method]($.touchEvents.start, handleTouchStart);
- col.container[method]($.touchEvents.move, handleTouchMove);
- col.container[method]($.touchEvents.end, handleTouchEnd);
- col.items[method]('click', handleClick);
- };
- col.destroyEvents = function () {
- col.initEvents(true);
- };
- col.container[0].f7DestroyPickerCol = function () {
- col.destroyEvents();
- };
- col.initEvents();
- };
- p.destroyPickerCol = function (colContainer) {
- colContainer = $(colContainer);
- if ('f7DestroyPickerCol' in colContainer[0]) colContainer[0].f7DestroyPickerCol();
- };
- // Resize cols
- function resizeCols() {
- if (!p.opened) return;
- for (var i = 0; i < p.cols.length; i++) {
- if (!p.cols[i].divider) {
- p.cols[i].calcSize();
- p.cols[i].setValue(p.cols[i].value, 0, false);
- }
- }
- }
- $(window).on('resize', resizeCols);
- // HTML Layout
- p.columnHTML = function (col, onlyItems) {
- var columnItemsHTML = '';
- var columnHTML = '';
- if (col.divider) {
- columnHTML += '<div class="picker-items-col picker-items-col-divider ' + (col.textAlign ? 'picker-items-col-' + col.textAlign : '') + ' ' + (col.cssClass || '') + '">' + col.content + '</div>';
- }
- else {
- for (var j = 0; j < col.values.length; j++) {
- columnItemsHTML += '<div class="picker-item" data-picker-value="' + col.values[j] + '">' + (col.displayValues ? col.displayValues[j] : col.values[j]) + '</div>';
- }
- columnHTML += '<div class="picker-items-col ' + (col.textAlign ? 'picker-items-col-' + col.textAlign : '') + ' ' + (col.cssClass || '') + '"><div class="picker-items-col-wrapper">' + columnItemsHTML + '</div></div>';
- }
- return onlyItems ? columnItemsHTML : columnHTML;
- };
- p.layout = function () {
- var pickerHTML = '';
- var pickerClass = '';
- var i;
- p.cols = [];
- var colsHTML = '';
- for (i = 0; i < p.params.cols.length; i++) {
- var col = p.params.cols[i];
- colsHTML += p.columnHTML(p.params.cols[i]);
- p.cols.push(col);
- }
- pickerClass = 'weui-picker-modal picker-columns ' + (p.params.cssClass || '') + (p.params.rotateEffect ? ' picker-3d' : '') + (p.params.cols.length === 1 ? ' picker-columns-single' : '');
- pickerHTML =
- '<div class="' + (pickerClass) + '">' +
- (p.params.toolbar ? p.params.toolbarTemplate.replace(/{{closeText}}/g, p.params.toolbarCloseText).replace(/{{title}}/g, p.params.title) : '') +
- '<div class="picker-modal-inner picker-items">' +
- colsHTML +
- '<div class="picker-center-highlight"></div>' +
- '</div>' +
- '</div>';
-
- p.pickerHTML = pickerHTML;
- };
- // Input Events
- function openOnInput(e) {
- e.preventDefault();
- if (p.opened) return;
- p.open();
- if (p.params.scrollToInput && !isPopover()) {
- var pageContent = p.input.parents('.content');
- if (pageContent.length === 0) return;
- var paddingTop = parseInt(pageContent.css('padding-top'), 10),
- paddingBottom = parseInt(pageContent.css('padding-bottom'), 10),
- pageHeight = pageContent[0].offsetHeight - paddingTop - p.container.height(),
- pageScrollHeight = pageContent[0].scrollHeight - paddingTop - p.container.height(),
- newPaddingBottom;
- var inputTop = p.input.offset().top - paddingTop + p.input[0].offsetHeight;
- if (inputTop > pageHeight) {
- var scrollTop = pageContent.scrollTop() + inputTop - pageHeight;
- if (scrollTop + pageHeight > pageScrollHeight) {
- newPaddingBottom = scrollTop + pageHeight - pageScrollHeight + paddingBottom;
- if (pageHeight === pageScrollHeight) {
- newPaddingBottom = p.container.height();
- }
- pageContent.css({'padding-bottom': (newPaddingBottom) + 'px'});
- }
- pageContent.scrollTop(scrollTop, 300);
- }
- }
- }
- function closeOnHTMLClick(e) {
- if (inPopover()) return;
- if (p.input && p.input.length > 0) {
- if (e.target !== p.input[0] && $(e.target).parents('.weui-picker-modal').length === 0) p.close();
- }
- else {
- if ($(e.target).parents('.weui-picker-modal').length === 0) p.close();
- }
- }
- if (p.params.input) {
- p.input = $(p.params.input);
- if (p.input.length > 0) {
- if (p.params.inputReadOnly) p.input.prop('readOnly', true);
- if (!p.inline) {
- p.input.on('click', openOnInput);
- }
- if (p.params.inputReadOnly) {
- p.input.on('focus mousedown', function (e) {
- e.preventDefault();
- });
- }
- }
-
- }
-
- if (!p.inline) $('html').on('click', closeOnHTMLClick);
- // Open
- function onPickerClose() {
- p.opened = false;
- if (p.input && p.input.length > 0) p.input.parents('.page-content').css({'padding-bottom': ''});
- if (p.params.onClose) p.params.onClose(p);
- // Destroy events
- p.container.find('.picker-items-col').each(function () {
- p.destroyPickerCol(this);
- });
- }
- p.opened = false;
- p.open = function () {
- var toPopover = isPopover();
- if (!p.opened) {
- // Layout
- p.layout();
- // Append
- if (toPopover) {
- p.pickerHTML = '<div class="popover popover-picker-columns"><div class="popover-inner">' + p.pickerHTML + '</div></div>';
- p.popover = $.popover(p.pickerHTML, p.params.input, true);
- p.container = $(p.popover).find('.weui-picker-modal');
- $(p.popover).on('close', function () {
- onPickerClose();
- });
- }
- else if (p.inline) {
- p.container = $(p.pickerHTML);
- p.container.addClass('picker-modal-inline');
- $(p.params.container).append(p.container);
- }
- else {
- p.container = $($.openPicker(p.pickerHTML));
- $(p.container)
- .on('close', function () {
- onPickerClose();
- });
- }
- // Store picker instance
- p.container[0].f7Picker = p;
- // Init Events
- p.container.find('.picker-items-col').each(function () {
- var updateItems = true;
- if ((!p.initialized && p.params.value) || (p.initialized && p.value)) updateItems = false;
- p.initPickerCol(this, updateItems);
- });
-
- // Set value
- if (!p.initialized) {
- if (p.params.value) {
- p.setValue(p.params.value, 0);
- }
- }
- else {
- if (p.value) p.setValue(p.value, 0);
- }
- }
- // Set flag
- p.opened = true;
- p.initialized = true;
- if (p.params.onOpen) p.params.onOpen(p);
- };
- // Close
- p.close = function (force) {
- if (!p.opened || p.inline) return;
- if (inPopover()) {
- $.closePicker(p.popover);
- return;
- }
- else {
- $.closePicker(p.container);
- return;
- }
- };
- // Destroy
- p.destroy = function () {
- p.close();
- if (p.params.input && p.input.length > 0) {
- p.input.off('click focus', openOnInput);
- $(p.input).data('picker', null);
- }
- $('html').off('click', closeOnHTMLClick);
- $(window).off('resize', resizeCols);
- };
- if (p.inline) {
- p.open();
- }
- return p;
- };
- $(document).on("click", ".close-picker", function() {
- var pickerToClose = $('.weui-picker-modal.weui-picker-modal-visible');
- if (pickerToClose.length > 0) {
- $.closePicker(pickerToClose);
- }
- });
- //修复picker会滚动页面的bug
- $(document).on($.touchEvents.move, ".picker-modal-inner", function(e) {
- e.preventDefault();
- });
- $.openPicker = function(tpl, className, callback) {
- if(typeof className === "function") {
- callback = className;
- className = undefined;
- }
- $.closePicker();
- var container = $("<div class='weui-picker-container "+ (className || "") + "'></div>").appendTo(document.body);
- container.show();
- container.addClass("weui-picker-container-visible");
- //关于布局的问题,如果直接放在body上,则做动画的时候会撑开body高度而导致滚动条变化。
- var dialog = $(tpl).appendTo(container);
-
- dialog.width(); //通过取一次CSS值,强制浏览器不能把上下两行代码合并执行,因为合并之后会导致无法出现动画。
- dialog.addClass("weui-picker-modal-visible");
- callback && container.on("close", callback);
- return dialog;
- }
- $.updatePicker = function(tpl) {
- var container = $(".weui-picker-container-visible");
- if(!container[0]) return false;
- container.html("");
- var dialog = $(tpl).appendTo(container);
- dialog.addClass("weui-picker-modal-visible");
- return dialog;
- }
- $.closePicker = function(container, callback) {
- console.log(1);
- if(typeof container === "function") callback = container;
- $(".weui-picker-modal-visible").removeClass("weui-picker-modal-visible").transitionEnd(function() {
- $(this).parent().remove();
- callback && callback();
- }).trigger("close");
- };
- $.fn.picker = function(params) {
- var args = arguments;
- return this.each(function() {
- if(!this) return;
- var $this = $(this);
-
- var picker = $this.data("picker");
- if(!picker) {
- params = $.extend({ input: this }, params || {}) // https://github.com/lihongxun945/jquery-weui/issues/432
- var inputValue = $this.val();
- if(params.value === undefined && inputValue !== "") {
- params.value = (params.cols && params.cols.length > 1) ? inputValue.split(" ") : [inputValue];
- }
- var p = $.extend({input: this}, params);
- picker = new Picker(p);
- $this.data("picker", picker);
- }
- if(typeof params === typeof "a") {
- picker[params].apply(picker, Array.prototype.slice.call(args, 1));
- }
- });
- };
- }($);
- /* global $:true */
- + function($) {
- "use strict";
- var defaults;
- var selects = [];
- var Select = function(input, config) {
- var self = this;
- this.config = config;
- //init empty data
- this.data = {
- values: '',
- titles: '',
- origins: [],
- length: 0
- };
- this.$input = $(input);
- this.$input.prop("readOnly", true);
- this.initConfig();
- config = this.config;
- this.$input.click($.proxy(this.open, this));
- selects.push(this)
- }
- Select.prototype.initConfig = function() {
- this.config = $.extend({}, defaults, this.config);
- var config = this.config;
- if(!config.items || !config.items.length) return;
- config.items = config.items.map(function(d, i) {
- if(typeof d == typeof "a") {
- return {
- title: d,
- value: d
- };
- }
- return d;
- });
- this.tpl = $.t7.compile("<div class='weui-picker-modal weui-select-modal'>" + config.toolbarTemplate + (config.multi ? config.checkboxTemplate : config.radioTemplate) + "</div>");
- if(config.input !== undefined) this.$input.val(config.input);
- this.parseInitValue();
- this._init = true;
- }
- Select.prototype.updateInputValue = function(values, titles) {
- var v, t;
- if(this.config.multi) {
- v = values.join(this.config.split);
- t = titles.join(this.config.split);
- } else {
- v = values[0];
- t = titles[0];
- }
- //caculate origin data
- var origins = [];
- this.config.items.forEach(function(d) {
- values.each(function(i, dd) {
- if(d.value == dd) origins.push(d);
- });
- });
- this.$input.val(t).data("values", v);
- this.$input.attr("value", t).attr("data-values", v);
- var data = {
- values: v,
- titles: t,
- valuesArray: values,
- titlesArray: titles,
- origins: origins,
- length: origins.length
- };
- this.data = data;
- this.$input.trigger("change", data);
- this.config.onChange && this.config.onChange.call(this, data);
- }
- Select.prototype.parseInitValue = function() {
- var value = this.$input.val();
- var items = this.config.items;
- //如果input为空,只有在第一次初始化的时候才保留默认选择。因为后来就是用户自己取消了全部选择,不能再为他选中默认值。
- if( !this._init && (value === undefined || value == null || value === "")) return;
- var titles = this.config.multi ? value.split(this.config.split) : [value];
- for(var i=0;i<items.length;i++) {
- items[i].checked = false;
- for(var j=0;j<titles.length;j++) {
- if(items[i].title === titles[j]) {
- items[i].checked = true;
- }
- }
- }
- }
- Select.prototype._bind = function(dialog) {
- var self = this,
- config = this.config;
- dialog.on("change", function(e) {
- var checked = dialog.find("input:checked");
- var values = checked.map(function() {
- return $(this).val();
- });
- var titles = checked.map(function() {
- return $(this).data("title");
- });
- self.updateInputValue(values, titles);
- if(config.autoClose && !config.multi) self.close();
- })
- .trigger('change')
- .on("click", ".close-select", function() {
- self.close();
- });
- }
- //更新数据
- Select.prototype.update = function(config) {
- this.config = $.extend({}, this.config, config);
- this.initConfig();
- if(this._open) {
- this._bind($.updatePicker(this.getHTML()));
- }
- }
-
- Select.prototype.open = function(values, titles) {
- if(this._open) return;
- // open picker 会默认关掉其他的,但是 onClose 不会被调用,所以这里先关掉其他select
- for (var i = 0; i < selects.length; i++ ) {
- var s = selects[i];
- if (s === this) continue;
- if (s._open) {
- if(!s.close()) return false; // 其他的select由于某些条件限制关闭失败。
- }
- }
- this.parseInitValue();
- var config = this.config;
- var dialog = this.dialog = $.openPicker(this.getHTML());
-
- this._bind(dialog);
- this._open = true;
- if(config.onOpen) config.onOpen(this);
- }
- Select.prototype.close = function(callback, force) {
- if (!this._open) return false;
- var self = this,
- beforeClose = this.config.beforeClose;
- if(typeof callback === typeof true) {
- force === callback;
- }
- if(!force) {
- if(beforeClose && typeof beforeClose === 'function' && beforeClose.call(this, this.data.values, this.data.titles) === false) {
- return false
- }
- if(this.config.multi) {
- if(this.config.min !== undefined && this.data.length < this.config.min) {
- $.toast("请至少选择"+this.config.min+"个", "text");
- return false
- }
- if(this.config.max !== undefined && this.data.length > this.config.max) {
- $.toast("最多只能选择"+this.config.max+"个", "text");
- return false
- }
- }
- }
- $.closePicker(function() {
- self.onClose();
- callback && callback();
- });
- return true
- }
- Select.prototype.onClose = function() {
- this._open = false;
- if(this.config.onClose) this.config.onClose(this);
- }
- Select.prototype.getHTML = function(callback) {
- var config = this.config;
- return this.tpl({
- items: config.items,
- title: config.title,
- closeText: config.closeText
- })
- }
- $.fn.select = function(params, args) {
- return this.each(function() {
- var $this = $(this);
- if(!$this.data("weui-select")) $this.data("weui-select", new Select(this, params));
- var select = $this.data("weui-select");
- if(typeof params === typeof "a") select[params].call(select, args);
- return select;
- });
- }
- defaults = $.fn.select.prototype.defaults = {
- items: [],
- input: undefined, //输入框的初始值
- title: "请选择",
- multi: false,
- closeText: "确定",
- autoClose: true, //是否选择完成后自动关闭,只有单选模式下才有效
- onChange: undefined, //function
- beforeClose: undefined, // function 关闭之前,如果返回false则阻止关闭
- onClose: undefined, //function
- onOpen: undefined, //function
- split: ",", //多选模式下的分隔符
- min: undefined, //多选模式下可用,最少选择数
- max: undefined, //单选模式下可用,最多选择数
- toolbarTemplate: '<div class="toolbar">\
- <div class="toolbar-inner">\
- <a href="javascript:;" class="picker-button close-select">{{closeText}}</a>\
- <h1 class="title">{{title}}</h1>\
- </div>\
- </div>',
- radioTemplate:
- '<div class="weui-cells weui-cells_radio">\
- {{#items}}\
- <label class="weui-cell weui-check_label" for="weui-select-id-{{this.title}}">\
- <div class="weui-cell__bd weui-cell_primary">\
- <p>{{this.title}}</p>\
- </div>\
- <div class="weui-cell__ft">\
- <input type="radio" class="weui-check" name="weui-select" id="weui-select-id-{{this.title}}" value="{{this.value}}" {{#if this.checked}}checked="checked"{{/if}} data-title="{{this.title}}">\
- <span class="weui-icon-checked"></span>\
- </div>\
- </label>\
- {{/items}}\
- </div>',
- checkboxTemplate:
- '<div class="weui-cells weui-cells_checkbox">\
- {{#items}}\
- <label class="weui-cell weui-check_label" for="weui-select-id-{{this.title}}">\
- <div class="weui-cell__bd weui-cell_primary">\
- <p>{{this.title}}</p>\
- </div>\
- <div class="weui-cell__ft">\
- <input type="checkbox" class="weui-check" name="weui-select" id="weui-select-id-{{this.title}}" value="{{this.value}}" {{#if this.checked}}checked="checked"{{/if}} data-title="{{this.title}}" >\
- <span class="weui-icon-checked"></span>\
- </div>\
- </label>\
- {{/items}}\
- </div>',
- }
- }($);
- /*======================================================
- ************ Calendar ************
- ======================================================*/
- /* global $:true */
- /*jshint unused: false*/
- +function ($) {
- "use strict";
- var rtl = false;
- var defaults;
- var isSameDate = function (a, b) {
- var a = new Date(a),
- b = new Date(b);
- return a.getFullYear() === b.getFullYear() && a.getMonth() === b.getMonth() && a.getDate() === b.getDate()
- }
- var Calendar = function (params) {
- var p = this;
- params = params || {};
- for (var def in defaults) {
- if (typeof params[def] === 'undefined') {
- params[def] = defaults[def];
- }
- }
- p.params = params;
- p.initialized = false;
- // Inline flag
- p.inline = p.params.container ? true : false;
- // Is horizontal
- p.isH = p.params.direction === 'horizontal';
- // RTL inverter
- var inverter = p.isH ? (rtl ? -1 : 1) : 1;
- // Animating flag
- p.animating = false;
- // Should be converted to popover
- function isPopover() {
- var toPopover = false;
- if (!p.params.convertToPopover && !p.params.onlyInPopover) return toPopover;
- if (!p.inline && p.params.input) {
- if (p.params.onlyInPopover) toPopover = true;
- else {
- if ($.device.ios) {
- toPopover = $.device.ipad ? true : false;
- }
- else {
- if ($(window).width() >= 768) toPopover = true;
- }
- }
- }
- return toPopover;
- }
- function inPopover() {
- if (p.opened && p.container && p.container.length > 0 && p.container.parents('.popover').length > 0) return true;
- else return false;
- }
- // Format date
- function formatDate(date) {
- date = new Date(date);
- var year = date.getFullYear();
- var month = date.getMonth();
- var month1 = month + 1;
- var day = date.getDate();
- var weekDay = date.getDay();
- return p.params.dateFormat
- .replace(/yyyy/g, year)
- .replace(/yy/g, (year + '').substring(2))
- .replace(/mm/g, month1 < 10 ? '0' + month1 : month1)
- .replace(/m/g, month1)
- .replace(/MM/g, p.params.monthNames[month])
- .replace(/M/g, p.params.monthNamesShort[month])
- .replace(/dd/g, day < 10 ? '0' + day : day)
- .replace(/d/g, day)
- .replace(/DD/g, p.params.dayNames[weekDay])
- .replace(/D/g, p.params.dayNamesShort[weekDay]);
- }
- // Value
- p.addValue = function (value) {
- if (p.params.multiple) {
- if (!p.value) p.value = [];
- var inValuesIndex;
- for (var i = 0; i < p.value.length; i++) {
- if (isSameDate(value, p.value[i])) {
- inValuesIndex = i;
- }
- }
- if (typeof inValuesIndex === 'undefined') {
- p.value.push(value);
- }
- else {
- p.value.splice(inValuesIndex, 1);
- }
- p.updateValue();
- }
- else {
- p.value = [value];
- p.updateValue();
- }
- };
- p.setValue = function (arrValues) {
- var date = new Date(arrValues[0]);
- p.setYearMonth(date.getFullYear(), date.getMonth());
- p.addValue(+ date);
- };
- p.updateValue = function () {
- p.wrapper.find('.picker-calendar-day-selected').removeClass('picker-calendar-day-selected');
- var i, inputValue;
- for (i = 0; i < p.value.length; i++) {
- var valueDate = new Date(p.value[i]);
- p.wrapper.find('.picker-calendar-day[data-date="' + valueDate.getFullYear() + '-' + valueDate.getMonth() + '-' + valueDate.getDate() + '"]').addClass('picker-calendar-day-selected');
- }
- if (p.params.onChange) {
- p.params.onChange(p, p.value.map(formatDate), p.value.map(function (d) {
- return + new Date(typeof d === typeof 'a' ? d.split(/\D/).filter(function (a) { return !!a; }).join("-") : d);
- }));
- }
- if (p.input && p.input.length > 0) {
- if (p.params.formatValue) inputValue = p.params.formatValue(p, p.value);
- else {
- inputValue = [];
- for (i = 0; i < p.value.length; i++) {
- inputValue.push(formatDate(p.value[i]));
- }
- inputValue = inputValue.join(', ');
- }
- $(p.input).val(inputValue);
- $(p.input).trigger('change');
- }
- };
- // Columns Handlers
- p.initCalendarEvents = function () {
- var col;
- var allowItemClick = true;
- var isTouched, isMoved, touchStartX, touchStartY, touchCurrentX, touchCurrentY, touchStartTime, touchEndTime, startTranslate, currentTranslate, wrapperWidth, wrapperHeight, percentage, touchesDiff, isScrolling;
- function handleTouchStart (e) {
- if (isMoved || isTouched) return;
- // e.preventDefault();
- isTouched = true;
- var position = $.getTouchPosition(e);
- touchStartX = touchCurrentY = position.x;
- touchStartY = touchCurrentY = position.y;
- touchStartTime = (new Date()).getTime();
- percentage = 0;
- allowItemClick = true;
- isScrolling = undefined;
- startTranslate = currentTranslate = p.monthsTranslate;
- }
- function handleTouchMove (e) {
- if (!isTouched) return;
- var position = $.getTouchPosition(e);
- touchCurrentX = position.x;
- touchCurrentY = position.y;
- if (typeof isScrolling === 'undefined') {
- isScrolling = !!(isScrolling || Math.abs(touchCurrentY - touchStartY) > Math.abs(touchCurrentX - touchStartX));
- }
- if (p.isH && isScrolling) {
- isTouched = false;
- return;
- }
- e.preventDefault();
- if (p.animating) {
- isTouched = false;
- return;
- }
- allowItemClick = false;
- if (!isMoved) {
- // First move
- isMoved = true;
- wrapperWidth = p.wrapper[0].offsetWidth;
- wrapperHeight = p.wrapper[0].offsetHeight;
- p.wrapper.transition(0);
- }
- e.preventDefault();
- touchesDiff = p.isH ? touchCurrentX - touchStartX : touchCurrentY - touchStartY;
- percentage = touchesDiff/(p.isH ? wrapperWidth : wrapperHeight);
- currentTranslate = (p.monthsTranslate * inverter + percentage) * 100;
- // Transform wrapper
- p.wrapper.transform('translate3d(' + (p.isH ? currentTranslate : 0) + '%, ' + (p.isH ? 0 : currentTranslate) + '%, 0)');
- }
- function handleTouchEnd (e) {
- if (!isTouched || !isMoved) {
- isTouched = isMoved = false;
- return;
- }
- isTouched = isMoved = false;
-
- touchEndTime = new Date().getTime();
- if (touchEndTime - touchStartTime < 300) {
- if (Math.abs(touchesDiff) < 10) {
- p.resetMonth();
- }
- else if (touchesDiff >= 10) {
- if (rtl) p.nextMonth();
- else p.prevMonth();
- }
- else {
- if (rtl) p.prevMonth();
- else p.nextMonth();
- }
- }
- else {
- if (percentage <= -0.5) {
- if (rtl) p.prevMonth();
- else p.nextMonth();
- }
- else if (percentage >= 0.5) {
- if (rtl) p.nextMonth();
- else p.prevMonth();
- }
- else {
- p.resetMonth();
- }
- }
- // Allow click
- setTimeout(function () {
- allowItemClick = true;
- }, 100);
- }
- function handleDayClick(e) {
- if (!allowItemClick) return;
- var day = $(e.target).parents('.picker-calendar-day');
- if (day.length === 0 && $(e.target).hasClass('picker-calendar-day')) {
- day = $(e.target);
- }
- if (day.length === 0) return;
- // if (day.hasClass('picker-calendar-day-selected') && !p.params.multiple) return;
- if (day.hasClass('picker-calendar-day-disabled')) return;
- if (day.hasClass('picker-calendar-day-next')) p.nextMonth();
- if (day.hasClass('picker-calendar-day-prev')) p.prevMonth();
- var dateYear = day.attr('data-year');
- var dateMonth = day.attr('data-month');
- var dateDay = day.attr('data-day');
- if (p.params.onDayClick) {
- p.params.onDayClick(p, day[0], dateYear, dateMonth, dateDay);
- }
- p.addValue(new Date(dateYear, dateMonth, dateDay).getTime());
- if (p.params.closeOnSelect && !p.params.multiple) p.close();
- }
- p.container.find('.picker-calendar-prev-month').on('click', p.prevMonth);
- p.container.find('.picker-calendar-next-month').on('click', p.nextMonth);
- p.container.find('.picker-calendar-prev-year').on('click', p.prevYear);
- p.container.find('.picker-calendar-next-year').on('click', p.nextYear);
- p.wrapper.on('click', handleDayClick);
- if (p.params.touchMove) {
- p.wrapper.on($.touchEvents.start, handleTouchStart);
- p.wrapper.on($.touchEvents.move, handleTouchMove);
- p.wrapper.on($.touchEvents.end, handleTouchEnd);
- }
-
- p.container[0].f7DestroyCalendarEvents = function () {
- p.container.find('.picker-calendar-prev-month').off('click', p.prevMonth);
- p.container.find('.picker-calendar-next-month').off('click', p.nextMonth);
- p.container.find('.picker-calendar-prev-year').off('click', p.prevYear);
- p.container.find('.picker-calendar-next-year').off('click', p.nextYear);
- p.wrapper.off('click', handleDayClick);
- if (p.params.touchMove) {
- p.wrapper.off($.touchEvents.start, handleTouchStart);
- p.wrapper.off($.touchEvents.move, handleTouchMove);
- p.wrapper.off($.touchEvents.end, handleTouchEnd);
- }
- };
-
- };
- p.destroyCalendarEvents = function (colContainer) {
- if ('f7DestroyCalendarEvents' in p.container[0]) p.container[0].f7DestroyCalendarEvents();
- };
- // Calendar Methods
- p.daysInMonth = function (date) {
- var d = new Date(date);
- return new Date(d.getFullYear(), d.getMonth() + 1, 0).getDate();
- };
- p.monthHTML = function (date, offset) {
- date = new Date(date);
- var year = date.getFullYear(),
- month = date.getMonth(),
- day = date.getDate();
- if (offset === 'next') {
- if (month === 11) date = new Date(year + 1, 0);
- else date = new Date(year, month + 1, 1);
- }
- if (offset === 'prev') {
- if (month === 0) date = new Date(year - 1, 11);
- else date = new Date(year, month - 1, 1);
- }
- if (offset === 'next' || offset === 'prev') {
- month = date.getMonth();
- year = date.getFullYear();
- }
- var daysInPrevMonth = p.daysInMonth(new Date(date.getFullYear(), date.getMonth()).getTime() - 10 * 24 * 60 * 60 * 1000),
- daysInMonth = p.daysInMonth(date),
- firstDayOfMonthIndex = new Date(date.getFullYear(), date.getMonth()).getDay();
- if (firstDayOfMonthIndex === 0) firstDayOfMonthIndex = 7;
-
- var dayDate, currentValues = [], i, j,
- rows = 6, cols = 7,
- monthHTML = '',
- dayIndex = 0 + (p.params.firstDay - 1),
- today = new Date().setHours(0,0,0,0),
- minDate = p.params.minDate ? new Date(p.params.minDate).getTime() : null,
- maxDate = p.params.maxDate ? new Date(p.params.maxDate).getTime() : null;
- if (p.value && p.value.length) {
- for (i = 0; i < p.value.length; i++) {
- currentValues.push(new Date(p.value[i]).setHours(0,0,0,0));
- }
- }
-
- for (i = 1; i <= rows; i++) {
- var rowHTML = '';
- var row = i;
- for (j = 1; j <= cols; j++) {
- var col = j;
- dayIndex ++;
- var dayNumber = dayIndex - firstDayOfMonthIndex;
- var addClass = '';
- if (dayNumber < 0) {
- dayNumber = daysInPrevMonth + dayNumber + 1;
- addClass += ' picker-calendar-day-prev';
- dayDate = new Date(month - 1 < 0 ? year - 1 : year, month - 1 < 0 ? 11 : month - 1, dayNumber).getTime();
- }
- else {
- dayNumber = dayNumber + 1;
- if (dayNumber > daysInMonth) {
- dayNumber = dayNumber - daysInMonth;
- addClass += ' picker-calendar-day-next';
- dayDate = new Date(month + 1 > 11 ? year + 1 : year, month + 1 > 11 ? 0 : month + 1, dayNumber).getTime();
- }
- else {
- dayDate = new Date(year, month, dayNumber).getTime();
- }
- }
- // Today
- if (dayDate === today) addClass += ' picker-calendar-day-today';
- // Selected
- if (currentValues.indexOf(dayDate) >= 0) addClass += ' picker-calendar-day-selected';
- // Weekend
- if (p.params.weekendDays.indexOf(col - 1) >= 0) {
- addClass += ' picker-calendar-day-weekend';
- }
- // Disabled
- if ((minDate && dayDate < minDate) || (maxDate && dayDate > maxDate)) {
- addClass += ' picker-calendar-day-disabled';
- }
- dayDate = new Date(dayDate);
- var dayYear = dayDate.getFullYear();
- var dayMonth = dayDate.getMonth();
- rowHTML += '<div data-year="' + dayYear + '" data-month="' + dayMonth + '" data-day="' + dayNumber + '" class="picker-calendar-day' + (addClass) + '" data-date="' + (dayYear + '-' + dayMonth + '-' + dayNumber) + '"><span>'+dayNumber+'</span></div>';
- }
- monthHTML += '<div class="picker-calendar-row">' + rowHTML + '</div>';
- }
- monthHTML = '<div class="picker-calendar-month" data-year="' + year + '" data-month="' + month + '">' + monthHTML + '</div>';
- return monthHTML;
- };
- p.animating = false;
- p.updateCurrentMonthYear = function (dir) {
- if (typeof dir === 'undefined') {
- p.currentMonth = parseInt(p.months.eq(1).attr('data-month'), 10);
- p.currentYear = parseInt(p.months.eq(1).attr('data-year'), 10);
- }
- else {
- p.currentMonth = parseInt(p.months.eq(dir === 'next' ? (p.months.length - 1) : 0).attr('data-month'), 10);
- p.currentYear = parseInt(p.months.eq(dir === 'next' ? (p.months.length - 1) : 0).attr('data-year'), 10);
- }
- p.container.find('.current-month-value').text(p.params.monthNames[p.currentMonth]);
- p.container.find('.current-year-value').text(p.currentYear);
-
- };
- p.onMonthChangeStart = function (dir) {
- p.updateCurrentMonthYear(dir);
- p.months.removeClass('picker-calendar-month-current picker-calendar-month-prev picker-calendar-month-next');
- var currentIndex = dir === 'next' ? p.months.length - 1 : 0;
- p.months.eq(currentIndex).addClass('picker-calendar-month-current');
- p.months.eq(dir === 'next' ? currentIndex - 1 : currentIndex + 1).addClass(dir === 'next' ? 'picker-calendar-month-prev' : 'picker-calendar-month-next');
- if (p.params.onMonthYearChangeStart) {
- p.params.onMonthYearChangeStart(p, p.currentYear, p.currentMonth);
- }
- };
- p.onMonthChangeEnd = function (dir, rebuildBoth) {
- p.animating = false;
- var nextMonthHTML, prevMonthHTML, newMonthHTML;
- p.wrapper.find('.picker-calendar-month:not(.picker-calendar-month-prev):not(.picker-calendar-month-current):not(.picker-calendar-month-next)').remove();
-
- if (typeof dir === 'undefined') {
- dir = 'next';
- rebuildBoth = true;
- }
- if (!rebuildBoth) {
- newMonthHTML = p.monthHTML(new Date(p.currentYear, p.currentMonth), dir);
- }
- else {
- p.wrapper.find('.picker-calendar-month-next, .picker-calendar-month-prev').remove();
- prevMonthHTML = p.monthHTML(new Date(p.currentYear, p.currentMonth), 'prev');
- nextMonthHTML = p.monthHTML(new Date(p.currentYear, p.currentMonth), 'next');
- }
- if (dir === 'next' || rebuildBoth) {
- p.wrapper.append(newMonthHTML || nextMonthHTML);
- }
- if (dir === 'prev' || rebuildBoth) {
- p.wrapper.prepend(newMonthHTML || prevMonthHTML);
- }
- p.months = p.wrapper.find('.picker-calendar-month');
- p.setMonthsTranslate(p.monthsTranslate);
- if (p.params.onMonthAdd) {
- p.params.onMonthAdd(p, dir === 'next' ? p.months.eq(p.months.length - 1)[0] : p.months.eq(0)[0]);
- }
- if (p.params.onMonthYearChangeEnd) {
- p.params.onMonthYearChangeEnd(p, p.currentYear, p.currentMonth);
- }
- };
- p.setMonthsTranslate = function (translate) {
- translate = translate || p.monthsTranslate || 0;
- if (typeof p.monthsTranslate === 'undefined') p.monthsTranslate = translate;
- p.months.removeClass('picker-calendar-month-current picker-calendar-month-prev picker-calendar-month-next');
- var prevMonthTranslate = -(translate + 1) * 100 * inverter;
- var currentMonthTranslate = -translate * 100 * inverter;
- var nextMonthTranslate = -(translate - 1) * 100 * inverter;
- p.months.eq(0).transform('translate3d(' + (p.isH ? prevMonthTranslate : 0) + '%, ' + (p.isH ? 0 : prevMonthTranslate) + '%, 0)').addClass('picker-calendar-month-prev');
- p.months.eq(1).transform('translate3d(' + (p.isH ? currentMonthTranslate : 0) + '%, ' + (p.isH ? 0 : currentMonthTranslate) + '%, 0)').addClass('picker-calendar-month-current');
- p.months.eq(2).transform('translate3d(' + (p.isH ? nextMonthTranslate : 0) + '%, ' + (p.isH ? 0 : nextMonthTranslate) + '%, 0)').addClass('picker-calendar-month-next');
- };
- p.nextMonth = function (transition) {
- if (typeof transition === 'undefined' || typeof transition === 'object') {
- transition = '';
- if (!p.params.animate) transition = 0;
- }
- var nextMonth = parseInt(p.months.eq(p.months.length - 1).attr('data-month'), 10);
- var nextYear = parseInt(p.months.eq(p.months.length - 1).attr('data-year'), 10);
- var nextDate = new Date(nextYear, nextMonth);
- var nextDateTime = nextDate.getTime();
- var transitionEndCallback = p.animating ? false : true;
- if (p.params.maxDate) {
- if (nextDateTime > new Date(p.params.maxDate).getTime()) {
- return p.resetMonth();
- }
- }
- p.monthsTranslate --;
- if (nextMonth === p.currentMonth) {
- var nextMonthTranslate = -(p.monthsTranslate) * 100 * inverter;
- var nextMonthHTML = $(p.monthHTML(nextDateTime, 'next')).transform('translate3d(' + (p.isH ? nextMonthTranslate : 0) + '%, ' + (p.isH ? 0 : nextMonthTranslate) + '%, 0)').addClass('picker-calendar-month-next');
- p.wrapper.append(nextMonthHTML[0]);
- p.months = p.wrapper.find('.picker-calendar-month');
- if (p.params.onMonthAdd) {
- p.params.onMonthAdd(p, p.months.eq(p.months.length - 1)[0]);
- }
- }
- p.animating = true;
- p.onMonthChangeStart('next');
- var translate = (p.monthsTranslate * 100) * inverter;
- p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? translate : 0) + '%, ' + (p.isH ? 0 : translate) + '%, 0)');
- if (transitionEndCallback) {
- p.wrapper.transitionEnd(function () {
- p.onMonthChangeEnd('next');
- });
- }
- if (!p.params.animate) {
- p.onMonthChangeEnd('next');
- }
- };
- p.prevMonth = function (transition) {
- if (typeof transition === 'undefined' || typeof transition === 'object') {
- transition = '';
- if (!p.params.animate) transition = 0;
- }
- var prevMonth = parseInt(p.months.eq(0).attr('data-month'), 10);
- var prevYear = parseInt(p.months.eq(0).attr('data-year'), 10);
- var prevDate = new Date(prevYear, prevMonth + 1, -1);
- var prevDateTime = prevDate.getTime();
- var transitionEndCallback = p.animating ? false : true;
- if (p.params.minDate) {
- if (prevDateTime < new Date(p.params.minDate).getTime()) {
- return p.resetMonth();
- }
- }
- p.monthsTranslate ++;
- if (prevMonth === p.currentMonth) {
- var prevMonthTranslate = -(p.monthsTranslate) * 100 * inverter;
- var prevMonthHTML = $(p.monthHTML(prevDateTime, 'prev')).transform('translate3d(' + (p.isH ? prevMonthTranslate : 0) + '%, ' + (p.isH ? 0 : prevMonthTranslate) + '%, 0)').addClass('picker-calendar-month-prev');
- p.wrapper.prepend(prevMonthHTML[0]);
- p.months = p.wrapper.find('.picker-calendar-month');
- if (p.params.onMonthAdd) {
- p.params.onMonthAdd(p, p.months.eq(0)[0]);
- }
- }
- p.animating = true;
- p.onMonthChangeStart('prev');
- var translate = (p.monthsTranslate * 100) * inverter;
- p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? translate : 0) + '%, ' + (p.isH ? 0 : translate) + '%, 0)');
- if (transitionEndCallback) {
- p.wrapper.transitionEnd(function () {
- p.onMonthChangeEnd('prev');
- });
- }
- if (!p.params.animate) {
- p.onMonthChangeEnd('prev');
- }
- };
- p.resetMonth = function (transition) {
- if (typeof transition === 'undefined') transition = '';
- var translate = (p.monthsTranslate * 100) * inverter;
- p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? translate : 0) + '%, ' + (p.isH ? 0 : translate) + '%, 0)');
- };
- p.setYearMonth = function (year, month, transition) {
- if (typeof year === 'undefined') year = p.currentYear;
- if (typeof month === 'undefined') month = p.currentMonth;
- if (typeof transition === 'undefined' || typeof transition === 'object') {
- transition = '';
- if (!p.params.animate) transition = 0;
- }
- var targetDate;
- if (year < p.currentYear) {
- targetDate = new Date(year, month + 1, -1).getTime();
- }
- else {
- targetDate = new Date(year, month).getTime();
- }
- if (p.params.maxDate && targetDate > new Date(p.params.maxDate).getTime()) {
- return false;
- }
- if (p.params.minDate && targetDate < new Date(p.params.minDate).getTime()) {
- return false;
- }
- var currentDate = new Date(p.currentYear, p.currentMonth).getTime();
- var dir = targetDate > currentDate ? 'next' : 'prev';
- var newMonthHTML = p.monthHTML(new Date(year, month));
- p.monthsTranslate = p.monthsTranslate || 0;
- var prevTranslate = p.monthsTranslate;
- var monthTranslate, wrapperTranslate;
- var transitionEndCallback = p.animating ? false : true;
- if (targetDate > currentDate) {
- // To next
- p.monthsTranslate --;
- if (!p.animating) p.months.eq(p.months.length - 1).remove();
- p.wrapper.append(newMonthHTML);
- p.months = p.wrapper.find('.picker-calendar-month');
- monthTranslate = -(prevTranslate - 1) * 100 * inverter;
- p.months.eq(p.months.length - 1).transform('translate3d(' + (p.isH ? monthTranslate : 0) + '%, ' + (p.isH ? 0 : monthTranslate) + '%, 0)').addClass('picker-calendar-month-next');
- }
- else {
- // To prev
- p.monthsTranslate ++;
- if (!p.animating) p.months.eq(0).remove();
- p.wrapper.prepend(newMonthHTML);
- p.months = p.wrapper.find('.picker-calendar-month');
- monthTranslate = -(prevTranslate + 1) * 100 * inverter;
- p.months.eq(0).transform('translate3d(' + (p.isH ? monthTranslate : 0) + '%, ' + (p.isH ? 0 : monthTranslate) + '%, 0)').addClass('picker-calendar-month-prev');
- }
- if (p.params.onMonthAdd) {
- p.params.onMonthAdd(p, dir === 'next' ? p.months.eq(p.months.length - 1)[0] : p.months.eq(0)[0]);
- }
- p.animating = true;
- p.onMonthChangeStart(dir);
- wrapperTranslate = (p.monthsTranslate * 100) * inverter;
- p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? wrapperTranslate : 0) + '%, ' + (p.isH ? 0 : wrapperTranslate) + '%, 0)');
- if (transitionEndCallback) {
- p.wrapper.transitionEnd(function () {
- p.onMonthChangeEnd(dir, true);
- });
- }
- if (!p.params.animate) {
- p.onMonthChangeEnd(dir);
- }
- };
- p.nextYear = function () {
- p.setYearMonth(p.currentYear + 1);
- };
- p.prevYear = function () {
- p.setYearMonth(p.currentYear - 1);
- };
-
- // HTML Layout
- p.layout = function () {
- var pickerHTML = '';
- var pickerClass = '';
- var i;
-
- var layoutDate = p.value && p.value.length ? p.value[0] : new Date().setHours(0,0,0,0);
- var prevMonthHTML = p.monthHTML(layoutDate, 'prev');
- var currentMonthHTML = p.monthHTML(layoutDate);
- var nextMonthHTML = p.monthHTML(layoutDate, 'next');
- var monthsHTML = '<div class="picker-calendar-months"><div class="picker-calendar-months-wrapper">' + (prevMonthHTML + currentMonthHTML + nextMonthHTML) + '</div></div>';
- // Week days header
- var weekHeaderHTML = '';
- if (p.params.weekHeader) {
- for (i = 0; i < 7; i++) {
- var weekDayIndex = (i + p.params.firstDay > 6) ? (i - 7 + p.params.firstDay) : (i + p.params.firstDay);
- var dayName = p.params.dayNamesShort[weekDayIndex];
- weekHeaderHTML += '<div class="picker-calendar-week-day ' + ((p.params.weekendDays.indexOf(weekDayIndex) >= 0) ? 'picker-calendar-week-day-weekend' : '') + '"> ' + dayName + '</div>';
-
- }
- weekHeaderHTML = '<div class="picker-calendar-week-days">' + weekHeaderHTML + '</div>';
- }
- pickerClass = 'weui-picker-calendar ' + (p.params.cssClass || '');
- if(!p.inline) pickerClass = 'weui-picker-modal ' + pickerClass;
- var toolbarHTML = p.params.toolbar ? p.params.toolbarTemplate.replace(/{{closeText}}/g, p.params.toolbarCloseText) : '';
- if (p.params.toolbar) {
- toolbarHTML = p.params.toolbarTemplate
- .replace(/{{closeText}}/g, p.params.toolbarCloseText)
- .replace(/{{monthPicker}}/g, (p.params.monthPicker ? p.params.monthPickerTemplate : ''))
- .replace(/{{yearPicker}}/g, (p.params.yearPicker ? p.params.yearPickerTemplate : ''));
- }
- pickerHTML =
- '<div class="' + (pickerClass) + '">' +
- toolbarHTML +
- '<div class="picker-modal-inner">' +
- weekHeaderHTML +
- monthsHTML +
- '</div>' +
- '</div>';
-
-
- p.pickerHTML = pickerHTML;
- };
- // Input Events
- function openOnInput(e) {
- e.preventDefault();
- if (p.opened) return;
- p.open();
- if (p.params.scrollToInput && !isPopover()) {
- var pageContent = p.input.parents('.page-content');
- if (pageContent.length === 0) return;
- var paddingTop = parseInt(pageContent.css('padding-top'), 10),
- paddingBottom = parseInt(pageContent.css('padding-bottom'), 10),
- pageHeight = pageContent[0].offsetHeight - paddingTop - p.container.height(),
- pageScrollHeight = pageContent[0].scrollHeight - paddingTop - p.container.height(),
- newPaddingBottom;
- var inputTop = p.input.offset().top - paddingTop + p.input[0].offsetHeight;
- if (inputTop > pageHeight) {
- var scrollTop = pageContent.scrollTop() + inputTop - pageHeight;
- if (scrollTop + pageHeight > pageScrollHeight) {
- newPaddingBottom = scrollTop + pageHeight - pageScrollHeight + paddingBottom;
- if (pageHeight === pageScrollHeight) {
- newPaddingBottom = p.container.height();
- }
- pageContent.css({'padding-bottom': (newPaddingBottom) + 'px'});
- }
- pageContent.scrollTop(scrollTop, 300);
- }
- }
- }
- function closeOnHTMLClick(e) {
- if (inPopover()) return;
- if (p.input && p.input.length > 0) {
- if (e.target !== p.input[0] && $(e.target).parents('.weui-picker-modal').length === 0) p.close();
- }
- else {
- if ($(e.target).parents('.weui-picker-modal').length === 0) p.close();
- }
- }
- if (p.params.input) {
- p.input = $(p.params.input);
- if (p.input.length > 0) {
- if (p.params.inputReadOnly) p.input.prop('readOnly', true);
- if (!p.inline) {
- p.input.on('click', openOnInput);
- }
- if (p.params.inputReadOnly) {
- p.input.on('focus mousedown', function (e) {
- e.preventDefault();
- });
- }
- }
-
- }
-
- //iphone 上无法正确触发 click,会导致点击外面无法关闭
- if (!p.inline) $(document).on('click touchend', closeOnHTMLClick);
- // Open
- function onPickerClose() {
- p.opened = false;
- if (p.input && p.input.length > 0) p.input.parents('.page-content').css({'padding-bottom': ''});
- if (p.params.onClose) p.params.onClose(p);
- // Destroy events
- p.destroyCalendarEvents();
- }
- p.opened = false;
- p.open = function () {
- var toPopover = isPopover() && false;
- var updateValue = false;
- if (!p.opened) {
- // Set date value
- if (!p.value) {
- if (p.params.value) {
- p.value = p.params.value;
- updateValue = true;
- }
- }
- // Layout
- p.layout();
- // Append
- if (toPopover) {
- p.pickerHTML = '<div class="popover popover-picker-calendar"><div class="popover-inner">' + p.pickerHTML + '</div></div>';
- p.popover = $.popover(p.pickerHTML, p.params.input, true);
- p.container = $(p.popover).find('.weui-picker-modal');
- $(p.popover).on('close', function () {
- onPickerClose();
- });
- }
- else if (p.inline) {
- p.container = $(p.pickerHTML);
- p.container.addClass('picker-modal-inline');
- $(p.params.container).append(p.container);
- }
- else {
- p.container = $($.openPicker(p.pickerHTML));
- $(p.container)
- .on('close', function () {
- onPickerClose();
- });
- }
- // Store calendar instance
- p.container[0].f7Calendar = p;
- p.wrapper = p.container.find('.picker-calendar-months-wrapper');
- // Months
- p.months = p.wrapper.find('.picker-calendar-month');
- // Update current month and year
- p.updateCurrentMonthYear();
- // Set initial translate
- p.monthsTranslate = 0;
- p.setMonthsTranslate();
- // Init events
- p.initCalendarEvents();
- // Update input value
- if (updateValue) p.updateValue();
-
- }
- // Set flag
- p.opened = true;
- p.initialized = true;
- if (p.params.onMonthAdd) {
- p.months.each(function () {
- p.params.onMonthAdd(p, this);
- });
- }
- if (p.params.onOpen) p.params.onOpen(p);
- };
- // Close
- p.close = function () {
- if (!p.opened || p.inline) return;
- p.animating = false; //有可能还有动画没做完,因此animating设置还没改。
- if (inPopover()) {
- $.closePicker(p.popover);
- return;
- }
- else {
- $.closePicker(p.container);
- return;
- }
- };
- // Destroy
- p.destroy = function () {
- p.close();
- if (p.params.input && p.input.length > 0) {
- p.input.off('click focus', openOnInput);
- p.input.data("calendar", null);
- }
- $('html').off('click', closeOnHTMLClick);
- };
- if (p.inline) {
- p.open();
- }
- return p;
- };
- var format = function(d) {
- return d < 10 ? "0"+d : d;
- }
- $.fn.calendar = function (params, args) {
- params = params || {};
- return this.each(function() {
- var $this = $(this);
- if(!$this[0]) return;
- var p = {};
- if($this[0].tagName.toUpperCase() === "INPUT") {
- p.input = $this;
- } else {
- p.container = $this;
- }
- var calendar = $this.data("calendar");
- if(!calendar) {
- if(typeof params === typeof "a") {
- } else {
- if(!params.value && $this.val()) params.value = [$this.val()];
- //默认显示今天
- if(!params.value) {
- var today = new Date();
- params.value = [today.getFullYear() + "/" + format(today.getMonth() + 1) + "/" + format(today.getDate())];
- }
- calendar = $this.data("calendar", new Calendar($.extend(p, params)));
- }
- }
- if(typeof params === typeof "a") {
- calendar[params].call(calendar, args);
- }
- });
- };
- defaults = $.fn.calendar.prototype.defaults = {
- value: undefined, // 通过JS赋值,注意是数组
- monthNames: ['一月', '二月', '三月', '四月', '五月', '六月', '七月', '八月', '九月', '十月', '十一月', '十二月'],
- monthNamesShort: ['一月', '二月', '三月', '四月', '五月', '六月', '七月', '八月', '九月', '十月', '十一月', '十二月'],
- dayNames: ['周日', '周一', '周二', '周三', '周四', '周五', '周六'],
- dayNamesShort: ['周日', '周一', '周二', '周三', '周四', '周五', '周六'],
- firstDay: 1, // First day of the week, Monday
- weekendDays: [0, 6], // Sunday and Saturday
- multiple: false,
- dateFormat: 'yyyy-mm-dd',
- direction: 'horizontal', // or 'vertical'
- minDate: null,
- maxDate: null,
- touchMove: true,
- animate: true,
- closeOnSelect: true,
- monthPicker: true,
- monthPickerTemplate:
- '<div class="picker-calendar-month-picker">' +
- '<a href="javascript:;" class="link icon-only picker-calendar-prev-month"><i class="icon icon-prev"></i></a>' +
- '<div class="current-month-value"></div>' +
- '<a href="javascript:;" class="link icon-only picker-calendar-next-month"><i class="icon icon-next"></i></a>' +
- '</div>',
- yearPicker: true,
- yearPickerTemplate:
- '<div class="picker-calendar-year-picker">' +
- '<a href="javascript:;" class="link icon-only picker-calendar-prev-year"><i class="icon icon-prev"></i></a>' +
- '<span class="current-year-value"></span>' +
- '<a href="javascript:;" class="link icon-only picker-calendar-next-year"><i class="icon icon-next"></i></a>' +
- '</div>',
- weekHeader: true,
- // Common settings
- scrollToInput: true,
- inputReadOnly: true,
- convertToPopover: true,
- onlyInPopover: false,
- toolbar: true,
- toolbarCloseText: 'Done',
- toolbarTemplate:
- '<div class="toolbar">' +
- '<div class="toolbar-inner">' +
- '{{yearPicker}}' +
- '{{monthPicker}}' +
- // '<a href="#" class="link close-picker">{{closeText}}</a>' +
- '</div>' +
- '</div>',
- /* Callbacks
- onMonthAdd
- onChange
- onOpen
- onClose
- onDayClick
- onMonthYearChangeStart
- onMonthYearChangeEnd
- */
- };
- }($);
- /* global $:true */
- /* jshint unused:false*/
- + function($) {
- "use strict";
- var defaults;
- var formatNumber = function (n) {
- return n < 10 ? "0" + n : n;
- }
- var Datetime = function(input, params) {
- this.input = $(input);
- this.params = params || {};
- this.initMonthes = params.monthes
- this.initYears = params.years
- var p = $.extend({}, params, this.getConfig());
- $(this.input).picker(p);
- }
- Datetime.prototype = {
- getDays : function(max) {
- var days = [];
- for(var i=1; i<= (max||31);i++) {
- days.push(i < 10 ? "0"+i : i);
- }
- return days;
- },
- getDaysByMonthAndYear : function(month, year) {
- var int_d = new Date(year, parseInt(month)+1-1, 1);
- var d = new Date(int_d - 1);
- return this.getDays(d.getDate());
- },
- getConfig: function() {
- var today = new Date(),
- params = this.params,
- self = this,
- lastValidValues;
- var config = {
- rotateEffect: false, //为了性能
- cssClass: 'datetime-picker',
- value: [today.getFullYear(), formatNumber(today.getMonth()+1), formatNumber(today.getDate()), formatNumber(today.getHours()), (formatNumber(today.getMinutes()))],
- onChange: function (picker, values, displayValues) {
- var cols = picker.cols;
- var days = self.getDaysByMonthAndYear(values[1], values[0]);
- var currentValue = values[2];
- if(currentValue > days.length) currentValue = days.length;
- picker.cols[4].setValue(currentValue);
- //check min and max
- var current = new Date(values[0]+'-'+values[1]+'-'+values[2]);
- var valid = true;
- if(params.min) {
- var min = new Date(typeof params.min === "function" ? params.min() : params.min);
- if(current < +min) {
- picker.setValue(lastValidValues);
- valid = false;
- }
- }
- if(params.max) {
- var max = new Date(typeof params.max === "function" ? params.max() : params.max);
- if(current > +max) {
- picker.setValue(lastValidValues);
- valid = false;
- }
- }
- valid && (lastValidValues = values);
- if (self.params.onChange) {
- self.params.onChange.apply(this, arguments);
- }
- },
- formatValue: function (p, values, displayValues) {
- return self.params.format(p, values, displayValues);
- },
- cols: [
- {
- values: this.initYears
- },
- {
- divider: true, // 这是一个分隔符
- content: params.yearSplit
- },
- {
- values: this.initMonthes
- },
- {
- divider: true, // 这是一个分隔符
- content: params.monthSplit
- },
- {
- values: (function () {
- var dates = [];
- for (var i=1; i<=31; i++) dates.push(formatNumber(i));
- return dates;
- })()
- },
-
- ]
- }
- if (params.dateSplit) {
- config.cols.push({
- divider: true,
- content: params.dateSplit
- })
- }
- config.cols.push({
- divider: true,
- content: params.datetimeSplit
- })
- var times = self.params.times();
- if (times && times.length) {
- config.cols = config.cols.concat(times);
- }
- var inputValue = this.input.val();
- if(inputValue) config.value = params.parse(inputValue);
- if(this.params.value) {
- this.input.val(this.params.value);
- config.value = params.parse(this.params.value);
- }
- return config;
- }
- }
- $.fn.datetimePicker = function(params) {
- params = $.extend({}, defaults, params);
- return this.each(function() {
- if(!this) return;
- var $this = $(this);
- var datetime = $this.data("datetime");
- if(!datetime) $this.data("datetime", new Datetime(this, params));
- return datetime;
- });
- };
- defaults = $.fn.datetimePicker.prototype.defaults = {
- input: undefined, // 默认值
- min: undefined, // YYYY-MM-DD 最大最小值只比较年月日,不比较时分秒
- max: undefined, // YYYY-MM-DD
- yearSplit: '-',
- monthSplit: '-',
- dateSplit: '', // 默认为空
- datetimeSplit: ' ', // 日期和时间之间的分隔符,不可为空
- monthes: ('01 02 03 04 05 06 07 08 09 10 11 12').split(' '),
- years: (function () {
- var arr = [];
- for (var i = 1950; i <= 2030; i++) { arr.push(i); }
- return arr;
- })(),
- times: function () {
- return [ // 自定义的时间
- {
- values: (function () {
- var hours = [];
- for (var i=0; i<24; i++) hours.push(formatNumber(i));
- return hours;
- })()
- },
- {
- divider: true, // 这是一个分隔符
- content: ':'
- },
- {
- values: (function () {
- var minutes = [];
- for (var i=0; i<60; i++) minutes.push(formatNumber(i));
- return minutes;
- })()
- }
- ];
- },
- format: function (p, values) { // 数组转换成字符串
- return p.cols.map(function (col) {
- return col.value || col.content;
- }).join('');
- },
- parse: function (str) {
- // 把字符串转换成数组,用来解析初始值
- // 如果你的定制的初始值格式无法被这个默认函数解析,请自定义这个函数。比如你的时间是 '子时' 那么默认情况这个'时'会被当做分隔符而导致错误,所以你需要自己定义parse函数
- // 默认兼容的分隔符
- var t = str.split(this.datetimeSplit);
- return t[0].split(/\D/).concat(t[1].split(/:|时|分|秒/)).filter(function (d) {
- return !!d;
- })
- }
- }
- }($);
- /*======================================================
- ************ Picker ************
- ======================================================*/
- /* global $:true */
- + function($) {
- "use strict";
- //Popup 和 picker 之类的不要共用一个弹出方法,因为这样会导致 在 popup 中再弹出 picker 的时候会有问题。
- $.openPopup = function(popup, className) {
- $.closePopup();
- popup = $(popup);
- popup.show();
- popup.width();
- popup.addClass("weui-popup__container--visible");
- var modal = popup.find(".weui-popup__modal");
- modal.width();
- modal.transitionEnd(function() {
- modal.trigger("open");
- });
- }
- $.closePopup = function(container, remove) {
- container = $(container || ".weui-popup__container--visible");
- container.find('.weui-popup__modal').transitionEnd(function() {
- var $this = $(this);
- $this.trigger("close");
- container.hide();
- remove && container.remove();
- })
- container.removeClass("weui-popup__container--visible")
- };
- $(document).on("click", ".close-popup, .weui-popup__overlay", function() {
- $.closePopup();
- })
- .on("click", ".open-popup", function() {
- $($(this).data("target")).popup();
- })
- .on("click", ".weui-popup__container", function(e) {
- if($(e.target).hasClass("weui-popup__container")) $.closePopup();
- })
- $.fn.popup = function() {
- return this.each(function() {
- $.openPopup(this);
- });
- };
- }($);
- /* ===============================================================================
- ************ Notification ************
- =============================================================================== */
- /* global $:true */
- +function ($) {
- "use strict";
- var noti, defaults, timeout, start, diffX, diffY;
- var touchStart = function(e) {
- var p = $.getTouchPosition(e);
- start = p;
- diffX = diffY = 0;
- noti.addClass("touching");
- };
- var touchMove = function(e) {
- if(!start) return false;
- e.preventDefault();
- e.stopPropagation();
- var p = $.getTouchPosition(e);
- diffX = p.x - start.x;
- diffY = p.y - start.y;
- if(diffY > 0) {
- diffY = Math.sqrt(diffY);
- }
- noti.css("transform", "translate3d(0, "+diffY+"px, 0)");
- };
- var touchEnd = function() {
- noti.removeClass("touching");
- noti.attr("style", "");
- if(diffY < 0 && (Math.abs(diffY) > noti.height()*0.38)) {
- $.closeNotification();
- }
- if(Math.abs(diffX) <= 1 && Math.abs(diffY) <= 1) {
- noti.trigger("noti-click");
- }
- start = false;
- };
- var attachEvents = function(el) {
- el.on($.touchEvents.start, touchStart);
- el.on($.touchEvents.move, touchMove);
- el.on($.touchEvents.end, touchEnd);
- };
- $.notification = $.noti = function(params) {
- params = $.extend({}, defaults, params);
- noti = $(".weui-notification");
- if(!noti[0]) { // create a new notification
- noti = $('<div class="weui-notification"></div>').appendTo(document.body);
- attachEvents(noti);
- }
- noti.off("noti-click"); //the click event is not correct sometime: it will trigger when user is draging.
- if(params.onClick) noti.on("noti-click", function() {
- params.onClick(params.data);
- });
- noti.html($.t7.compile(params.tpl)(params));
- noti.show();
- noti.addClass("weui-notification--in");
- noti.data("params", params);
- var startTimeout = function() {
- if(timeout) {
- clearTimeout(timeout);
- timeout = null;
- }
- timeout = setTimeout(function() {
- if(noti.hasClass("weui-notification--touching")) {
- startTimeout();
- } else {
- $.closeNotification();
- }
- }, params.time);
- };
- startTimeout();
- };
- $.closeNotification = function() {
- timeout && clearTimeout(timeout);
- timeout = null;
- var noti = $(".weui-notification").removeClass("weui-notification--in").transitionEnd(function() {
- $(this).remove();
- });
- if(noti[0]) {
- var params = $(".weui-notification").data("params");
- if(params && params.onClose) {
- params.onClose(params.data);
- }
- }
- };
- defaults = $.noti.prototype.defaults = {
- title: undefined,
- text: undefined,
- media: undefined,
- time: 4000,
- onClick: undefined,
- onClose: undefined,
- data: undefined,
- tpl: '<div class="weui-notification__inner">' +
- '{{#if media}}<div class="weui-notification__media">{{media}}</div>{{/if}}' +
- '<div class="weui-notification__content">' +
- '{{#if title}}<div class="weui-notification__title">{{title}}</div>{{/if}}' +
- '{{#if text}}<div class="weui-notification__text">{{text}}</div>{{/if}}' +
- '</div>' +
- '<div class="weui-notification__handle-bar"></div>' +
- '</div>'
- };
- }($);
- + function($) {
- "use strict";
- var timeout;
- $.toptip = function(text, duration, type) {
- if(!text) return;
- if(typeof duration === typeof "a") {
- type = duration;
- duration = undefined;
- }
- duration = duration || 3000;
- var className = type ? 'bg-' + type : 'bg-danger';
- var $t = $('.weui-toptips').remove();
- $t = $('<div class="weui-toptips"></div>').appendTo(document.body);
- $t.html(text);
- $t[0].className = 'weui-toptips ' + className
- clearTimeout(timeout);
- if(!$t.hasClass('weui-toptips_visible')) {
- $t.show().width();
- $t.addClass('weui-toptips_visible');
- }
- timeout = setTimeout(function() {
- $t.removeClass('weui-toptips_visible').transitionEnd(function() {
- $t.remove();
- });
- }, duration);
- }
- }($);
- /* global $:true */
- + function($) {
- "use strict";
- var Slider = function (container, arg) {
- this.container = $(container);
- this.handler = this.container.find('.weui-slider__handler')
- this.track = this.container.find('.weui-slider__track')
- this.value = this.container.find('.weui-slider-box__value')
- this.bind()
- if (typeof arg === 'function') {
- this.callback = arg
- }
- }
- Slider.prototype.bind = function () {
- this.container
- .on($.touchEvents.start, $.proxy(this.touchStart, this))
- .on($.touchEvents.end, $.proxy(this.touchEnd, this));
- $(document.body).on($.touchEvents.move, $.proxy(this.touchMove, this)) // move even outside container
- }
- Slider.prototype.touchStart = function (e) {
- e.preventDefault()
- this.start = $.getTouchPosition(e)
- this.width = this.container.find('.weui-slider__inner').width()
- this.left = parseInt(this.container.find('.weui-slider__handler').css('left'))
- this.touching = true
- }
- Slider.prototype.touchMove = function (e) {
- if (!this.touching) return true
- var p = $.getTouchPosition(e)
- var distance = p.x - this.start.x
- var left = distance + this.left
- var per = parseInt(left / this.width * 100)
- if (per < 0) per = 0
- if (per > 100) per = 100
- this.handler.css('left', per + '%')
- this.track.css('width', per + '%')
- this.value.text(per)
- this.callback && this.callback.call(this, per)
- this.container.trigger('change', per)
- }
- Slider.prototype.touchEnd = function (e) {
- this.touching = false
- }
- $.fn.slider = function (arg) {
- this.each(function () {
- var $this = $(this)
- var slider = $this.data('slider')
- if (slider) return slider;
- else $this.data('slider', new Slider(this, arg))
- })
- }
- }($);
- /* ===============================================================================
- ************ Swipeout ************
- =============================================================================== */
- /* global $:true */
- +function ($) {
- "use strict";
- var cache = [];
- var TOUCHING = 'swipeout-touching'
- var Swipeout = function(el) {
- this.container = $(el);
- this.mover = this.container.find('>.weui-cell__bd')
- this.attachEvents();
- cache.push(this)
- }
- Swipeout.prototype.touchStart = function(e) {
- var p = $.getTouchPosition(e);
- this.container.addClass(TOUCHING);
- this.start = p;
- this.startX = 0;
- this.startTime = + new Date;
- var transform = this.mover.css('transform').match(/-?[\d\.]+/g)
- if (transform && transform.length) this.startX = parseInt(transform[4])
- this.diffX = this.diffY = 0;
- this._closeOthers()
- this.limit = this.container.find('>.weui-cell__ft').width() || 68; // 因为有的时候初始化的时候元素是隐藏的(比如在对话框内),所以在touchstart的时候计算宽度而不是初始化的时候
- };
- Swipeout.prototype.touchMove= function(e) {
- if(!this.start) return true;
- var p = $.getTouchPosition(e);
- this.diffX = p.x - this.start.x;
- this.diffY = p.y - this.start.y;
- if (Math.abs(this.diffX) < Math.abs(this.diffY)) { // 说明是上下方向在拖动
- this.close()
- this.start = false
- return true;
- }
- e.preventDefault();
- e.stopPropagation();
- var x = this.diffX + this.startX
- if (x > 0) x = 0;
- if (Math.abs(x) > this.limit) x = - (Math.pow(-(x+this.limit), .7) + this.limit)
- this.mover.css("transform", "translate3d("+x+"px, 0, 0)");
- };
- Swipeout.prototype.touchEnd = function() {
- if (!this.start) return true;
- this.start = false;
- var x = this.diffX + this.startX
- var t = new Date - this.startTime;
- if (this.diffX < -5 && t < 200) { // 向左快速滑动,则打开
- this.open()
- } else if (this.diffX >= 0 && t < 200) { // 向右快速滑动,或者单击,则关闭
- this.close()
- } else if (x > 0 || -x <= this.limit / 2) {
- this.close()
- } else {
- this.open()
- }
- };
- Swipeout.prototype.close = function() {
- this.container.removeClass(TOUCHING);
- this.mover.css("transform", "translate3d(0, 0, 0)");
- this.container.trigger('swipeout-close');
- }
- Swipeout.prototype.open = function() {
- this.container.removeClass(TOUCHING);
- this._closeOthers()
- this.mover.css("transform", "translate3d(" + (-this.limit) + "px, 0, 0)");
- this.container.trigger('swipeout-open');
- }
- Swipeout.prototype.attachEvents = function() {
- var el = this.mover;
- el.on($.touchEvents.start, $.proxy(this.touchStart, this));
- el.on($.touchEvents.move, $.proxy(this.touchMove, this));
- el.on($.touchEvents.end, $.proxy(this.touchEnd, this));
- }
- Swipeout.prototype._closeOthers = function() {
- //close others
- var self = this
- cache.forEach(function (s) {
- if (s !== self) s.close()
- })
- }
- var swipeout = function(el) {
- return new Swipeout(el);
- };
- $.fn.swipeout = function (arg) {
- return this.each(function() {
- var $this = $(this)
- var s = $this.data('swipeout') || swipeout(this);
- $this.data('swipeout', s);
- if (typeof arg === typeof 'a') {
- s[arg]()
- }
- });
- }
- $('.weui-cell_swiped').swipeout() // auto init
- }($);
|