123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804980598069807980898099810981198129813981498159816981798189819982098219822982398249825982698279828982998309831983298339834983598369837983898399840984198429843984498459846984798489849985098519852985398549855985698579858985998609861986298639864986598669867986898699870987198729873987498759876987798789879988098819882988398849885988698879888988998909891989298939894989598969897989898999900990199029903990499059906990799089909991099119912991399149915991699179918991999209921992299239924992599269927992899299930993199329933993499359936993799389939994099419942994399449945994699479948994999509951995299539954995599569957995899599960996199629963996499659966996799689969997099719972997399749975997699779978997999809981998299839984998599869987998899899990999199929993999499959996999799989999100001000110002100031000410005100061000710008100091001010011100121001310014100151001610017100181001910020100211002210023100241002510026100271002810029100301003110032100331003410035100361003710038100391004010041100421004310044100451004610047100481004910050100511005210053100541005510056100571005810059100601006110062100631006410065100661006710068100691007010071100721007310074100751007610077100781007910080100811008210083100841008510086100871008810089100901009110092100931009410095100961009710098100991010010101101021010310104101051010610107101081010910110101111011210113101141011510116101171011810119101201012110122101231012410125101261012710128101291013010131101321013310134101351013610137101381013910140101411014210143101441014510146101471014810149101501015110152101531015410155101561015710158101591016010161101621016310164101651016610167101681016910170101711017210173101741017510176101771017810179101801018110182101831018410185101861018710188101891019010191101921019310194101951019610197101981019910200102011020210203102041020510206102071020810209102101021110212102131021410215102161021710218102191022010221102221022310224102251022610227102281022910230102311023210233102341023510236102371023810239102401024110242102431024410245102461024710248102491025010251102521025310254102551025610257102581025910260102611026210263102641026510266102671026810269102701027110272102731027410275102761027710278102791028010281102821028310284102851028610287102881028910290102911029210293102941029510296102971029810299103001030110302103031030410305103061030710308103091031010311103121031310314103151031610317103181031910320103211032210323103241032510326103271032810329103301033110332103331033410335103361033710338103391034010341103421034310344103451034610347103481034910350103511035210353103541035510356103571035810359103601036110362103631036410365103661036710368103691037010371103721037310374103751037610377103781037910380103811038210383103841038510386103871038810389103901039110392103931039410395103961039710398103991040010401104021040310404104051040610407104081040910410104111041210413104141041510416104171041810419104201042110422104231042410425104261042710428104291043010431104321043310434104351043610437104381043910440104411044210443104441044510446104471044810449104501045110452104531045410455104561045710458104591046010461104621046310464104651046610467104681046910470104711047210473104741047510476104771047810479104801048110482104831048410485104861048710488104891049010491104921049310494104951049610497104981049910500105011050210503105041050510506105071050810509105101051110512105131051410515105161051710518105191052010521105221052310524105251052610527105281052910530105311053210533105341053510536105371053810539105401054110542105431054410545105461054710548105491055010551105521055310554105551055610557105581055910560105611056210563105641056510566105671056810569105701057110572105731057410575105761057710578105791058010581105821058310584105851058610587105881058910590105911059210593105941059510596105971059810599106001060110602106031060410605106061060710608106091061010611106121061310614106151061610617106181061910620106211062210623106241062510626106271062810629106301063110632106331063410635106361063710638106391064010641106421064310644106451064610647106481064910650106511065210653106541065510656106571065810659106601066110662106631066410665106661066710668106691067010671106721067310674106751067610677106781067910680106811068210683106841068510686106871068810689106901069110692106931069410695106961069710698106991070010701107021070310704107051070610707107081070910710107111071210713107141071510716107171071810719107201072110722107231072410725107261072710728107291073010731107321073310734107351073610737107381073910740107411074210743107441074510746107471074810749107501075110752107531075410755107561075710758107591076010761107621076310764107651076610767107681076910770107711077210773107741077510776107771077810779107801078110782107831078410785107861078710788107891079010791107921079310794107951079610797107981079910800108011080210803108041080510806108071080810809108101081110812108131081410815108161081710818108191082010821108221082310824108251082610827108281082910830108311083210833108341083510836108371083810839108401084110842108431084410845108461084710848108491085010851108521085310854108551085610857108581085910860108611086210863108641086510866108671086810869108701087110872108731087410875108761087710878108791088010881108821088310884108851088610887108881088910890108911089210893108941089510896108971089810899109001090110902109031090410905109061090710908109091091010911109121091310914109151091610917109181091910920109211092210923109241092510926109271092810929109301093110932109331093410935109361093710938109391094010941109421094310944109451094610947109481094910950109511095210953109541095510956109571095810959109601096110962109631096410965109661096710968109691097010971109721097310974109751097610977109781097910980109811098210983109841098510986109871098810989109901099110992109931099410995109961099710998109991100011001110021100311004110051100611007110081100911010110111101211013110141101511016110171101811019110201102111022110231102411025110261102711028110291103011031110321103311034110351103611037110381103911040110411104211043110441104511046110471104811049110501105111052110531105411055110561105711058110591106011061110621106311064110651106611067110681106911070110711107211073110741107511076110771107811079110801108111082110831108411085110861108711088110891109011091110921109311094110951109611097110981109911100111011110211103111041110511106111071110811109111101111111112111131111411115111161111711118111191112011121111221112311124111251112611127111281112911130111311113211133111341113511136111371113811139111401114111142111431114411145111461114711148111491115011151111521115311154111551115611157111581115911160111611116211163111641116511166111671116811169111701117111172111731117411175111761117711178111791118011181111821118311184111851118611187111881118911190111911119211193111941119511196111971119811199112001120111202112031120411205112061120711208112091121011211112121121311214112151121611217112181121911220112211122211223112241122511226112271122811229112301123111232112331123411235112361123711238112391124011241112421124311244112451124611247112481124911250112511125211253112541125511256112571125811259112601126111262112631126411265112661126711268112691127011271112721127311274112751127611277112781127911280112811128211283112841128511286112871128811289112901129111292112931129411295112961129711298112991130011301113021130311304113051130611307113081130911310113111131211313113141131511316113171131811319113201132111322113231132411325113261132711328113291133011331113321133311334113351133611337113381133911340113411134211343113441134511346113471134811349113501135111352113531135411355113561135711358113591136011361113621136311364113651136611367113681136911370113711137211373113741137511376113771137811379113801138111382113831138411385113861138711388113891139011391113921139311394113951139611397113981139911400114011140211403114041140511406114071140811409114101141111412114131141411415114161141711418114191142011421114221142311424114251142611427114281142911430114311143211433114341143511436114371143811439114401144111442114431144411445114461144711448114491145011451114521145311454114551145611457114581145911460114611146211463114641146511466114671146811469114701147111472114731147411475114761147711478114791148011481114821148311484114851148611487114881148911490114911149211493114941149511496114971149811499115001150111502115031150411505115061150711508115091151011511115121151311514115151151611517115181151911520115211152211523115241152511526115271152811529115301153111532115331153411535115361153711538115391154011541115421154311544115451154611547115481154911550115511155211553115541155511556115571155811559115601156111562115631156411565115661156711568115691157011571115721157311574115751157611577115781157911580115811158211583115841158511586115871158811589115901159111592115931159411595115961159711598115991160011601116021160311604116051160611607116081160911610116111161211613116141161511616116171161811619116201162111622116231162411625116261162711628116291163011631116321163311634116351163611637116381163911640116411164211643116441164511646116471164811649116501165111652116531165411655116561165711658116591166011661116621166311664116651166611667116681166911670116711167211673116741167511676116771167811679116801168111682116831168411685116861168711688116891169011691116921169311694116951169611697116981169911700117011170211703117041170511706117071170811709117101171111712117131171411715117161171711718117191172011721117221172311724117251172611727117281172911730117311173211733117341173511736117371173811739117401174111742117431174411745117461174711748117491175011751117521175311754117551175611757117581175911760117611176211763117641176511766117671176811769117701177111772117731177411775117761177711778117791178011781117821178311784117851178611787117881178911790117911179211793117941179511796117971179811799118001180111802118031180411805118061180711808118091181011811118121181311814118151181611817118181181911820118211182211823118241182511826118271182811829118301183111832118331183411835118361183711838118391184011841118421184311844118451184611847118481184911850118511185211853118541185511856118571185811859118601186111862118631186411865118661186711868118691187011871118721187311874118751187611877118781187911880118811188211883118841188511886118871188811889118901189111892118931189411895118961189711898118991190011901119021190311904119051190611907119081190911910119111191211913119141191511916119171191811919119201192111922119231192411925119261192711928119291193011931119321193311934119351193611937119381193911940119411194211943119441194511946119471194811949119501195111952119531195411955119561195711958119591196011961119621196311964119651196611967119681196911970119711197211973119741197511976119771197811979119801198111982119831198411985119861198711988119891199011991119921199311994119951199611997119981199912000120011200212003120041200512006120071200812009120101201112012120131201412015120161201712018120191202012021120221202312024120251202612027120281202912030120311203212033120341203512036120371203812039120401204112042120431204412045120461204712048120491205012051120521205312054120551205612057120581205912060120611206212063120641206512066120671206812069120701207112072120731207412075120761207712078120791208012081120821208312084120851208612087120881208912090120911209212093120941209512096120971209812099121001210112102121031210412105121061210712108121091211012111121121211312114121151211612117121181211912120121211212212123121241212512126121271212812129121301213112132121331213412135121361213712138121391214012141121421214312144121451214612147121481214912150121511215212153121541215512156121571215812159121601216112162121631216412165121661216712168121691217012171121721217312174121751217612177121781217912180121811218212183121841218512186121871218812189121901219112192121931219412195121961219712198121991220012201122021220312204122051220612207122081220912210122111221212213122141221512216122171221812219122201222112222122231222412225122261222712228122291223012231122321223312234122351223612237122381223912240122411224212243122441224512246122471224812249122501225112252122531225412255122561225712258122591226012261122621226312264122651226612267122681226912270122711227212273122741227512276122771227812279122801228112282122831228412285122861228712288122891229012291122921229312294122951229612297122981229912300123011230212303123041230512306123071230812309123101231112312123131231412315123161231712318123191232012321123221232312324123251232612327123281232912330123311233212333123341233512336123371233812339123401234112342123431234412345123461234712348123491235012351123521235312354123551235612357123581235912360123611236212363123641236512366123671236812369123701237112372123731237412375123761237712378123791238012381123821238312384123851238612387123881238912390123911239212393123941239512396123971239812399124001240112402124031240412405124061240712408124091241012411124121241312414124151241612417124181241912420124211242212423124241242512426124271242812429124301243112432124331243412435124361243712438124391244012441124421244312444124451244612447124481244912450124511245212453124541245512456124571245812459124601246112462124631246412465124661246712468124691247012471124721247312474124751247612477124781247912480124811248212483124841248512486124871248812489124901249112492124931249412495124961249712498124991250012501125021250312504125051250612507125081250912510125111251212513125141251512516125171251812519125201252112522125231252412525125261252712528125291253012531125321253312534125351253612537125381253912540125411254212543125441254512546125471254812549125501255112552125531255412555125561255712558125591256012561125621256312564125651256612567125681256912570125711257212573125741257512576125771257812579125801258112582125831258412585125861258712588125891259012591125921259312594125951259612597125981259912600126011260212603126041260512606126071260812609126101261112612126131261412615126161261712618126191262012621126221262312624126251262612627126281262912630126311263212633126341263512636126371263812639126401264112642126431264412645126461264712648126491265012651126521265312654126551265612657126581265912660126611266212663126641266512666126671266812669126701267112672126731267412675126761267712678126791268012681126821268312684126851268612687126881268912690126911269212693126941269512696126971269812699127001270112702127031270412705127061270712708127091271012711127121271312714127151271612717127181271912720127211272212723127241272512726127271272812729127301273112732127331273412735127361273712738127391274012741127421274312744127451274612747127481274912750127511275212753127541275512756127571275812759127601276112762127631276412765127661276712768127691277012771127721277312774127751277612777127781277912780127811278212783127841278512786127871278812789127901279112792127931279412795127961279712798127991280012801128021280312804128051280612807128081280912810128111281212813128141281512816128171281812819128201282112822128231282412825128261282712828128291283012831128321283312834128351283612837128381283912840128411284212843128441284512846128471284812849128501285112852128531285412855128561285712858128591286012861128621286312864128651286612867128681286912870128711287212873128741287512876128771287812879128801288112882128831288412885128861288712888128891289012891128921289312894128951289612897128981289912900129011290212903129041290512906129071290812909129101291112912129131291412915129161291712918129191292012921129221292312924129251292612927129281292912930129311293212933129341293512936129371293812939129401294112942129431294412945129461294712948129491295012951129521295312954129551295612957129581295912960129611296212963129641296512966129671296812969129701297112972129731297412975129761297712978129791298012981129821298312984129851298612987129881298912990129911299212993129941299512996129971299812999130001300113002130031300413005130061300713008130091301013011130121301313014130151301613017130181301913020130211302213023130241302513026130271302813029130301303113032130331303413035130361303713038130391304013041130421304313044130451304613047130481304913050130511305213053130541305513056130571305813059130601306113062130631306413065130661306713068130691307013071130721307313074130751307613077130781307913080130811308213083130841308513086130871308813089130901309113092130931309413095130961309713098130991310013101131021310313104131051310613107131081310913110131111311213113131141311513116131171311813119131201312113122131231312413125131261312713128131291313013131131321313313134131351313613137131381313913140131411314213143131441314513146131471314813149131501315113152131531315413155131561315713158131591316013161131621316313164131651316613167131681316913170131711317213173131741317513176131771317813179131801318113182131831318413185131861318713188131891319013191131921319313194131951319613197131981319913200132011320213203132041320513206132071320813209132101321113212132131321413215132161321713218132191322013221132221322313224132251322613227132281322913230132311323213233132341323513236132371323813239132401324113242132431324413245132461324713248132491325013251132521325313254132551325613257132581325913260132611326213263132641326513266132671326813269132701327113272132731327413275132761327713278132791328013281132821328313284132851328613287132881328913290132911329213293132941329513296132971329813299133001330113302133031330413305133061330713308133091331013311133121331313314133151331613317133181331913320133211332213323133241332513326133271332813329133301333113332133331333413335133361333713338133391334013341133421334313344133451334613347133481334913350133511335213353133541335513356133571335813359133601336113362133631336413365133661336713368133691337013371133721337313374133751337613377133781337913380133811338213383133841338513386133871338813389133901339113392133931339413395133961339713398133991340013401134021340313404134051340613407134081340913410134111341213413134141341513416134171341813419134201342113422134231342413425134261342713428134291343013431134321343313434134351343613437134381343913440134411344213443134441344513446134471344813449134501345113452134531345413455134561345713458134591346013461134621346313464134651346613467134681346913470134711347213473134741347513476134771347813479134801348113482134831348413485134861348713488134891349013491134921349313494134951349613497134981349913500135011350213503135041350513506135071350813509135101351113512135131351413515135161351713518135191352013521135221352313524135251352613527135281352913530135311353213533135341353513536135371353813539135401354113542135431354413545135461354713548135491355013551135521355313554135551355613557135581355913560135611356213563135641356513566135671356813569135701357113572135731357413575135761357713578135791358013581135821358313584135851358613587135881358913590135911359213593135941359513596135971359813599136001360113602136031360413605136061360713608136091361013611136121361313614136151361613617136181361913620136211362213623136241362513626136271362813629136301363113632136331363413635136361363713638136391364013641136421364313644136451364613647136481364913650136511365213653136541365513656136571365813659136601366113662136631366413665136661366713668136691367013671136721367313674136751367613677136781367913680136811368213683136841368513686136871368813689136901369113692136931369413695136961369713698136991370013701137021370313704137051370613707137081370913710137111371213713137141371513716137171371813719137201372113722137231372413725137261372713728137291373013731137321373313734137351373613737137381373913740137411374213743137441374513746137471374813749137501375113752137531375413755137561375713758137591376013761137621376313764137651376613767137681376913770137711377213773137741377513776137771377813779137801378113782137831378413785137861378713788137891379013791137921379313794137951379613797137981379913800138011380213803138041380513806138071380813809138101381113812138131381413815138161381713818138191382013821138221382313824138251382613827138281382913830138311383213833138341383513836138371383813839138401384113842138431384413845138461384713848138491385013851138521385313854138551385613857138581385913860138611386213863138641386513866138671386813869138701387113872138731387413875138761387713878138791388013881138821388313884138851388613887138881388913890138911389213893138941389513896138971389813899139001390113902139031390413905139061390713908139091391013911139121391313914139151391613917139181391913920139211392213923139241392513926139271392813929139301393113932139331393413935139361393713938139391394013941139421394313944139451394613947139481394913950139511395213953139541395513956139571395813959139601396113962139631396413965139661396713968139691397013971139721397313974139751397613977139781397913980139811398213983139841398513986139871398813989139901399113992139931399413995139961399713998139991400014001140021400314004140051400614007140081400914010140111401214013140141401514016140171401814019140201402114022140231402414025140261402714028140291403014031140321403314034140351403614037140381403914040140411404214043140441404514046140471404814049140501405114052140531405414055140561405714058140591406014061140621406314064140651406614067140681406914070140711407214073140741407514076140771407814079140801408114082140831408414085140861408714088140891409014091140921409314094140951409614097140981409914100141011410214103141041410514106141071410814109141101411114112141131411414115141161411714118141191412014121141221412314124141251412614127141281412914130141311413214133141341413514136141371413814139141401414114142141431414414145141461414714148141491415014151141521415314154141551415614157141581415914160141611416214163141641416514166141671416814169141701417114172141731417414175141761417714178141791418014181141821418314184141851418614187141881418914190141911419214193141941419514196141971419814199142001420114202142031420414205142061420714208142091421014211142121421314214142151421614217142181421914220142211422214223142241422514226142271422814229142301423114232142331423414235142361423714238142391424014241142421424314244142451424614247142481424914250142511425214253142541425514256142571425814259142601426114262142631426414265142661426714268142691427014271142721427314274142751427614277142781427914280142811428214283142841428514286142871428814289142901429114292142931429414295142961429714298142991430014301143021430314304143051430614307143081430914310143111431214313143141431514316143171431814319143201432114322143231432414325143261432714328143291433014331143321433314334143351433614337143381433914340143411434214343143441434514346143471434814349143501435114352143531435414355143561435714358143591436014361143621436314364143651436614367143681436914370143711437214373143741437514376143771437814379143801438114382143831438414385143861438714388143891439014391143921439314394143951439614397143981439914400144011440214403144041440514406144071440814409144101441114412144131441414415144161441714418144191442014421144221442314424144251442614427144281442914430144311443214433144341443514436144371443814439144401444114442144431444414445144461444714448144491445014451144521445314454144551445614457144581445914460144611446214463144641446514466144671446814469144701447114472144731447414475144761447714478144791448014481144821448314484144851448614487144881448914490144911449214493144941449514496144971449814499145001450114502145031450414505145061450714508145091451014511145121451314514145151451614517145181451914520145211452214523145241452514526145271452814529145301453114532145331453414535145361453714538145391454014541145421454314544145451454614547145481454914550145511455214553145541455514556145571455814559145601456114562145631456414565145661456714568145691457014571145721457314574145751457614577145781457914580145811458214583145841458514586145871458814589145901459114592145931459414595145961459714598145991460014601146021460314604146051460614607146081460914610146111461214613146141461514616146171461814619146201462114622146231462414625146261462714628146291463014631146321463314634146351463614637146381463914640146411464214643146441464514646146471464814649146501465114652146531465414655146561465714658146591466014661146621466314664146651466614667146681466914670146711467214673146741467514676146771467814679146801468114682146831468414685146861468714688146891469014691146921469314694146951469614697146981469914700147011470214703147041470514706147071470814709147101471114712147131471414715147161471714718147191472014721147221472314724147251472614727147281472914730147311473214733147341473514736147371473814739147401474114742147431474414745147461474714748147491475014751147521475314754147551475614757147581475914760147611476214763147641476514766147671476814769147701477114772147731477414775147761477714778147791478014781147821478314784147851478614787147881478914790147911479214793147941479514796147971479814799148001480114802148031480414805148061480714808148091481014811148121481314814148151481614817148181481914820148211482214823148241482514826148271482814829148301483114832148331483414835148361483714838148391484014841148421484314844148451484614847148481484914850148511485214853148541485514856148571485814859148601486114862148631486414865148661486714868148691487014871148721487314874148751487614877148781487914880148811488214883148841488514886148871488814889148901489114892148931489414895148961489714898148991490014901149021490314904149051490614907149081490914910149111491214913149141491514916149171491814919149201492114922149231492414925149261492714928149291493014931149321493314934149351493614937149381493914940149411494214943149441494514946149471494814949149501495114952149531495414955149561495714958149591496014961149621496314964149651496614967149681496914970149711497214973149741497514976149771497814979149801498114982149831498414985149861498714988149891499014991149921499314994149951499614997149981499915000150011500215003150041500515006150071500815009150101501115012150131501415015150161501715018150191502015021150221502315024150251502615027150281502915030150311503215033150341503515036150371503815039150401504115042150431504415045150461504715048150491505015051150521505315054150551505615057150581505915060150611506215063150641506515066150671506815069150701507115072150731507415075150761507715078150791508015081150821508315084150851508615087150881508915090150911509215093150941509515096150971509815099151001510115102151031510415105151061510715108151091511015111151121511315114151151511615117151181511915120151211512215123151241512515126151271512815129151301513115132151331513415135151361513715138151391514015141151421514315144151451514615147151481514915150151511515215153151541515515156151571515815159151601516115162151631516415165151661516715168151691517015171151721517315174151751517615177151781517915180151811518215183151841518515186151871518815189151901519115192151931519415195151961519715198151991520015201152021520315204152051520615207152081520915210152111521215213152141521515216152171521815219152201522115222152231522415225152261522715228152291523015231152321523315234152351523615237152381523915240152411524215243152441524515246152471524815249152501525115252152531525415255152561525715258152591526015261152621526315264152651526615267152681526915270152711527215273152741527515276152771527815279152801528115282152831528415285152861528715288152891529015291152921529315294152951529615297152981529915300153011530215303153041530515306153071530815309153101531115312153131531415315153161531715318153191532015321153221532315324153251532615327153281532915330153311533215333153341533515336153371533815339153401534115342153431534415345153461534715348153491535015351153521535315354153551535615357153581535915360153611536215363153641536515366153671536815369153701537115372153731537415375153761537715378153791538015381153821538315384153851538615387153881538915390153911539215393153941539515396153971539815399154001540115402154031540415405154061540715408154091541015411154121541315414154151541615417154181541915420154211542215423154241542515426154271542815429154301543115432154331543415435154361543715438154391544015441154421544315444154451544615447154481544915450154511545215453154541545515456154571545815459154601546115462154631546415465154661546715468154691547015471154721547315474154751547615477154781547915480154811548215483154841548515486154871548815489154901549115492154931549415495154961549715498154991550015501155021550315504155051550615507155081550915510155111551215513155141551515516155171551815519155201552115522155231552415525155261552715528155291553015531155321553315534155351553615537155381553915540155411554215543155441554515546155471554815549155501555115552155531555415555155561555715558155591556015561155621556315564155651556615567155681556915570155711557215573155741557515576155771557815579155801558115582155831558415585155861558715588155891559015591155921559315594155951559615597155981559915600156011560215603156041560515606156071560815609156101561115612156131561415615156161561715618156191562015621156221562315624156251562615627156281562915630156311563215633156341563515636156371563815639156401564115642156431564415645156461564715648156491565015651156521565315654156551565615657156581565915660156611566215663156641566515666156671566815669156701567115672156731567415675156761567715678156791568015681156821568315684156851568615687156881568915690156911569215693156941569515696156971569815699157001570115702157031570415705157061570715708157091571015711157121571315714157151571615717157181571915720157211572215723157241572515726157271572815729157301573115732157331573415735157361573715738157391574015741157421574315744157451574615747157481574915750157511575215753157541575515756157571575815759157601576115762157631576415765157661576715768157691577015771157721577315774157751577615777157781577915780157811578215783157841578515786157871578815789157901579115792157931579415795157961579715798157991580015801158021580315804158051580615807158081580915810158111581215813158141581515816158171581815819158201582115822158231582415825158261582715828158291583015831158321583315834158351583615837158381583915840158411584215843158441584515846158471584815849158501585115852158531585415855158561585715858158591586015861158621586315864158651586615867158681586915870158711587215873158741587515876158771587815879158801588115882158831588415885158861588715888158891589015891158921589315894158951589615897158981589915900159011590215903159041590515906159071590815909159101591115912159131591415915159161591715918159191592015921159221592315924159251592615927159281592915930159311593215933159341593515936159371593815939159401594115942159431594415945159461594715948159491595015951159521595315954159551595615957159581595915960159611596215963159641596515966159671596815969159701597115972159731597415975159761597715978159791598015981159821598315984159851598615987159881598915990159911599215993159941599515996159971599815999160001600116002160031600416005160061600716008160091601016011160121601316014160151601616017160181601916020160211602216023160241602516026160271602816029160301603116032160331603416035160361603716038160391604016041160421604316044160451604616047160481604916050160511605216053160541605516056160571605816059160601606116062160631606416065160661606716068160691607016071160721607316074160751607616077160781607916080160811608216083160841608516086160871608816089160901609116092160931609416095160961609716098160991610016101161021610316104161051610616107161081610916110161111611216113161141611516116161171611816119161201612116122161231612416125161261612716128161291613016131161321613316134161351613616137161381613916140161411614216143161441614516146161471614816149161501615116152161531615416155161561615716158161591616016161161621616316164161651616616167161681616916170161711617216173161741617516176161771617816179161801618116182161831618416185161861618716188161891619016191161921619316194161951619616197161981619916200162011620216203162041620516206162071620816209162101621116212162131621416215162161621716218162191622016221162221622316224162251622616227162281622916230162311623216233162341623516236162371623816239162401624116242162431624416245162461624716248162491625016251162521625316254162551625616257162581625916260162611626216263162641626516266162671626816269162701627116272162731627416275162761627716278162791628016281162821628316284162851628616287162881628916290162911629216293162941629516296162971629816299163001630116302163031630416305163061630716308163091631016311163121631316314163151631616317163181631916320163211632216323163241632516326163271632816329163301633116332163331633416335163361633716338163391634016341163421634316344163451634616347163481634916350163511635216353163541635516356163571635816359163601636116362163631636416365163661636716368163691637016371163721637316374163751637616377163781637916380163811638216383163841638516386163871638816389163901639116392163931639416395163961639716398163991640016401164021640316404164051640616407164081640916410164111641216413164141641516416164171641816419164201642116422164231642416425164261642716428164291643016431164321643316434164351643616437164381643916440164411644216443164441644516446164471644816449164501645116452164531645416455164561645716458164591646016461164621646316464164651646616467164681646916470164711647216473164741647516476164771647816479164801648116482164831648416485164861648716488164891649016491164921649316494164951649616497164981649916500165011650216503165041650516506165071650816509165101651116512165131651416515165161651716518165191652016521165221652316524165251652616527165281652916530165311653216533165341653516536165371653816539165401654116542165431654416545165461654716548165491655016551165521655316554165551655616557165581655916560165611656216563165641656516566165671656816569165701657116572165731657416575165761657716578165791658016581165821658316584165851658616587165881658916590165911659216593165941659516596165971659816599166001660116602166031660416605166061660716608166091661016611166121661316614166151661616617166181661916620166211662216623166241662516626166271662816629166301663116632166331663416635166361663716638166391664016641166421664316644166451664616647166481664916650166511665216653166541665516656166571665816659166601666116662166631666416665166661666716668166691667016671166721667316674166751667616677166781667916680166811668216683166841668516686166871668816689166901669116692166931669416695166961669716698166991670016701167021670316704167051670616707167081670916710167111671216713167141671516716167171671816719167201672116722167231672416725167261672716728167291673016731167321673316734167351673616737167381673916740167411674216743167441674516746167471674816749167501675116752167531675416755167561675716758167591676016761167621676316764167651676616767167681676916770167711677216773167741677516776167771677816779167801678116782167831678416785167861678716788167891679016791167921679316794167951679616797167981679916800 |
- /*! Amaze UI v2.7.2 | by Amaze UI Team | (c) 2016 AllMobilize, Inc. | Licensed under MIT | 2016-12-16T16:25:57+0800 */
- (function webpackUniversalModuleDefinition(root, factory) {
- if(typeof exports === 'object' && typeof module === 'object')
- module.exports = factory(require("jquery"));
- else if(typeof define === 'function' && define.amd)
- define(["jquery"], factory);
- else if(typeof exports === 'object')
- exports["AMUI"] = factory(require("jquery"));
- else
- root["AMUI"] = factory(root["jQuery"]);
- })(this, function(__WEBPACK_EXTERNAL_MODULE_1__) {
- return /******/ (function(modules) { // webpackBootstrap
- /******/ // The module cache
- /******/ var installedModules = {};
- /******/ // The require function
- /******/ function __webpack_require__(moduleId) {
- /******/ // Check if module is in cache
- /******/ if(installedModules[moduleId])
- /******/ return installedModules[moduleId].exports;
- /******/ // Create a new module (and put it into the cache)
- /******/ var module = installedModules[moduleId] = {
- /******/ exports: {},
- /******/ id: moduleId,
- /******/ loaded: false
- /******/ };
- /******/ // Execute the module function
- /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
- /******/ // Flag the module as loaded
- /******/ module.loaded = true;
- /******/ // Return the exports of the module
- /******/ return module.exports;
- /******/ }
- /******/ // expose the modules object (__webpack_modules__)
- /******/ __webpack_require__.m = modules;
- /******/ // expose the module cache
- /******/ __webpack_require__.c = installedModules;
- /******/ // __webpack_public_path__
- /******/ __webpack_require__.p = "";
- /******/ // Load entry module and return exports
- /******/ return __webpack_require__(0);
- /******/ })
- /************************************************************************/
- /******/ ([
- /* 0 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(3);
- __webpack_require__(4);
- __webpack_require__(5);
- __webpack_require__(6);
- __webpack_require__(7);
- __webpack_require__(8);
- __webpack_require__(9);
- __webpack_require__(10);
- __webpack_require__(11);
- __webpack_require__(14);
- __webpack_require__(15);
- __webpack_require__(16);
- __webpack_require__(17);
- __webpack_require__(18);
- __webpack_require__(19);
- __webpack_require__(20);
- __webpack_require__(21);
- __webpack_require__(22);
- __webpack_require__(24);
- __webpack_require__(25);
- __webpack_require__(23);
- __webpack_require__(27);
- __webpack_require__(28);
- __webpack_require__(29);
- __webpack_require__(30);
- __webpack_require__(31);
- __webpack_require__(32);
- __webpack_require__(33);
- __webpack_require__(26);
- __webpack_require__(34);
- __webpack_require__(35);
- __webpack_require__(36);
- __webpack_require__(37);
- __webpack_require__(38);
- __webpack_require__(39);
- __webpack_require__(40);
- __webpack_require__(41);
- __webpack_require__(42);
- __webpack_require__(43);
- __webpack_require__(44);
- __webpack_require__(45);
- __webpack_require__(46);
- __webpack_require__(47);
- __webpack_require__(48);
- __webpack_require__(49);
- __webpack_require__(50);
- __webpack_require__(51);
- __webpack_require__(52);
- __webpack_require__(53);
- __webpack_require__(54);
- module.exports = $.AMUI = UI;
- /***/ },
- /* 1 */
- /***/ function(module, exports) {
- module.exports = __WEBPACK_EXTERNAL_MODULE_1__;
- /***/ },
- /* 2 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- if (typeof $ === 'undefined') {
- throw new Error('Amaze UI 2.x requires jQuery :-(\n' +
- '\u7231\u4e0a\u4e00\u5339\u91ce\u9a6c\uff0c\u53ef\u4f60' +
- '\u7684\u5bb6\u91cc\u6ca1\u6709\u8349\u539f\u2026');
- }
- var UI = $.AMUI || {};
- var $win = $(window);
- var doc = window.document;
- var $html = $('html');
- UI.VERSION = '2.7.2';
- UI.support = {};
- UI.support.transition = (function() {
- var transitionEnd = (function() {
- // https://developer.mozilla.org/en-US/docs/Web/Events/transitionend#Browser_compatibility
- var element = doc.body || doc.documentElement;
- var transEndEventNames = {
- WebkitTransition: 'webkitTransitionEnd',
- MozTransition: 'transitionend',
- OTransition: 'oTransitionEnd otransitionend',
- transition: 'transitionend'
- };
- for (var name in transEndEventNames) {
- if (element.style[name] !== undefined) {
- return transEndEventNames[name];
- }
- }
- })();
- return transitionEnd && {end: transitionEnd};
- })();
- UI.support.animation = (function() {
- var animationEnd = (function() {
- var element = doc.body || doc.documentElement;
- var animEndEventNames = {
- WebkitAnimation: 'webkitAnimationEnd',
- MozAnimation: 'animationend',
- OAnimation: 'oAnimationEnd oanimationend',
- animation: 'animationend'
- };
- for (var name in animEndEventNames) {
- if (element.style[name] !== undefined) {
- return animEndEventNames[name];
- }
- }
- })();
- return animationEnd && {end: animationEnd};
- })();
- /* eslint-disable dot-notation */
- UI.support.touch = (
- ('ontouchstart' in window &&
- navigator.userAgent.toLowerCase().match(/mobile|tablet/)) ||
- (window.DocumentTouch && document instanceof window.DocumentTouch) ||
- (window.navigator['msPointerEnabled'] &&
- window.navigator['msMaxTouchPoints'] > 0) || // IE 10
- (window.navigator['pointerEnabled'] &&
- window.navigator['maxTouchPoints'] > 0) || // IE >=11
- false);
- /* eslint-enable dot-notation */
- // https://developer.mozilla.org/zh-CN/docs/DOM/MutationObserver
- UI.support.mutationobserver = (window.MutationObserver ||
- window.WebKitMutationObserver || null);
- // https://github.com/Modernizr/Modernizr/blob/924c7611c170ef2dc502582e5079507aff61e388/feature-detects/forms/validation.js#L20
- UI.support.formValidation = (typeof document.createElement('form').
- checkValidity === 'function');
- UI.utils = {};
- /**
- * Debounce function
- *
- * @param {function} func Function to be debounced
- * @param {number} wait Function execution threshold in milliseconds
- * @param {bool} immediate Whether the function should be called at
- * the beginning of the delay instead of the
- * end. Default is false.
- * @description Executes a function when it stops being invoked for n seconds
- * @see _.debounce() http://underscorejs.org
- */
- UI.utils.debounce = function(func, wait, immediate) {
- var timeout;
- return function() {
- var context = this;
- var args = arguments;
- var later = function() {
- timeout = null;
- if (!immediate) {
- func.apply(context, args);
- }
- };
- var callNow = immediate && !timeout;
- clearTimeout(timeout);
- timeout = setTimeout(later, wait);
- if (callNow) {
- func.apply(context, args);
- }
- };
- };
- UI.utils.isInView = function(element, options) {
- var $element = $(element);
- var visible = !!($element.width() || $element.height()) &&
- $element.css('display') !== 'none';
- if (!visible) {
- return false;
- }
- var windowLeft = $win.scrollLeft();
- var windowTop = $win.scrollTop();
- var offset = $element.offset();
- var left = offset.left;
- var top = offset.top;
- options = $.extend({topOffset: 0, leftOffset: 0}, options);
- return (top + $element.height() >= windowTop &&
- top - options.topOffset <= windowTop + $win.height() &&
- left + $element.width() >= windowLeft &&
- left - options.leftOffset <= windowLeft + $win.width());
- };
- UI.utils.parseOptions = UI.utils.options = function(string) {
- if ($.isPlainObject(string)) {
- return string;
- }
- var start = (string ? string.indexOf('{') : -1);
- var options = {};
- if (start != -1) {
- try {
- options = (new Function('',
- 'var json = ' + string.substr(start) +
- '; return JSON.parse(JSON.stringify(json));'))();
- } catch (e) {
- }
- }
- return options;
- };
- UI.utils.generateGUID = function(namespace) {
- var uid = namespace + '-' || 'am-';
- do {
- uid += Math.random().toString(36).substring(2, 7);
- } while (document.getElementById(uid));
- return uid;
- };
- // @see https://davidwalsh.name/get-absolute-url
- UI.utils.getAbsoluteUrl = (function() {
- var a;
- return function(url) {
- if (!a) {
- a = document.createElement('a');
- }
- a.href = url;
- return a.href;
- };
- })();
- /**
- * Plugin AMUI Component to jQuery
- *
- * @param {String} name - plugin name
- * @param {Function} Component - plugin constructor
- * @param {Object} [pluginOption]
- * @param {String} pluginOption.dataOptions
- * @param {Function} pluginOption.methodCall - custom method call
- * @param {Function} pluginOption.before
- * @param {Function} pluginOption.after
- * @since v2.4.1
- */
- UI.plugin = function UIPlugin(name, Component, pluginOption) {
- var old = $.fn[name];
- pluginOption = pluginOption || {};
- $.fn[name] = function(option) {
- var allArgs = Array.prototype.slice.call(arguments, 0);
- var args = allArgs.slice(1);
- var propReturn;
- var $set = this.each(function() {
- var $this = $(this);
- var dataName = 'amui.' + name;
- var dataOptionsName = pluginOption.dataOptions || ('data-am-' + name);
- var instance = $this.data(dataName);
- var options = $.extend({},
- UI.utils.parseOptions($this.attr(dataOptionsName)),
- typeof option === 'object' && option);
- if (!instance && option === 'destroy') {
- return;
- }
- if (!instance) {
- $this.data(dataName, (instance = new Component(this, options)));
- }
- // custom method call
- if (pluginOption.methodCall) {
- pluginOption.methodCall.call($this, allArgs, instance);
- } else {
- // before method call
- pluginOption.before &&
- pluginOption.before.call($this, allArgs, instance);
- if (typeof option === 'string') {
- propReturn = typeof instance[option] === 'function' ?
- instance[option].apply(instance, args) : instance[option];
- }
- // after method call
- pluginOption.after && pluginOption.after.call($this, allArgs, instance);
- }
- });
- return (propReturn === undefined) ? $set : propReturn;
- };
- $.fn[name].Constructor = Component;
- // no conflict
- $.fn[name].noConflict = function() {
- $.fn[name] = old;
- return this;
- };
- UI[name] = Component;
- };
- // http://blog.alexmaccaw.com/css-transitions
- $.fn.emulateTransitionEnd = function(duration) {
- var called = false;
- var $el = this;
- $(this).one(UI.support.transition.end, function() {
- called = true;
- });
- var callback = function() {
- if (!called) {
- $($el).trigger(UI.support.transition.end);
- }
- $el.transitionEndTimmer = undefined;
- };
- this.transitionEndTimmer = setTimeout(callback, duration);
- return this;
- };
- $.fn.redraw = function() {
- return this.each(function() {
- /* eslint-disable */
- var redraw = this.offsetHeight;
- /* eslint-enable */
- });
- };
- $.fn.transitionEnd = function(callback) {
- var endEvent = UI.support.transition.end;
- var dom = this;
- function fireCallBack(e) {
- callback.call(this, e);
- endEvent && dom.off(endEvent, fireCallBack);
- }
- if (callback && endEvent) {
- dom.on(endEvent, fireCallBack);
- }
- return this;
- };
- $.fn.removeClassRegEx = function() {
- return this.each(function(regex) {
- var classes = $(this).attr('class');
- if (!classes || !regex) {
- return false;
- }
- var classArray = [];
- classes = classes.split(' ');
- for (var i = 0, len = classes.length; i < len; i++) {
- if (!classes[i].match(regex)) {
- classArray.push(classes[i]);
- }
- }
- $(this).attr('class', classArray.join(' '));
- });
- };
- //
- $.fn.alterClass = function(removals, additions) {
- var self = this;
- if (removals.indexOf('*') === -1) {
- // Use native jQuery methods if there is no wildcard matching
- self.removeClass(removals);
- return !additions ? self : self.addClass(additions);
- }
- var classPattern = new RegExp('\\s' +
- removals.
- replace(/\*/g, '[A-Za-z0-9-_]+').
- split(' ').
- join('\\s|\\s') +
- '\\s', 'g');
- self.each(function(i, it) {
- var cn = ' ' + it.className + ' ';
- while (classPattern.test(cn)) {
- cn = cn.replace(classPattern, ' ');
- }
- it.className = $.trim(cn);
- });
- return !additions ? self : self.addClass(additions);
- };
- // handle multiple browsers for requestAnimationFrame()
- // http://www.paulirish.com/2011/requestanimationframe-for-smart-animating/
- // https://github.com/gnarf/jquery-requestAnimationFrame
- UI.utils.rAF = (function() {
- return window.requestAnimationFrame ||
- window.webkitRequestAnimationFrame ||
- window.mozRequestAnimationFrame ||
- window.oRequestAnimationFrame ||
- // if all else fails, use setTimeout
- function(callback) {
- return window.setTimeout(callback, 1000 / 60); // shoot for 60 fps
- };
- })();
- // handle multiple browsers for cancelAnimationFrame()
- UI.utils.cancelAF = (function() {
- return window.cancelAnimationFrame ||
- window.webkitCancelAnimationFrame ||
- window.mozCancelAnimationFrame ||
- window.oCancelAnimationFrame ||
- function(id) {
- window.clearTimeout(id);
- };
- })();
- // via http://davidwalsh.name/detect-scrollbar-width
- UI.utils.measureScrollbar = function() {
- if (document.body.clientWidth >= window.innerWidth) {
- return 0;
- }
- // if ($html.width() >= window.innerWidth) return;
- // var scrollbarWidth = window.innerWidth - $html.width();
- var $measure = $('<div ' +
- 'style="width: 100px;height: 100px;overflow: scroll;' +
- 'position: absolute;top: -9999px;"></div>');
- $(document.body).append($measure);
- var scrollbarWidth = $measure[0].offsetWidth - $measure[0].clientWidth;
- $measure.remove();
- return scrollbarWidth;
- };
- UI.utils.imageLoader = function($image, callback) {
- function loaded() {
- callback($image[0]);
- }
- function bindLoad() {
- this.one('load', loaded);
- if (/MSIE (\d+\.\d+);/.test(navigator.userAgent)) {
- var src = this.attr('src');
- var param = src.match(/\?/) ? '&' : '?';
- param += 'random=' + (new Date()).getTime();
- this.attr('src', src + param);
- }
- }
- if (!$image.attr('src')) {
- loaded();
- return;
- }
- if ($image[0].complete || $image[0].readyState === 4) {
- loaded();
- } else {
- bindLoad.call($image);
- }
- };
- /**
- * @see https://github.com/cho45/micro-template.js
- * (c) cho45 http://cho45.github.com/mit-license
- */
- UI.template = function(id, data) {
- var me = UI.template;
- if (!me.cache[id]) {
- me.cache[id] = (function() {
- var name = id;
- var string = /^[\w\-]+$/.test(id) ?
- me.get(id) : (name = 'template(string)', id); // no warnings
- var line = 1;
- /* eslint-disable max-len, quotes */
- var body = ('try { ' + (me.variable ?
- 'var ' + me.variable + ' = this.stash;' : 'with (this.stash) { ') +
- "this.ret += '" +
- string.
- replace(/<%/g, '\x11').replace(/%>/g, '\x13'). // if you want other tag, just edit this line
- replace(/'(?![^\x11\x13]+?\x13)/g, '\\x27').
- replace(/^\s*|\s*$/g, '').
- replace(/\n/g, function() {
- return "';\nthis.line = " + (++line) + "; this.ret += '\\n";
- }).
- replace(/\x11-(.+?)\x13/g, "' + ($1) + '").
- replace(/\x11=(.+?)\x13/g, "' + this.escapeHTML($1) + '").
- replace(/\x11(.+?)\x13/g, "'; $1; this.ret += '") +
- "'; " + (me.variable ? "" : "}") + "return this.ret;" +
- "} catch (e) { throw 'TemplateError: ' + e + ' (on " + name +
- "' + ' line ' + this.line + ')'; } " +
- "//@ sourceURL=" + name + "\n" // source map
- ).replace(/this\.ret \+= '';/g, '');
- /* eslint-enable max-len, quotes */
- var func = new Function(body);
- var map = {
- '&': '&',
- '<': '<',
- '>': '>',
- '\x22': '"',
- '\x27': '''
- };
- var escapeHTML = function(string) {
- return ('' + string).replace(/[&<>\'\"]/g, function(_) {
- return map[_];
- });
- };
- return function(stash) {
- return func.call(me.context = {
- escapeHTML: escapeHTML,
- line: 1,
- ret: '',
- stash: stash
- });
- };
- })();
- }
- return data ? me.cache[id](data) : me.cache[id];
- };
- UI.template.cache = {};
- UI.template.get = function(id) {
- if (id) {
- var element = document.getElementById(id);
- return element && element.innerHTML || '';
- }
- };
- // Dom mutation watchers
- UI.DOMWatchers = [];
- UI.DOMReady = false;
- UI.ready = function(callback) {
- UI.DOMWatchers.push(callback);
- if (UI.DOMReady) {
- // console.log('Ready call');
- callback(document);
- }
- };
- UI.DOMObserve = function(elements, options, callback) {
- var Observer = UI.support.mutationobserver;
- if (!Observer) {
- return;
- }
- options = $.isPlainObject(options) ?
- options : {childList: true, subtree: true};
- callback = typeof callback === 'function' && callback || function() {
- };
- $(elements).each(function() {
- var element = this;
- var $element = $(element);
- if ($element.data('am.observer')) {
- return;
- }
- try {
- var observer = new Observer(UI.utils.debounce(
- function(mutations, instance) {
- callback.call(element, mutations, instance);
- // trigger this event manually if MutationObserver not supported
- $element.trigger('changed.dom.amui');
- }, 50));
- observer.observe(element, options);
- $element.data('am.observer', observer);
- } catch (e) {
- }
- });
- };
- $.fn.DOMObserve = function(options, callback) {
- return this.each(function() {
- /* eslint-disable new-cap */
- UI.DOMObserve(this, options, callback);
- /* eslint-enable new-cap */
- });
- };
- if (UI.support.touch) {
- $html.addClass('am-touch');
- }
- $(document).on('changed.dom.amui', function(e) {
- var element = e.target;
- // TODO: just call changed element's watcher
- // every watcher callback should have a key
- // use like this: <div data-am-observe='key1, key2'>
- // get keys via $(element).data('amObserve')
- // call functions store with these keys
- $.each(UI.DOMWatchers, function(i, watcher) {
- watcher(element);
- });
- });
- $(function() {
- var $body = $(document.body);
- UI.DOMReady = true;
- // Run default init
- $.each(UI.DOMWatchers, function(i, watcher) {
- watcher(document);
- });
- // watches DOM
- /* eslint-disable new-cap */
- UI.DOMObserve('[data-am-observe]');
- /* eslint-enable */
- $html.removeClass('no-js').addClass('js');
- UI.support.animation && $html.addClass('cssanimations');
- // iOS standalone mode
- if (window.navigator.standalone) {
- $html.addClass('am-standalone');
- }
- $('.am-topbar-fixed-top').length &&
- $body.addClass('am-with-topbar-fixed-top');
- $('.am-topbar-fixed-bottom').length &&
- $body.addClass('am-with-topbar-fixed-bottom');
- // Remove responsive classes in .am-layout
- var $layout = $('.am-layout');
- $layout.find('[class*="md-block-grid"]').alterClass('md-block-grid-*');
- $layout.find('[class*="lg-block-grid"]').alterClass('lg-block-grid');
- // widgets not in .am-layout
- $('[data-am-widget]').each(function() {
- var $widget = $(this);
- // console.log($widget.parents('.am-layout').length)
- if ($widget.parents('.am-layout').length === 0) {
- $widget.addClass('am-no-layout');
- }
- });
- });
- module.exports = UI;
- /***/ },
- /* 3 */
- /***/ function(module, exports, __webpack_require__) {
- /*! Hammer.JS - v2.0.8 - 2016-04-22
- * http://hammerjs.github.io/
- *
- * Copyright (c) 2016 Jorik Tangelder;
- * Licensed under the MIT license */
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- 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.7';
- /**
- * 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
- });
- // jquery.hammer.js
- // This jQuery plugin is just a small wrapper around the Hammer() class.
- // It also extends the Manager.emit method by triggering jQuery events.
- // $(element).hammer(options).bind("pan", myPanHandler);
- // The Hammer instance is stored at $element.data("hammer").
- // https://github.com/hammerjs/jquery.hammer.js
- (function($, Hammer) {
- function hammerify(el, options) {
- var $el = $(el);
- if (!$el.data('hammer')) {
- $el.data('hammer', new Hammer($el[0], options));
- }
- }
- $.fn.hammer = function(options) {
- return this.each(function() {
- hammerify(this, options);
- });
- };
- // extend the emit method to also trigger jQuery events
- Hammer.Manager.prototype.emit = (function(originalEmit) {
- return function(type, data) {
- originalEmit.call(this, type, data);
- $(this.element).trigger({
- type: type,
- gesture: data
- });
- };
- })(Hammer.Manager.prototype.emit);
- })($, Hammer);
- module.exports = UI.Hammer = Hammer;
- /***/ },
- /* 4 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- /**
- * Add to Homescreen v3.2.2
- * (c) 2015 Matteo Spinelli
- * @license: http://cubiq.org/license
- */
- // Check for addEventListener browser support (prevent errors in IE<9)
- var _eventListener = 'addEventListener' in window;
- // Check if document is loaded, needed by autostart
- var _DOMReady = false;
- if (document.readyState === 'complete') {
- _DOMReady = true;
- } else if (_eventListener) {
- window.addEventListener('load', loaded, false);
- }
- function loaded() {
- window.removeEventListener('load', loaded, false);
- _DOMReady = true;
- }
- // regex used to detect if app has been added to the homescreen
- var _reSmartURL = /\/ath(\/)?$/;
- var _reQueryString = /([\?&]ath=[^&]*$|&ath=[^&]*(&))/;
- // singleton
- var _instance;
- function ath(options) {
- _instance = _instance || new ath.Class(options);
- return _instance;
- }
- // message in all supported languages
- ath.intl = {
- en_us: {
- ios: 'To add this web app to the home screen: tap %icon and then <strong>Add to Home Screen</strong>.',
- android: 'To add this web app to the home screen open the browser option menu and tap on <strong>Add to homescreen</strong>. <small>The menu can be accessed by pressing the menu hardware button if your device has one, or by tapping the top right menu icon <span class="ath-action-icon">icon</span>.</small>'
- },
- zh_cn: {
- ios: '如要把应用程式加至主屏幕,请点击%icon, 然后<strong>加至主屏幕</strong>',
- android: 'To add this web app to the home screen open the browser option menu and tap on <strong>Add to homescreen</strong>. <small>The menu can be accessed by pressing the menu hardware button if your device has one, or by tapping the top right menu icon <span class="ath-action-icon">icon</span>.</small>'
- },
- zh_tw: {
- ios: '如要把應用程式加至主屏幕, 請點擊%icon, 然後<strong>加至主屏幕</strong>.',
- android: 'To add this web app to the home screen open the browser option menu and tap on <strong>Add to homescreen</strong>. <small>The menu can be accessed by pressing the menu hardware button if your device has one, or by tapping the top right menu icon <span class="ath-action-icon">icon</span>.</small>'
- }
- };
- // Add 2 characters language support (Android mostly)
- for (var lang in ath.intl) {
- ath.intl[lang.substr(0, 2)] = ath.intl[lang];
- }
- // default options
- ath.defaults = {
- appID: 'org.cubiq.addtohome', // local storage name (no need to change)
- fontSize: 15, // base font size, used to properly resize the popup based on viewport scale factor
- debug: false, // override browser checks
- logging: false, // log reasons for showing or not showing to js console; defaults to true when debug is true
- modal: false, // prevent further actions until the message is closed
- mandatory: false, // you can't proceed if you don't add the app to the homescreen
- autostart: true, // show the message automatically
- skipFirstVisit: false, // show only to returning visitors (ie: skip the first time you visit)
- startDelay: 1, // display the message after that many seconds from page load
- lifespan: 15, // life of the message in seconds
- displayPace: 1440, // minutes before the message is shown again (0: display every time, default 24 hours)
- maxDisplayCount: 0, // absolute maximum number of times the message will be shown to the user (0: no limit)
- icon: true, // add touch icon to the message
- message: '', // the message can be customized
- validLocation: [], // list of pages where the message will be shown (array of regexes)
- onInit: null, // executed on instance creation
- onShow: null, // executed when the message is shown
- onRemove: null, // executed when the message is removed
- onAdd: null, // when the application is launched the first time from the homescreen (guesstimate)
- onPrivate: null, // executed if user is in private mode
- privateModeOverride: false, // show the message even in private mode (very rude)
- detectHomescreen: false // try to detect if the site has been added to the homescreen (false | true | 'hash' | 'queryString' | 'smartURL')
- };
- // browser info and capability
- var _ua = window.navigator.userAgent;
- var _nav = window.navigator;
- _extend(ath, {
- hasToken: document.location.hash == '#ath' || _reSmartURL.test(document.location.href) || _reQueryString.test(document.location.search),
- isRetina: window.devicePixelRatio && window.devicePixelRatio > 1,
- isIDevice: (/iphone|ipod|ipad/i).test(_ua),
- isMobileChrome: _ua.indexOf('Android') > -1 && (/Chrome\/[.0-9]*/).test(_ua) && _ua.indexOf("Version") == -1,
- isMobileIE: _ua.indexOf('Windows Phone') > -1,
- language: _nav.language && _nav.language.toLowerCase().replace('-', '_') || ''
- });
- // falls back to en_us if language is unsupported
- ath.language = ath.language && ath.language in ath.intl ? ath.language : 'en_us';
- ath.isMobileSafari = ath.isIDevice && _ua.indexOf('Safari') > -1 && _ua.indexOf('CriOS') < 0;
- ath.OS = ath.isIDevice ? 'ios' : ath.isMobileChrome ? 'android' : ath.isMobileIE ? 'windows' : 'unsupported';
- ath.OSVersion = _ua.match(/(OS|Android) (\d+[_\.]\d+)/);
- ath.OSVersion = ath.OSVersion && ath.OSVersion[2] ? +ath.OSVersion[2].replace('_', '.') : 0;
- ath.isStandalone = 'standalone' in window.navigator && window.navigator.standalone;
- ath.isTablet = (ath.isMobileSafari && _ua.indexOf('iPad') > -1) || (ath.isMobileChrome && _ua.indexOf('Mobile') < 0);
- ath.isCompatible = (ath.isMobileSafari && ath.OSVersion >= 6) || ath.isMobileChrome; // TODO: add winphone
- var _defaultSession = {
- lastDisplayTime: 0, // last time we displayed the message
- returningVisitor: false, // is this the first time you visit
- displayCount: 0, // number of times the message has been shown
- optedout: false, // has the user opted out
- added: false // has been actually added to the homescreen
- };
- ath.removeSession = function(appID) {
- try {
- if (!localStorage) {
- throw new Error('localStorage is not defined');
- }
- localStorage.removeItem(appID || ath.defaults.appID);
- } catch (e) {
- // we are most likely in private mode
- }
- };
- ath.doLog = function(logStr) {
- if (this.options.logging) {
- console.log(logStr);
- }
- };
- ath.Class = function(options) {
- // class methods
- this.doLog = ath.doLog;
- // merge default options with user config
- this.options = _extend({}, ath.defaults);
- _extend(this.options, options);
- // override defaults that are dependent on each other
- if (this.options.debug) {
- this.options.logging = true;
- }
- // IE<9 so exit (I hate you, really)
- if (!_eventListener) {
- return;
- }
- // normalize some options
- this.options.mandatory = this.options.mandatory && ( 'standalone' in window.navigator || this.options.debug );
- this.options.modal = this.options.modal || this.options.mandatory;
- if (this.options.mandatory) {
- this.options.startDelay = -0.5; // make the popup hasty
- }
- this.options.detectHomescreen = this.options.detectHomescreen === true ? 'hash' : this.options.detectHomescreen;
- // setup the debug environment
- if (this.options.debug) {
- ath.isCompatible = true;
- ath.OS = typeof this.options.debug == 'string' ? this.options.debug : ath.OS == 'unsupported' ? 'android' : ath.OS;
- ath.OSVersion = ath.OS == 'ios' ? '8' : '4';
- }
- // the element the message will be appended to
- this.container = document.documentElement;
- // load session
- this.session = this.getItem(this.options.appID);
- this.session = this.session ? JSON.parse(this.session) : undefined;
- // user most likely came from a direct link containing our token, we don't need it and we remove it
- if (ath.hasToken && ( !ath.isCompatible || !this.session )) {
- ath.hasToken = false;
- _removeToken();
- }
- // the device is not supported
- if (!ath.isCompatible) {
- this.doLog("Add to homescreen: not displaying callout because device not supported");
- return;
- }
- this.session = this.session || _defaultSession;
- // check if we can use the local storage
- try {
- if (!localStorage) {
- throw new Error('localStorage is not defined');
- }
- localStorage.setItem(this.options.appID, JSON.stringify(this.session));
- ath.hasLocalStorage = true;
- } catch (e) {
- // we are most likely in private mode
- ath.hasLocalStorage = false;
- if (this.options.onPrivate) {
- this.options.onPrivate.call(this);
- }
- }
- // check if this is a valid location
- var isValidLocation = !this.options.validLocation.length;
- for (var i = this.options.validLocation.length; i--;) {
- if (this.options.validLocation[i].test(document.location.href)) {
- isValidLocation = true;
- break;
- }
- }
- // check compatibility with old versions of add to homescreen. Opt-out if an old session is found
- if (this.getItem('addToHome')) {
- this.optOut();
- }
- // critical errors:
- if (this.session.optedout) {
- this.doLog("Add to homescreen: not displaying callout because user opted out");
- return;
- }
- if (this.session.added) {
- this.doLog("Add to homescreen: not displaying callout because already added to the homescreen");
- return;
- }
- if (!isValidLocation) {
- this.doLog("Add to homescreen: not displaying callout because not a valid location");
- return;
- }
- // check if the app is in stand alone mode
- if (ath.isStandalone) {
- // execute the onAdd event if we haven't already
- if (!this.session.added) {
- this.session.added = true;
- this.updateSession();
- if (this.options.onAdd && ath.hasLocalStorage) { // double check on localstorage to avoid multiple calls to the custom event
- this.options.onAdd.call(this);
- }
- }
- this.doLog("Add to homescreen: not displaying callout because in standalone mode");
- return;
- }
- // (try to) check if the page has been added to the homescreen
- if (this.options.detectHomescreen) {
- // the URL has the token, we are likely coming from the homescreen
- if (ath.hasToken) {
- _removeToken(); // we don't actually need the token anymore, we remove it to prevent redistribution
- // this is called the first time the user opens the app from the homescreen
- if (!this.session.added) {
- this.session.added = true;
- this.updateSession();
- if (this.options.onAdd && ath.hasLocalStorage) { // double check on localstorage to avoid multiple calls to the custom event
- this.options.onAdd.call(this);
- }
- }
- this.doLog("Add to homescreen: not displaying callout because URL has token, so we are likely coming from homescreen");
- return;
- }
- // URL doesn't have the token, so add it
- if (this.options.detectHomescreen == 'hash') {
- history.replaceState('', window.document.title, document.location.href + '#ath');
- } else if (this.options.detectHomescreen == 'smartURL') {
- history.replaceState('', window.document.title, document.location.href.replace(/(\/)?$/, '/ath$1'));
- } else {
- history.replaceState('', window.document.title, document.location.href + (document.location.search ? '&' : '?' ) + 'ath=');
- }
- }
- // check if this is a returning visitor
- if (!this.session.returningVisitor) {
- this.session.returningVisitor = true;
- this.updateSession();
- // we do not show the message if this is your first visit
- if (this.options.skipFirstVisit) {
- this.doLog("Add to homescreen: not displaying callout because skipping first visit");
- return;
- }
- }
- // we do no show the message in private mode
- if (!this.options.privateModeOverride && !ath.hasLocalStorage) {
- this.doLog("Add to homescreen: not displaying callout because browser is in private mode");
- return;
- }
- // all checks passed, ready to display
- this.ready = true;
- if (this.options.onInit) {
- this.options.onInit.call(this);
- }
- if (this.options.autostart) {
- this.doLog("Add to homescreen: autostart displaying callout");
- this.show();
- }
- };
- ath.Class.prototype = {
- // event type to method conversion
- events: {
- load: '_delayedShow',
- error: '_delayedShow',
- orientationchange: 'resize',
- resize: 'resize',
- scroll: 'resize',
- click: 'remove',
- touchmove: '_preventDefault',
- transitionend: '_removeElements',
- webkitTransitionEnd: '_removeElements',
- MSTransitionEnd: '_removeElements'
- },
- handleEvent: function(e) {
- var type = this.events[e.type];
- if (type) {
- this[type](e);
- }
- },
- show: function(force) {
- // in autostart mode wait for the document to be ready
- if (this.options.autostart && !_DOMReady) {
- setTimeout(this.show.bind(this), 50);
- // we are not displaying callout because DOM not ready, but don't log that because
- // it would log too frequently
- return;
- }
- // message already on screen
- if (this.shown) {
- this.doLog("Add to homescreen: not displaying callout because already shown on screen");
- return;
- }
- var now = Date.now();
- var lastDisplayTime = this.session.lastDisplayTime;
- if (force !== true) {
- // this is needed if autostart is disabled and you programmatically call the show() method
- if (!this.ready) {
- this.doLog("Add to homescreen: not displaying callout because not ready");
- return;
- }
- // we obey the display pace (prevent the message to popup too often)
- if (now - lastDisplayTime < this.options.displayPace * 60000) {
- this.doLog("Add to homescreen: not displaying callout because displayed recently");
- return;
- }
- // obey the maximum number of display count
- if (this.options.maxDisplayCount && this.session.displayCount >= this.options.maxDisplayCount) {
- this.doLog("Add to homescreen: not displaying callout because displayed too many times already");
- return;
- }
- }
- this.shown = true;
- // increment the display count
- this.session.lastDisplayTime = now;
- this.session.displayCount++;
- this.updateSession();
- // try to get the highest resolution application icon
- if (!this.applicationIcon) {
- if (ath.OS == 'ios') {
- this.applicationIcon = document.querySelector('head link[rel^=apple-touch-icon][sizes="152x152"],head link[rel^=apple-touch-icon][sizes="144x144"],head link[rel^=apple-touch-icon][sizes="120x120"],head link[rel^=apple-touch-icon][sizes="114x114"],head link[rel^=apple-touch-icon]');
- } else {
- this.applicationIcon = document.querySelector('head link[rel^="shortcut icon"][sizes="196x196"],head link[rel^=apple-touch-icon]');
- }
- }
- var message = '';
- if (typeof this.options.message == 'object' && ath.language in this.options.message) { // use custom language message
- message = this.options.message[ath.language][ath.OS];
- } else if (typeof this.options.message == 'object' && ath.OS in this.options.message) { // use custom os message
- message = this.options.message[ath.OS];
- } else if (this.options.message in ath.intl) { // you can force the locale
- message = ath.intl[this.options.message][ath.OS];
- } else if (this.options.message !== '') { // use a custom message
- message = this.options.message;
- } else if (ath.OS in ath.intl[ath.language]) { // otherwise we use our message
- message = ath.intl[ath.language][ath.OS];
- }
- // add the action icon
- message = '<p>' + message.replace('%icon', '<span class="ath-action-icon">icon</span>') + '</p>';
- // create the message container
- this.viewport = document.createElement('div');
- this.viewport.className = 'ath-viewport';
- if (this.options.modal) {
- this.viewport.className += ' ath-modal';
- }
- if (this.options.mandatory) {
- this.viewport.className += ' ath-mandatory';
- }
- this.viewport.style.position = 'absolute';
- // create the actual message element
- this.element = document.createElement('div');
- this.element.className = 'ath-container ath-' + ath.OS + ' ath-' + ath.OS + (ath.OSVersion + '').substr(0, 1) + ' ath-' + (ath.isTablet ? 'tablet' : 'phone');
- this.element.style.cssText = '-webkit-transition-property:-webkit-transform,opacity;-webkit-transition-duration:0s;-webkit-transition-timing-function:ease-out;transition-property:transform,opacity;transition-duration:0s;transition-timing-function:ease-out;';
- this.element.style.webkitTransform = 'translate3d(0,-' + window.innerHeight + 'px,0)';
- this.element.style.transform = 'translate3d(0,-' + window.innerHeight + 'px,0)';
- // add the application icon
- if (this.options.icon && this.applicationIcon) {
- this.element.className += ' ath-icon';
- this.img = document.createElement('img');
- this.img.className = 'ath-application-icon';
- this.img.addEventListener('load', this, false);
- this.img.addEventListener('error', this, false);
- this.img.src = this.applicationIcon.href;
- this.element.appendChild(this.img);
- }
- this.element.innerHTML += message;
- // we are not ready to show, place the message out of sight
- this.viewport.style.left = '-99999em';
- // attach all elements to the DOM
- this.viewport.appendChild(this.element);
- this.container.appendChild(this.viewport);
- // if we don't have to wait for an image to load, show the message right away
- if (this.img) {
- this.doLog("Add to homescreen: not displaying callout because waiting for img to load");
- } else {
- this._delayedShow();
- }
- },
- _delayedShow: function(e) {
- setTimeout(this._show.bind(this), this.options.startDelay * 1000 + 500);
- },
- _show: function() {
- var that = this;
- // update the viewport size and orientation
- this.updateViewport();
- // reposition/resize the message on orientation change
- window.addEventListener('resize', this, false);
- window.addEventListener('scroll', this, false);
- window.addEventListener('orientationchange', this, false);
- if (this.options.modal) {
- // lock any other interaction
- document.addEventListener('touchmove', this, true);
- }
- // Enable closing after 1 second
- if (!this.options.mandatory) {
- setTimeout(function() {
- that.element.addEventListener('click', that, true);
- }, 1000);
- }
- // kick the animation
- setTimeout(function() {
- that.element.style.webkitTransitionDuration = '1.2s';
- that.element.style.transitionDuration = '1.2s';
- that.element.style.webkitTransform = 'translate3d(0,0,0)';
- that.element.style.transform = 'translate3d(0,0,0)';
- }, 0);
- // set the destroy timer
- if (this.options.lifespan) {
- this.removeTimer = setTimeout(this.remove.bind(this), this.options.lifespan * 1000);
- }
- // fire the custom onShow event
- if (this.options.onShow) {
- this.options.onShow.call(this);
- }
- },
- remove: function() {
- clearTimeout(this.removeTimer);
- // clear up the event listeners
- if (this.img) {
- this.img.removeEventListener('load', this, false);
- this.img.removeEventListener('error', this, false);
- }
- window.removeEventListener('resize', this, false);
- window.removeEventListener('scroll', this, false);
- window.removeEventListener('orientationchange', this, false);
- document.removeEventListener('touchmove', this, true);
- this.element.removeEventListener('click', this, true);
- // remove the message element on transition end
- this.element.addEventListener('transitionend', this, false);
- this.element.addEventListener('webkitTransitionEnd', this, false);
- this.element.addEventListener('MSTransitionEnd', this, false);
- // start the fade out animation
- this.element.style.webkitTransitionDuration = '0.3s';
- this.element.style.opacity = '0';
- },
- _removeElements: function() {
- this.element.removeEventListener('transitionend', this, false);
- this.element.removeEventListener('webkitTransitionEnd', this, false);
- this.element.removeEventListener('MSTransitionEnd', this, false);
- // remove the message from the DOM
- this.container.removeChild(this.viewport);
- this.shown = false;
- // fire the custom onRemove event
- if (this.options.onRemove) {
- this.options.onRemove.call(this);
- }
- },
- updateViewport: function() {
- if (!this.shown) {
- return;
- }
- this.viewport.style.width = window.innerWidth + 'px';
- this.viewport.style.height = window.innerHeight + 'px';
- this.viewport.style.left = window.scrollX + 'px';
- this.viewport.style.top = window.scrollY + 'px';
- var clientWidth = document.documentElement.clientWidth;
- this.orientation = clientWidth > document.documentElement.clientHeight ? 'landscape' : 'portrait';
- var screenWidth = ath.OS == 'ios' ? this.orientation == 'portrait' ? screen.width : screen.height : screen.width;
- this.scale = screen.width > clientWidth ? 1 : screenWidth / window.innerWidth;
- this.element.style.fontSize = this.options.fontSize / this.scale + 'px';
- },
- resize: function() {
- clearTimeout(this.resizeTimer);
- this.resizeTimer = setTimeout(this.updateViewport.bind(this), 100);
- },
- updateSession: function() {
- if (ath.hasLocalStorage === false) {
- return;
- }
- if (localStorage) {
- localStorage.setItem(this.options.appID, JSON.stringify(this.session));
- }
- },
- clearSession: function() {
- this.session = _defaultSession;
- this.updateSession();
- },
- getItem: function(item) {
- try {
- if (!localStorage) {
- throw new Error('localStorage is not defined');
- }
- return localStorage.getItem(item);
- } catch (e) {
- // Preventing exception for some browsers when fetching localStorage key
- ath.hasLocalStorage = false;
- }
- },
- optOut: function() {
- this.session.optedout = true;
- this.updateSession();
- },
- optIn: function() {
- this.session.optedout = false;
- this.updateSession();
- },
- clearDisplayCount: function() {
- this.session.displayCount = 0;
- this.updateSession();
- },
- _preventDefault: function(e) {
- e.preventDefault();
- e.stopPropagation();
- }
- };
- // utility
- function _extend(target, obj) {
- for (var i in obj) {
- target[i] = obj[i];
- }
- return target;
- }
- function _removeToken() {
- if (document.location.hash == '#ath') {
- history.replaceState('', window.document.title, document.location.href.split('#')[0]);
- }
- if (_reSmartURL.test(document.location.href)) {
- history.replaceState('', window.document.title, document.location.href.replace(_reSmartURL, '$1'));
- }
- if (_reQueryString.test(document.location.search)) {
- history.replaceState('', window.document.title, document.location.href.replace(_reQueryString, '$2'));
- }
- }
- /* jshint +W101, +W106 */
- ath.VERSION = '3.2.2';
- module.exports = UI.addToHomescreen = ath;
- /***/ },
- /* 5 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * @via https://github.com/Minwe/bootstrap/blob/master/js/alert.js
- * @copyright Copyright 2013 Twitter, Inc.
- * @license Apache 2.0
- */
- // Alert Class
- // NOTE: removeElement option is unavailable now
- var Alert = function(element, options) {
- var _this = this;
- this.options = $.extend({}, Alert.DEFAULTS, options);
- this.$element = $(element);
- this.$element
- .addClass('am-fade am-in')
- .on('click.alert.amui', '.am-close', function() {
- _this.close();
- });
- };
- Alert.DEFAULTS = {
- removeElement: true
- };
- Alert.prototype.close = function() {
- var $element = this.$element;
- $element.trigger('close.alert.amui').removeClass('am-in');
- function processAlert() {
- $element.trigger('closed.alert.amui').remove();
- }
- UI.support.transition && $element.hasClass('am-fade') ?
- $element
- .one(UI.support.transition.end, processAlert)
- .emulateTransitionEnd(200) :
- processAlert();
- };
- // plugin
- UI.plugin('alert', Alert);
- // Init code
- $(document).on('click.alert.amui.data-api', '[data-am-alert]', function(e) {
- var $target = $(e.target);
- $target.is('.am-close') && $(this).alert('close');
- });
- module.exports = Alert;
- /***/ },
- /* 6 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * @via https://github.com/twbs/bootstrap/blob/master/js/button.js
- * @copyright (c) 2011-2014 Twitter, Inc
- * @license The MIT License
- */
- var Button = function(element, options) {
- this.$element = $(element);
- this.options = $.extend({}, Button.DEFAULTS, options);
- this.isLoading = false;
- this.hasSpinner = false;
- };
- Button.DEFAULTS = {
- loadingText: 'loading...',
- disabledClassName: 'am-disabled',
- activeClassName: 'am-active',
- spinner: undefined
- };
- Button.prototype.setState = function(state, stateText) {
- var $element = this.$element;
- var disabled = 'disabled';
- var data = $element.data();
- var options = this.options;
- var val = $element.is('input') ? 'val' : 'html';
- var stateClassName = 'am-btn-' + state + ' ' + options.disabledClassName;
- state += 'Text';
- if (!options.resetText) {
- options.resetText = $element[val]();
- }
- // add spinner for element with html()
- if (UI.support.animation && options.spinner &&
- val === 'html' && !this.hasSpinner) {
- options.loadingText = '<span class="am-icon-' + options.spinner +
- ' am-icon-spin"></span>' + options.loadingText;
- this.hasSpinner = true;
- }
- stateText = stateText ||
- (data[state] === undefined ? options[state] : data[state]);
- $element[val](stateText);
- // push to event loop to allow forms to submit
- setTimeout($.proxy(function() {
- // TODO: add stateClass for other states
- if (state === 'loadingText') {
- $element.addClass(stateClassName).attr(disabled, disabled);
- this.isLoading = true;
- } else if (this.isLoading) {
- $element.removeClass(stateClassName).removeAttr(disabled);
- this.isLoading = false;
- }
- }, this), 0);
- };
- Button.prototype.toggle = function() {
- var changed = true;
- var $element = this.$element;
- var $parent = this.$element.parent('[class*="am-btn-group"]');
- var activeClassName = Button.DEFAULTS.activeClassName;
- if ($parent.length) {
- var $input = this.$element.find('input');
- if ($input.prop('type') == 'radio') {
- if ($input.prop('checked') && $element.hasClass(activeClassName)) {
- changed = false;
- } else {
- $parent.find('.' + activeClassName).removeClass(activeClassName);
- }
- }
- if (changed) {
- $input.prop('checked',
- !$element.hasClass(activeClassName)).trigger('change');
- }
- }
- if (changed) {
- $element.toggleClass(activeClassName);
- if (!$element.hasClass(activeClassName)) {
- $element.blur();
- }
- }
- };
- UI.plugin('button', Button, {
- dataOptions: 'data-am-loading',
- methodCall: function(args, instance) {
- if (args[0] === 'toggle') {
- instance.toggle();
- } else if (typeof args[0] === 'string') {
- instance.setState.apply(instance, args);
- }
- }
- });
- // Init code
- $(document).on('click.button.amui.data-api', '[data-am-button]', function(e) {
- e.preventDefault();
- var $btn = $(e.target);
- if (!$btn.hasClass('am-btn')) {
- $btn = $btn.closest('.am-btn');
- }
- $btn.button('toggle');
- });
- UI.ready(function(context) {
- $('[data-am-loading]', context).button();
- // resolves #866
- $('[data-am-button]', context).find('input:checked').each(function() {
- $(this).parent('label').addClass(Button.DEFAULTS.activeClassName);
- });
- });
- module.exports = UI.button = Button;
- /***/ },
- /* 7 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * @via https://github.com/twbs/bootstrap/blob/master/js/collapse.js
- * @copyright (c) 2011-2014 Twitter, Inc
- * @license The MIT License
- */
- var Collapse = function(element, options) {
- this.$element = $(element);
- this.options = $.extend({}, Collapse.DEFAULTS, options);
- this.transitioning = null;
- if (this.options.parent) {
- this.$parent = $(this.options.parent);
- }
- if (this.options.toggle) {
- this.toggle();
- }
- };
- Collapse.DEFAULTS = {
- toggle: true
- };
- Collapse.prototype.open = function() {
- if (this.transitioning || this.$element.hasClass('am-in')) {
- return;
- }
- var startEvent = $.Event('open.collapse.amui');
- this.$element.trigger(startEvent);
- if (startEvent.isDefaultPrevented()) {
- return;
- }
- var actives = this.$parent && this.$parent.find('> .am-panel > .am-in');
- if (actives && actives.length) {
- var hasData = actives.data('amui.collapse');
- if (hasData && hasData.transitioning) {
- return;
- }
- Plugin.call(actives, 'close');
- hasData || actives.data('amui.collapse', null);
- }
- this.$element
- .removeClass('am-collapse')
- .addClass('am-collapsing').height(0);
- this.transitioning = 1;
- var complete = function() {
- this.$element
- .removeClass('am-collapsing')
- .addClass('am-collapse am-in')
- .height('')
- .trigger('opened.collapse.amui');
- this.transitioning = 0;
- };
- if (!UI.support.transition) {
- return complete.call(this);
- }
- var scrollHeight = this.$element[0].scrollHeight;
- this.$element
- .one(UI.support.transition.end, $.proxy(complete, this))
- .emulateTransitionEnd(300)
- .css({height: scrollHeight}); // 当折叠的容器有 padding 时,如果用 height() 只能设置内容的宽度
- };
- Collapse.prototype.close = function() {
- if (this.transitioning || !this.$element.hasClass('am-in')) {
- return;
- }
- var startEvent = $.Event('close.collapse.amui');
- this.$element.trigger(startEvent);
- if (startEvent.isDefaultPrevented()) {
- return;
- }
- this.$element.height(this.$element.height()).redraw();
- this.$element.addClass('am-collapsing').
- removeClass('am-collapse am-in');
- this.transitioning = 1;
- var complete = function() {
- this.transitioning = 0;
- this.$element
- .trigger('closed.collapse.amui')
- .removeClass('am-collapsing')
- .addClass('am-collapse');
- // css({height: '0'});
- };
- if (!UI.support.transition) {
- return complete.call(this);
- }
- this.$element.height(0)
- .one(UI.support.transition.end, $.proxy(complete, this))
- .emulateTransitionEnd(300);
- };
- Collapse.prototype.toggle = function() {
- this[this.$element.hasClass('am-in') ? 'close' : 'open']();
- };
- // Collapse Plugin
- function Plugin(option) {
- return this.each(function() {
- var $this = $(this);
- var data = $this.data('amui.collapse');
- var options = $.extend({}, Collapse.DEFAULTS,
- UI.utils.options($this.attr('data-am-collapse')),
- typeof option == 'object' && option);
- if (!data && options.toggle && option === 'open') {
- option = !option;
- }
- if (!data) {
- $this.data('amui.collapse', (data = new Collapse(this, options)));
- }
- if (typeof option == 'string') {
- data[option]();
- }
- });
- }
- $.fn.collapse = Plugin;
- // Init code
- $(document).on('click.collapse.amui.data-api', '[data-am-collapse]',
- function(e) {
- var href;
- var $this = $(this);
- var options = UI.utils.options($this.attr('data-am-collapse'));
- var target = options.target ||
- e.preventDefault() ||
- (href = $this.attr('href')) &&
- href.replace(/.*(?=#[^\s]+$)/, '');
- var $target = $(target);
- var data = $target.data('amui.collapse');
- var option = data ? 'toggle' : options;
- var parent = options.parent;
- var $parent = parent && $(parent);
- if (!data || !data.transitioning) {
- if ($parent) {
- // '[data-am-collapse*="{parent: \'' + parent + '"]
- $parent.find('[data-am-collapse]').not($this).addClass('am-collapsed');
- }
- $this[$target.hasClass('am-in') ?
- 'addClass' : 'removeClass']('am-collapsed');
- }
- Plugin.call($target, option);
- });
- module.exports = UI.collapse = Collapse;
- // TODO: 更好的 target 选择方式
- // 折叠的容器必须没有 border/padding 才能正常处理,否则动画会有一些小问题
- // 寻找更好的未知高度 transition 动画解决方案,max-height 之类的就算了
- /***/ },
- /* 8 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var $doc = $(document);
- /**
- * bootstrap-datepicker.js
- * @via http://www.eyecon.ro/bootstrap-datepicker
- * @license http://www.apache.org/licenses/LICENSE-2.0
- */
- var Datepicker = function(element, options) {
- this.$element = $(element);
- this.options = $.extend({}, Datepicker.DEFAULTS, options);
- this.format = DPGlobal.parseFormat(this.options.format);
- this.$element.data('date', this.options.date);
- this.language = this.getLocale(this.options.locale);
- this.theme = this.options.theme;
- this.$picker = $(DPGlobal.template).appendTo('body').on({
- click: $.proxy(this.click, this)
- // mousedown: $.proxy(this.mousedown, this)
- });
- this.isInput = this.$element.is('input');
- this.component = this.$element.is('.am-datepicker-date') ?
- this.$element.find('.am-datepicker-add-on') : false;
- if (this.isInput) {
- this.$element.on({
- 'click.datepicker.amui': $.proxy(this.open, this),
- // blur: $.proxy(this.close, this),
- 'keyup.datepicker.amui': $.proxy(this.update, this)
- });
- } else {
- if (this.component) {
- this.component.on('click.datepicker.amui', $.proxy(this.open, this));
- } else {
- this.$element.on('click.datepicker.amui', $.proxy(this.open, this));
- }
- }
- this.minViewMode = this.options.minViewMode;
- if (typeof this.minViewMode === 'string') {
- switch (this.minViewMode) {
- case 'months':
- this.minViewMode = 1;
- break;
- case 'years':
- this.minViewMode = 2;
- break;
- default:
- this.minViewMode = 0;
- break;
- }
- }
- this.viewMode = this.options.viewMode;
- if (typeof this.viewMode === 'string') {
- switch (this.viewMode) {
- case 'months':
- this.viewMode = 1;
- break;
- case 'years':
- this.viewMode = 2;
- break;
- default:
- this.viewMode = 0;
- break;
- }
- }
- this.startViewMode = this.viewMode;
- this.weekStart = ((this.options.weekStart ||
- Datepicker.locales[this.language].weekStart || 0) % 7);
- this.weekEnd = ((this.weekStart + 6) % 7);
- this.onRender = this.options.onRender;
- this.setTheme();
- this.fillDow();
- this.fillMonths();
- this.update();
- this.showMode();
- };
- Datepicker.DEFAULTS = {
- locale: 'zh_CN',
- format: 'yyyy-mm-dd',
- weekStart: undefined,
- viewMode: 0,
- minViewMode: 0,
- date: '',
- theme: '',
- autoClose: 1,
- onRender: function(date) {
- return '';
- }
- };
- Datepicker.prototype.open = function(e) {
- this.$picker.show();
- this.height = this.component ?
- this.component.outerHeight() : this.$element.outerHeight();
- this.place();
- $(window).on('resize.datepicker.amui', $.proxy(this.place, this));
- if (e) {
- e.stopPropagation();
- e.preventDefault();
- }
- var that = this;
- $doc.on('mousedown.datapicker.amui touchstart.datepicker.amui', function(ev) {
- if ($(ev.target).closest('.am-datepicker').length === 0) {
- that.close();
- }
- });
- this.$element.trigger({
- type: 'open.datepicker.amui',
- date: this.date
- });
- };
- Datepicker.prototype.close = function() {
- this.$picker.hide();
- $(window).off('resize.datepicker.amui', this.place);
- this.viewMode = this.startViewMode;
- this.showMode();
- if (!this.isInput) {
- $(document).off('mousedown.datapicker.amui touchstart.datepicker.amui',
- this.close);
- }
- // this.set();
- this.$element.trigger({
- type: 'close.datepicker.amui',
- date: this.date
- });
- };
- Datepicker.prototype.set = function() {
- var formatted = DPGlobal.formatDate(this.date, this.format);
- var $input;
- if (!this.isInput) {
- if (this.component) {
- $input = this.$element.find('input').attr('value', formatted);
- }
- this.$element.data('date', formatted);
- } else {
- $input = this.$element.attr('value', formatted);
- }
- // fixes https://github.com/amazeui/amazeui/issues/711
- $input && $input.trigger('change');
- };
- Datepicker.prototype.setValue = function(newDate) {
- if (typeof newDate === 'string') {
- this.date = DPGlobal.parseDate(newDate, this.format);
- } else {
- this.date = new Date(newDate);
- }
- this.set();
- this.viewDate = new Date(this.date.getFullYear(),
- this.date.getMonth(), 1, 0, 0, 0, 0);
- this.fill();
- };
- Datepicker.prototype.place = function() {
- var offset = this.component ?
- this.component.offset() : this.$element.offset();
- var $width = this.component ?
- this.component.width() : this.$element.width();
- var top = offset.top + this.height;
- var left = offset.left;
- var right = $doc.width() - offset.left - $width;
- var isOutView = this.isOutView();
- this.$picker.removeClass('am-datepicker-right');
- this.$picker.removeClass('am-datepicker-up');
- if ($doc.width() > 640) {
- if (isOutView.outRight) {
- this.$picker.addClass('am-datepicker-right');
- this.$picker.css({
- top: top,
- left: 'auto',
- right: right
- });
- return;
- }
- if (isOutView.outBottom) {
- this.$picker.addClass('am-datepicker-up');
- top = offset.top - this.$picker.outerHeight(true);
- }
- } else {
- left = 0;
- }
- this.$picker.css({
- top: top,
- left: left
- });
- };
- Datepicker.prototype.update = function(newDate) {
- this.date = DPGlobal.parseDate(
- typeof newDate === 'string' ? newDate : (this.isInput ?
- this.$element.prop('value') : this.$element.data('date')),
- this.format
- );
- this.viewDate = new Date(this.date.getFullYear(),
- this.date.getMonth(), 1, 0, 0, 0, 0);
- this.fill();
- };
- // Days of week
- Datepicker.prototype.fillDow = function() {
- var dowCount = this.weekStart;
- var html = '<tr>';
- while (dowCount < this.weekStart + 7) {
- // NOTE: do % then add 1
- html += '<th class="am-datepicker-dow">' +
- Datepicker.locales[this.language].daysMin[(dowCount++) % 7] +
- '</th>';
- }
- html += '</tr>';
- this.$picker.find('.am-datepicker-days thead').append(html);
- };
- Datepicker.prototype.fillMonths = function() {
- var html = '';
- var i = 0;
- while (i < 12) {
- html += '<span class="am-datepicker-month">' +
- Datepicker.locales[this.language].monthsShort[i++] + '</span>';
- }
- this.$picker.find('.am-datepicker-months td').append(html);
- };
- Datepicker.prototype.fill = function() {
- var d = new Date(this.viewDate);
- var year = d.getFullYear();
- var month = d.getMonth();
- var currentDate = this.date.valueOf();
- var prevMonth = new Date(year, month - 1, 28, 0, 0, 0, 0);
- var day = DPGlobal
- .getDaysInMonth(prevMonth.getFullYear(), prevMonth.getMonth());
- var daysSelect = this.$picker
- .find('.am-datepicker-days .am-datepicker-select');
- if (this.language === 'zh_CN') {
- daysSelect.text(year + Datepicker.locales[this.language].year[0] +
- ' ' + Datepicker.locales[this.language].months[month]);
- } else {
- daysSelect.text(Datepicker.locales[this.language].months[month] +
- ' ' + year);
- }
- prevMonth.setDate(day);
- prevMonth.setDate(day - (prevMonth.getDay() - this.weekStart + 7) % 7);
- var nextMonth = new Date(prevMonth);
- nextMonth.setDate(nextMonth.getDate() + 42);
- nextMonth = nextMonth.valueOf();
- var html = [];
- var className;
- var prevY;
- var prevM;
- while (prevMonth.valueOf() < nextMonth) {
- if (prevMonth.getDay() === this.weekStart) {
- html.push('<tr>');
- }
- className = this.onRender(prevMonth, 0);
- prevY = prevMonth.getFullYear();
- prevM = prevMonth.getMonth();
- if ((prevM < month && prevY === year) || prevY < year) {
- className += ' am-datepicker-old';
- } else if ((prevM > month && prevY === year) || prevY > year) {
- className += ' am-datepicker-new';
- }
- if (prevMonth.valueOf() === currentDate) {
- className += ' am-active';
- }
- html.push('<td class="am-datepicker-day ' +
- className + '">' + prevMonth.getDate() + '</td>');
- if (prevMonth.getDay() === this.weekEnd) {
- html.push('</tr>');
- }
- prevMonth.setDate(prevMonth.getDate() + 1);
- }
- this.$picker.find('.am-datepicker-days tbody')
- .empty().append(html.join(''));
- var currentYear = this.date.getFullYear();
- var months = this.$picker.find('.am-datepicker-months')
- .find('.am-datepicker-select')
- .text(year);
- months = months.end()
- .find('span').removeClass('am-active').removeClass('am-disabled');
- var monthLen = 0;
- while (monthLen < 12) {
- if (this.onRender(d.setFullYear(year, monthLen), 1)) {
- months.eq(monthLen).addClass('am-disabled');
- }
- monthLen++;
- }
- if (currentYear === year) {
- months.eq(this.date.getMonth())
- .removeClass('am-disabled')
- .addClass('am-active');
- }
- html = '';
- year = parseInt(year / 10, 10) * 10;
- var yearCont = this.$picker
- .find('.am-datepicker-years')
- .find('.am-datepicker-select')
- .text(year + '-' + (year + 9))
- .end()
- .find('td');
- var yearClassName;
- // fixes https://github.com/amazeui/amazeui/issues/770
- // maybe not need now
- var viewDate = new Date(this.viewDate);
- year -= 1;
- for (var i = -1; i < 11; i++) {
- yearClassName = this.onRender(viewDate.setFullYear(year), 2);
- html += '<span class="' + yearClassName + '' +
- (i === -1 || i === 10 ? ' am-datepicker-old' : '') +
- (currentYear === year ? ' am-active' : '') + '">' + year + '</span>';
- year += 1;
- }
- yearCont.html(html);
- };
- Datepicker.prototype.click = function(event) {
- event.stopPropagation();
- event.preventDefault();
- var month;
- var year;
- var $dayActive = this.$picker.find('.am-datepicker-days').find('.am-active');
- var $months = this.$picker.find('.am-datepicker-months');
- var $monthIndex = $months.find('.am-active').index();
- var $target = $(event.target).closest('span, td, th');
- if ($target.length === 1) {
- switch ($target[0].nodeName.toLowerCase()) {
- case 'th':
- switch ($target[0].className) {
- case 'am-datepicker-switch':
- this.showMode(1);
- break;
- case 'am-datepicker-prev':
- case 'am-datepicker-next':
- this.viewDate['set' + DPGlobal.modes[this.viewMode].navFnc].call(
- this.viewDate,
- this.viewDate
- ['get' + DPGlobal.modes[this.viewMode].navFnc]
- .call(this.viewDate) +
- DPGlobal.modes[this.viewMode].navStep *
- ($target[0].className === 'am-datepicker-prev' ? -1 : 1)
- );
- this.fill();
- this.set();
- break;
- }
- break;
- case 'span':
- if ($target.is('.am-disabled')) {
- return;
- }
- if ($target.is('.am-datepicker-month')) {
- month = $target.parent().find('span').index($target);
- if ($target.is('.am-active')) {
- this.viewDate.setMonth(month, $dayActive.text());
- } else {
- this.viewDate.setMonth(month);
- }
- } else {
- year = parseInt($target.text(), 10) || 0;
- if ($target.is('.am-active')) {
- this.viewDate.setFullYear(year, $monthIndex, $dayActive.text());
- } else {
- this.viewDate.setFullYear(year);
- }
- }
- if (this.viewMode !== 0) {
- this.date = new Date(this.viewDate);
- this.$element.trigger({
- type: 'changeDate.datepicker.amui',
- date: this.date,
- viewMode: DPGlobal.modes[this.viewMode].clsName
- });
- }
- this.showMode(-1);
- this.fill();
- this.set();
- break;
- case 'td':
- if ($target.is('.am-datepicker-day') && !$target.is('.am-disabled')) {
- var day = parseInt($target.text(), 10) || 1;
- month = this.viewDate.getMonth();
- if ($target.is('.am-datepicker-old')) {
- month -= 1;
- } else if ($target.is('.am-datepicker-new')) {
- month += 1;
- }
- year = this.viewDate.getFullYear();
- this.date = new Date(year, month, day, 0, 0, 0, 0);
- this.viewDate = new Date(year, month, Math.min(28, day), 0, 0, 0, 0);
- this.fill();
- this.set();
- this.$element.trigger({
- type: 'changeDate.datepicker.amui',
- date: this.date,
- viewMode: DPGlobal.modes[this.viewMode].clsName
- });
- this.options.autoClose && this.close();
- }
- break;
- }
- }
- };
- Datepicker.prototype.mousedown = function(event) {
- event.stopPropagation();
- event.preventDefault();
- };
- Datepicker.prototype.showMode = function(dir) {
- if (dir) {
- this.viewMode = Math.max(this.minViewMode,
- Math.min(2, this.viewMode + dir));
- }
- this.$picker.find('>div').hide().
- filter('.am-datepicker-' + DPGlobal.modes[this.viewMode].clsName).show();
- };
- Datepicker.prototype.isOutView = function() {
- var offset = this.component ?
- this.component.offset() : this.$element.offset();
- var isOutView = {
- outRight: false,
- outBottom: false
- };
- var $picker = this.$picker;
- var width = offset.left + $picker.outerWidth(true);
- var height = offset.top + $picker.outerHeight(true) +
- this.$element.innerHeight();
- if (width > $doc.width()) {
- isOutView.outRight = true;
- }
- if (height > $doc.height()) {
- isOutView.outBottom = true;
- }
- return isOutView;
- };
- Datepicker.prototype.getLocale = function(locale) {
- if (!locale) {
- locale = navigator.language && navigator.language.split('-');
- locale[1] = locale[1].toUpperCase();
- locale = locale.join('_');
- }
- if (!Datepicker.locales[locale]) {
- locale = 'en_US';
- }
- return locale;
- };
- Datepicker.prototype.setTheme = function() {
- if (this.theme) {
- this.$picker.addClass('am-datepicker-' + this.theme);
- }
- };
- // Datepicker locales
- Datepicker.locales = {
- en_US: {
- days: ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday',
- 'Friday', 'Saturday'],
- daysShort: ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'],
- daysMin: ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'],
- months: ['January', 'February', 'March', 'April', 'May', 'June',
- 'July', 'August', 'September', 'October', 'November', 'December'],
- monthsShort: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun',
- 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'],
- weekStart: 0
- },
- zh_CN: {
- days: ['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六'],
- daysShort: ['周日', '周一', '周二', '周三', '周四', '周五', '周六'],
- daysMin: ['日', '一', '二', '三', '四', '五', '六'],
- months: ['一月', '二月', '三月', '四月', '五月', '六月', '七月',
- '八月', '九月', '十月', '十一月', '十二月'],
- monthsShort: ['一月', '二月', '三月', '四月', '五月', '六月',
- '七月', '八月', '九月', '十月', '十一月', '十二月'],
- weekStart: 1,
- year: ['年']
- }
- };
- var DPGlobal = {
- modes: [
- {
- clsName: 'days',
- navFnc: 'Month',
- navStep: 1
- },
- {
- clsName: 'months',
- navFnc: 'FullYear',
- navStep: 1
- },
- {
- clsName: 'years',
- navFnc: 'FullYear',
- navStep: 10
- }
- ],
- isLeapYear: function(year) {
- return (((year % 4 === 0) && (year % 100 !== 0)) || (year % 400 === 0));
- },
- getDaysInMonth: function(year, month) {
- return [31, (DPGlobal.isLeapYear(year) ? 29 : 28),
- 31, 30, 31, 30, 31, 31, 30, 31, 30, 31][month];
- },
- parseFormat: function(format) {
- var separator = format.match(/[.\/\-\s].*?/);
- var parts = format.split(/\W+/);
- if (!separator || !parts || parts.length === 0) {
- throw new Error('Invalid date format.');
- }
- return {
- separator: separator,
- parts: parts
- };
- },
- parseDate: function(date, format) {
- var parts = date.split(format.separator);
- var val;
- date = new Date();
- date.setHours(0);
- date.setMinutes(0);
- date.setSeconds(0);
- date.setMilliseconds(0);
- if (parts.length === format.parts.length) {
- var year = date.getFullYear();
- var day = date.getDate();
- var month = date.getMonth();
- for (var i = 0, cnt = format.parts.length; i < cnt; i++) {
- val = parseInt(parts[i], 10) || 1;
- switch (format.parts[i]) {
- case 'dd':
- case 'd':
- day = val;
- date.setDate(val);
- break;
- case 'mm':
- case 'm':
- month = val - 1;
- date.setMonth(val - 1);
- break;
- case 'yy':
- year = 2000 + val;
- date.setFullYear(2000 + val);
- break;
- case 'yyyy':
- year = val;
- date.setFullYear(val);
- break;
- }
- }
- date = new Date(year, month, day, 0, 0, 0);
- }
- return date;
- },
- formatDate: function(date, format) {
- var val = {
- d: date.getDate(),
- m: date.getMonth() + 1,
- yy: date.getFullYear().toString().substring(2),
- yyyy: date.getFullYear()
- };
- var dateArray = [];
- val.dd = (val.d < 10 ? '0' : '') + val.d;
- val.mm = (val.m < 10 ? '0' : '') + val.m;
- for (var i = 0, cnt = format.parts.length; i < cnt; i++) {
- dateArray.push(val[format.parts[i]]);
- }
- return dateArray.join(format.separator);
- },
- headTemplate: '<thead>' +
- '<tr class="am-datepicker-header">' +
- '<th class="am-datepicker-prev">' +
- '<i class="am-datepicker-prev-icon"></i></th>' +
- '<th colspan="5" class="am-datepicker-switch">' +
- '<div class="am-datepicker-select"></div></th>' +
- '<th class="am-datepicker-next"><i class="am-datepicker-next-icon"></i>' +
- '</th></tr></thead>',
- contTemplate: '<tbody><tr><td colspan="7"></td></tr></tbody>'
- };
- DPGlobal.template = '<div class="am-datepicker am-datepicker-dropdown">' +
- '<div class="am-datepicker-caret"></div>' +
- '<div class="am-datepicker-days">' +
- '<table class="am-datepicker-table">' +
- DPGlobal.headTemplate +
- '<tbody></tbody>' +
- '</table>' +
- '</div>' +
- '<div class="am-datepicker-months">' +
- '<table class="am-datepicker-table">' +
- DPGlobal.headTemplate +
- DPGlobal.contTemplate +
- '</table>' +
- '</div>' +
- '<div class="am-datepicker-years">' +
- '<table class="am-datepicker-table">' +
- DPGlobal.headTemplate +
- DPGlobal.contTemplate +
- '</table>' +
- '</div>' +
- '</div>';
- // jQuery plugin
- UI.plugin('datepicker', Datepicker);
- // Init code
- UI.ready(function(context) {
- $('[data-am-datepicker]').datepicker();
- });
- module.exports = UI.datepicker = Datepicker;
- // TODO: 1. 载入动画
- // 2. less 优化
- /***/ },
- /* 9 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var $doc = $(document);
- var transition = UI.support.transition;
- var Dimmer = function() {
- this.id = UI.utils.generateGUID('am-dimmer');
- this.$element = $(Dimmer.DEFAULTS.tpl, {
- id: this.id
- });
- this.inited = false;
- this.scrollbarWidth = 0;
- this.$used = $([]);
- };
- Dimmer.DEFAULTS = {
- tpl: '<div class="am-dimmer" data-am-dimmer></div>'
- };
- Dimmer.prototype.init = function() {
- if (!this.inited) {
- $(document.body).append(this.$element);
- this.inited = true;
- $doc.trigger('init.dimmer.amui');
- this.$element.on('touchmove.dimmer.amui', function(e) {
- e.preventDefault();
- });
- }
- return this;
- };
- Dimmer.prototype.open = function(relatedElement) {
- if (!this.inited) {
- this.init();
- }
- var $element = this.$element;
- // 用于多重调用
- if (relatedElement) {
- this.$used = this.$used.add($(relatedElement));
- }
- this.checkScrollbar().setScrollbar();
- $element.show().trigger('open.dimmer.amui');
- transition && $element.off(transition.end);
- setTimeout(function() {
- $element.addClass('am-active');
- }, 0);
- return this;
- };
- Dimmer.prototype.close = function(relatedElement, force) {
- this.$used = this.$used.not($(relatedElement));
- if (!force && this.$used.length) {
- return this;
- }
- var $element = this.$element;
- $element.removeClass('am-active').trigger('close.dimmer.amui');
- function complete() {
- $element.hide();
- this.resetScrollbar();
- }
- // transition ? $element.one(transition.end, $.proxy(complete, this)) :
- complete.call(this);
- return this;
- };
- Dimmer.prototype.checkScrollbar = function() {
- this.scrollbarWidth = UI.utils.measureScrollbar();
- return this;
- };
- Dimmer.prototype.setScrollbar = function() {
- var $body = $(document.body);
- var bodyPaddingRight = parseInt(($body.css('padding-right') || 0), 10);
- if (this.scrollbarWidth) {
- $body.css('padding-right', bodyPaddingRight + this.scrollbarWidth);
- }
- $body.addClass('am-dimmer-active');
- return this;
- };
- Dimmer.prototype.resetScrollbar = function() {
- $(document.body).css('padding-right', '').removeClass('am-dimmer-active');
- return this;
- };
- module.exports = UI.dimmer = new Dimmer();
- /***/ },
- /* 10 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var animation = UI.support.animation;
- /**
- * @via https://github.com/Minwe/bootstrap/blob/master/js/dropdown.js
- * @copyright (c) 2011-2014 Twitter, Inc
- * @license The MIT License
- */
- // var toggle = '[data-am-dropdown] > .am-dropdown-toggle';
- var Dropdown = function(element, options) {
- this.options = $.extend({}, Dropdown.DEFAULTS, options);
- options = this.options;
- this.$element = $(element);
- this.$toggle = this.$element.find(options.selector.toggle);
- this.$dropdown = this.$element.find(options.selector.dropdown);
- this.$boundary = (options.boundary === window) ? $(window) :
- this.$element.closest(options.boundary);
- this.$justify = (options.justify && $(options.justify).length &&
- $(options.justify)) || undefined;
- !this.$boundary.length && (this.$boundary = $(window));
- this.active = this.$element.hasClass('am-active') ? true : false;
- this.animating = null;
- this.events();
- };
- Dropdown.DEFAULTS = {
- animation: 'am-animation-slide-top-fixed',
- boundary: window,
- justify: undefined,
- selector: {
- dropdown: '.am-dropdown-content',
- toggle: '.am-dropdown-toggle'
- },
- trigger: 'click'
- };
- Dropdown.prototype.toggle = function() {
- this.clear();
- if (this.animating) {
- return;
- }
- this[this.active ? 'close' : 'open']();
- };
- Dropdown.prototype.open = function(e) {
- var $toggle = this.$toggle;
- var $element = this.$element;
- var $dropdown = this.$dropdown;
- if ($toggle.is('.am-disabled, :disabled')) {
- return;
- }
- if (this.active) {
- return;
- }
- $element.trigger('open.dropdown.amui').addClass('am-active');
- $toggle.trigger('focus');
- this.checkDimensions(e);
- var complete = $.proxy(function() {
- $element.trigger('opened.dropdown.amui');
- this.active = true;
- this.animating = 0;
- }, this);
- if (animation) {
- this.animating = 1;
- $dropdown.addClass(this.options.animation).
- on(animation.end + '.open.dropdown.amui', $.proxy(function() {
- complete();
- $dropdown.removeClass(this.options.animation);
- }, this));
- } else {
- complete();
- }
- };
- Dropdown.prototype.close = function() {
- if (!this.active) {
- return;
- }
- // fix #165
- // var animationName = this.options.animation + ' am-animation-reverse';
- var animationName = 'am-dropdown-animation';
- var $element = this.$element;
- var $dropdown = this.$dropdown;
- $element.trigger('close.dropdown.amui');
- var complete = $.proxy(function complete() {
- $element.
- removeClass('am-active').
- trigger('closed.dropdown.amui');
- this.active = false;
- this.animating = 0;
- this.$toggle.blur();
- }, this);
- if (animation) {
- $dropdown.removeClass(this.options.animation);
- $dropdown.addClass(animationName);
- this.animating = 1;
- // animation
- $dropdown.one(animation.end + '.close.dropdown.amui', function() {
- $dropdown.removeClass(animationName);
- complete();
- });
- } else {
- complete();
- }
- };
- Dropdown.prototype.enable = function() {
- this.$toggle.prop('disabled', false);
- },
- Dropdown.prototype.disable = function() {
- this.$toggle.prop('disabled', true);
- },
- Dropdown.prototype.checkDimensions = function(e) {
- if (!this.$dropdown.length) {
- return;
- }
- var $dropdown = this.$dropdown;
-
- // @see #873
- if (e && e.offset) {
- $dropdown.offset(e.offset);
- }
- var offset = $dropdown.offset();
- var width = $dropdown.outerWidth();
- var boundaryWidth = this.$boundary.width();
- var boundaryOffset = $.isWindow(this.boundary) && this.$boundary.offset() ?
- this.$boundary.offset().left : 0;
- if (this.$justify) {
- // jQuery.fn.width() is really...
- $dropdown.css({'min-width': this.$justify.css('width')});
- }
- if ((width + (offset.left - boundaryOffset)) > boundaryWidth) {
- this.$element.addClass('am-dropdown-flip');
- }
- };
- Dropdown.prototype.clear = function() {
- $('[data-am-dropdown]').not(this.$element).each(function() {
- var data = $(this).data('amui.dropdown');
- data && data.close();
- });
- };
- Dropdown.prototype.events = function() {
- var eventNS = 'dropdown.amui';
- // triggers = this.options.trigger.split(' '),
- var $toggle = this.$toggle;
- $toggle.on('click.' + eventNS, $.proxy(function(e) {
- e.preventDefault();
- this.toggle();
- }, this));
- /*for (var i = triggers.length; i--;) {
- var trigger = triggers[i];
- if (trigger === 'click') {
- $toggle.on('click.' + eventNS, $.proxy(this.toggle, this))
- }
- if (trigger === 'focus' || trigger === 'hover') {
- var eventIn = trigger == 'hover' ? 'mouseenter' : 'focusin';
- var eventOut = trigger == 'hover' ? 'mouseleave' : 'focusout';
- this.$element.on(eventIn + '.' + eventNS, $.proxy(this.open, this))
- .on(eventOut + '.' + eventNS, $.proxy(this.close, this));
- }
- }*/
- $(document).on('keydown.dropdown.amui', $.proxy(function(e) {
- e.keyCode === 27 && this.active && this.close();
- }, this)).on('click.outer.dropdown.amui', $.proxy(function(e) {
- // var $target = $(e.target);
- if (this.active &&
- (this.$element[0] === e.target || !this.$element.find(e.target).length)) {
- this.close();
- }
- }, this));
- };
- // Dropdown Plugin
- UI.plugin('dropdown', Dropdown);
- // Init code
- UI.ready(function(context) {
- $('[data-am-dropdown]', context).dropdown();
- });
- $(document).on('click.dropdown.amui.data-api', '.am-dropdown form',
- function(e) {
- e.stopPropagation();
- });
- module.exports = UI.dropdown = Dropdown;
- // TODO: 1. 处理链接 focus
- // 2. 增加 mouseenter / mouseleave 选项
- // 3. 宽度适应
- /***/ },
- /* 11 */
- /***/ function(module, exports, __webpack_require__) {
- /* WEBPACK VAR INJECTION */(function(setImmediate) {var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- // MODIFIED:
- // - LINE 252: add `<i></i>`
- // - namespace
- // - Init code
- // TODO: start after x ms when pause on actions
- /*
- * jQuery FlexSlider v2.6.1
- * Copyright 2012 WooThemes
- * Contributing Author: Tyler Smith
- */
- var focused = true;
- // FlexSlider: Object Instance
- $.flexslider = function(el, options) {
- var slider = $(el);
- // making variables public
- slider.vars = $.extend({}, $.flexslider.defaults, options);
- var namespace = slider.vars.namespace,
- msGesture = window.navigator && window.navigator.msPointerEnabled && window.MSGesture,
- touch = (( "ontouchstart" in window ) || msGesture || window.DocumentTouch && document instanceof DocumentTouch) && slider.vars.touch,
- // depricating this idea, as devices are being released with both of these events
- eventType = "click touchend MSPointerUp keyup",
- watchedEvent = "",
- watchedEventClearTimer,
- vertical = slider.vars.direction === "vertical",
- reverse = slider.vars.reverse,
- carousel = (slider.vars.itemWidth > 0),
- fade = slider.vars.animation === "fade",
- asNav = slider.vars.asNavFor !== "",
- methods = {};
- // Store a reference to the slider object
- $.data(el, 'flexslider', slider);
- // Private slider methods
- methods = {
- init: function() {
- slider.animating = false;
- // Get current slide and make sure it is a number
- slider.currentSlide = parseInt((slider.vars.startAt ? slider.vars.startAt : 0), 10);
- if (isNaN(slider.currentSlide)) {
- slider.currentSlide = 0;
- }
- slider.animatingTo = slider.currentSlide;
- slider.atEnd = (slider.currentSlide === 0 || slider.currentSlide === slider.last);
- slider.containerSelector = slider.vars.selector.substr(0, slider.vars.selector.search(' '));
- slider.slides = $(slider.vars.selector, slider);
- slider.container = $(slider.containerSelector, slider);
- slider.count = slider.slides.length;
- // SYNC:
- slider.syncExists = $(slider.vars.sync).length > 0;
- // SLIDE:
- if (slider.vars.animation === "slide") {
- slider.vars.animation = "swing";
- }
- slider.prop = (vertical) ? "top" : "marginLeft";
- slider.args = {};
- // SLIDESHOW:
- slider.manualPause = false;
- slider.stopped = false;
- //PAUSE WHEN INVISIBLE
- slider.started = false;
- slider.startTimeout = null;
- // TOUCH/USECSS:
- slider.transitions = !slider.vars.video && !fade && slider.vars.useCSS && (function() {
- var obj = document.createElement('div'),
- props = ['perspectiveProperty', 'WebkitPerspective', 'MozPerspective', 'OPerspective', 'msPerspective'];
- for (var i in props) {
- if (obj.style[props[i]] !== undefined) {
- slider.pfx = props[i].replace('Perspective', '').toLowerCase();
- slider.prop = "-" + slider.pfx + "-transform";
- return true;
- }
- }
- return false;
- }());
- slider.ensureAnimationEnd = '';
- // CONTROLSCONTAINER:
- if (slider.vars.controlsContainer !== "") slider.controlsContainer = $(slider.vars.controlsContainer).length > 0 && $(slider.vars.controlsContainer);
- // MANUAL:
- if (slider.vars.manualControls !== "") slider.manualControls = $(slider.vars.manualControls).length > 0 && $(slider.vars.manualControls);
- // CUSTOM DIRECTION NAV:
- if (slider.vars.customDirectionNav !== "") slider.customDirectionNav = $(slider.vars.customDirectionNav).length === 2 && $(slider.vars.customDirectionNav);
- // RANDOMIZE:
- if (slider.vars.randomize) {
- slider.slides.sort(function() {
- return (Math.round(Math.random()) - 0.5);
- });
- slider.container.empty().append(slider.slides);
- }
- slider.doMath();
- // INIT
- slider.setup("init");
- // CONTROLNAV:
- if (slider.vars.controlNav) {
- methods.controlNav.setup();
- }
- // DIRECTIONNAV:
- if (slider.vars.directionNav) {
- methods.directionNav.setup();
- }
- // KEYBOARD:
- if (slider.vars.keyboard && ($(slider.containerSelector).length === 1 || slider.vars.multipleKeyboard)) {
- $(document).bind('keyup', function(event) {
- var keycode = event.keyCode;
- if (!slider.animating && (keycode === 39 || keycode === 37)) {
- var target = (keycode === 39) ? slider.getTarget('next') :
- (keycode === 37) ? slider.getTarget('prev') : false;
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- }
- });
- }
- // MOUSEWHEEL:
- if (slider.vars.mousewheel) {
- slider.bind('mousewheel', function(event, delta, deltaX, deltaY) {
- event.preventDefault();
- var target = (delta < 0) ? slider.getTarget('next') : slider.getTarget('prev');
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- });
- }
- // PAUSEPLAY
- if (slider.vars.pausePlay) {
- methods.pausePlay.setup();
- }
- //PAUSE WHEN INVISIBLE
- if (slider.vars.slideshow && slider.vars.pauseInvisible) {
- methods.pauseInvisible.init();
- }
- // SLIDSESHOW
- if (slider.vars.slideshow) {
- if (slider.vars.pauseOnHover) {
- slider.hover(function() {
- if (!slider.manualPlay && !slider.manualPause) {slider.pause();}
- }, function() {
- if (!slider.manualPause && !slider.manualPlay && !slider.stopped) {slider.play();}
- });
- }
- // initialize animation
- // If we're visible, or we don't use PageVisibility API
- if (!slider.vars.pauseInvisible || !methods.pauseInvisible.isHidden()) {
- (slider.vars.initDelay > 0) ? slider.startTimeout = setTimeout(slider.play, slider.vars.initDelay) : slider.play();
- }
- }
- // ASNAV:
- if (asNav) {methods.asNav.setup();}
- // TOUCH
- if (touch && slider.vars.touch) {methods.touch();}
- // FADE&&SMOOTHHEIGHT || SLIDE:
- if (!fade || (fade && slider.vars.smoothHeight)) {$(window).bind("resize orientationchange focus", methods.resize);}
- slider.find("img").attr("draggable", "false");
- // API: start() Callback
- setTimeout(function() {
- slider.vars.start(slider);
- }, 200);
- },
- asNav: {
- setup: function() {
- slider.asNav = true;
- slider.animatingTo = Math.floor(slider.currentSlide / slider.move);
- slider.currentItem = slider.currentSlide;
- slider.slides.removeClass(namespace + "active-slide").eq(slider.currentItem).addClass(namespace + "active-slide");
- if (!msGesture) {
- slider.slides.on(eventType, function(e) {
- e.preventDefault();
- var $slide = $(this),
- target = $slide.index();
- var posFromLeft = $slide.offset().left - $(slider).scrollLeft(); // Find position of slide relative to left of slider container
- if (posFromLeft <= 0 && $slide.hasClass(namespace + 'active-slide')) {
- slider.flexAnimate(slider.getTarget("prev"), true);
- } else if (!$(slider.vars.asNavFor).data('flexslider').animating && !$slide.hasClass(namespace + "active-slide")) {
- slider.direction = (slider.currentItem < target) ? "next" : "prev";
- slider.flexAnimate(target, slider.vars.pauseOnAction, false, true, true);
- }
- });
- } else {
- el._slider = slider;
- slider.slides.each(function() {
- var that = this;
- that._gesture = new MSGesture();
- that._gesture.target = that;
- that.addEventListener("MSPointerDown", function(e) {
- e.preventDefault();
- if (e.currentTarget._gesture) {
- e.currentTarget._gesture.addPointer(e.pointerId);
- }
- }, false);
- that.addEventListener("MSGestureTap", function(e) {
- e.preventDefault();
- var $slide = $(this),
- target = $slide.index();
- if (!$(slider.vars.asNavFor).data('flexslider').animating && !$slide.hasClass('active')) {
- slider.direction = (slider.currentItem < target) ? "next" : "prev";
- slider.flexAnimate(target, slider.vars.pauseOnAction, false, true, true);
- }
- });
- });
- }
- }
- },
- controlNav: {
- setup: function() {
- if (!slider.manualControls) {
- methods.controlNav.setupPaging();
- } else { // MANUALCONTROLS:
- methods.controlNav.setupManual();
- }
- },
- setupPaging: function() {
- var type = (slider.vars.controlNav === "thumbnails") ? 'control-thumbs' : 'control-paging',
- j = 1,
- item,
- slide;
- slider.controlNavScaffold = $('<ol class="' + namespace + 'control-nav ' + namespace + type + '"></ol>');
- if (slider.pagingCount > 1) {
- for (var i = 0; i < slider.pagingCount; i++) {
- slide = slider.slides.eq(i);
- if (undefined === slide.attr('data-thumb-alt')) {
- slide.attr('data-thumb-alt', '');
- }
- var altText = ('' !== slide.attr('data-thumb-alt')) ? altText = ' alt="' + slide.attr('data-thumb-alt') + '"' : '';
- item = (slider.vars.controlNav === "thumbnails") ? '<img src="' + slide.attr( 'data-thumb' ) + '"' + altText + '/>' : '<a href="#">' + j + '</a>';
- if ('thumbnails' === slider.vars.controlNav && true === slider.vars.thumbCaptions) {
- var captn = slide.attr('data-thumbcaption');
- if ('' !== captn && undefined !== captn) {item += '<span class="' + namespace + 'caption">' + captn + '</span>';}
- }
- // slider.controlNavScaffold.append('<li>' + item + '</li>');
- slider.controlNavScaffold.append('<li>' + item + '<i></i></li>');
- j++;
- }
- }
- // CONTROLSCONTAINER:
- (slider.controlsContainer) ? $(slider.controlsContainer).append(slider.controlNavScaffold) : slider.append(slider.controlNavScaffold);
- methods.controlNav.set();
- methods.controlNav.active();
- slider.controlNavScaffold.delegate('a, img', eventType, function(event) {
- event.preventDefault();
- if (watchedEvent === "" || watchedEvent === event.type) {
- var $this = $(this),
- target = slider.controlNav.index($this);
- if (!$this.hasClass(namespace + 'active')) {
- slider.direction = (target > slider.currentSlide) ? "next" : "prev";
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- }
- }
- // setup flags to prevent event duplication
- if (watchedEvent === "") {
- watchedEvent = event.type;
- }
- methods.setToClearWatchedEvent();
- });
- },
- setupManual: function() {
- slider.controlNav = slider.manualControls;
- methods.controlNav.active();
- slider.controlNav.bind(eventType, function(event) {
- event.preventDefault();
- if (watchedEvent === "" || watchedEvent === event.type) {
- var $this = $(this),
- target = slider.controlNav.index($this);
- if (!$this.hasClass(namespace + 'active')) {
- (target > slider.currentSlide) ? slider.direction = "next" : slider.direction = "prev";
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- }
- }
- // setup flags to prevent event duplication
- if (watchedEvent === "") {
- watchedEvent = event.type;
- }
- methods.setToClearWatchedEvent();
- });
- },
- set: function() {
- var selector = (slider.vars.controlNav === "thumbnails") ? 'img' : 'a';
- slider.controlNav = $('.' + namespace + 'control-nav li ' + selector, (slider.controlsContainer) ? slider.controlsContainer : slider);
- },
- active: function() {
- slider.controlNav.removeClass(namespace + "active").eq(slider.animatingTo).addClass(namespace + "active");
- },
- update: function(action, pos) {
- if (slider.pagingCount > 1 && action === "add") {
- slider.controlNavScaffold.append($('<li><a href="#">' + slider.count + '</a></li>'));
- } else if (slider.pagingCount === 1) {
- slider.controlNavScaffold.find('li').remove();
- } else {
- slider.controlNav.eq(pos).closest('li').remove();
- }
- methods.controlNav.set();
- (slider.pagingCount > 1 && slider.pagingCount !== slider.controlNav.length) ? slider.update(pos, action) : methods.controlNav.active();
- }
- },
- directionNav: {
- setup: function() {
- var directionNavScaffold = $('<ul class="' + namespace + 'direction-nav"><li class="' + namespace + 'nav-prev"><a class="' + namespace + 'prev" href="#">' + slider.vars.prevText + '</a></li><li class="' + namespace + 'nav-next"><a class="' + namespace + 'next" href="#">' + slider.vars.nextText + '</a></li></ul>');
- // CUSTOM DIRECTION NAV:
- if (slider.customDirectionNav) {
- slider.directionNav = slider.customDirectionNav;
- } else if (slider.controlsContainer) { // CONTROLSCONTAINER:
- $(slider.controlsContainer).append(directionNavScaffold);
- slider.directionNav = $('.' + namespace + 'direction-nav li a', slider.controlsContainer);
- } else {
- slider.append(directionNavScaffold);
- slider.directionNav = $('.' + namespace + 'direction-nav li a', slider);
- }
- methods.directionNav.update();
- slider.directionNav.bind(eventType, function(event) {
- event.preventDefault();
- var target;
- if (watchedEvent === "" || watchedEvent === event.type) {
- target = ($(this).hasClass(namespace + 'next')) ? slider.getTarget('next') : slider.getTarget('prev');
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- }
- // setup flags to prevent event duplication
- if (watchedEvent === "") {
- watchedEvent = event.type;
- }
- methods.setToClearWatchedEvent();
- });
- },
- update: function() {
- var disabledClass = namespace + 'disabled';
- if (slider.pagingCount === 1) {
- slider.directionNav.addClass(disabledClass).attr('tabindex', '-1');
- } else if (!slider.vars.animationLoop) {
- if (slider.animatingTo === 0) {
- slider.directionNav.removeClass(disabledClass).filter('.' + namespace + "prev").addClass(disabledClass).attr('tabindex', '-1');
- } else if (slider.animatingTo === slider.last) {
- slider.directionNav.removeClass(disabledClass).filter('.' + namespace + "next").addClass(disabledClass).attr('tabindex', '-1');
- } else {
- slider.directionNav.removeClass(disabledClass).removeAttr('tabindex');
- }
- } else {
- slider.directionNav.removeClass(disabledClass).removeAttr('tabindex');
- }
- }
- },
- pausePlay: {
- setup: function() {
- var pausePlayScaffold = $('<div class="' + namespace + 'pauseplay"><a href="#"></a></div>');
- // CONTROLSCONTAINER:
- if (slider.controlsContainer) {
- slider.controlsContainer.append(pausePlayScaffold);
- slider.pausePlay = $('.' + namespace + 'pauseplay a', slider.controlsContainer);
- } else {
- slider.append(pausePlayScaffold);
- slider.pausePlay = $('.' + namespace + 'pauseplay a', slider);
- }
- methods.pausePlay.update((slider.vars.slideshow) ? namespace + 'pause' : namespace + 'play');
- slider.pausePlay.bind(eventType, function(event) {
- event.preventDefault();
- if (watchedEvent === "" || watchedEvent === event.type) {
- if ($(this).hasClass(namespace + 'pause')) {
- slider.manualPause = true;
- slider.manualPlay = false;
- slider.pause();
- } else {
- slider.manualPause = false;
- slider.manualPlay = true;
- slider.play();
- }
- }
- // setup flags to prevent event duplication
- if (watchedEvent === "") {
- watchedEvent = event.type;
- }
- methods.setToClearWatchedEvent();
- });
- },
- update: function(state) {
- (state === "play") ? slider.pausePlay.removeClass(namespace + 'pause').addClass(namespace + 'play').html(slider.vars.playText) : slider.pausePlay.removeClass(namespace + 'play').addClass(namespace + 'pause').html(slider.vars.pauseText);
- }
- },
- touch: function() {
- var startX,
- startY,
- offset,
- cwidth,
- dx,
- startT,
- onTouchStart,
- onTouchMove,
- onTouchEnd,
- scrolling = false,
- localX = 0,
- localY = 0,
- accDx = 0;
- if (!msGesture) {
- onTouchStart = function(e) {
- if (slider.animating) {
- e.preventDefault();
- } else if (( window.navigator.msPointerEnabled ) || e.touches.length === 1) {
- slider.pause();
- // CAROUSEL:
- cwidth = (vertical) ? slider.h : slider.w;
- startT = Number(new Date());
- // CAROUSEL:
- // Local vars for X and Y points.
- localX = e.touches[0].pageX;
- localY = e.touches[0].pageY;
- offset = (carousel && reverse && slider.animatingTo === slider.last) ? 0 :
- (carousel && reverse) ? slider.limit - (((slider.itemW + slider.vars.itemMargin) * slider.move) * slider.animatingTo) :
- (carousel && slider.currentSlide === slider.last) ? slider.limit :
- (carousel) ? ((slider.itemW + slider.vars.itemMargin) * slider.move) * slider.currentSlide :
- (reverse) ? (slider.last - slider.currentSlide + slider.cloneOffset) * cwidth : (slider.currentSlide + slider.cloneOffset) * cwidth;
- startX = (vertical) ? localY : localX;
- startY = (vertical) ? localX : localY;
- el.addEventListener('touchmove', onTouchMove, false);
- el.addEventListener('touchend', onTouchEnd, false);
- }
- };
- onTouchMove = function(e) {
- // Local vars for X and Y points.
- localX = e.touches[0].pageX;
- localY = e.touches[0].pageY;
- dx = (vertical) ? startX - localY : startX - localX;
- scrolling = (vertical) ? (Math.abs(dx) < Math.abs(localX - startY)) : (Math.abs(dx) < Math.abs(localY - startY));
- var fxms = 500;
- if (!scrolling || Number(new Date()) - startT > fxms) {
- e.preventDefault();
- if (!fade && slider.transitions) {
- if (!slider.vars.animationLoop) {
- dx = dx / ((slider.currentSlide === 0 && dx < 0 || slider.currentSlide === slider.last && dx > 0) ? (Math.abs(dx) / cwidth + 2) : 1);
- }
- slider.setProps(offset + dx, "setTouch");
- }
- }
- };
- onTouchEnd = function(e) {
- // finish the touch by undoing the touch session
- el.removeEventListener('touchmove', onTouchMove, false);
- if (slider.animatingTo === slider.currentSlide && !scrolling && !(dx === null)) {
- var updateDx = (reverse) ? -dx : dx,
- target = (updateDx > 0) ? slider.getTarget('next') : slider.getTarget('prev');
- if (slider.canAdvance(target) && (Number(new Date()) - startT < 550 && Math.abs(updateDx) > 50 || Math.abs(updateDx) > cwidth / 2)) {
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- } else {
- if (!fade) {slider.flexAnimate(slider.currentSlide, slider.vars.pauseOnAction, true);}
- }
- }
- el.removeEventListener('touchend', onTouchEnd, false);
- startX = null;
- startY = null;
- dx = null;
- offset = null;
- };
- el.addEventListener('touchstart', onTouchStart, false);
- } else {
- el.style.msTouchAction = "none";
- el._gesture = new MSGesture();
- el._gesture.target = el;
- el.addEventListener("MSPointerDown", onMSPointerDown, false);
- el._slider = slider;
- el.addEventListener("MSGestureChange", onMSGestureChange, false);
- el.addEventListener("MSGestureEnd", onMSGestureEnd, false);
- function onMSPointerDown(e) {
- e.stopPropagation();
- if (slider.animating) {
- e.preventDefault();
- } else {
- slider.pause();
- el._gesture.addPointer(e.pointerId);
- accDx = 0;
- cwidth = (vertical) ? slider.h : slider.w;
- startT = Number(new Date());
- // CAROUSEL:
- offset = (carousel && reverse && slider.animatingTo === slider.last) ? 0 :
- (carousel && reverse) ? slider.limit - (((slider.itemW + slider.vars.itemMargin) * slider.move) * slider.animatingTo) :
- (carousel && slider.currentSlide === slider.last) ? slider.limit :
- (carousel) ? ((slider.itemW + slider.vars.itemMargin) * slider.move) * slider.currentSlide :
- (reverse) ? (slider.last - slider.currentSlide + slider.cloneOffset) * cwidth : (slider.currentSlide + slider.cloneOffset) * cwidth;
- }
- }
- function onMSGestureChange(e) {
- e.stopPropagation();
- var slider = e.target._slider;
- if (!slider) {
- return;
- }
- var transX = -e.translationX,
- transY = -e.translationY;
- //Accumulate translations.
- accDx = accDx + ((vertical) ? transY : transX);
- dx = accDx;
- scrolling = (vertical) ? (Math.abs(accDx) < Math.abs(-transX)) : (Math.abs(accDx) < Math.abs(-transY));
- if (e.detail === e.MSGESTURE_FLAG_INERTIA) {
- setImmediate(function() {
- el._gesture.stop();
- });
- return;
- }
- if (!scrolling || Number(new Date()) - startT > 500) {
- e.preventDefault();
- if (!fade && slider.transitions) {
- if (!slider.vars.animationLoop) {
- dx = accDx / ((slider.currentSlide === 0 && accDx < 0 || slider.currentSlide === slider.last && accDx > 0) ? (Math.abs(accDx) / cwidth + 2) : 1);
- }
- slider.setProps(offset + dx, "setTouch");
- }
- }
- }
- function onMSGestureEnd(e) {
- e.stopPropagation();
- var slider = e.target._slider;
- if (!slider) {
- return;
- }
- if (slider.animatingTo === slider.currentSlide && !scrolling && !(dx === null)) {
- var updateDx = (reverse) ? -dx : dx,
- target = (updateDx > 0) ? slider.getTarget('next') : slider.getTarget('prev');
- if (slider.canAdvance(target) && (Number(new Date()) - startT < 550 && Math.abs(updateDx) > 50 || Math.abs(updateDx) > cwidth / 2)) {
- slider.flexAnimate(target, slider.vars.pauseOnAction);
- } else {
- if (!fade) {slider.flexAnimate(slider.currentSlide, slider.vars.pauseOnAction, true);}
- }
- }
- startX = null;
- startY = null;
- dx = null;
- offset = null;
- accDx = 0;
- }
- }
- },
- resize: function() {
- if (!slider.animating && slider.is(':visible')) {
- if (!carousel) {slider.doMath()};
- if (fade) {
- // SMOOTH HEIGHT:
- methods.smoothHeight();
- } else if (carousel) { //CAROUSEL:
- slider.slides.width(slider.computedW);
- slider.update(slider.pagingCount);
- slider.setProps();
- }
- else if (vertical) { //VERTICAL:
- slider.viewport.height(slider.h);
- slider.setProps(slider.h, "setTotal");
- } else {
- // SMOOTH HEIGHT:
- if (slider.vars.smoothHeight) {methods.smoothHeight();}
- slider.newSlides.width(slider.computedW);
- slider.setProps(slider.computedW, "setTotal");
- }
- }
- },
- smoothHeight: function(dur) {
- if (!vertical || fade) {
- var $obj = (fade) ? slider : slider.viewport;
- (dur) ? $obj.animate({"height": slider.slides.eq(slider.animatingTo).innerHeight()}, dur) : $obj.innerHeight(slider.slides.eq(slider.animatingTo).innerHeight());
- }
- },
- sync: function(action) {
- var $obj = $(slider.vars.sync).data("flexslider"),
- target = slider.animatingTo;
- switch (action) {
- case "animate":
- $obj.flexAnimate(target, slider.vars.pauseOnAction, false, true);
- break;
- case "play":
- if (!$obj.playing && !$obj.asNav) {
- $obj.play();
- }
- break;
- case "pause":
- $obj.pause();
- break;
- }
- },
- uniqueID: function($clone) {
- // Append _clone to current level and children elements with id attributes
- $clone.filter('[id]').add($clone.find('[id]')).each(function() {
- var $this = $(this);
- $this.attr('id', $this.attr('id') + '_clone');
- });
- return $clone;
- },
- pauseInvisible: {
- visProp: null,
- init: function() {
- var visProp = methods.pauseInvisible.getHiddenProp();
- if (visProp) {
- var evtname = visProp.replace(/[H|h]idden/,'') + 'visibilitychange';
- document.addEventListener(evtname, function() {
- if (methods.pauseInvisible.isHidden()) {
- if(slider.startTimeout) {
- clearTimeout(slider.startTimeout); //If clock is ticking, stop timer and prevent from starting while invisible
- } else {
- slider.pause(); //Or just pause
- }
- }
- else {
- if(slider.started) {
- slider.play(); //Initiated before, just play
- } else {
- if (slider.vars.initDelay > 0) {
- setTimeout(slider.play, slider.vars.initDelay);
- } else {
- slider.play(); //Didn't init before: simply init or wait for it
- }
- }
- }
- });
- }
- },
- isHidden: function() {
- var prop = methods.pauseInvisible.getHiddenProp();
- if (!prop) {
- return false;
- }
- return document[prop];
- },
- getHiddenProp: function() {
- var prefixes = ['webkit','moz','ms','o'];
- // if 'hidden' is natively supported just return it
- if ('hidden' in document) {
- return 'hidden';
- }
- // otherwise loop over all the known prefixes until we find one
- for (var i = 0; i < prefixes.length; i++ ) {
- if ((prefixes[i] + 'Hidden') in document) {
- return prefixes[i] + 'Hidden';
- }
- }
- // otherwise it's not supported
- return null;
- }
- },
- setToClearWatchedEvent: function() {
- clearTimeout(watchedEventClearTimer);
- watchedEventClearTimer = setTimeout(function() {
- watchedEvent = "";
- }, 3000);
- }
- };
- // public methods
- slider.flexAnimate = function(target, pause, override, withSync, fromNav) {
- if (!slider.vars.animationLoop && target !== slider.currentSlide) {
- slider.direction = (target > slider.currentSlide) ? "next" : "prev";
- }
- if (asNav && slider.pagingCount === 1) slider.direction = (slider.currentItem < target) ? "next" : "prev";
- if (!slider.animating && (slider.canAdvance(target, fromNav) || override) && slider.is(":visible")) {
- if (asNav && withSync) {
- var master = $(slider.vars.asNavFor).data('flexslider');
- slider.atEnd = target === 0 || target === slider.count - 1;
- master.flexAnimate(target, true, false, true, fromNav);
- slider.direction = (slider.currentItem < target) ? "next" : "prev";
- master.direction = slider.direction;
- if (Math.ceil((target + 1) / slider.visible) - 1 !== slider.currentSlide && target !== 0) {
- slider.currentItem = target;
- slider.slides.removeClass(namespace + "active-slide").eq(target).addClass(namespace + "active-slide");
- target = Math.floor(target / slider.visible);
- } else {
- slider.currentItem = target;
- slider.slides.removeClass(namespace + "active-slide").eq(target).addClass(namespace + "active-slide");
- return false;
- }
- }
- slider.animating = true;
- slider.animatingTo = target;
- // SLIDESHOW:
- if (pause) {slider.pause();}
- // API: before() animation Callback
- slider.vars.before(slider);
- // SYNC:
- if (slider.syncExists && !fromNav) {methods.sync("animate");}
- // CONTROLNAV
- if (slider.vars.controlNav) {methods.controlNav.active();}
- // !CAROUSEL:
- // CANDIDATE: slide active class (for add/remove slide)
- if (!carousel) {slider.slides.removeClass(namespace + 'active-slide').eq(target).addClass(namespace + 'active-slide');}
- // INFINITE LOOP:
- // CANDIDATE: atEnd
- slider.atEnd = target === 0 || target === slider.last;
- // DIRECTIONNAV:
- if (slider.vars.directionNav) {methods.directionNav.update();}
- if (target === slider.last) {
- // API: end() of cycle Callback
- slider.vars.end(slider);
- // SLIDESHOW && !INFINITE LOOP:
- if (!slider.vars.animationLoop) {slider.pause();}
- }
- // SLIDE:
- if (!fade) {
- var dimension = (vertical) ? slider.slides.filter(':first').height() : slider.computedW,
- margin, slideString, calcNext;
- // INFINITE LOOP / REVERSE:
- if (carousel) {
- //margin = (slider.vars.itemWidth > slider.w) ? slider.vars.itemMargin * 2 : slider.vars.itemMargin;
- margin = slider.vars.itemMargin;
- calcNext = ((slider.itemW + margin) * slider.move) * slider.animatingTo;
- slideString = (calcNext > slider.limit && slider.visible !== 1) ? slider.limit : calcNext;
- } else if (slider.currentSlide === 0 && target === slider.count - 1 && slider.vars.animationLoop && slider.direction !== "next") {
- slideString = (reverse) ? (slider.count + slider.cloneOffset) * dimension : 0;
- } else if (slider.currentSlide === slider.last && target === 0 && slider.vars.animationLoop && slider.direction !== "prev") {
- slideString = (reverse) ? 0 : (slider.count + 1) * dimension;
- } else {
- slideString = (reverse) ? ((slider.count - 1) - target + slider.cloneOffset) * dimension : (target + slider.cloneOffset) * dimension;
- }
- slider.setProps(slideString, "", slider.vars.animationSpeed);
- if (slider.transitions) {
- if (!slider.vars.animationLoop || !slider.atEnd) {
- slider.animating = false;
- slider.currentSlide = slider.animatingTo;
- }
- // Unbind previous transitionEnd events and re-bind new transitionEnd event
- slider.container.unbind("webkitTransitionEnd transitionend");
- slider.container.bind("webkitTransitionEnd transitionend", function() {
- clearTimeout(slider.ensureAnimationEnd);
- slider.wrapup(dimension);
- });
- // Insurance for the ever-so-fickle transitionEnd event
- clearTimeout(slider.ensureAnimationEnd);
- slider.ensureAnimationEnd = setTimeout(function() {
- slider.wrapup(dimension);
- }, slider.vars.animationSpeed + 100);
- } else {
- slider.container.animate(slider.args, slider.vars.animationSpeed, slider.vars.easing, function(){
- slider.wrapup(dimension);
- });
- }
- } else { // FADE:
- if (!touch) {
- //slider.slides.eq(slider.currentSlide).fadeOut(slider.vars.animationSpeed, slider.vars.easing);
- //slider.slides.eq(target).fadeIn(slider.vars.animationSpeed, slider.vars.easing, slider.wrapup);
- slider.slides.eq(slider.currentSlide).css({"zIndex": 1}).animate({"opacity": 0}, slider.vars.animationSpeed, slider.vars.easing);
- slider.slides.eq(target).css({"zIndex": 2}).animate({"opacity": 1}, slider.vars.animationSpeed, slider.vars.easing, slider.wrapup);
- } else {
- slider.slides.eq(slider.currentSlide).css({
- "opacity": 0,
- "zIndex": 1
- });
- slider.slides.eq(target).css({"opacity": 1, "zIndex": 2});
- slider.wrapup(dimension);
- }
- }
- // SMOOTH HEIGHT:
- if (slider.vars.smoothHeight) {methods.smoothHeight(slider.vars.animationSpeed)};
- }
- };
- slider.wrapup = function(dimension) {
- // SLIDE:
- if (!fade && !carousel) {
- if (slider.currentSlide === 0 && slider.animatingTo === slider.last && slider.vars.animationLoop) {
- slider.setProps(dimension, "jumpEnd");
- } else if (slider.currentSlide === slider.last && slider.animatingTo === 0 && slider.vars.animationLoop) {
- slider.setProps(dimension, "jumpStart");
- }
- }
- slider.animating = false;
- slider.currentSlide = slider.animatingTo;
- // API: after() animation Callback
- slider.vars.after(slider);
- };
- // SLIDESHOW:
- slider.animateSlides = function() {
- if (!slider.animating && focused) {slider.flexAnimate(slider.getTarget("next"));}
- };
- // SLIDESHOW:
- slider.pause = function() {
- clearInterval(slider.animatedSlides);
- slider.animatedSlides = null;
- slider.playing = false;
- // PAUSEPLAY:
- if (slider.vars.pausePlay) {methods.pausePlay.update("play");}
- // SYNC:
- if (slider.syncExists) {methods.sync("pause");}
- };
- // SLIDESHOW:
- slider.play = function() {
- if (slider.playing) {clearInterval(slider.animatedSlides);}
- slider.animatedSlides = slider.animatedSlides || setInterval(slider.animateSlides, slider.vars.slideshowSpeed);
- slider.started = slider.playing = true;
- // PAUSEPLAY:
- if (slider.vars.pausePlay) {methods.pausePlay.update("pause");}
- // SYNC:
- if (slider.syncExists) {methods.sync("play");}
- };
- // STOP:
- slider.stop = function() {
- slider.pause();
- slider.stopped = true;
- };
- slider.canAdvance = function(target, fromNav) {
- // ASNAV:
- var last = (asNav) ? slider.pagingCount - 1 : slider.last;
- return (fromNav) ? true :
- (asNav && slider.currentItem === slider.count - 1 && target === 0 && slider.direction === "prev") ? true :
- (asNav && slider.currentItem === 0 && target === slider.pagingCount - 1 && slider.direction !== "next") ? false :
- (target === slider.currentSlide && !asNav) ? false :
- (slider.vars.animationLoop) ? true :
- (slider.atEnd && slider.currentSlide === 0 && target === last && slider.direction !== "next") ? false :
- (slider.atEnd && slider.currentSlide === last && target === 0 && slider.direction === "next") ? false :
- true;
- };
- slider.getTarget = function(dir) {
- slider.direction = dir;
- if (dir === "next") {
- return (slider.currentSlide === slider.last) ? 0 : slider.currentSlide + 1;
- } else {
- return (slider.currentSlide === 0) ? slider.last : slider.currentSlide - 1;
- }
- };
- // SLIDE:
- slider.setProps = function(pos, special, dur) {
- var target = (function() {
- var posCheck = (pos) ? pos : ((slider.itemW + slider.vars.itemMargin) * slider.move) * slider.animatingTo,
- posCalc = (function() {
- if (carousel) {
- return (special === "setTouch") ? pos :
- (reverse && slider.animatingTo === slider.last) ? 0 :
- (reverse) ? slider.limit - (((slider.itemW + slider.vars.itemMargin) * slider.move) * slider.animatingTo) :
- (slider.animatingTo === slider.last) ? slider.limit : posCheck;
- } else {
- switch (special) {
- case "setTotal":
- return (reverse) ? ((slider.count - 1) - slider.currentSlide + slider.cloneOffset) * pos : (slider.currentSlide + slider.cloneOffset) * pos;
- case "setTouch":
- return (reverse) ? pos : pos;
- case "jumpEnd":
- return (reverse) ? pos : slider.count * pos;
- case "jumpStart":
- return (reverse) ? slider.count * pos : pos;
- default:
- return pos;
- }
- }
- }());
- return (posCalc * -1) + "px";
- }());
- if (slider.transitions) {
- target = (vertical) ? "translate3d(0," + target + ",0)" : "translate3d(" + target + ",0,0)";
- dur = (dur !== undefined) ? (dur / 1000) + "s" : "0s";
- slider.container.css("-" + slider.pfx + "-transition-duration", dur);
- slider.container.css("transition-duration", dur);
- }
- slider.args[slider.prop] = target;
- if (slider.transitions || dur === undefined) {slider.container.css(slider.args);}
- slider.container.css('transform', target);
- };
- slider.setup = function(type) {
- // SLIDE:
- if (!fade) {
- var sliderOffset, arr;
- if (type === "init") {
- slider.viewport = $('<div class="' + namespace + 'viewport"></div>').css({
- "overflow": "hidden",
- "position": "relative"
- }).appendTo(slider).append(slider.container);
- // INFINITE LOOP:
- slider.cloneCount = 0;
- slider.cloneOffset = 0;
- // REVERSE:
- if (reverse) {
- arr = $.makeArray(slider.slides).reverse();
- slider.slides = $(arr);
- slider.container.empty().append(slider.slides);
- }
- }
- // INFINITE LOOP && !CAROUSEL:
- if (slider.vars.animationLoop && !carousel) {
- slider.cloneCount = 2;
- slider.cloneOffset = 1;
- // clear out old clones
- if (type !== "init") { slider.container.find('.clone').remove(); }
- slider.container.append(methods.uniqueID(slider.slides.first().clone().addClass('clone')).attr('aria-hidden', 'true'))
- .prepend(methods.uniqueID(slider.slides.last().clone().addClass('clone')).attr('aria-hidden', 'true'));
- }
- slider.newSlides = $(slider.vars.selector, slider);
- sliderOffset = (reverse) ? slider.count - 1 - slider.currentSlide + slider.cloneOffset : slider.currentSlide + slider.cloneOffset;
- // VERTICAL:
- if (vertical && !carousel) {
- slider.container.height((slider.count + slider.cloneCount) * 200 + "%").css("position", "absolute").width("100%");
- setTimeout(function() {
- slider.newSlides.css({"display": "block"});
- slider.doMath();
- slider.viewport.height(slider.h);
- slider.setProps(sliderOffset * slider.h, "init");
- }, (type === "init") ? 100 : 0);
- } else {
- slider.container.width((slider.count + slider.cloneCount) * 200 + "%");
- slider.setProps(sliderOffset * slider.computedW, "init");
- setTimeout(function() {
- slider.doMath();
- slider.newSlides.css({"width": slider.computedW, "marginRight" : slider.computedM, "float": "left", "display": "block"});
- // SMOOTH HEIGHT:
- if (slider.vars.smoothHeight) {methods.smoothHeight();}
- }, (type === "init") ? 100 : 0);
- }
- } else { // FADE:
- slider.slides.css({
- "width": "100%",
- "float": "left",
- "marginRight": "-100%",
- "position": "relative"
- });
- if (type === "init") {
- if (!touch) {
- //slider.slides.eq(slider.currentSlide).fadeIn(slider.vars.animationSpeed, slider.vars.easing);
- if (slider.vars.fadeFirstSlide == false) {
- slider.slides.css({ "opacity": 0, "display": "block", "zIndex": 1 }).eq(slider.currentSlide).css({"zIndex": 2}).css({"opacity": 1});
- } else {
- slider.slides.css({ "opacity": 0, "display": "block", "zIndex": 1 }).eq(slider.currentSlide).css({"zIndex": 2}).animate({"opacity": 1},slider.vars.animationSpeed,slider.vars.easing);
- }
- } else {
- slider.slides.css({ "opacity": 0, "display": "block", "webkitTransition": "opacity " + slider.vars.animationSpeed / 1000 + "s ease", "zIndex": 1 }).eq(slider.currentSlide).css({ "opacity": 1, "zIndex": 2});
- }
- }
- // SMOOTH HEIGHT:
- if (slider.vars.smoothHeight) {methods.smoothHeight();}
- }
- // !CAROUSEL:
- // CANDIDATE: active slide
- if (!carousel) {slider.slides.removeClass(namespace + "active-slide").eq(slider.currentSlide).addClass(namespace + "active-slide");}
- //FlexSlider: init() Callback
- slider.vars.init(slider);
- };
- slider.doMath = function() {
- var slide = slider.slides.first(),
- slideMargin = slider.vars.itemMargin,
- minItems = slider.vars.minItems,
- maxItems = slider.vars.maxItems;
- slider.w = (slider.viewport === undefined) ? slider.width() : slider.viewport.width();
- slider.h = slide.height();
- slider.boxPadding = slide.outerWidth() - slide.width();
- // CAROUSEL:
- if (carousel) {
- slider.itemT = slider.vars.itemWidth + slideMargin;
- slider.itemM = slideMargin;
- slider.minW = (minItems) ? minItems * slider.itemT : slider.w;
- slider.maxW = (maxItems) ? (maxItems * slider.itemT) - slideMargin : slider.w;
- slider.itemW = (slider.minW > slider.w) ? (slider.w - (slideMargin * (minItems - 1))) / minItems :
- (slider.maxW < slider.w) ? (slider.w - (slideMargin * (maxItems - 1))) / maxItems :
- (slider.vars.itemWidth > slider.w) ? slider.w : slider.vars.itemWidth;
- slider.visible = Math.floor(slider.w / (slider.itemW));
- slider.move = (slider.vars.move > 0 && slider.vars.move < slider.visible ) ? slider.vars.move : slider.visible;
- slider.pagingCount = Math.ceil(((slider.count - slider.visible) / slider.move) + 1);
- slider.last = slider.pagingCount - 1;
- slider.limit = (slider.pagingCount === 1) ? 0 :
- (slider.vars.itemWidth > slider.w) ? (slider.itemW * (slider.count - 1)) + (slideMargin * (slider.count - 1)) : ((slider.itemW + slideMargin) * slider.count) - slider.w - slideMargin;
- } else {
- slider.itemW = slider.w;
- slider.itemM = slideMargin;
- slider.pagingCount = slider.count;
- slider.last = slider.count - 1;
- }
- slider.computedW = slider.itemW - slider.boxPadding;
- slider.computedM = slider.itemM;
- };
- slider.update = function(pos, action) {
- slider.doMath();
- // update currentSlide and slider.animatingTo if necessary
- if (!carousel) {
- if (pos < slider.currentSlide) {
- slider.currentSlide += 1;
- } else if (pos <= slider.currentSlide && pos !== 0) {
- slider.currentSlide -= 1;
- }
- slider.animatingTo = slider.currentSlide;
- }
- // update controlNav
- if (slider.vars.controlNav && !slider.manualControls) {
- if ((action === "add" && !carousel) || slider.pagingCount > slider.controlNav.length) {
- methods.controlNav.update("add");
- } else if ((action === "remove" && !carousel) || slider.pagingCount < slider.controlNav.length) {
- if (carousel && slider.currentSlide > slider.last) {
- slider.currentSlide -= 1;
- slider.animatingTo -= 1;
- }
- methods.controlNav.update("remove", slider.last);
- }
- }
- // update directionNav
- if (slider.vars.directionNav) {methods.directionNav.update();}
- };
- slider.addSlide = function(obj, pos) {
- var $obj = $(obj);
- slider.count += 1;
- slider.last = slider.count - 1;
- // append new slide
- if (vertical && reverse) {
- (pos !== undefined) ? slider.slides.eq(slider.count - pos).after($obj) : slider.container.prepend($obj);
- } else {
- (pos !== undefined) ? slider.slides.eq(pos).before($obj) : slider.container.append($obj);
- }
- // update currentSlide, animatingTo, controlNav, and directionNav
- slider.update(pos, "add");
- // update slider.slides
- slider.slides = $(slider.vars.selector + ':not(.clone)', slider);
- // re-setup the slider to accomdate new slide
- slider.setup();
- //FlexSlider: added() Callback
- slider.vars.added(slider);
- };
- slider.removeSlide = function(obj) {
- var pos = (isNaN(obj)) ? slider.slides.index($(obj)) : obj;
- // update count
- slider.count -= 1;
- slider.last = slider.count - 1;
- // remove slide
- if (isNaN(obj)) {
- $(obj, slider.slides).remove();
- } else {
- (vertical && reverse) ? slider.slides.eq(slider.last).remove() : slider.slides.eq(obj).remove();
- }
- // update currentSlide, animatingTo, controlNav, and directionNav
- slider.doMath();
- slider.update(pos, "remove");
- // update slider.slides
- slider.slides = $(slider.vars.selector + ':not(.clone)', slider);
- // re-setup the slider to accomdate new slide
- slider.setup();
- // FlexSlider: removed() Callback
- slider.vars.removed(slider);
- };
- //FlexSlider: Initialize
- methods.init();
- };
- // Ensure the slider isn't focussed if the window loses focus.
- $(window).blur(function(e) {
- focused = false;
- }).focus(function(e) {
- focused = true;
- });
- // FlexSlider: Default Settings
- $.flexslider.defaults = {
- namespace: 'am-', // {NEW} String: Prefix string attached to the class of every element generated by the plugin
- selector: '.am-slides > li', // {NEW} Selector: Must match a simple pattern. '{container} > {slide}' -- Ignore pattern at your own peril
- animation: 'slide', // String: Select your animation type, 'fade' or 'slide'
- easing: 'swing', // {NEW} String: Determines the easing method used in jQuery transitions. jQuery easing plugin is supported!
- direction: 'horizontal', // String: Select the sliding direction, "horizontal" or "vertical"
- reverse: false, // {NEW} Boolean: Reverse the animation direction
- animationLoop: true, // Boolean: Should the animation loop? If false, directionNav will received "disable" classes at either end
- smoothHeight: false, // {NEW} Boolean: Allow height of the slider to animate smoothly in horizontal mode
- startAt: 0, // Integer: The slide that the slider should start on. Array notation (0 = first slide)
- slideshow: true, // Boolean: Animate slider automatically
- slideshowSpeed: 5000, // Integer: Set the speed of the slideshow cycling, in milliseconds
- animationSpeed: 600, // Integer: Set the speed of animations, in milliseconds
- initDelay: 0, // {NEW} Integer: Set an initialization delay, in milliseconds
- randomize: false, // Boolean: Randomize slide order
- fadeFirstSlide: true, // Boolean: Fade in the first slide when animation type is "fade"
- thumbCaptions: false, // Boolean: Whether or not to put captions on thumbnails when using the "thumbnails" controlNav.
- // Usability features
- pauseOnAction: true, // Boolean: Pause the slideshow when interacting with control elements, highly recommended.
- pauseOnHover: false, // Boolean: Pause the slideshow when hovering over slider, then resume when no longer hovering
- pauseInvisible: true, // {NEW} Boolean: Pause the slideshow when tab is invisible, resume when visible. Provides better UX, lower CPU usage.
- useCSS: true, // {NEW} Boolean: Slider will use CSS3 transitions if available
- touch: true, // {NEW} Boolean: Allow touch swipe navigation of the slider on touch-enabled devices
- video: false, // {NEW} Boolean: If using video in the slider, will prevent CSS3 3D Transforms to avoid graphical glitches
- // Primary Controls
- controlNav: true, // Boolean: Create navigation for paging control of each slide? Note: Leave true for manualControls usage
- directionNav: true, // Boolean: Create navigation for previous/next navigation? (true/false)
- prevText: ' ', // String: Set the text for the "previous" directionNav item
- nextText: ' ', // String: Set the text for the "next" directionNav item
- // Secondary Navigation
- keyboard: true, // Boolean: Allow slider navigating via keyboard left/right keys
- multipleKeyboard: false, // {NEW} Boolean: Allow keyboard navigation to affect multiple sliders. Default behavior cuts out keyboard navigation with more than one slider present.
- mousewheel: false, // {UPDATED} Boolean: Requires jquery.mousewheel.js (https://github.com/brandonaaron/jquery-mousewheel) - Allows slider navigating via mousewheel
- pausePlay: false, // Boolean: Create pause/play dynamic element
- pauseText: 'Pause', // String: Set the text for the 'pause' pausePlay item
- playText: 'Play', // String: Set the text for the 'play' pausePlay item
- // Special properties
- controlsContainer: '', // {UPDATED} jQuery Object/Selector: Declare which container the navigation elements should be appended too. Default container is the FlexSlider element. Example use would be $('.flexslider-container'). Property is ignored if given element is not found.
- manualControls: '', // {UPDATED} jQuery Object/Selector: Declare custom control navigation. Examples would be $(".flex-control-nav li") or "#tabs-nav li img", etc. The number of elements in your controlNav should match the number of slides/tabs.
- customDirectionNav: '', // {NEW} jQuery Object/Selector: Custom prev / next button. Must be two jQuery elements. In order to make the events work they have to have the classes "prev" and "next" (plus namespace)
- sync: '', // {NEW} Selector: Mirror the actions performed on this slider with another slider. Use with care.
- asNavFor: '', // {NEW} Selector: Internal property exposed for turning the slider into a thumbnail navigation for another slider
- // Carousel Options
- itemWidth: 0, // {NEW} Integer: Box-model width of individual carousel items, including horizontal borders and padding.
- itemMargin: 0, // {NEW} Integer: Margin between carousel items.
- minItems: 1, // {NEW} Integer: Minimum number of carousel items that should be visible. Items will resize fluidly when below this.
- maxItems: 0, // {NEW} Integer: Maxmimum number of carousel items that should be visible. Items will resize fluidly when above this limit.
- move: 0, // {NEW} Integer: Number of carousel items that should move on animation. If 0, slider will move all visible items.
- allowOneSlide: true, // {NEW} Boolean: Whether or not to allow a slider comprised of a single slide
- // Callback API
- start: function() {
- }, // Callback: function(slider) - Fires when the slider loads the first slide
- before: function() {
- }, // Callback: function(slider) - Fires asynchronously with each slider animation
- after: function() {
- }, // Callback: function(slider) - Fires after each slider animation completes
- end: function() {
- }, // Callback: function(slider) - Fires when the slider reaches the last slide (asynchronous)
- added: function() {
- }, // {NEW} Callback: function(slider) - Fires after a slide is added
- removed: function() {
- }, // {NEW} Callback: function(slider) - Fires after a slide is removed
- init: function() {
- } // {NEW} Callback: function(slider) - Fires after the slider is initially setup
- };
- // FlexSlider: Plugin Function
- $.fn.flexslider = function(options) {
- var args = Array.prototype.slice.call(arguments, 1);
- if (options === undefined) {options = {};}
- if (typeof options === 'object') {
- return this.each(function() {
- var $this = $(this);
- var selector = (options.selector) ? options.selector : '.am-slides > li';
- var $slides = $this.find(selector);
- if (( $slides.length === 1 && options.allowOneSlide === false) || $slides.length === 0) {
- $slides.fadeIn(400);
- if (options.start) {options.start($this);}
- } else if ($this.data('flexslider') === undefined) {
- new $.flexslider(this, options);
- }
- });
- } else {
- // Helper strings to quickly pecdrform functions on the slider
- var $slider = $(this).data('flexslider');
- var methodReturn;
- switch (options) {
- case 'next':
- $slider.flexAnimate($slider.getTarget('next'), true);
- break;
- case 'prev':
- case 'previous':
- $slider.flexAnimate($slider.getTarget('prev'), true);
- break;
- default:
- if (typeof options === 'number') {
- $slider.flexAnimate(options, true);
- } else if (typeof options === 'string') {
- methodReturn = (typeof $slider[options] === 'function') ?
- $slider[options].apply($slider, args) : $slider[options];
- }
- }
- return methodReturn === undefined ? this : methodReturn;
- }
- };
- // Init code
- UI.ready(function(context) {
- $('[data-am-flexslider]', context).each(function(i, item) {
- var $slider = $(item);
- var options = UI.utils.parseOptions($slider.data('amFlexslider'));
- options.before = function(slider) {
- if (slider._pausedTimer) {
- window.clearTimeout(slider._pausedTimer);
- slider._pausedTimer = null;
- }
- };
- options.after = function(slider) {
- var pauseTime = slider.vars.playAfterPaused;
- if (pauseTime && !isNaN(pauseTime) && !slider.playing) {
- if (!slider.manualPause && !slider.manualPlay && !slider.stopped) {
- slider._pausedTimer = window.setTimeout(function() {
- slider.play();
- }, pauseTime);
- }
- }
- };
- $slider.flexslider(options);
- });
- });
- module.exports = $.flexslider;
- /* WEBPACK VAR INJECTION */}.call(exports, __webpack_require__(12).setImmediate))
- /***/ },
- /* 12 */
- /***/ function(module, exports, __webpack_require__) {
- /* WEBPACK VAR INJECTION */(function(setImmediate, clearImmediate) {var nextTick = __webpack_require__(13).nextTick;
- var apply = Function.prototype.apply;
- var slice = Array.prototype.slice;
- var immediateIds = {};
- var nextImmediateId = 0;
- // DOM APIs, for completeness
- exports.setTimeout = function() {
- return new Timeout(apply.call(setTimeout, window, arguments), clearTimeout);
- };
- exports.setInterval = function() {
- return new Timeout(apply.call(setInterval, window, arguments), clearInterval);
- };
- exports.clearTimeout =
- exports.clearInterval = function(timeout) { timeout.close(); };
- function Timeout(id, clearFn) {
- this._id = id;
- this._clearFn = clearFn;
- }
- Timeout.prototype.unref = Timeout.prototype.ref = function() {};
- Timeout.prototype.close = function() {
- this._clearFn.call(window, this._id);
- };
- // Does not start the time, just sets up the members needed.
- exports.enroll = function(item, msecs) {
- clearTimeout(item._idleTimeoutId);
- item._idleTimeout = msecs;
- };
- exports.unenroll = function(item) {
- clearTimeout(item._idleTimeoutId);
- item._idleTimeout = -1;
- };
- exports._unrefActive = exports.active = function(item) {
- clearTimeout(item._idleTimeoutId);
- var msecs = item._idleTimeout;
- if (msecs >= 0) {
- item._idleTimeoutId = setTimeout(function onTimeout() {
- if (item._onTimeout)
- item._onTimeout();
- }, msecs);
- }
- };
- // That's not how node.js implements it but the exposed api is the same.
- exports.setImmediate = typeof setImmediate === "function" ? setImmediate : function(fn) {
- var id = nextImmediateId++;
- var args = arguments.length < 2 ? false : slice.call(arguments, 1);
- immediateIds[id] = true;
- nextTick(function onNextTick() {
- if (immediateIds[id]) {
- // fn.call() is faster so we optimize for the common use-case
- // @see http://jsperf.com/call-apply-segu
- if (args) {
- fn.apply(null, args);
- } else {
- fn.call(null);
- }
- // Prevent ids from leaking
- exports.clearImmediate(id);
- }
- });
- return id;
- };
- exports.clearImmediate = typeof clearImmediate === "function" ? clearImmediate : function(id) {
- delete immediateIds[id];
- };
- /* WEBPACK VAR INJECTION */}.call(exports, __webpack_require__(12).setImmediate, __webpack_require__(12).clearImmediate))
- /***/ },
- /* 13 */
- /***/ function(module, exports) {
- // shim for using process in browser
- var process = module.exports = {};
- // cached from whatever global is present so that test runners that stub it
- // don't break things. But we need to wrap it in a try catch in case it is
- // wrapped in strict mode code which doesn't define any globals. It's inside a
- // function because try/catches deoptimize in certain engines.
- var cachedSetTimeout;
- var cachedClearTimeout;
- function defaultSetTimout() {
- throw new Error('setTimeout has not been defined');
- }
- function defaultClearTimeout () {
- throw new Error('clearTimeout has not been defined');
- }
- (function () {
- try {
- if (typeof setTimeout === 'function') {
- cachedSetTimeout = setTimeout;
- } else {
- cachedSetTimeout = defaultSetTimout;
- }
- } catch (e) {
- cachedSetTimeout = defaultSetTimout;
- }
- try {
- if (typeof clearTimeout === 'function') {
- cachedClearTimeout = clearTimeout;
- } else {
- cachedClearTimeout = defaultClearTimeout;
- }
- } catch (e) {
- cachedClearTimeout = defaultClearTimeout;
- }
- } ())
- function runTimeout(fun) {
- if (cachedSetTimeout === setTimeout) {
- //normal enviroments in sane situations
- return setTimeout(fun, 0);
- }
- // if setTimeout wasn't available but was latter defined
- if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {
- cachedSetTimeout = setTimeout;
- return setTimeout(fun, 0);
- }
- try {
- // when when somebody has screwed with setTimeout but no I.E. maddness
- return cachedSetTimeout(fun, 0);
- } catch(e){
- try {
- // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
- return cachedSetTimeout.call(null, fun, 0);
- } catch(e){
- // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error
- return cachedSetTimeout.call(this, fun, 0);
- }
- }
- }
- function runClearTimeout(marker) {
- if (cachedClearTimeout === clearTimeout) {
- //normal enviroments in sane situations
- return clearTimeout(marker);
- }
- // if clearTimeout wasn't available but was latter defined
- if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {
- cachedClearTimeout = clearTimeout;
- return clearTimeout(marker);
- }
- try {
- // when when somebody has screwed with setTimeout but no I.E. maddness
- return cachedClearTimeout(marker);
- } catch (e){
- try {
- // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
- return cachedClearTimeout.call(null, marker);
- } catch (e){
- // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.
- // Some versions of I.E. have different rules for clearTimeout vs setTimeout
- return cachedClearTimeout.call(this, marker);
- }
- }
- }
- var queue = [];
- var draining = false;
- var currentQueue;
- var queueIndex = -1;
- function cleanUpNextTick() {
- if (!draining || !currentQueue) {
- return;
- }
- draining = false;
- if (currentQueue.length) {
- queue = currentQueue.concat(queue);
- } else {
- queueIndex = -1;
- }
- if (queue.length) {
- drainQueue();
- }
- }
- function drainQueue() {
- if (draining) {
- return;
- }
- var timeout = runTimeout(cleanUpNextTick);
- draining = true;
- var len = queue.length;
- while(len) {
- currentQueue = queue;
- queue = [];
- while (++queueIndex < len) {
- if (currentQueue) {
- currentQueue[queueIndex].run();
- }
- }
- queueIndex = -1;
- len = queue.length;
- }
- currentQueue = null;
- draining = false;
- runClearTimeout(timeout);
- }
- process.nextTick = function (fun) {
- var args = new Array(arguments.length - 1);
- if (arguments.length > 1) {
- for (var i = 1; i < arguments.length; i++) {
- args[i - 1] = arguments[i];
- }
- }
- queue.push(new Item(fun, args));
- if (queue.length === 1 && !draining) {
- runTimeout(drainQueue);
- }
- };
- // v8 likes predictible objects
- function Item(fun, array) {
- this.fun = fun;
- this.array = array;
- }
- Item.prototype.run = function () {
- this.fun.apply(null, this.array);
- };
- process.title = 'browser';
- process.browser = true;
- process.env = {};
- process.argv = [];
- process.version = ''; // empty string to avoid regexp issues
- process.versions = {};
- function noop() {}
- process.on = noop;
- process.addListener = noop;
- process.once = noop;
- process.off = noop;
- process.removeListener = noop;
- process.removeAllListeners = noop;
- process.emit = noop;
- process.binding = function (name) {
- throw new Error('process.binding is not supported');
- };
- process.cwd = function () { return '/' };
- process.chdir = function (dir) {
- throw new Error('process.chdir is not supported');
- };
- process.umask = function() { return 0; };
- /***/ },
- /* 14 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- /*! iScroll v5.2.0
- * (c) 2008-2016 Matteo Spinelli
- * http://cubiq.org/license
- */
- var rAF = UI.utils.rAF;
- var utils = (function() {
- var me = {};
- var _elementStyle = document.createElement('div').style;
- var _vendor = (function() {
- var vendors = ['t', 'webkitT', 'MozT', 'msT', 'OT'],
- transform,
- i = 0,
- l = vendors.length;
- for (; i < l; i++) {
- transform = vendors[i] + 'ransform';
- if (transform in _elementStyle) return vendors[i].substr(0, vendors[i].length - 1);
- }
- return false;
- })();
- function _prefixStyle(style) {
- if (_vendor === false) return false;
- if (_vendor === '') return style;
- return _vendor + style.charAt(0).toUpperCase() + style.substr(1);
- }
- me.getTime = Date.now || function getTime() {
- return new Date().getTime();
- };
- me.extend = function(target, obj) {
- for (var i in obj) {
- target[i] = obj[i];
- }
- };
- me.addEvent = function(el, type, fn, capture) {
- el.addEventListener(type, fn, !!capture);
- };
- me.removeEvent = function(el, type, fn, capture) {
- el.removeEventListener(type, fn, !!capture);
- };
- me.prefixPointerEvent = function(pointerEvent) {
- return window.MSPointerEvent ?
- 'MSPointer' + pointerEvent.charAt(7)
- .toUpperCase() + pointerEvent.substr(8) :
- pointerEvent;
- };
- me.momentum = function(current, start, time, lowerMargin, wrapperSize, deceleration) {
- var distance = current - start,
- speed = Math.abs(distance) / time,
- destination,
- duration;
- deceleration = deceleration === undefined ? 0.0006 : deceleration;
- destination = current + ( speed * speed ) / ( 2 * deceleration ) * ( distance < 0 ? -1 : 1 );
- duration = speed / deceleration;
- if (destination < lowerMargin) {
- destination = wrapperSize ? lowerMargin - ( wrapperSize / 2.5 * ( speed / 8 ) ) : lowerMargin;
- distance = Math.abs(destination - current);
- duration = distance / speed;
- } else if (destination > 0) {
- destination = wrapperSize ? wrapperSize / 2.5 * ( speed / 8 ) : 0;
- distance = Math.abs(current) + destination;
- duration = distance / speed;
- }
- return {
- destination: Math.round(destination),
- duration: duration
- };
- };
- var _transform = _prefixStyle('transform');
- me.extend(me, {
- hasTransform: _transform !== false,
- hasPerspective: _prefixStyle('perspective') in _elementStyle,
- hasTouch: 'ontouchstart' in window,
- hasPointer: !!(window.PointerEvent || window.MSPointerEvent), // IE10 is prefixed
- hasTransition: _prefixStyle('transition') in _elementStyle
- });
- /*
- This should find all Android browsers lower than build 535.19 (both stock browser and webview)
- - galaxy S2 is ok
- - 2.3.6 : `AppleWebKit/533.1 (KHTML, like Gecko) Version/4.0 Mobile Safari/533.1`
- - 4.0.4 : `AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Mobile Safari/534.30`
- - galaxy S3 is badAndroid (stock brower, webview)
- `AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Mobile Safari/534.30`
- - galaxy S4 is badAndroid (stock brower, webview)
- `AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Mobile Safari/534.30`
- - galaxy S5 is OK
- `AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Mobile Safari/537.36 (Chrome/)`
- - galaxy S6 is OK
- `AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Mobile Safari/537.36 (Chrome/)`
- */
- me.isBadAndroid = (function() {
- var appVersion = window.navigator.appVersion;
- // Android browser is not a chrome browser.
- if (/Android/.test(appVersion) && !(/Chrome\/\d/.test(appVersion))) {
- var safariVersion = appVersion.match(/Safari\/(\d+.\d)/);
- if (safariVersion && typeof safariVersion === "object" && safariVersion.length >= 2) {
- return parseFloat(safariVersion[1]) < 535.19;
- } else {
- return true;
- }
- } else {
- return false;
- }
- })();
- me.extend(me.style = {}, {
- transform: _transform,
- transitionTimingFunction: _prefixStyle('transitionTimingFunction'),
- transitionDuration: _prefixStyle('transitionDuration'),
- transitionDelay: _prefixStyle('transitionDelay'),
- transformOrigin: _prefixStyle('transformOrigin')
- });
- me.hasClass = function(e, c) {
- var re = new RegExp("(^|\\s)" + c + "(\\s|$)");
- return re.test(e.className);
- };
- me.addClass = function(e, c) {
- if (me.hasClass(e, c)) {
- return;
- }
- var newclass = e.className.split(' ');
- newclass.push(c);
- e.className = newclass.join(' ');
- };
- me.removeClass = function(e, c) {
- if (!me.hasClass(e, c)) {
- return;
- }
- var re = new RegExp("(^|\\s)" + c + "(\\s|$)", 'g');
- e.className = e.className.replace(re, ' ');
- };
- me.offset = function(el) {
- var left = -el.offsetLeft,
- top = -el.offsetTop;
- // jshint -W084
- while (el = el.offsetParent) {
- left -= el.offsetLeft;
- top -= el.offsetTop;
- }
- // jshint +W084
- return {
- left: left,
- top: top
- };
- };
- me.preventDefaultException = function(el, exceptions) {
- for (var i in exceptions) {
- if (exceptions[i].test(el[i])) {
- return true;
- }
- }
- return false;
- };
- me.extend(me.eventType = {}, {
- touchstart: 1,
- touchmove: 1,
- touchend: 1,
- mousedown: 2,
- mousemove: 2,
- mouseup: 2,
- pointerdown: 3,
- pointermove: 3,
- pointerup: 3,
- MSPointerDown: 3,
- MSPointerMove: 3,
- MSPointerUp: 3
- });
- me.extend(me.ease = {}, {
- quadratic: {
- style: 'cubic-bezier(0.25, 0.46, 0.45, 0.94)',
- fn: function(k) {
- return k * ( 2 - k );
- }
- },
- circular: {
- style: 'cubic-bezier(0.1, 0.57, 0.1, 1)', // Not properly "circular" but this looks better, it should be (0.075, 0.82, 0.165, 1)
- fn: function(k) {
- return Math.sqrt(1 - ( --k * k ));
- }
- },
- back: {
- style: 'cubic-bezier(0.175, 0.885, 0.32, 1.275)',
- fn: function(k) {
- var b = 4;
- return ( k = k - 1 ) * k * ( ( b + 1 ) * k + b ) + 1;
- }
- },
- bounce: {
- style: '',
- fn: function(k) {
- if (( k /= 1 ) < ( 1 / 2.75 )) {
- return 7.5625 * k * k;
- } else if (k < ( 2 / 2.75 )) {
- return 7.5625 * ( k -= ( 1.5 / 2.75 ) ) * k + 0.75;
- } else if (k < ( 2.5 / 2.75 )) {
- return 7.5625 * ( k -= ( 2.25 / 2.75 ) ) * k + 0.9375;
- } else {
- return 7.5625 * ( k -= ( 2.625 / 2.75 ) ) * k + 0.984375;
- }
- }
- },
- elastic: {
- style: '',
- fn: function(k) {
- var f = 0.22,
- e = 0.4;
- if (k === 0) {
- return 0;
- }
- if (k == 1) {
- return 1;
- }
- return ( e * Math.pow(2, -10 * k) * Math.sin(( k - f / 4 ) * ( 2 * Math.PI ) / f) + 1 );
- }
- }
- });
- me.tap = function(e, eventName) {
- var ev = document.createEvent('Event');
- ev.initEvent(eventName, true, true);
- ev.pageX = e.pageX;
- ev.pageY = e.pageY;
- e.target.dispatchEvent(ev);
- };
- me.click = function(e) {
- var target = e.target,
- ev;
- if (!(/(SELECT|INPUT|TEXTAREA)/i).test(target.tagName)) {
- // https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/initMouseEvent
- // initMouseEvent is deprecated.
- ev = document.createEvent(window.MouseEvent ? 'MouseEvents' : 'Event');
- ev.initEvent('click', true, true);
- ev.view = e.view || window;
- ev.detail = 1;
- ev.screenX = target.screenX || 0;
- ev.screenY = target.screenY || 0;
- ev.clientX = target.clientX || 0;
- ev.clientY = target.clientY || 0;
- ev.ctrlKey = !!e.ctrlKey;
- ev.altKey = !!e.altKey;
- ev.shiftKey = !!e.shiftKey;
- ev.metaKey = !!e.metaKey;
- ev.button = 0;
- ev.relatedTarget = null;
- ev._constructed = true;
- target.dispatchEvent(ev);
- }
- };
- return me;
- })();
- function IScroll(el, options) {
- this.wrapper = typeof el == 'string' ? document.querySelector(el) : el;
- this.scroller = this.wrapper.children[0];
- this.scrollerStyle = this.scroller.style; // cache style for better performance
- this.options = {
- // INSERT POINT: OPTIONS
- disablePointer: !utils.hasPointer,
- disableTouch: utils.hasPointer || !utils.hasTouch,
- disableMouse: utils.hasPointer || utils.hasTouch,
- startX: 0,
- startY: 0,
- scrollY: true,
- directionLockThreshold: 5,
- momentum: true,
- bounce: true,
- bounceTime: 600,
- bounceEasing: '',
- preventDefault: true,
- preventDefaultException: {tagName: /^(INPUT|TEXTAREA|BUTTON|SELECT)$/},
- HWCompositing: true,
- useTransition: true,
- useTransform: true,
- bindToWrapper: typeof window.onmousedown === "undefined"
- };
- for (var i in options) {
- this.options[i] = options[i];
- }
- // Normalize options
- this.translateZ = this.options.HWCompositing && utils.hasPerspective ? ' translateZ(0)' : '';
- this.options.useTransition = utils.hasTransition && this.options.useTransition;
- this.options.useTransform = utils.hasTransform && this.options.useTransform;
- this.options.eventPassthrough = this.options.eventPassthrough === true ? 'vertical' : this.options.eventPassthrough;
- this.options.preventDefault = !this.options.eventPassthrough && this.options.preventDefault;
- // If you want eventPassthrough I have to lock one of the axes
- this.options.scrollY = this.options.eventPassthrough == 'vertical' ? false : this.options.scrollY;
- this.options.scrollX = this.options.eventPassthrough == 'horizontal' ? false : this.options.scrollX;
- // With eventPassthrough we also need lockDirection mechanism
- this.options.freeScroll = this.options.freeScroll && !this.options.eventPassthrough;
- this.options.directionLockThreshold = this.options.eventPassthrough ? 0 : this.options.directionLockThreshold;
- this.options.bounceEasing = typeof this.options.bounceEasing == 'string' ? utils.ease[this.options.bounceEasing] || utils.ease.circular : this.options.bounceEasing;
- this.options.resizePolling = this.options.resizePolling === undefined ? 60 : this.options.resizePolling;
- if (this.options.tap === true) {
- this.options.tap = 'tap';
- }
- // https://github.com/cubiq/iscroll/issues/1029
- if (!this.options.useTransition && !this.options.useTransform) {
- if (!(/relative|absolute/i).test(this.scrollerStyle.position)) {
- this.scrollerStyle.position = "relative";
- }
- }
- // INSERT POINT: NORMALIZATION
- // Some defaults
- this.x = 0;
- this.y = 0;
- this.directionX = 0;
- this.directionY = 0;
- this._events = {};
- // INSERT POINT: DEFAULTS
- this._init();
- this.refresh();
- this.scrollTo(this.options.startX, this.options.startY);
- this.enable();
- }
- IScroll.prototype = {
- version: '5.2.0',
- _init: function() {
- this._initEvents();
- // INSERT POINT: _init
- },
- destroy: function() {
- this._initEvents(true);
- clearTimeout(this.resizeTimeout);
- this.resizeTimeout = null;
- this._execEvent('destroy');
- },
- _transitionEnd: function(e) {
- if (e.target != this.scroller || !this.isInTransition) {
- return;
- }
- this._transitionTime();
- if (!this.resetPosition(this.options.bounceTime)) {
- this.isInTransition = false;
- this._execEvent('scrollEnd');
- }
- },
- _start: function(e) {
- // React to left mouse button only
- if (utils.eventType[e.type] != 1) {
- // for button property
- // http://unixpapa.com/js/mouse.html
- var button;
- if (!e.which) {
- /* IE case */
- button = (e.button < 2) ? 0 :
- ((e.button == 4) ? 1 : 2);
- } else {
- /* All others */
- button = e.button;
- }
- if (button !== 0) {
- return;
- }
- }
- if (!this.enabled || (this.initiated && utils.eventType[e.type] !== this.initiated)) {
- return;
- }
- if (this.options.preventDefault && !utils.isBadAndroid && !utils.preventDefaultException(e.target, this.options.preventDefaultException)) {
- e.preventDefault();
- }
- var point = e.touches ? e.touches[0] : e,
- pos;
- this.initiated = utils.eventType[e.type];
- this.moved = false;
- this.distX = 0;
- this.distY = 0;
- this.directionX = 0;
- this.directionY = 0;
- this.directionLocked = 0;
- this.startTime = utils.getTime();
- if (this.options.useTransition && this.isInTransition) {
- this._transitionTime();
- this.isInTransition = false;
- pos = this.getComputedPosition();
- this._translate(Math.round(pos.x), Math.round(pos.y));
- this._execEvent('scrollEnd');
- } else if (!this.options.useTransition && this.isAnimating) {
- this.isAnimating = false;
- this._execEvent('scrollEnd');
- }
- this.startX = this.x;
- this.startY = this.y;
- this.absStartX = this.x;
- this.absStartY = this.y;
- this.pointX = point.pageX;
- this.pointY = point.pageY;
- this._execEvent('beforeScrollStart');
- },
- _move: function(e) {
- if (!this.enabled || utils.eventType[e.type] !== this.initiated) {
- return;
- }
- if (this.options.preventDefault) { // increases performance on Android? TODO: check!
- e.preventDefault();
- }
- var point = e.touches ? e.touches[0] : e,
- deltaX = point.pageX - this.pointX,
- deltaY = point.pageY - this.pointY,
- timestamp = utils.getTime(),
- newX, newY,
- absDistX, absDistY;
- this.pointX = point.pageX;
- this.pointY = point.pageY;
- this.distX += deltaX;
- this.distY += deltaY;
- absDistX = Math.abs(this.distX);
- absDistY = Math.abs(this.distY);
- // We need to move at least 10 pixels for the scrolling to initiate
- if (timestamp - this.endTime > 300 && (absDistX < 10 && absDistY < 10)) {
- return;
- }
- // If you are scrolling in one direction lock the other
- if (!this.directionLocked && !this.options.freeScroll) {
- if (absDistX > absDistY + this.options.directionLockThreshold) {
- this.directionLocked = 'h'; // lock horizontally
- } else if (absDistY >= absDistX + this.options.directionLockThreshold) {
- this.directionLocked = 'v'; // lock vertically
- } else {
- this.directionLocked = 'n'; // no lock
- }
- }
- if (this.directionLocked == 'h') {
- if (this.options.eventPassthrough == 'vertical') {
- e.preventDefault();
- } else if (this.options.eventPassthrough == 'horizontal') {
- this.initiated = false;
- return;
- }
- deltaY = 0;
- } else if (this.directionLocked == 'v') {
- if (this.options.eventPassthrough == 'horizontal') {
- e.preventDefault();
- } else if (this.options.eventPassthrough == 'vertical') {
- this.initiated = false;
- return;
- }
- deltaX = 0;
- }
- deltaX = this.hasHorizontalScroll ? deltaX : 0;
- deltaY = this.hasVerticalScroll ? deltaY : 0;
- newX = this.x + deltaX;
- newY = this.y + deltaY;
- // Slow down if outside of the boundaries
- if (newX > 0 || newX < this.maxScrollX) {
- newX = this.options.bounce ? this.x + deltaX / 3 : newX > 0 ? 0 : this.maxScrollX;
- }
- if (newY > 0 || newY < this.maxScrollY) {
- newY = this.options.bounce ? this.y + deltaY / 3 : newY > 0 ? 0 : this.maxScrollY;
- }
- this.directionX = deltaX > 0 ? -1 : deltaX < 0 ? 1 : 0;
- this.directionY = deltaY > 0 ? -1 : deltaY < 0 ? 1 : 0;
- if (!this.moved) {
- this._execEvent('scrollStart');
- }
- this.moved = true;
- this._translate(newX, newY);
- /* REPLACE START: _move */
- if (timestamp - this.startTime > 300) {
- this.startTime = timestamp;
- this.startX = this.x;
- this.startY = this.y;
- }
- /* REPLACE END: _move */
- },
- _end: function(e) {
- if (!this.enabled || utils.eventType[e.type] !== this.initiated) {
- return;
- }
- if (this.options.preventDefault && !utils.preventDefaultException(e.target, this.options.preventDefaultException)) {
- e.preventDefault();
- }
- var point = e.changedTouches ? e.changedTouches[0] : e,
- momentumX,
- momentumY,
- duration = utils.getTime() - this.startTime,
- newX = Math.round(this.x),
- newY = Math.round(this.y),
- distanceX = Math.abs(newX - this.startX),
- distanceY = Math.abs(newY - this.startY),
- time = 0,
- easing = '';
- this.isInTransition = 0;
- this.initiated = 0;
- this.endTime = utils.getTime();
- // reset if we are outside of the boundaries
- if (this.resetPosition(this.options.bounceTime)) {
- return;
- }
- this.scrollTo(newX, newY); // ensures that the last position is rounded
- // we scrolled less than 10 pixels
- if (!this.moved) {
- if (this.options.tap) {
- utils.tap(e, this.options.tap);
- }
- if (this.options.click) {
- utils.click(e);
- }
- this._execEvent('scrollCancel');
- return;
- }
- if (this._events.flick && duration < 200 && distanceX < 100 && distanceY < 100) {
- this._execEvent('flick');
- return;
- }
- // start momentum animation if needed
- if (this.options.momentum && duration < 300) {
- momentumX = this.hasHorizontalScroll ? utils.momentum(this.x, this.startX, duration, this.maxScrollX, this.options.bounce ? this.wrapperWidth : 0, this.options.deceleration) : {
- destination: newX,
- duration: 0
- };
- momentumY = this.hasVerticalScroll ? utils.momentum(this.y, this.startY, duration, this.maxScrollY, this.options.bounce ? this.wrapperHeight : 0, this.options.deceleration) : {
- destination: newY,
- duration: 0
- };
- newX = momentumX.destination;
- newY = momentumY.destination;
- time = Math.max(momentumX.duration, momentumY.duration);
- this.isInTransition = 1;
- }
- // INSERT POINT: _end
- if (newX != this.x || newY != this.y) {
- // change easing function when scroller goes out of the boundaries
- if (newX > 0 || newX < this.maxScrollX || newY > 0 || newY < this.maxScrollY) {
- easing = utils.ease.quadratic;
- }
- this.scrollTo(newX, newY, time, easing);
- return;
- }
- this._execEvent('scrollEnd');
- },
- _resize: function() {
- var that = this;
- clearTimeout(this.resizeTimeout);
- this.resizeTimeout = setTimeout(function() {
- that.refresh();
- }, this.options.resizePolling);
- },
- resetPosition: function(time) {
- var x = this.x,
- y = this.y;
- time = time || 0;
- if (!this.hasHorizontalScroll || this.x > 0) {
- x = 0;
- } else if (this.x < this.maxScrollX) {
- x = this.maxScrollX;
- }
- if (!this.hasVerticalScroll || this.y > 0) {
- y = 0;
- } else if (this.y < this.maxScrollY) {
- y = this.maxScrollY;
- }
- if (x == this.x && y == this.y) {
- return false;
- }
- this.scrollTo(x, y, time, this.options.bounceEasing);
- return true;
- },
- disable: function() {
- this.enabled = false;
- },
- enable: function() {
- this.enabled = true;
- },
- refresh: function() {
- var rf = this.wrapper.offsetHeight; // Force reflow
- this.wrapperWidth = this.wrapper.clientWidth;
- this.wrapperHeight = this.wrapper.clientHeight;
- /* REPLACE START: refresh */
- this.scrollerWidth = this.scroller.offsetWidth;
- this.scrollerHeight = this.scroller.offsetHeight;
- this.maxScrollX = this.wrapperWidth - this.scrollerWidth;
- this.maxScrollY = this.wrapperHeight - this.scrollerHeight;
- /* REPLACE END: refresh */
- this.hasHorizontalScroll = this.options.scrollX && this.maxScrollX < 0;
- this.hasVerticalScroll = this.options.scrollY && this.maxScrollY < 0;
- if (!this.hasHorizontalScroll) {
- this.maxScrollX = 0;
- this.scrollerWidth = this.wrapperWidth;
- }
- if (!this.hasVerticalScroll) {
- this.maxScrollY = 0;
- this.scrollerHeight = this.wrapperHeight;
- }
- this.endTime = 0;
- this.directionX = 0;
- this.directionY = 0;
- this.wrapperOffset = utils.offset(this.wrapper);
- this._execEvent('refresh');
- this.resetPosition();
- // INSERT POINT: _refresh
- },
- on: function(type, fn) {
- if (!this._events[type]) {
- this._events[type] = [];
- }
- this._events[type].push(fn);
- },
- off: function(type, fn) {
- if (!this._events[type]) {
- return;
- }
- var index = this._events[type].indexOf(fn);
- if (index > -1) {
- this._events[type].splice(index, 1);
- }
- },
- _execEvent: function(type) {
- if (!this._events[type]) {
- return;
- }
- var i = 0,
- l = this._events[type].length;
- if (!l) {
- return;
- }
- for (; i < l; i++) {
- this._events[type][i].apply(this, [].slice.call(arguments, 1));
- }
- },
- scrollBy: function(x, y, time, easing) {
- x = this.x + x;
- y = this.y + y;
- time = time || 0;
- this.scrollTo(x, y, time, easing);
- },
- scrollTo: function(x, y, time, easing) {
- easing = easing || utils.ease.circular;
- this.isInTransition = this.options.useTransition && time > 0;
- var transitionType = this.options.useTransition && easing.style;
- if (!time || transitionType) {
- if (transitionType) {
- this._transitionTimingFunction(easing.style);
- this._transitionTime(time);
- }
- this._translate(x, y);
- } else {
- this._animate(x, y, time, easing.fn);
- }
- },
- scrollToElement: function(el, time, offsetX, offsetY, easing) {
- el = el.nodeType ? el : this.scroller.querySelector(el);
- if (!el) {
- return;
- }
- var pos = utils.offset(el);
- pos.left -= this.wrapperOffset.left;
- pos.top -= this.wrapperOffset.top;
- // if offsetX/Y are true we center the element to the screen
- if (offsetX === true) {
- offsetX = Math.round(el.offsetWidth / 2 - this.wrapper.offsetWidth / 2);
- }
- if (offsetY === true) {
- offsetY = Math.round(el.offsetHeight / 2 - this.wrapper.offsetHeight / 2);
- }
- pos.left -= offsetX || 0;
- pos.top -= offsetY || 0;
- pos.left = pos.left > 0 ? 0 : pos.left < this.maxScrollX ? this.maxScrollX : pos.left;
- pos.top = pos.top > 0 ? 0 : pos.top < this.maxScrollY ? this.maxScrollY : pos.top;
- time = time === undefined || time === null || time === 'auto' ? Math.max(Math.abs(this.x - pos.left), Math.abs(this.y - pos.top)) : time;
- this.scrollTo(pos.left, pos.top, time, easing);
- },
- _transitionTime: function(time) {
- if (!this.options.useTransition) {
- return;
- }
- time = time || 0;
- var durationProp = utils.style.transitionDuration;
- if (!durationProp) {
- return;
- }
- this.scrollerStyle[durationProp] = time + 'ms';
- if (!time && utils.isBadAndroid) {
- this.scrollerStyle[durationProp] = '0.0001ms';
- // remove 0.0001ms
- var self = this;
- rAF(function() {
- if (self.scrollerStyle[durationProp] === '0.0001ms') {
- self.scrollerStyle[durationProp] = '0s';
- }
- });
- }
- // INSERT POINT: _transitionTime
- },
- _transitionTimingFunction: function(easing) {
- this.scrollerStyle[utils.style.transitionTimingFunction] = easing;
- // INSERT POINT: _transitionTimingFunction
- },
- _translate: function(x, y) {
- if (this.options.useTransform) {
- /* REPLACE START: _translate */
- this.scrollerStyle[utils.style.transform] = 'translate(' + x + 'px,' + y + 'px)' + this.translateZ;
- /* REPLACE END: _translate */
- } else {
- x = Math.round(x);
- y = Math.round(y);
- this.scrollerStyle.left = x + 'px';
- this.scrollerStyle.top = y + 'px';
- }
- this.x = x;
- this.y = y;
- // INSERT POINT: _translate
- },
- _initEvents: function(remove) {
- var eventType = remove ? utils.removeEvent : utils.addEvent,
- target = this.options.bindToWrapper ? this.wrapper : window;
- eventType(window, 'orientationchange', this);
- eventType(window, 'resize', this);
- if (this.options.click) {
- eventType(this.wrapper, 'click', this, true);
- }
- if (!this.options.disableMouse) {
- eventType(this.wrapper, 'mousedown', this);
- eventType(target, 'mousemove', this);
- eventType(target, 'mousecancel', this);
- eventType(target, 'mouseup', this);
- }
- if (utils.hasPointer && !this.options.disablePointer) {
- eventType(this.wrapper, utils.prefixPointerEvent('pointerdown'), this);
- eventType(target, utils.prefixPointerEvent('pointermove'), this);
- eventType(target, utils.prefixPointerEvent('pointercancel'), this);
- eventType(target, utils.prefixPointerEvent('pointerup'), this);
- }
- if (utils.hasTouch && !this.options.disableTouch) {
- eventType(this.wrapper, 'touchstart', this);
- eventType(target, 'touchmove', this);
- eventType(target, 'touchcancel', this);
- eventType(target, 'touchend', this);
- }
- eventType(this.scroller, 'transitionend', this);
- eventType(this.scroller, 'webkitTransitionEnd', this);
- eventType(this.scroller, 'oTransitionEnd', this);
- eventType(this.scroller, 'MSTransitionEnd', this);
- },
- getComputedPosition: function() {
- var matrix = window.getComputedStyle(this.scroller, null),
- x, y;
- if (this.options.useTransform) {
- matrix = matrix[utils.style.transform].split(')')[0].split(', ');
- x = +(matrix[12] || matrix[4]);
- y = +(matrix[13] || matrix[5]);
- } else {
- x = +matrix.left.replace(/[^-\d.]/g, '');
- y = +matrix.top.replace(/[^-\d.]/g, '');
- }
- return {x: x, y: y};
- },
- _animate: function(destX, destY, duration, easingFn) {
- var that = this,
- startX = this.x,
- startY = this.y,
- startTime = utils.getTime(),
- destTime = startTime + duration;
- function step() {
- var now = utils.getTime(),
- newX, newY,
- easing;
- if (now >= destTime) {
- that.isAnimating = false;
- that._translate(destX, destY);
- if (!that.resetPosition(that.options.bounceTime)) {
- that._execEvent('scrollEnd');
- }
- return;
- }
- now = ( now - startTime ) / duration;
- easing = easingFn(now);
- newX = ( destX - startX ) * easing + startX;
- newY = ( destY - startY ) * easing + startY;
- that._translate(newX, newY);
- if (that.isAnimating) {
- rAF(step);
- }
- }
- this.isAnimating = true;
- step();
- },
- handleEvent: function(e) {
- switch (e.type) {
- case 'touchstart':
- case 'pointerdown':
- case 'MSPointerDown':
- case 'mousedown':
- this._start(e);
- break;
- case 'touchmove':
- case 'pointermove':
- case 'MSPointerMove':
- case 'mousemove':
- this._move(e);
- break;
- case 'touchend':
- case 'pointerup':
- case 'MSPointerUp':
- case 'mouseup':
- case 'touchcancel':
- case 'pointercancel':
- case 'MSPointerCancel':
- case 'mousecancel':
- this._end(e);
- break;
- case 'orientationchange':
- case 'resize':
- this._resize();
- break;
- case 'transitionend':
- case 'webkitTransitionEnd':
- case 'oTransitionEnd':
- case 'MSTransitionEnd':
- this._transitionEnd(e);
- break;
- case 'wheel':
- case 'DOMMouseScroll':
- case 'mousewheel':
- this._wheel(e);
- break;
- case 'keydown':
- this._key(e);
- break;
- case 'click':
- if (this.enabled && !e._constructed) {
- e.preventDefault();
- e.stopPropagation();
- }
- break;
- }
- }
- };
- IScroll.utils = utils;
- module.exports = UI.iScroll = IScroll;
- /* jshint unused: true */
- /* jshint +W101, +W116, +W109 */
- /***/ },
- /* 15 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var dimmer = __webpack_require__(9);
- var $doc = $(document);
- var supportTransition = UI.support.transition;
- /**
- * @reference https://github.com/nolimits4web/Framework7/blob/master/src/js/modals.js
- * @license https://github.com/nolimits4web/Framework7/blob/master/LICENSE
- */
- var Modal = function(element, options) {
- this.options = $.extend({}, Modal.DEFAULTS, options || {});
- this.$element = $(element);
- this.$dialog = this.$element.find('.am-modal-dialog');
- if (!this.$element.attr('id')) {
- this.$element.attr('id', UI.utils.generateGUID('am-modal'));
- }
- this.isPopup = this.$element.hasClass('am-popup');
- this.isActions = this.$element.hasClass('am-modal-actions');
- this.isPrompt = this.$element.hasClass('am-modal-prompt');
- this.isLoading = this.$element.hasClass('am-modal-loading');
- this.active = this.transitioning = this.relatedTarget = null;
- this.dimmer = this.options.dimmer ? dimmer : {
- open: function() {
- },
- close: function() {
- }
- };
- this.events();
- };
- Modal.DEFAULTS = {
- className: {
- active: 'am-modal-active',
- out: 'am-modal-out'
- },
- selector: {
- modal: '.am-modal',
- active: '.am-modal-active'
- },
- closeViaDimmer: true,
- cancelable: true,
- onConfirm: function() {
- },
- onCancel: function() {
- },
- closeOnCancel: true,
- closeOnConfirm: true,
- dimmer: true,
- height: undefined,
- width: undefined,
- duration: 300, // must equal the CSS transition duration
- transitionEnd: supportTransition && supportTransition.end + '.modal.amui'
- };
- Modal.prototype.toggle = function(relatedTarget) {
- return this.active ? this.close() : this.open(relatedTarget);
- };
- Modal.prototype.open = function(relatedTarget) {
- var $element = this.$element;
- var options = this.options;
- var isPopup = this.isPopup;
- var width = options.width;
- var height = options.height;
- var style = {};
- if (this.active) {
- return;
- }
- if (!this.$element.length) {
- return;
- }
- // callback hook
- relatedTarget && (this.relatedTarget = relatedTarget);
- // 判断如果还在动画,就先触发之前的closed事件
- if (this.transitioning) {
- clearTimeout($element.transitionEndTimmer);
- $element.transitionEndTimmer = null;
- $element.trigger(options.transitionEnd)
- .off(options.transitionEnd);
- }
- isPopup && this.$element.show();
- this.active = true;
- $element.trigger($.Event('open.modal.amui', {relatedTarget: relatedTarget}));
- this.dimmer.open($element);
- $element.show().redraw();
- // apply Modal width/height if set
- if (!isPopup && !this.isActions) {
- if (width) {
- style.width = parseInt(width, 10) + 'px';
- }
- if (height) {
- style.height = parseInt(height, 10) + 'px';
- }
- this.$dialog.css(style);
- }
- $element
- .removeClass(options.className.out)
- .addClass(options.className.active);
- this.transitioning = 1;
- var complete = function() {
- $element.trigger($.Event('opened.modal.amui', {
- relatedTarget: relatedTarget
- }));
- this.transitioning = 0;
- // Prompt auto focus
- if (this.isPrompt) {
- this.$dialog.find('input').eq(0).focus();
- }
- };
- if (!supportTransition) {
- return complete.call(this);
- }
- $element
- .one(options.transitionEnd, $.proxy(complete, this))
- .emulateTransitionEnd(options.duration);
- };
- Modal.prototype.close = function(relatedTarget) {
- if (!this.active) {
- return;
- }
- var $element = this.$element;
- var options = this.options;
- var isPopup = this.isPopup;
- // 判断如果还在动画,就先触发之前的opened事件
- if (this.transitioning) {
- clearTimeout($element.transitionEndTimmer);
- $element.transitionEndTimmer = null;
- $element.trigger(options.transitionEnd).off(options.transitionEnd);
- this.dimmer.close($element, true);
- }
- this.$element.trigger($.Event('close.modal.amui', {
- relatedTarget: relatedTarget
- }));
- this.transitioning = 1;
- var complete = function() {
- $element.trigger('closed.modal.amui');
- isPopup && $element.removeClass(options.className.out);
- $element.hide();
- this.transitioning = 0;
- // 不强制关闭 Dimmer,以便多个 Modal 可以共享 Dimmer
- this.dimmer.close($element, false);
- this.active = false;
- };
- $element.removeClass(options.className.active)
- .addClass(options.className.out);
- if (!supportTransition) {
- return complete.call(this);
- }
- $element.one(options.transitionEnd, $.proxy(complete, this))
- .emulateTransitionEnd(options.duration);
- };
- Modal.prototype.events = function() {
- var _this = this;
- var options = this.options;
- var $element = this.$element;
- var $dimmer = this.dimmer.$element;
- var $ipt = $element.find('.am-modal-prompt-input');
- var $confirm = $element.find('[data-am-modal-confirm]');
- var $cancel = $element.find('[data-am-modal-cancel]');
- var getData = function() {
- var data = [];
- $ipt.each(function() {
- data.push($(this).val());
- });
- return (data.length === 0) ? undefined :
- ((data.length === 1) ? data[0] : data);
- };
- // close via Esc key
- if (this.options.cancelable) {
- $element.on('keyup.modal.amui', function(e) {
- if (_this.active && e.which === 27) {
- $element.trigger('cancel.modal.amui');
- _this.close();
- }
- });
- }
- // Close Modal when dimmer clicked
- if (this.options.dimmer && this.options.closeViaDimmer && !this.isLoading) {
- $dimmer.on('click.dimmer.modal.amui', function() {
- _this.close();
- });
- }
- // Close Modal when button clicked
- $element.on(
- 'click.close.modal.amui',
- '[data-am-modal-close], .am-modal-btn',
- function(e) {
- e.preventDefault();
- var $this = $(this);
- if ($this.is($confirm)) {
- options.closeOnConfirm && _this.close();
- } else if ($this.is($cancel)) {
- options.closeOnCancel && _this.close();
- } else {
- _this.close();
- }
- }
- )
- // trigger dimmer click event if non-dialog area clicked
- // fixes #882 caused by https://github.com/amazeui/amazeui/commit/b6be7719681193f1c4cb04af89cb9fd9f4422163
- .on('click', function(e) {
- // fixes #900
- // e.stopPropagation();
- $(e.target).is($element) && $dimmer.trigger('click.dimmer.modal.amui');
- });
- $confirm.on('click.confirm.modal.amui',
- function() {
- $element.trigger($.Event('confirm.modal.amui', {
- trigger: this
- }));
- });
- $cancel.on('click.cancel.modal.amui', function() {
- $element.trigger($.Event('cancel.modal.amui', {
- trigger: this
- }));
- });
- $element.on('confirm.modal.amui', function(e) {
- e.data = getData();
- _this.options.onConfirm.call(_this, e);
- }).on('cancel.modal.amui', function(e) {
- e.data = getData();
- _this.options.onCancel.call(_this, e);
- });
- };
- function Plugin(option, relatedTarget) {
- return this.each(function() {
- var $this = $(this);
- var data = $this.data('amui.modal');
- var options = typeof option == 'object' && option;
- if (!data) {
- $this.data('amui.modal', (data = new Modal(this, options)));
- }
- if (typeof option == 'string') {
- data[option] && data[option](relatedTarget);
- } else {
- data.toggle(option && option.relatedTarget || undefined);
- }
- });
- }
- $.fn.modal = Plugin;
- // Init
- $doc.on('click.modal.amui.data-api', '[data-am-modal]', function() {
- var $this = $(this);
- var options = UI.utils.parseOptions($this.attr('data-am-modal'));
- var $target = $(options.target ||
- (this.href && this.href.replace(/.*(?=#[^\s]+$)/, '')));
- var option = $target.data('amui.modal') ? 'toggle' : options;
- Plugin.call($target, option, this);
- });
- module.exports = UI.modal = Modal;
- /***/ },
- /* 16 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(3);
- var $win = $(window);
- var $doc = $(document);
- var scrollPos;
- /**
- * @via https://github.com/uikit/uikit/blob/master/src/js/offcanvas.js
- * @license https://github.com/uikit/uikit/blob/master/LICENSE.md
- */
- var OffCanvas = function(element, options) {
- this.$element = $(element);
- this.options = $.extend({}, OffCanvas.DEFAULTS, options);
- this.active = null;
- this.bindEvents();
- };
- OffCanvas.DEFAULTS = {
- duration: 300,
- effect: 'overlay' // {push|overlay}, push is too expensive
- };
- OffCanvas.prototype.open = function(relatedElement) {
- var _this = this;
- var $element = this.$element;
- if (!$element.length || $element.hasClass('am-active')) {
- return;
- }
- var effect = this.options.effect;
- var $html = $('html');
- var $body = $('body');
- var $bar = $element.find('.am-offcanvas-bar').first();
- var dir = $bar.hasClass('am-offcanvas-bar-flip') ? -1 : 1;
- $bar.addClass('am-offcanvas-bar-' + effect);
- scrollPos = {x: window.scrollX, y: window.scrollY};
- $element.addClass('am-active');
- $body.css({
- width: window.innerWidth,
- height: $win.height()
- }).addClass('am-offcanvas-page');
- if (effect !== 'overlay') {
- $body.css({
- 'margin-left': $bar.outerWidth() * dir
- }).width(); // force redraw
- }
- $html.css('margin-top', scrollPos.y * -1);
- setTimeout(function() {
- $bar.addClass('am-offcanvas-bar-active').width();
- }, 0);
- $element.trigger('open.offcanvas.amui');
- this.active = 1;
- // Close OffCanvas when none content area clicked
- $element.on('click.offcanvas.amui', function(e) {
- var $target = $(e.target);
- if ($target.hasClass('am-offcanvas-bar')) {
- return;
- }
- if ($target.parents('.am-offcanvas-bar').first().length) {
- return;
- }
- // https://developer.mozilla.org/zh-CN/docs/DOM/event.stopImmediatePropagation
- e.stopImmediatePropagation();
- _this.close();
- });
- $html.on('keydown.offcanvas.amui', function(e) {
- (e.keyCode === 27) && _this.close();
- });
- };
- OffCanvas.prototype.close = function(relatedElement) {
- var _this = this;
- var $html = $('html');
- var $body = $('body');
- var $element = this.$element;
- var $bar = $element.find('.am-offcanvas-bar').first();
- if (!$element.length || !this.active || !$element.hasClass('am-active')) {
- return;
- }
- $element.trigger('close.offcanvas.amui');
- function complete() {
- $body
- .removeClass('am-offcanvas-page')
- .css({
- width: '',
- height: '',
- 'margin-left': '',
- 'margin-right': ''
- });
- $element.removeClass('am-active');
- $bar.removeClass('am-offcanvas-bar-active');
- $html.css('margin-top', '');
- window.scrollTo(scrollPos.x, scrollPos.y);
- $element.trigger('closed.offcanvas.amui');
- _this.active = 0;
- }
- if (UI.support.transition) {
- setTimeout(function() {
- $bar.removeClass('am-offcanvas-bar-active');
- }, 0);
- $body.css('margin-left', '').one(UI.support.transition.end, function() {
- complete();
- }).emulateTransitionEnd(this.options.duration);
- } else {
- complete();
- }
- $element.off('click.offcanvas.amui');
- $html.off('.offcanvas.amui');
- };
- OffCanvas.prototype.bindEvents = function() {
- var _this = this;
- $doc.on('click.offcanvas.amui', '[data-am-dismiss="offcanvas"]', function(e) {
- e.preventDefault();
- _this.close();
- });
- $win.on('resize.offcanvas.amui orientationchange.offcanvas.amui',
- function() {
- _this.active && _this.close();
- });
- this.$element.hammer().on('swipeleft swipeleft', function(e) {
- e.preventDefault();
- _this.close();
- });
- return this;
- };
- function Plugin(option, relatedElement) {
- var args = Array.prototype.slice.call(arguments, 1);
- return this.each(function() {
- var $this = $(this);
- var data = $this.data('amui.offcanvas');
- var options = $.extend({}, typeof option == 'object' && option);
- if (!data) {
- $this.data('amui.offcanvas', (data = new OffCanvas(this, options)));
- (!option || typeof option == 'object') && data.open(relatedElement);
- }
- if (typeof option == 'string') {
- data[option] && data[option].apply(data, args);
- }
- });
- }
- $.fn.offCanvas = Plugin;
- // Init code
- $doc.on('click.offcanvas.amui', '[data-am-offcanvas]', function(e) {
- e.preventDefault();
- var $this = $(this);
- var options = UI.utils.parseOptions($this.data('amOffcanvas'));
- var $target = $(options.target ||
- (this.href && this.href.replace(/.*(?=#[^\s]+$)/, '')));
- var option = $target.data('amui.offcanvas') ? 'open' : options;
- Plugin.call($target, option, this);
- });
- module.exports = UI.offcanvas = OffCanvas;
- // TODO: 优化动画效果
- // http://dbushell.github.io/Responsive-Off-Canvas-Menu/step4.html
- /***/ },
- /* 17 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var requestAnimationFrame = UI.utils.rAF;
- /**
- * @via https://github.com/manuelstofer/pinchzoom/blob/master/src/pinchzoom.js
- * @license the MIT License.
- */
- var definePinchZoom = function($) {
- /**
- * Pinch zoom using jQuery
- * @version 0.0.2
- * @author Manuel Stofer <mst@rtp.ch>
- * @param el
- * @param options
- * @constructor
- */
- var PinchZoom = function(el, options) {
- this.el = $(el);
- this.zoomFactor = 1;
- this.lastScale = 1;
- this.offset = {
- x: 0,
- y: 0
- };
- this.options = $.extend({}, this.defaults, options);
- this.setupMarkup();
- this.bindEvents();
- this.update();
- // default enable.
- this.enable();
- },
- sum = function(a, b) {
- return a + b;
- },
- isCloseTo = function(value, expected) {
- return value > expected - 0.01 && value < expected + 0.01;
- };
- PinchZoom.prototype = {
- defaults: {
- tapZoomFactor: 2,
- zoomOutFactor: 1.3,
- animationDuration: 300,
- maxZoom: 4,
- minZoom: 0.5,
- lockDragAxis: false,
- use2d: true,
- zoomStartEventName: 'pz_zoomstart',
- zoomEndEventName: 'pz_zoomend',
- dragStartEventName: 'pz_dragstart',
- dragEndEventName: 'pz_dragend',
- doubleTapEventName: 'pz_doubletap'
- },
- /**
- * Event handler for 'dragstart'
- * @param event
- */
- handleDragStart: function(event) {
- this.el.trigger(this.options.dragStartEventName);
- this.stopAnimation();
- this.lastDragPosition = false;
- this.hasInteraction = true;
- this.handleDrag(event);
- },
- /**
- * Event handler for 'drag'
- * @param event
- */
- handleDrag: function(event) {
- if (this.zoomFactor > 1.0) {
- var touch = this.getTouches(event)[0];
- this.drag(touch, this.lastDragPosition);
- this.offset = this.sanitizeOffset(this.offset);
- this.lastDragPosition = touch;
- }
- },
- handleDragEnd: function() {
- this.el.trigger(this.options.dragEndEventName);
- this.end();
- },
- /**
- * Event handler for 'zoomstart'
- * @param event
- */
- handleZoomStart: function(event) {
- this.el.trigger(this.options.zoomStartEventName);
- this.stopAnimation();
- this.lastScale = 1;
- this.nthZoom = 0;
- this.lastZoomCenter = false;
- this.hasInteraction = true;
- },
- /**
- * Event handler for 'zoom'
- * @param event
- */
- handleZoom: function(event, newScale) {
- // a relative scale factor is used
- var touchCenter = this.getTouchCenter(this.getTouches(event)),
- scale = newScale / this.lastScale;
- this.lastScale = newScale;
- // the first touch events are thrown away since they are not precise
- this.nthZoom += 1;
- if (this.nthZoom > 3) {
- this.scale(scale, touchCenter);
- this.drag(touchCenter, this.lastZoomCenter);
- }
- this.lastZoomCenter = touchCenter;
- },
- handleZoomEnd: function() {
- this.el.trigger(this.options.zoomEndEventName);
- this.end();
- },
- /**
- * Event handler for 'doubletap'
- * @param event
- */
- handleDoubleTap: function(event) {
- var center = this.getTouches(event)[0],
- zoomFactor = this.zoomFactor > 1 ? 1 : this.options.tapZoomFactor,
- startZoomFactor = this.zoomFactor,
- updateProgress = (function(progress) {
- this.scaleTo(startZoomFactor + progress * (zoomFactor - startZoomFactor), center);
- }).bind(this);
- if (this.hasInteraction) {
- return;
- }
- if (startZoomFactor > zoomFactor) {
- center = this.getCurrentZoomCenter();
- }
- this.animate(this.options.animationDuration, updateProgress, this.swing);
- this.el.trigger(this.options.doubleTapEventName);
- },
- /**
- * Max / min values for the offset
- * @param offset
- * @return {Object} the sanitized offset
- */
- sanitizeOffset: function(offset) {
- var maxX = (this.zoomFactor - 1) * this.getContainerX(),
- maxY = (this.zoomFactor - 1) * this.getContainerY(),
- maxOffsetX = Math.max(maxX, 0),
- maxOffsetY = Math.max(maxY, 0),
- minOffsetX = Math.min(maxX, 0),
- minOffsetY = Math.min(maxY, 0);
- return {
- x: Math.min(Math.max(offset.x, minOffsetX), maxOffsetX),
- y: Math.min(Math.max(offset.y, minOffsetY), maxOffsetY)
- };
- },
- /**
- * Scale to a specific zoom factor (not relative)
- * @param zoomFactor
- * @param center
- */
- scaleTo: function(zoomFactor, center) {
- this.scale(zoomFactor / this.zoomFactor, center);
- },
- /**
- * Scales the element from specified center
- * @param scale
- * @param center
- */
- scale: function(scale, center) {
- scale = this.scaleZoomFactor(scale);
- this.addOffset({
- x: (scale - 1) * (center.x + this.offset.x),
- y: (scale - 1) * (center.y + this.offset.y)
- });
- },
- /**
- * Scales the zoom factor relative to current state
- * @param scale
- * @return the actual scale (can differ because of max min zoom factor)
- */
- scaleZoomFactor: function(scale) {
- var originalZoomFactor = this.zoomFactor;
- this.zoomFactor *= scale;
- this.zoomFactor = Math.min(this.options.maxZoom, Math.max(this.zoomFactor, this.options.minZoom));
- return this.zoomFactor / originalZoomFactor;
- },
- /**
- * Drags the element
- * @param center
- * @param lastCenter
- */
- drag: function(center, lastCenter) {
- if (lastCenter) {
- if (this.options.lockDragAxis) {
- // lock scroll to position that was changed the most
- if (Math.abs(center.x - lastCenter.x) > Math.abs(center.y - lastCenter.y)) {
- this.addOffset({
- x: -(center.x - lastCenter.x),
- y: 0
- });
- }
- else {
- this.addOffset({
- y: -(center.y - lastCenter.y),
- x: 0
- });
- }
- }
- else {
- this.addOffset({
- y: -(center.y - lastCenter.y),
- x: -(center.x - lastCenter.x)
- });
- }
- }
- },
- /**
- * Calculates the touch center of multiple touches
- * @param touches
- * @return {Object}
- */
- getTouchCenter: function(touches) {
- return this.getVectorAvg(touches);
- },
- /**
- * Calculates the average of multiple vectors (x, y values)
- */
- getVectorAvg: function(vectors) {
- return {
- x: vectors.map(function(v) {
- return v.x;
- }).reduce(sum) / vectors.length,
- y: vectors.map(function(v) {
- return v.y;
- }).reduce(sum) / vectors.length
- };
- },
- /**
- * Adds an offset
- * @param offset the offset to add
- * @return return true when the offset change was accepted
- */
- addOffset: function(offset) {
- this.offset = {
- x: this.offset.x + offset.x,
- y: this.offset.y + offset.y
- };
- },
- sanitize: function() {
- if (this.zoomFactor < this.options.zoomOutFactor) {
- this.zoomOutAnimation();
- } else if (this.isInsaneOffset(this.offset)) {
- this.sanitizeOffsetAnimation();
- }
- },
- /**
- * Checks if the offset is ok with the current zoom factor
- * @param offset
- * @return {Boolean}
- */
- isInsaneOffset: function(offset) {
- var sanitizedOffset = this.sanitizeOffset(offset);
- return sanitizedOffset.x !== offset.x ||
- sanitizedOffset.y !== offset.y;
- },
- /**
- * Creates an animation moving to a sane offset
- */
- sanitizeOffsetAnimation: function() {
- var targetOffset = this.sanitizeOffset(this.offset),
- startOffset = {
- x: this.offset.x,
- y: this.offset.y
- },
- updateProgress = (function(progress) {
- this.offset.x = startOffset.x + progress * (targetOffset.x - startOffset.x);
- this.offset.y = startOffset.y + progress * (targetOffset.y - startOffset.y);
- this.update();
- }).bind(this);
- this.animate(
- this.options.animationDuration,
- updateProgress,
- this.swing
- );
- },
- /**
- * Zooms back to the original position,
- * (no offset and zoom factor 1)
- */
- zoomOutAnimation: function() {
- var startZoomFactor = this.zoomFactor,
- zoomFactor = 1,
- center = this.getCurrentZoomCenter(),
- updateProgress = (function(progress) {
- this.scaleTo(startZoomFactor + progress * (zoomFactor - startZoomFactor), center);
- }).bind(this);
- this.animate(
- this.options.animationDuration,
- updateProgress,
- this.swing
- );
- },
- /**
- * Updates the aspect ratio
- */
- updateAspectRatio: function() {
- this.setContainerY(this.getContainerX() / this.getAspectRatio());
- },
- /**
- * Calculates the initial zoom factor (for the element to fit into the container)
- * @return the initial zoom factor
- */
- getInitialZoomFactor: function() {
- // use .offsetWidth instead of width()
- // because jQuery-width() return the original width but Zepto-width() will calculate width with transform.
- // the same as .height()
- return this.container[0].offsetWidth / this.el[0].offsetWidth;
- },
- /**
- * Calculates the aspect ratio of the element
- * @return the aspect ratio
- */
- getAspectRatio: function() {
- return this.el[0].offsetWidth / this.el[0].offsetHeight;
- },
- /**
- * Calculates the virtual zoom center for the current offset and zoom factor
- * (used for reverse zoom)
- * @return {Object} the current zoom center
- */
- getCurrentZoomCenter: function() {
- // uses following formula to calculate the zoom center x value
- // offset_left / offset_right = zoomcenter_x / (container_x - zoomcenter_x)
- var length = this.container[0].offsetWidth * this.zoomFactor,
- offsetLeft = this.offset.x,
- offsetRight = length - offsetLeft - this.container[0].offsetWidth,
- widthOffsetRatio = offsetLeft / offsetRight,
- centerX = widthOffsetRatio * this.container[0].offsetWidth / (widthOffsetRatio + 1),
- // the same for the zoomcenter y
- height = this.container[0].offsetHeight * this.zoomFactor,
- offsetTop = this.offset.y,
- offsetBottom = height - offsetTop - this.container[0].offsetHeight,
- heightOffsetRatio = offsetTop / offsetBottom,
- centerY = heightOffsetRatio * this.container[0].offsetHeight / (heightOffsetRatio + 1);
- // prevents division by zero
- if (offsetRight === 0) {
- centerX = this.container[0].offsetWidth;
- }
- if (offsetBottom === 0) {
- centerY = this.container[0].offsetHeight;
- }
- return {
- x: centerX,
- y: centerY
- };
- },
- canDrag: function() {
- return !isCloseTo(this.zoomFactor, 1);
- },
- /**
- * Returns the touches of an event relative to the container offset
- * @param event
- * @return array touches
- */
- getTouches: function(event) {
- var position = this.container.offset();
- return Array.prototype.slice.call(event.touches).map(function(touch) {
- return {
- x: touch.pageX - position.left,
- y: touch.pageY - position.top
- };
- });
- },
- /**
- * Animation loop
- * does not support simultaneous animations
- * @param duration
- * @param framefn
- * @param timefn
- * @param callback
- */
- animate: function(duration, framefn, timefn, callback) {
- var startTime = new Date().getTime(),
- renderFrame = (function() {
- if (!this.inAnimation) {
- return;
- }
- var frameTime = new Date().getTime() - startTime,
- progress = frameTime / duration;
- if (frameTime >= duration) {
- framefn(1);
- if (callback) {
- callback();
- }
- this.update();
- this.stopAnimation();
- this.update();
- } else {
- if (timefn) {
- progress = timefn(progress);
- }
- framefn(progress);
- this.update();
- requestAnimationFrame(renderFrame);
- }
- }).bind(this);
- this.inAnimation = true;
- requestAnimationFrame(renderFrame);
- },
- /**
- * Stops the animation
- */
- stopAnimation: function() {
- this.inAnimation = false;
- },
- /**
- * Swing timing function for animations
- * @param p
- * @return {Number}
- */
- swing: function(p) {
- return -Math.cos(p * Math.PI) / 2 + 0.5;
- },
- getContainerX: function() {
- return this.container[0].offsetWidth;
- },
- getContainerY: function() {
- return this.container[0].offsetHeight;
- },
- setContainerY: function(y) {
- return this.container.height(y);
- },
- /**
- * Creates the expected html structure
- */
- setupMarkup: function() {
- this.container = $('<div class="pinch-zoom-container"></div>');
- this.el.before(this.container);
- this.container.append(this.el);
- this.container.css({
- 'overflow': 'hidden',
- 'position': 'relative'
- });
- // Zepto doesn't recognize `webkitTransform..` style
- this.el.css({
- '-webkit-transform-origin': '0% 0%',
- '-moz-transform-origin': '0% 0%',
- '-ms-transform-origin': '0% 0%',
- '-o-transform-origin': '0% 0%',
- 'transform-origin': '0% 0%',
- 'position': 'absolute'
- });
- },
- end: function() {
- this.hasInteraction = false;
- this.sanitize();
- this.update();
- },
- /**
- * Binds all required event listeners
- */
- bindEvents: function() {
- detectGestures(this.container.get(0), this);
- // Zepto and jQuery both know about `on`
- $(window).on('resize', this.update.bind(this));
- $(this.el).find('img').on('load', this.update.bind(this));
- },
- /**
- * Updates the css values according to the current zoom factor and offset
- */
- update: function() {
- if (this.updatePlaned) {
- return;
- }
- this.updatePlaned = true;
- setTimeout((function() {
- this.updatePlaned = false;
- this.updateAspectRatio();
- var zoomFactor = this.getInitialZoomFactor() * this.zoomFactor,
- offsetX = -this.offset.x / zoomFactor,
- offsetY = -this.offset.y / zoomFactor,
- transform3d = 'scale3d(' + zoomFactor + ', ' + zoomFactor + ',1) ' +
- 'translate3d(' + offsetX + 'px,' + offsetY + 'px,0px)',
- transform2d = 'scale(' + zoomFactor + ', ' + zoomFactor + ') ' +
- 'translate(' + offsetX + 'px,' + offsetY + 'px)',
- removeClone = (function() {
- if (this.clone) {
- this.clone.remove();
- delete this.clone;
- }
- }).bind(this);
- // Scale 3d and translate3d are faster (at least on ios)
- // but they also reduce the quality.
- // PinchZoom uses the 3d transformations during interactions
- // after interactions it falls back to 2d transformations
- if (!this.options.use2d || this.hasInteraction || this.inAnimation) {
- this.is3d = true;
- removeClone();
- this.el.css({
- '-webkit-transform': transform3d,
- '-o-transform': transform2d,
- '-ms-transform': transform2d,
- '-moz-transform': transform2d,
- 'transform': transform3d
- });
- } else {
- // When changing from 3d to 2d transform webkit has some glitches.
- // To avoid this, a copy of the 3d transformed element is displayed in the
- // foreground while the element is converted from 3d to 2d transform
- if (this.is3d) {
- this.clone = this.el.clone();
- this.clone.css('pointer-events', 'none');
- this.clone.appendTo(this.container);
- setTimeout(removeClone, 200);
- }
- this.el.css({
- '-webkit-transform': transform2d,
- '-o-transform': transform2d,
- '-ms-transform': transform2d,
- '-moz-transform': transform2d,
- 'transform': transform2d
- });
- this.is3d = false;
- }
- }).bind(this), 0);
- },
- /**
- * Enables event handling for gestures
- */
- enable: function() {
- this.enabled = true;
- },
- /**
- * Disables event handling for gestures
- */
- disable: function() {
- this.enabled = false;
- }
- };
- var detectGestures = function(el, target) {
- var interaction = null,
- fingers = 0,
- lastTouchStart = null,
- startTouches = null,
- setInteraction = function(newInteraction, event) {
- if (interaction !== newInteraction) {
- if (interaction && !newInteraction) {
- switch (interaction) {
- case "zoom":
- target.handleZoomEnd(event);
- break;
- case 'drag':
- target.handleDragEnd(event);
- break;
- }
- }
- switch (newInteraction) {
- case 'zoom':
- target.handleZoomStart(event);
- break;
- case 'drag':
- target.handleDragStart(event);
- break;
- }
- }
- interaction = newInteraction;
- },
- updateInteraction = function(event) {
- if (fingers === 2) {
- setInteraction('zoom');
- } else if (fingers === 1 && target.canDrag()) {
- setInteraction('drag', event);
- } else {
- setInteraction(null, event);
- }
- },
- targetTouches = function(touches) {
- return Array.prototype.slice.call(touches).map(function(touch) {
- return {
- x: touch.pageX,
- y: touch.pageY
- };
- });
- },
- getDistance = function(a, b) {
- var x, y;
- x = a.x - b.x;
- y = a.y - b.y;
- return Math.sqrt(x * x + y * y);
- },
- calculateScale = function(startTouches, endTouches) {
- var startDistance = getDistance(startTouches[0], startTouches[1]),
- endDistance = getDistance(endTouches[0], endTouches[1]);
- return endDistance / startDistance;
- },
- cancelEvent = function(event) {
- event.stopPropagation();
- event.preventDefault();
- },
- detectDoubleTap = function(event) {
- var time = (new Date()).getTime();
- if (fingers > 1) {
- lastTouchStart = null;
- }
- if (time - lastTouchStart < 300) {
- cancelEvent(event);
- target.handleDoubleTap(event);
- switch (interaction) {
- case 'zoom':
- target.handleZoomEnd(event);
- break;
- case 'drag':
- target.handleDragEnd(event);
- break;
- }
- }
- if (fingers === 1) {
- lastTouchStart = time;
- }
- },
- firstMove = true;
- el.addEventListener('touchstart', function(event) {
- if (target.enabled) {
- firstMove = true;
- fingers = event.touches.length;
- detectDoubleTap(event);
- }
- });
- el.addEventListener('touchmove', function(event) {
- if (target.enabled) {
- if (firstMove) {
- updateInteraction(event);
- if (interaction) {
- cancelEvent(event);
- }
- startTouches = targetTouches(event.touches);
- } else {
- switch (interaction) {
- case 'zoom':
- target.handleZoom(event, calculateScale(startTouches, targetTouches(event.touches)));
- break;
- case 'drag':
- target.handleDrag(event);
- break;
- }
- if (interaction) {
- cancelEvent(event);
- target.update();
- }
- }
- firstMove = false;
- }
- });
- el.addEventListener('touchend', function(event) {
- if (target.enabled) {
- fingers = event.touches.length;
- updateInteraction(event);
- }
- });
- };
- return PinchZoom;
- };
- module.exports = UI.pichzoom = definePinchZoom($);
- /***/ },
- /* 18 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var $w = $(window);
- /**
- * @reference https://github.com/nolimits4web/Framework7/blob/master/src/js/modals.js
- * @license https://github.com/nolimits4web/Framework7/blob/master/LICENSE
- */
- var Popover = function(element, options) {
- this.options = $.extend({}, Popover.DEFAULTS, options);
- this.$element = $(element);
- this.active = null;
- this.$popover = (this.options.target && $(this.options.target)) || null;
- this.init();
- this._bindEvents();
- };
- Popover.DEFAULTS = {
- theme: null,
- trigger: 'click',
- content: '',
- open: false,
- target: null,
- tpl: '<div class="am-popover">' +
- '<div class="am-popover-inner"></div>' +
- '<div class="am-popover-caret"></div></div>'
- };
- Popover.prototype.init = function() {
- var _this = this;
- var $element = this.$element;
- var $popover;
- if (!this.options.target) {
- this.$popover = this.getPopover();
- this.setContent();
- }
- $popover = this.$popover;
- $popover.appendTo($('body'));
- this.sizePopover();
- function sizePopover() {
- _this.sizePopover();
- }
- // TODO: 监听页面内容变化,重新调整位置
- $element.on('open.popover.amui', function() {
- $(window).on('resize.popover.amui', UI.utils.debounce(sizePopover, 50));
- });
- $element.on('close.popover.amui', function() {
- $(window).off('resize.popover.amui', sizePopover);
- });
- this.options.open && this.open();
- };
- Popover.prototype.sizePopover = function sizePopover() {
- var $element = this.$element;
- var $popover = this.$popover;
- if (!$popover || !$popover.length) {
- return;
- }
- var popWidth = $popover.outerWidth();
- var popHeight = $popover.outerHeight();
- var $popCaret = $popover.find('.am-popover-caret');
- var popCaretSize = ($popCaret.outerWidth() / 2) || 8;
- // 取不到 $popCaret.outerHeight() 的值,所以直接加 8
- var popTotalHeight = popHeight + 8; // $popCaret.outerHeight();
- var triggerWidth = $element.outerWidth();
- var triggerHeight = $element.outerHeight();
- var triggerOffset = $element.offset();
- var triggerRect = $element[0].getBoundingClientRect();
- var winHeight = $w.height();
- var winWidth = $w.width();
- var popTop = 0;
- var popLeft = 0;
- var diff = 0;
- var spacing = 2;
- var popPosition = 'top';
- $popover.css({left: '', top: ''}).removeClass('am-popover-left ' +
- 'am-popover-right am-popover-top am-popover-bottom');
- // $popCaret.css({left: '', top: ''});
- if (popTotalHeight - spacing < triggerRect.top + spacing) {
- // Popover on the top of trigger
- popTop = triggerOffset.top - popTotalHeight - spacing;
- } else if (popTotalHeight <
- winHeight - triggerRect.top - triggerRect.height) {
- // On bottom
- popPosition = 'bottom';
- popTop = triggerOffset.top + triggerHeight + popCaretSize + spacing;
- } else { // On middle
- popPosition = 'middle';
- popTop = triggerHeight / 2 + triggerOffset.top - popHeight / 2;
- }
- // Horizontal Position
- if (popPosition === 'top' || popPosition === 'bottom') {
- popLeft = triggerWidth / 2 + triggerOffset.left - popWidth / 2;
- diff = popLeft;
- if (popLeft < 5) {
- popLeft = 5;
- }
- if (popLeft + popWidth > winWidth) {
- popLeft = (winWidth - popWidth - 20);
- // console.log('left %d, win %d, popw %d', popLeft, winWidth, popWidth);
- }
- if (popPosition === 'top') {
- // This is the Popover position, NOT caret position
- // Popover on the Top of trigger, caret on the bottom of Popover
- $popover.addClass('am-popover-top');
- }
- if (popPosition === 'bottom') {
- $popover.addClass('am-popover-bottom');
- }
- diff = diff - popLeft;
- // $popCaret.css({left: (popWidth / 2 - popCaretSize + diff) + 'px'});
- } else if (popPosition === 'middle') {
- popLeft = triggerOffset.left - popWidth - popCaretSize;
- $popover.addClass('am-popover-left');
- if (popLeft < 5) {
- popLeft = triggerOffset.left + triggerWidth + popCaretSize;
- $popover.removeClass('am-popover-left').addClass('am-popover-right');
- }
- if (popLeft + popWidth > winWidth) {
- popLeft = winWidth - popWidth - 5;
- $popover.removeClass('am-popover-left').addClass('am-popover-right');
- }
- // $popCaret.css({top: (popHeight / 2 - popCaretSize / 2) + 'px'});
- }
- // Apply position style
- $popover.css({top: popTop + 'px', left: popLeft + 'px'});
- };
- Popover.prototype.toggle = function() {
- return this[this.active ? 'close' : 'open']();
- };
- Popover.prototype.open = function() {
- var $popover = this.$popover;
- this.$element.trigger('open.popover.amui');
- this.sizePopover();
- $popover.show().addClass('am-active');
- this.active = true;
- };
- Popover.prototype.close = function() {
- var $popover = this.$popover;
- this.$element.trigger('close.popover.amui');
- $popover
- .removeClass('am-active')
- .trigger('closed.popover.amui')
- .hide();
- this.active = false;
- };
- Popover.prototype.getPopover = function() {
- var uid = UI.utils.generateGUID('am-popover');
- var theme = [];
- if (this.options.theme) {
- $.each(this.options.theme.split(' '), function(i, item) {
- theme.push('am-popover-' + $.trim(item));
- });
- }
- return $(this.options.tpl).attr('id', uid).addClass(theme.join(' '));
- };
- Popover.prototype.setContent = function(content) {
- content = content || this.options.content;
- this.$popover && this.$popover.find('.am-popover-inner')
- .empty().html(content);
- };
- Popover.prototype._bindEvents = function() {
- var eventNS = 'popover.amui';
- var triggers = this.options.trigger.split(' ');
- for (var i = triggers.length; i--;) {
- var trigger = triggers[i];
- if (trigger === 'click') {
- this.$element.on('click.' + eventNS, $.proxy(this.toggle, this));
- } else { // hover or focus
- var eventIn = trigger == 'hover' ? 'mouseenter' : 'focusin';
- var eventOut = trigger == 'hover' ? 'mouseleave' : 'focusout';
- this.$element.on(eventIn + '.' + eventNS, $.proxy(this.open, this));
- this.$element.on(eventOut + '.' + eventNS, $.proxy(this.close, this));
- }
- }
- };
- Popover.prototype.destroy = function() {
- this.$element.off('.popover.amui').removeData('amui.popover');
- this.$popover.remove();
- };
- UI.plugin('popover', Popover);
- // Init code
- UI.ready(function(context) {
- $('[data-am-popover]', context).popover();
- });
- module.exports = Popover;
- // TODO: 允许用户定义位置
- /***/ },
- /* 19 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- var Progress = (function() {
- /**
- * NProgress (c) 2013, Rico Sta. Cruz
- * @via http://ricostacruz.com/nprogress
- */
- var NProgress = {};
- NProgress.version = '0.2.0';
- var Settings = NProgress.settings = {
- minimum: 0.08,
- easing: 'ease',
- positionUsing: '',
- speed: 200,
- trickle: true,
- trickleRate: 0.02,
- trickleSpeed: 800,
- showSpinner: true,
- parent: 'body',
- barSelector: '[role="nprogress-bar"]',
- spinnerSelector: '[role="nprogress-spinner"]',
- template: '<div class="nprogress-bar" role="nprogress-bar">' +
- '<div class="nprogress-peg"></div></div>' +
- '<div class="nprogress-spinner" role="nprogress-spinner">' +
- '<div class="nprogress-spinner-icon"></div></div>'
- };
- /**
- * Updates configuration.
- *
- * NProgress.configure({
- * minimum: 0.1
- * });
- */
- NProgress.configure = function(options) {
- var key, value;
- for (key in options) {
- value = options[key];
- if (value !== undefined && options.hasOwnProperty(key)) Settings[key] = value;
- }
- return this;
- };
- /**
- * Last number.
- */
- NProgress.status = null;
- /**
- * Sets the progress bar status, where `n` is a number from `0.0` to `1.0`.
- *
- * NProgress.set(0.4);
- * NProgress.set(1.0);
- */
- NProgress.set = function(n) {
- var started = NProgress.isStarted();
- n = clamp(n, Settings.minimum, 1);
- NProgress.status = (n === 1 ? null : n);
- var progress = NProgress.render(!started),
- bar = progress.querySelector(Settings.barSelector),
- speed = Settings.speed,
- ease = Settings.easing;
- progress.offsetWidth; /* Repaint */
- queue(function(next) {
- // Set positionUsing if it hasn't already been set
- if (Settings.positionUsing === '') Settings.positionUsing = NProgress.getPositioningCSS();
- // Add transition
- css(bar, barPositionCSS(n, speed, ease));
- if (n === 1) {
- // Fade out
- css(progress, {
- transition: 'none',
- opacity: 1
- });
- progress.offsetWidth; /* Repaint */
- setTimeout(function() {
- css(progress, {
- transition: 'all ' + speed + 'ms linear',
- opacity: 0
- });
- setTimeout(function() {
- NProgress.remove();
- next();
- }, speed);
- }, speed);
- } else {
- setTimeout(next, speed);
- }
- });
- return this;
- };
- NProgress.isStarted = function() {
- return typeof NProgress.status === 'number';
- };
- /**
- * Shows the progress bar.
- * This is the same as setting the status to 0%, except that it doesn't go backwards.
- *
- * NProgress.start();
- *
- */
- NProgress.start = function() {
- if (!NProgress.status) NProgress.set(0);
- var work = function() {
- setTimeout(function() {
- if (!NProgress.status) return;
- NProgress.trickle();
- work();
- }, Settings.trickleSpeed);
- };
- if (Settings.trickle) work();
- return this;
- };
- /**
- * Hides the progress bar.
- * This is the *sort of* the same as setting the status to 100%, with the
- * difference being `done()` makes some placebo effect of some realistic motion.
- *
- * NProgress.done();
- *
- * If `true` is passed, it will show the progress bar even if its hidden.
- *
- * NProgress.done(true);
- */
- NProgress.done = function(force) {
- if (!force && !NProgress.status) return this;
- return NProgress.inc(0.3 + 0.5 * Math.random()).set(1);
- };
- /**
- * Increments by a random amount.
- */
- NProgress.inc = function(amount) {
- var n = NProgress.status;
- if (!n) {
- return NProgress.start();
- } else {
- if (typeof amount !== 'number') {
- amount = (1 - n) * clamp(Math.random() * n, 0.1, 0.95);
- }
- n = clamp(n + amount, 0, 0.994);
- return NProgress.set(n);
- }
- };
- NProgress.trickle = function() {
- return NProgress.inc(Math.random() * Settings.trickleRate);
- };
- /**
- * Waits for all supplied jQuery promises and
- * increases the progress as the promises resolve.
- *
- * @param $promise jQUery Promise
- */
- (function() {
- var initial = 0, current = 0;
- NProgress.promise = function($promise) {
- if (!$promise || $promise.state() === "resolved") {
- return this;
- }
- if (current === 0) {
- NProgress.start();
- }
- initial++;
- current++;
- $promise.always(function() {
- current--;
- if (current === 0) {
- initial = 0;
- NProgress.done();
- } else {
- NProgress.set((initial - current) / initial);
- }
- });
- return this;
- };
- })();
- /**
- * (Internal) renders the progress bar markup based on the `template`
- * setting.
- */
- NProgress.render = function(fromStart) {
- if (NProgress.isRendered()) return document.getElementById('nprogress');
- addClass(document.documentElement, 'nprogress-busy');
- var progress = document.createElement('div');
- progress.id = 'nprogress';
- progress.innerHTML = Settings.template;
- var bar = progress.querySelector(Settings.barSelector),
- perc = fromStart ? '-100' : toBarPerc(NProgress.status || 0),
- parent = document.querySelector(Settings.parent),
- spinner;
- css(bar, {
- transition: 'all 0 linear',
- transform: 'translate3d(' + perc + '%,0,0)'
- });
- if (!Settings.showSpinner) {
- spinner = progress.querySelector(Settings.spinnerSelector);
- spinner && removeElement(spinner);
- }
- if (parent != document.body) {
- addClass(parent, 'nprogress-custom-parent');
- }
- parent.appendChild(progress);
- return progress;
- };
- /**
- * Removes the element. Opposite of render().
- */
- NProgress.remove = function() {
- removeClass(document.documentElement, 'nprogress-busy');
- removeClass(document.querySelector(Settings.parent), 'nprogress-custom-parent');
- var progress = document.getElementById('nprogress');
- progress && removeElement(progress);
- };
- /**
- * Checks if the progress bar is rendered.
- */
- NProgress.isRendered = function() {
- return !!document.getElementById('nprogress');
- };
- /**
- * Determine which positioning CSS rule to use.
- */
- NProgress.getPositioningCSS = function() {
- // Sniff on document.body.style
- var bodyStyle = document.body.style;
- // Sniff prefixes
- var vendorPrefix = ('WebkitTransform' in bodyStyle) ? 'Webkit' :
- ('MozTransform' in bodyStyle) ? 'Moz' :
- ('msTransform' in bodyStyle) ? 'ms' :
- ('OTransform' in bodyStyle) ? 'O' : '';
- if (vendorPrefix + 'Perspective' in bodyStyle) {
- // Modern browsers with 3D support, e.g. Webkit, IE10
- return 'translate3d';
- } else if (vendorPrefix + 'Transform' in bodyStyle) {
- // Browsers without 3D support, e.g. IE9
- return 'translate';
- } else {
- // Browsers without translate() support, e.g. IE7-8
- return 'margin';
- }
- };
- /**
- * Helpers
- */
- function clamp(n, min, max) {
- if (n < min) return min;
- if (n > max) return max;
- return n;
- }
- /**
- * (Internal) converts a percentage (`0..1`) to a bar translateX
- * percentage (`-100%..0%`).
- */
- function toBarPerc(n) {
- return (-1 + n) * 100;
- }
- /**
- * (Internal) returns the correct CSS for changing the bar's
- * position given an n percentage, and speed and ease from Settings
- */
- function barPositionCSS(n, speed, ease) {
- var barCSS;
- if (Settings.positionUsing === 'translate3d') {
- barCSS = { transform: 'translate3d('+toBarPerc(n)+'%,0,0)' };
- } else if (Settings.positionUsing === 'translate') {
- barCSS = { transform: 'translate('+toBarPerc(n)+'%,0)' };
- } else {
- barCSS = { 'margin-left': toBarPerc(n)+'%' };
- }
- barCSS.transition = 'all '+speed+'ms '+ease;
- return barCSS;
- }
- /**
- * (Internal) Queues a function to be executed.
- */
- var queue = (function() {
- var pending = [];
- function next() {
- var fn = pending.shift();
- if (fn) {
- fn(next);
- }
- }
- return function(fn) {
- pending.push(fn);
- if (pending.length == 1) next();
- };
- })();
- /**
- * (Internal) Applies css properties to an element, similar to the jQuery
- * css method.
- *
- * While this helper does assist with vendor prefixed property names, it
- * does not perform any manipulation of values prior to setting styles.
- */
- var css = (function() {
- var cssPrefixes = [ 'Webkit', 'O', 'Moz', 'ms' ],
- cssProps = {};
- function camelCase(string) {
- return string.replace(/^-ms-/, 'ms-').replace(/-([\da-z])/gi, function(match, letter) {
- return letter.toUpperCase();
- });
- }
- function getVendorProp(name) {
- var style = document.body.style;
- if (name in style) return name;
- var i = cssPrefixes.length,
- capName = name.charAt(0).toUpperCase() + name.slice(1),
- vendorName;
- while (i--) {
- vendorName = cssPrefixes[i] + capName;
- if (vendorName in style) return vendorName;
- }
- return name;
- }
- function getStyleProp(name) {
- name = camelCase(name);
- return cssProps[name] || (cssProps[name] = getVendorProp(name));
- }
- function applyCss(element, prop, value) {
- prop = getStyleProp(prop);
- element.style[prop] = value;
- }
- return function(element, properties) {
- var args = arguments,
- prop,
- value;
- if (args.length == 2) {
- for (prop in properties) {
- value = properties[prop];
- if (value !== undefined && properties.hasOwnProperty(prop)) applyCss(element, prop, value);
- }
- } else {
- applyCss(element, args[1], args[2]);
- }
- }
- })();
- /**
- * (Internal) Determines if an element or space separated list of class names contains a class name.
- */
- function hasClass(element, name) {
- var list = typeof element == 'string' ? element : classList(element);
- return list.indexOf(' ' + name + ' ') >= 0;
- }
- /**
- * (Internal) Adds a class to an element.
- */
- function addClass(element, name) {
- var oldList = classList(element),
- newList = oldList + name;
- if (hasClass(oldList, name)) return;
- // Trim the opening space.
- element.className = newList.substring(1);
- }
- /**
- * (Internal) Removes a class from an element.
- */
- function removeClass(element, name) {
- var oldList = classList(element),
- newList;
- if (!hasClass(element, name)) return;
- // Replace the class name.
- newList = oldList.replace(' ' + name + ' ', ' ');
- // Trim the opening and closing spaces.
- element.className = newList.substring(1, newList.length - 1);
- }
- /**
- * (Internal) Gets a space separated list of the class names on the element.
- * The list is wrapped with a single space on each end to facilitate finding
- * matches within the list.
- */
- function classList(element) {
- return (' ' + (element.className || '') + ' ').replace(/\s+/gi, ' ');
- }
- /**
- * (Internal) Removes an element from the DOM.
- */
- function removeElement(element) {
- element && element.parentNode && element.parentNode.removeChild(element);
- }
- return NProgress;
- })();
- module.exports = UI.progress = Progress;
- /***/ },
- /* 20 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var PinchZoom = __webpack_require__(17);
- var Hammer = __webpack_require__(3);
- var animation = UI.support.animation;
- var transition = UI.support.transition;
- /**
- * PureView
- * @desc Image browser for Mobile
- * @param element
- * @param options
- * @constructor
- */
- var PureView = function(element, options) {
- this.$element = $(element);
- this.$body = $(document.body);
- this.options = $.extend({}, PureView.DEFAULTS, options);
- this.$pureview = $(this.options.tpl).attr('id',
- UI.utils.generateGUID('am-pureview'));
- this.$slides = null;
- this.transitioning = null;
- this.scrollbarWidth = 0;
- this.init();
- };
- PureView.DEFAULTS = {
- tpl: '<div class="am-pureview am-pureview-bar-active">' +
- '<ul class="am-pureview-slider"></ul>' +
- '<ul class="am-pureview-direction">' +
- '<li class="am-pureview-prev"><a href=""></a></li>' +
- '<li class="am-pureview-next"><a href=""></a></li></ul>' +
- '<ol class="am-pureview-nav"></ol>' +
- '<div class="am-pureview-bar am-active">' +
- '<span class="am-pureview-title"></span>' +
- '<div class="am-pureview-counter"><span class="am-pureview-current"></span> / ' +
- '<span class="am-pureview-total"></span></div></div>' +
- '<div class="am-pureview-actions am-active">' +
- '<a href="javascript: void(0)" class="am-icon-chevron-left" ' +
- 'data-am-close="pureview"></a></div>' +
- '</div>',
- className: {
- prevSlide: 'am-pureview-slide-prev',
- nextSlide: 'am-pureview-slide-next',
- onlyOne: 'am-pureview-only',
- active: 'am-active',
- barActive: 'am-pureview-bar-active',
- activeBody: 'am-pureview-active'
- },
- selector: {
- slider: '.am-pureview-slider',
- close: '[data-am-close="pureview"]',
- total: '.am-pureview-total',
- current: '.am-pureview-current',
- title: '.am-pureview-title',
- actions: '.am-pureview-actions',
- bar: '.am-pureview-bar',
- pinchZoom: '.am-pinch-zoom',
- nav: '.am-pureview-nav'
- },
- shareBtn: false,
- // press to toggle Toolbar
- toggleToolbar: true,
- // 从何处获取图片,img 可以使用 data-rel 指定大图
- target: 'img',
- // 微信 Webview 中调用微信的图片浏览器
- // 实现图片保存、分享好友、收藏图片等功能
- weChatImagePreview: true
- };
- PureView.prototype.init = function() {
- var _this = this;
- var options = this.options;
- var $element = this.$element;
- var $pureview = this.$pureview;
- this.refreshSlides();
- $('body').append($pureview);
- this.$title = $pureview.find(options.selector.title);
- this.$current = $pureview.find(options.selector.current);
- this.$bar = $pureview.find(options.selector.bar);
- this.$actions = $pureview.find(options.selector.actions);
- if (options.shareBtn) {
- this.$actions.append('<a href="javascript: void(0)" ' +
- 'class="am-icon-share-square-o" data-am-toggle="share"></a>');
- }
- this.$element.on('click.pureview.amui', options.target, function(e) {
- e.preventDefault();
- var clicked = _this.$images.index(this);
- // Invoke WeChat ImagePreview in WeChat
- // TODO: detect WeChat before init
- if (options.weChatImagePreview && window.WeixinJSBridge) {
- window.WeixinJSBridge.invoke('imagePreview', {
- current: _this.imgUrls[clicked],
- urls: _this.imgUrls
- });
- } else {
- _this.open(clicked);
- }
- });
- $pureview.find('.am-pureview-direction').
- on('click.direction.pureview.amui', 'li', function(e) {
- e.preventDefault();
- if ($(this).is('.am-pureview-prev')) {
- _this.prevSlide();
- } else {
- _this.nextSlide();
- }
- });
- // Nav Contorl
- $pureview.find(options.selector.nav).on('click.nav.pureview.amui', 'li',
- function() {
- var index = _this.$navItems.index($(this));
- _this.activate(_this.$slides.eq(index));
- });
- // Close Icon
- $pureview.find(options.selector.close).
- on('click.close.pureview.amui', function(e) {
- e.preventDefault();
- _this.close();
- });
- this.$slider.hammer().on('swipeleft.pureview.amui', function(e) {
- e.preventDefault();
- _this.nextSlide();
- }).on('swiperight.pureview.amui', function(e) {
- e.preventDefault();
- _this.prevSlide();
- }).on('press.pureview.amui', function(e) {
- e.preventDefault();
- options.toggleToolbar && _this.toggleToolBar();
- });
- this.$slider.data('hammer').get('swipe').set({
- direction: Hammer.DIRECTION_HORIZONTAL,
- velocity: 0.35
- });
- // Observe DOM
- $element.DOMObserve({
- childList: true,
- subtree: true
- }, function(mutations, observer) {
- // _this.refreshSlides();
- // console.log('mutations[0].type);
- });
- // NOTE:
- // trigger this event manually if MutationObserver not supported
- // when new images appended, or call refreshSlides()
- // if (!UI.support.mutationobserver) $element.trigger('changed.dom.amui')
- $element.on('changed.dom.amui', function(e) {
- e.stopPropagation();
- _this.refreshSlides();
- });
- $(document).on('keydown.pureview.amui', $.proxy(function(e) {
- var keyCode = e.keyCode;
- if (keyCode == 37) {
- this.prevSlide();
- } else if (keyCode == 39) {
- this.nextSlide();
- } else if (keyCode == 27) {
- this.close();
- }
- }, this));
- };
- PureView.prototype.refreshSlides = function() {
- // update images collections
- this.$images = this.$element.find(this.options.target);
- var _this = this;
- var options = this.options;
- var $pureview = this.$pureview;
- var $slides = $([]);
- var $navItems = $([]);
- var $images = this.$images;
- var total = $images.length;
- this.$slider = $pureview.find(options.selector.slider);
- this.$nav = $pureview.find(options.selector.nav);
- var viewedFlag = 'data-am-pureviewed';
- // for WeChat Image Preview
- this.imgUrls = this.imgUrls || [];
- if (!total) {
- return;
- }
- if (total === 1) {
- $pureview.addClass(options.className.onlyOne);
- }
- $images.not('[' + viewedFlag + ']').each(function(i, item) {
- var src;
- var title;
- // get image URI from link's href attribute
- if (item.nodeName === 'A') {
- src = item.href; // to absolute path
- title = item.title || '';
- } else {
- // NOTE: `data-rel` should be a full URL, otherwise,
- // WeChat images preview will not work
- src = $(item).data('rel') || item.src; // <img src='' data-rel='' />
- src = UI.utils.getAbsoluteUrl(src);
- title = $(item).attr('alt') || '';
- }
- // add pureviewed flag
- item.setAttribute(viewedFlag, '1');
- // hide bar: wechat_webview_type=1
- // http://tmt.io/wechat/ not working?
- _this.imgUrls.push(src);
- $slides = $slides.add($('<li data-src="' + src + '" data-title="' + title +
- '"></li>'));
- $navItems = $navItems.add($('<li>' + (i + 1) + '</li>'));
- });
- $pureview.find(options.selector.total).text(total);
- this.$slider.append($slides);
- this.$nav.append($navItems);
- this.$navItems = this.$nav.find('li');
- this.$slides = this.$slider.find('li');
- };
- PureView.prototype.loadImage = function($slide, callback) {
- var appendedFlag = 'image-appended';
- if (!$slide.data(appendedFlag)) {
- var $img = $('<img>', {
- src: $slide.data('src'),
- alt: $slide.data('title')
- });
- $slide.html($img).wrapInner('<div class="am-pinch-zoom"></div>').redraw();
- var $pinchWrapper = $slide.find(this.options.selector.pinchZoom);
- $pinchWrapper.data('amui.pinchzoom', new PinchZoom($pinchWrapper[0], {}));
- $slide.data('image-appended', true);
- }
- callback && callback.call(this);
- };
- PureView.prototype.activate = function($slide) {
- var options = this.options;
- var $slides = this.$slides;
- var activeIndex = $slides.index($slide);
- var title = $slide.data('title') || '';
- var active = options.className.active;
- if ($slides.find('.' + active).is($slide)) {
- return;
- }
- if (this.transitioning) {
- return;
- }
- this.loadImage($slide, function() {
- UI.utils.imageLoader($slide.find('img'), function(image) {
- $slide.find('.am-pinch-zoom').addClass('am-pureview-loaded');
- $(image).addClass('am-img-loaded');
- });
- });
- this.transitioning = 1;
- this.$title.text(title);
- this.$current.text(activeIndex + 1);
- $slides.removeClass();
- $slide.addClass(active);
- $slides.eq(activeIndex - 1).addClass(options.className.prevSlide);
- $slides.eq(activeIndex + 1).addClass(options.className.nextSlide);
- this.$navItems.removeClass().
- eq(activeIndex).addClass(options.className.active);
- if (transition) {
- $slide.one(transition.end, $.proxy(function() {
- this.transitioning = 0;
- }, this)).emulateTransitionEnd(300);
- } else {
- this.transitioning = 0;
- }
- // TODO: pre-load next image
- };
- PureView.prototype.nextSlide = function() {
- if (this.$slides.length === 1) {
- return;
- }
- var $slides = this.$slides;
- var $active = $slides.filter('.am-active');
- var activeIndex = $slides.index($active);
- var rightSpring = 'am-animation-right-spring';
- if (activeIndex + 1 >= $slides.length) { // last one
- animation && $active.addClass(rightSpring).on(animation.end, function() {
- $active.removeClass(rightSpring);
- });
- } else {
- this.activate($slides.eq(activeIndex + 1));
- }
- };
- PureView.prototype.prevSlide = function() {
- if (this.$slides.length === 1) {
- return;
- }
- var $slides = this.$slides;
- var $active = $slides.filter('.am-active');
- var activeIndex = this.$slides.index(($active));
- var leftSpring = 'am-animation-left-spring';
- if (activeIndex === 0) { // first one
- animation && $active.addClass(leftSpring).on(animation.end, function() {
- $active.removeClass(leftSpring);
- });
- } else {
- this.activate($slides.eq(activeIndex - 1));
- }
- };
- PureView.prototype.toggleToolBar = function() {
- this.$pureview.toggleClass(this.options.className.barActive);
- };
- PureView.prototype.open = function(index) {
- var active = index || 0;
- this.checkScrollbar();
- this.setScrollbar();
- this.activate(this.$slides.eq(active));
- this.$pureview.show().redraw().addClass(this.options.className.active);
- this.$body.addClass(this.options.className.activeBody);
- };
- PureView.prototype.close = function() {
- var options = this.options;
- this.$pureview.removeClass(options.className.active);
- this.$slides.removeClass();
- function resetBody() {
- this.$pureview.hide();
- this.$body.removeClass(options.className.activeBody);
- this.resetScrollbar();
- }
- if (transition) {
- this.$pureview.one(transition.end, $.proxy(resetBody, this)).
- emulateTransitionEnd(300);
- } else {
- resetBody.call(this);
- }
- };
- PureView.prototype.checkScrollbar = function() {
- this.scrollbarWidth = UI.utils.measureScrollbar();
- };
- PureView.prototype.setScrollbar = function() {
- var bodyPaddingRight = parseInt((this.$body.css('padding-right') || 0), 10);
- if (this.scrollbarWidth) {
- this.$body.css('padding-right', bodyPaddingRight + this.scrollbarWidth);
- }
- };
- PureView.prototype.resetScrollbar = function() {
- this.$body.css('padding-right', '');
- };
- UI.plugin('pureview', PureView);
- // Init code
- UI.ready(function(context) {
- $('[data-am-pureview]', context).pureview();
- });
- module.exports = PureView;
- // TODO: 1. 动画改进
- // 2. 改变图片的时候恢复 Zoom
- // 3. 选项
- // 4. 图片高度问题:由于 PinchZoom 的原因,过高的图片如果设置看了滚动,则放大以后显示不全
- /***/ },
- /* 21 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * @via https://github.com/uikit/uikit/blob/master/src/js/scrollspy.js
- * @license https://github.com/uikit/uikit/blob/master/LICENSE.md
- */
- var ScrollSpy = function(element, options) {
- if (!UI.support.animation) {
- return;
- }
- this.options = $.extend({}, ScrollSpy.DEFAULTS, options);
- this.$element = $(element);
- var checkViewRAF = function() {
- UI.utils.rAF.call(window, $.proxy(this.checkView, this));
- }.bind(this);
- this.$window = $(window).on('scroll.scrollspy.amui', checkViewRAF)
- .on('resize.scrollspy.amui orientationchange.scrollspy.amui',
- UI.utils.debounce(checkViewRAF, 50));
- this.timer = this.inViewState = this.initInView = null;
- checkViewRAF();
- };
- ScrollSpy.DEFAULTS = {
- animation: 'fade',
- className: {
- inView: 'am-scrollspy-inview',
- init: 'am-scrollspy-init'
- },
- repeat: true,
- delay: 0,
- topOffset: 0,
- leftOffset: 0
- };
- ScrollSpy.prototype.checkView = function() {
- var $element = this.$element;
- var options = this.options;
- var inView = UI.utils.isInView($element, options);
- var animation = options.animation ?
- ' am-animation-' + options.animation : '';
- if (inView && !this.inViewState) {
- if (this.timer) {
- clearTimeout(this.timer);
- }
- if (!this.initInView) {
- $element.addClass(options.className.init);
- this.offset = $element.offset();
- this.initInView = true;
- $element.trigger('init.scrollspy.amui');
- }
- this.timer = setTimeout(function() {
- if (inView) {
- $element.addClass(options.className.inView + animation).width();
- }
- }, options.delay);
- this.inViewState = true;
- $element.trigger('inview.scrollspy.amui');
- }
- if (!inView && this.inViewState && options.repeat) {
- $element.removeClass(options.className.inView + animation);
- this.inViewState = false;
- $element.trigger('outview.scrollspy.amui');
- }
- };
- ScrollSpy.prototype.check = function() {
- UI.utils.rAF.call(window, $.proxy(this.checkView, this));
- };
- // Sticky Plugin
- UI.plugin('scrollspy', ScrollSpy);
- // Init code
- UI.ready(function(context) {
- $('[data-am-scrollspy]', context).scrollspy();
- });
- module.exports = ScrollSpy;
- /***/ },
- /* 22 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(23);
- /**
- * @via https://github.com/uikit/uikit/
- * @license https://github.com/uikit/uikit/blob/master/LICENSE.md
- */
- // ScrollSpyNav Class
- var ScrollSpyNav = function(element, options) {
- this.options = $.extend({}, ScrollSpyNav.DEFAULTS, options);
- this.$element = $(element);
- this.anchors = [];
- this.$links = this.$element.find('a[href^="#"]').each(function(i, link) {
- this.anchors.push($(link).attr('href'));
- }.bind(this));
- this.$targets = $(this.anchors.join(', '));
- var processRAF = function() {
- UI.utils.rAF.call(window, $.proxy(this.process, this));
- }.bind(this);
- this.$window = $(window).on('scroll.scrollspynav.amui', processRAF)
- .on('resize.scrollspynav.amui orientationchange.scrollspynav.amui',
- UI.utils.debounce(processRAF, 50));
- processRAF();
- this.scrollProcess();
- };
- ScrollSpyNav.DEFAULTS = {
- className: {
- active: 'am-active'
- },
- closest: false,
- smooth: true,
- offsetTop: 0
- };
- ScrollSpyNav.prototype.process = function() {
- var scrollTop = this.$window.scrollTop();
- var options = this.options;
- var inViews = [];
- var $links = this.$links;
- var $targets = this.$targets;
- $targets.each(function(i, target) {
- if (UI.utils.isInView(target, options)) {
- inViews.push(target);
- }
- });
- // console.log(inViews.length);
- if (inViews.length) {
- var $target;
- $.each(inViews, function(i, item) {
- if ($(item).offset().top >= scrollTop) {
- $target = $(item);
- return false; // break
- }
- });
- if (!$target) {
- return;
- }
- if (options.closest) {
- $links.closest(options.closest).removeClass(options.className.active);
- $links.filter('a[href="#' + $target.attr('id') + '"]').
- closest(options.closest).addClass(options.className.active);
- } else {
- $links.removeClass(options.className.active).
- filter('a[href="#' + $target.attr('id') + '"]').
- addClass(options.className.active);
- }
- }
- };
- ScrollSpyNav.prototype.scrollProcess = function() {
- var $links = this.$links;
- var options = this.options;
- // smoothScroll
- if (options.smooth && $.fn.smoothScroll) {
- $links.on('click', function(e) {
- e.preventDefault();
- var $this = $(this);
- var $target = $($this.attr('href'));
- if (!$target) {
- return;
- }
- var offsetTop = options.offsetTop &&
- !isNaN(parseInt(options.offsetTop)) && parseInt(options.offsetTop) || 0;
- $(window).smoothScroll({position: $target.offset().top - offsetTop});
- });
- }
- };
- // ScrollSpyNav Plugin
- UI.plugin('scrollspynav', ScrollSpyNav);
- // Init code
- UI.ready(function(context) {
- $('[data-am-scrollspynav]', context).scrollspynav();
- });
- module.exports = ScrollSpyNav;
- // TODO: 1. 算法改进
- // 2. 多级菜单支持
- // 3. smooth scroll pushState
- /***/ },
- /* 23 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var rAF = UI.utils.rAF;
- var cAF = UI.utils.cancelAF;
- /**
- * Smooth Scroll
- * @param position
- * @via http://mir.aculo.us/2014/01/19/scrolling-dom-elements-to-the-top-a-zepto-plugin/
- */
- // Usage: $(window).smoothScroll([options])
- // only allow one scroll to top operation to be in progress at a time,
- // which is probably what you want
- var smoothScrollInProgress = false;
- var SmoothScroll = function(element, options) {
- options = options || {};
- var $this = $(element);
- var targetY = parseInt(options.position) || SmoothScroll.DEFAULTS.position;
- var initialY = $this.scrollTop();
- var lastY = initialY;
- var delta = targetY - initialY;
- // duration in ms, make it a bit shorter for short distances
- // this is not scientific and you might want to adjust this for
- // your preferences
- var speed = options.speed ||
- Math.min(750, Math.min(1500, Math.abs(initialY - targetY)));
- // temp variables (t will be a position between 0 and 1, y is the calculated scrollTop)
- var start;
- var t;
- var y;
- var cancelScroll = function() {
- abort();
- };
- // abort if already in progress or nothing to scroll
- if (smoothScrollInProgress) {
- return;
- }
- if (delta === 0) {
- return;
- }
- // quint ease-in-out smoothing, from
- // https://github.com/madrobby/scripty2/blob/master/src/effects/transitions/penner.js#L127-L136
- function smooth(pos) {
- if ((pos /= 0.5) < 1) {
- return 0.5 * Math.pow(pos, 5);
- }
- return 0.5 * (Math.pow((pos - 2), 5) + 2);
- }
- function abort() {
- $this.off('touchstart.smoothscroll.amui', cancelScroll);
- smoothScrollInProgress = false;
- }
- // when there's a touch detected while scrolling is in progress, abort
- // the scrolling (emulates native scrolling behavior)
- $this.on('touchstart.smoothscroll.amui', cancelScroll);
- smoothScrollInProgress = true;
- // start rendering away! note the function given to frame
- // is named "render" so we can reference it again further down
- function render(now) {
- if (!smoothScrollInProgress) {
- return;
- }
- if (!start) {
- start = now;
- }
- // calculate t, position of animation in [0..1]
- t = Math.min(1, Math.max((now - start) / speed, 0));
- // calculate the new scrollTop position (don't forget to smooth)
- y = Math.round(initialY + delta * smooth(t));
- // bracket scrollTop so we're never over-scrolling
- if (delta > 0 && y > targetY) {
- y = targetY;
- }
- if (delta < 0 && y < targetY) {
- y = targetY;
- }
- // only actually set scrollTop if there was a change fromt he last frame
- if (lastY != y) {
- $this.scrollTop(y);
- }
- lastY = y;
- // if we're not done yet, queue up an other frame to render,
- // or clean up
- if (y !== targetY) {
- cAF(scrollRAF);
- scrollRAF = rAF(render);
- } else {
- cAF(scrollRAF);
- abort();
- }
- }
- var scrollRAF = rAF(render);
- };
- SmoothScroll.DEFAULTS = {
- position: 0
- };
- $.fn.smoothScroll = function(option) {
- return this.each(function() {
- new SmoothScroll(this, option);
- });
- };
- // Init code
- $(document).on('click.smoothScroll.amui.data-api', '[data-am-smooth-scroll]',
- function(e) {
- e.preventDefault();
- var options = UI.utils.parseOptions($(this).data('amSmoothScroll'));
- $(window).smoothScroll(options);
- });
- module.exports = SmoothScroll;
- /***/ },
- /* 24 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- // require('./ui.dropdown');
- // Make jQuery :contains Case-Insensitive
- $.expr[':'].containsNC = function(elem, i, match, array) {
- return (elem.textContent || elem.innerText || '').toLowerCase().
- indexOf((match[3] || '').toLowerCase()) >= 0;
- };
- /**
- * Selected
- * @desc HTML select replacer
- * @via https://github.com/silviomoreto/bootstrap-select
- * @license https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE
- * @param element
- * @param options
- * @constructor
- */
- var Selected = function(element, options) {
- this.$element = $(element);
- this.options = $.extend({}, Selected.DEFAULTS, {
- placeholder: element.getAttribute('placeholder') ||
- Selected.DEFAULTS.placeholder
- }, options);
- this.$originalOptions = this.$element.find('option');
- this.multiple = element.multiple;
- this.$selector = null;
- this.initialized = false;
- this.init();
- };
- Selected.DEFAULTS = {
- btnWidth: null,
- btnSize: null,
- btnStyle: 'default',
- dropUp: 0,
- maxHeight: null,
- maxChecked: null,
- placeholder: '点击选择...',
- selectedClass: 'am-checked',
- disabledClass: 'am-disabled',
- searchBox: false,
- tpl: '<div class="am-selected am-dropdown ' +
- '<%= dropUp ? \'am-dropdown-up\': \'\' %>" id="<%= id %>" data-am-dropdown>' +
- ' <button type="button" class="am-selected-btn am-btn am-dropdown-toggle">' +
- ' <span class="am-selected-status am-fl"></span>' +
- ' <i class="am-selected-icon am-icon-caret-' +
- '<%= dropUp ? \'up\' : \'down\' %>"></i>' +
- ' </button>' +
- ' <div class="am-selected-content am-dropdown-content">' +
- ' <h2 class="am-selected-header">' +
- '<span class="am-icon-chevron-left">返回</span></h2>' +
- ' <% if (searchBox) { %>' +
- ' <div class="am-selected-search">' +
- ' <input autocomplete="off" class="am-form-field am-input-sm" />' +
- ' </div>' +
- ' <% } %>' +
- ' <ul class="am-selected-list">' +
- ' <% for (var i = 0; i < options.length; i++) { %>' +
- ' <% var option = options[i] %>' +
- ' <% if (option.header) { %>' +
- ' <li data-group="<%= option.group %>" class="am-selected-list-header">' +
- ' <%= option.text %></li>' +
- ' <% } else { %>' +
- ' <li class="<%= option.classNames%>" ' +
- ' data-index="<%= option.index %>" ' +
- ' data-group="<%= option.group || 0 %>" ' +
- ' data-value="<%= option.value %>" >' +
- ' <span class="am-selected-text"><%= option.text %></span>' +
- ' <i class="am-icon-check"></i></li>' +
- ' <% } %>' +
- ' <% } %>' +
- ' </ul>' +
- ' <div class="am-selected-hint"></div>' +
- ' </div>' +
- '</div>',
- listTpl: '<% for (var i = 0; i < options.length; i++) { %>' +
- ' <% var option = options[i] %>' +
- ' <% if (option.header) { %>' +
- ' <li data-group="<%= option.group %>" class="am-selected-list-header">' +
- ' <%= option.text %></li>' +
- ' <% } else { %>' +
- ' <li class="<%= option.classNames %>" ' +
- ' data-index="<%= option.index %>" ' +
- ' data-group="<%= option.group || 0 %>" ' +
- ' data-value="<%= option.value %>" >' +
- ' <span class="am-selected-text"><%= option.text %></span>' +
- ' <i class="am-icon-check"></i></li>' +
- ' <% } %>' +
- ' <% } %>'
- };
- Selected.prototype.init = function() {
- var _this = this;
- var $element = this.$element;
- var options = this.options;
- $element.hide();
- var data = {
- id: UI.utils.generateGUID('am-selected'),
- multiple: this.multiple,
- options: [],
- searchBox: options.searchBox,
- dropUp: options.dropUp,
- placeholder: options.placeholder
- };
- this.$selector = $(UI.template(this.options.tpl, data));
- // set select button styles
- this.$selector.css({width: this.options.btnWidth});
- this.$list = this.$selector.find('.am-selected-list');
- this.$searchField = this.$selector.find('.am-selected-search input');
- this.$hint = this.$selector.find('.am-selected-hint');
- var $selectorBtn = this.$selector.find('.am-selected-btn');
- var btnClassNames = [];
- options.btnSize && btnClassNames.push('am-btn-' + options.btnSize);
- options.btnStyle && btnClassNames.push('am-btn-' + options.btnStyle);
- $selectorBtn.addClass(btnClassNames.join(' '));
- this.$selector.dropdown({
- justify: $selectorBtn
- });
- // disable Selected instance if <selected> is disabled
- // should call .disable() after Dropdown initialed
- if ($element[0].disabled) {
- this.disable();
- }
- // set list height
- if (options.maxHeight) {
- this.$selector.find('.am-selected-list').css({
- 'max-height': options.maxHeight,
- 'overflow-y': 'scroll'
- });
- }
- // set hint text
- var hint = [];
- var min = $element.attr('minchecked');
- var max = $element.attr('maxchecked') || options.maxChecked;
- this.maxChecked = max || Infinity;
- if ($element[0].required) {
- hint.push('必选');
- }
- if (min || max) {
- min && hint.push('至少选择 ' + min + ' 项');
- max && hint.push('至多选择 ' + max + ' 项');
- }
- this.$hint.text(hint.join(','));
- // render dropdown list
- this.renderOptions();
- // append $selector after <select>
- this.$element.after(this.$selector);
- this.dropdown = this.$selector.data('amui.dropdown');
- this.$status = this.$selector.find('.am-selected-status');
- // #try to fixes #476
- setTimeout(function() {
- _this.syncData();
- _this.initialized = true;
- }, 0);
- this.bindEvents();
- };
- Selected.prototype.renderOptions = function() {
- var $element = this.$element;
- var options = this.options;
- var optionItems = [];
- var $optgroup = $element.find('optgroup');
- this.$originalOptions = this.$element.find('option');
- // 单选框使用 JS 禁用已经选择的 option 以后,
- // 浏览器会重新选定第一个 option,但有一定延迟,致使 JS 获取 value 时返回 null
- if (!this.multiple && ($element.val() === null)) {
- this.$originalOptions.length &&
- (this.$originalOptions.get(0).selected = true);
- }
- function pushOption(index, item, group) {
- if (item.value === '') {
- // skip to next iteration
- // @see http://stackoverflow.com/questions/481601/how-to-skip-to-next-iteration-in-jquery-each-util
- return true;
- }
- var classNames = '';
- item.disabled && (classNames += options.disabledClass);
- !item.disabled && item.selected && (classNames += options.selectedClass);
- optionItems.push({
- group: group,
- index: index,
- classNames: classNames,
- text: item.text,
- value: item.value
- });
- }
- // select with option groups
- if ($optgroup.length) {
- $optgroup.each(function(i) {
- // push group name
- optionItems.push({
- header: true,
- group: i + 1,
- text: this.label
- });
- $optgroup.eq(i).find('option').each(function(index, item) {
- pushOption(index, item, i);
- });
- });
- } else {
- // without option groups
- this.$originalOptions.each(function(index, item) {
- pushOption(index, item, null);
- });
- }
- this.$list.html(UI.template(options.listTpl, {options: optionItems}));
- this.$shadowOptions = this.$list.find('> li').
- not('.am-selected-list-header');
- };
- Selected.prototype.setChecked = function(item) {
- var options = this.options;
- var $item = $(item);
- var isChecked = $item.hasClass(options.selectedClass);
- if (this.multiple) {
- // multiple
- var checkedLength = this.$list.find('.' + options.selectedClass).length;
- if (!isChecked && this.maxChecked <= checkedLength) {
- this.$element.trigger('checkedOverflow.selected.amui', {
- selected: this
- });
- return false;
- }
- } else {
- // close dropdown whether item is checked or not
- // @see #860
- this.dropdown.close();
- if (isChecked) {
- return false;
- }
- this.$shadowOptions.not($item).removeClass(options.selectedClass);
- }
- $item.toggleClass(options.selectedClass);
- this.syncData(item);
- };
- /**
- * syncData
- *
- * @description if `item` set, only sync `item` related option
- * @param {Object} [item]
- */
- Selected.prototype.syncData = function(item) {
- var _this = this;
- var options = this.options;
- var status = [];
- var $checked = $([]);
- this.$shadowOptions.filter('.' + options.selectedClass).each(function() {
- var $this = $(this);
- status.push($this.find('.am-selected-text').text());
- if (!item) {
- $checked = $checked.add(_this.$originalOptions
- .filter('[value="' + $this.data('value') + '"]')
- .prop('selected', true));
- }
- });
- if (item) {
- var $item = $(item);
- this.$originalOptions
- .filter('[value="' + $item.data('value') + '"]')
- .prop('selected', $item.hasClass(options.selectedClass));
- } else {
- this.$originalOptions.not($checked).prop('selected', false);
- }
- // nothing selected
- if (!this.$element.val()) {
- status = [options.placeholder];
- }
- this.$status.text(status.join(', '));
- // Do not trigger change event on initializing
- this.initialized && this.$element.trigger('change');
- };
- Selected.prototype.bindEvents = function() {
- var _this = this;
- var header = 'am-selected-list-header';
- var handleKeyup = UI.utils.debounce(function(e) {
- _this.$shadowOptions.not('.' + header).hide().
- filter(':containsNC("' + e.target.value + '")').show();
- }, 100);
- this.$list.on('click', '> li', function(e) {
- var $this = $(this);
- !$this.hasClass(_this.options.disabledClass) &&
- !$this.hasClass(header) && _this.setChecked(this);
- });
- // simple search with jQuery :contains
- this.$searchField.on('keyup.selected.amui', handleKeyup);
- // empty search keywords
- this.$selector.on('closed.dropdown.amui', function() {
- _this.$searchField.val('');
- _this.$shadowOptions.css({display: ''});
- });
- // work with Validator
- // @since 2.5
- this.$element.on('validated.field.validator.amui', function(e) {
- if (e.validity) {
- var valid = e.validity.valid;
- var errorClassName = 'am-invalid';
- _this.$selector[(!valid ? 'add' : 'remove') + 'Class'](errorClassName);
- }
- });
- // observe DOM
- if (UI.support.mutationobserver) {
- this.observer = new UI.support.mutationobserver(function() {
- _this.$element.trigger('changed.selected.amui');
- });
- this.observer.observe(this.$element[0], {
- childList: true,
- subtree: true,
- characterData: true
- });
- }
- // custom event
- this.$element.on('changed.selected.amui', function() {
- _this.renderOptions();
- _this.syncData();
- });
- };
- // @since: 2.5
- Selected.prototype.select = function(item) {
- var $item;
- if (typeof item === 'number') {
- $item = this.$list.find('> li').not('.am-selected-list-header').eq(item);
- } else if (typeof item === 'string') {
- $item = this.$list.find(item);
- } else {
- $item = $(item);
- }
- $item.trigger('click');
- };
- // @since: 2.5
- Selected.prototype.enable = function() {
- this.$element.prop('disable', false);
- this.$selector.dropdown('enable');
- };
- // @since: 2.5
- Selected.prototype.disable = function() {
- this.$element.prop('disable', true);
- this.$selector.dropdown('disable');
- };
- Selected.prototype.destroy = function() {
- this.$element.removeData('amui.selected').show();
- this.$selector.remove();
- };
- UI.plugin('selected', Selected);
- // Conflict with jQuery form
- // https://github.com/malsup/form/blob/6bf24a5f6d8be65f4e5491863180c09356d9dadd/jquery.form.js#L1240-L1258
- // https://github.com/allmobilize/amazeui/issues/379
- // @deprecated: $.fn.selected = $.fn.selectIt = Plugin;
- // New way to resolve conflict:
- // @see https://github.com/amazeui/amazeui/issues/781#issuecomment-158873541
- UI.ready(function(context) {
- $('[data-am-selected]', context).selected();
- });
- module.exports = Selected;
- /***/ },
- /* 25 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- __webpack_require__(15);
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var QRCode = __webpack_require__(26);
- var doc = document;
- var $doc = $(doc);
- var Share = function(options) {
- this.options = $.extend({}, Share.DEFAULTS, options || {});
- this.$element = null;
- this.$wechatQr = null;
- this.pics = null;
- this.inited = false;
- this.active = false;
- // this.init();
- };
- Share.DEFAULTS = {
- sns: ['weibo', 'qq', 'qzone', 'tqq', 'wechat', 'renren'],
- title: '分享到',
- cancel: '取消',
- closeOnShare: true,
- id: UI.utils.generateGUID('am-share'),
- desc: 'Hi,孤夜观天象,发现一个不错的西西,分享一下下 ;-)',
- via: 'Amaze UI',
- tpl: '<div class="am-share am-modal-actions" id="<%= id %>">' +
- '<h3 class="am-share-title"><%= title %></h3>' +
- '<ul class="am-share-sns am-avg-sm-3">' +
- '<% for(var i = 0; i < sns.length; i++) {%>' +
- '<li>' +
- '<a href="<%= sns[i].shareUrl %>" ' +
- 'data-am-share-to="<%= sns[i].id %>" >' +
- '<i class="am-icon-<%= sns[i].icon %>"></i>' +
- '<span><%= sns[i].title %></span>' +
- '</a></li>' +
- '<% } %></ul>' +
- '<div class="am-share-footer">' +
- '<button class="am-btn am-btn-default am-btn-block" ' +
- 'data-am-share-close><%= cancel %></button></div>' +
- '</div>'
- };
- Share.SNS = {
- weibo: {
- title: '新浪微博',
- url: 'http://service.weibo.com/share/share.php',
- width: 620,
- height: 450,
- icon: 'weibo'
- },
- // url 链接地址
- // title:”, 分享的文字内容(可选,默认为所在页面的title)
- // appkey:”, 您申请的应用appkey,显示分享来源(可选)
- // pic:”, 分享图片的路径(可选)
- // ralateUid:”, 关联用户的UID,分享微博会@该用户(可选)
- // NOTE: 会自动抓取图片,不用指定 pic
- qq: {
- title: 'QQ 好友',
- url: 'http://connect.qq.com/widget/shareqq/index.html',
- icon: 'qq'
- },
- // url:,
- // title:'', 分享标题(可选)
- // pics:'', 分享图片的路径(可选)
- // summary:'', 分享摘要(可选)
- // site:'', 分享来源 如:腾讯网(可选)
- // desc: '' 发送给用户的消息
- // NOTE: 经过测试,最终发给用户的只有 url 和 desc
- qzone: {
- title: 'QQ 空间',
- url: 'http://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey',
- icon: 'star'
- },
- // http://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey?url=xxx&title=xxx&desc=&summary=&site=
- // url:,
- // title:'', 分享标题(可选)
- // desc:'', 默认分享理由(可选)
- // summary:'', 分享摘要(可选)
- // site:'', 分享来源 如:腾讯网(可选)
- // pics:'', 分享图片的路径(可选),不会自动抓取,多个图片用|分隔
- tqq: {
- title: '腾讯微博',
- url: 'http://v.t.qq.com/share/share.php',
- icon: 'tencent-weibo'
- },
- // url=xx&title=&appkey=801cf76d3cfc44ada52ec13114e84a96
- // url
- // title
- // pic 多个图片用 | 分隔
- // appkey
- // NOTE: 不会自动抓取图片
- wechat: {
- title: '微信',
- url: '[qrcode]',
- icon: 'wechat'
- },
- // 生成一个二维码 供用户扫描
- // 相关接口 https://github.com/zxlie/WeixinApi
- renren: {
- title: '人人网',
- url: 'http://widget.renren.com/dialog/share',
- icon: 'renren'
- },
- // http://widget.renren.com/dialog/share?resourceUrl=www&srcUrl=www&title=ww&description=xxx
- // 550 * 400
- // resourceUrl : '', // 分享的资源Url
- // srcUrl : '', // 分享的资源来源Url,
- // // 默认为header中的Referer,如果分享失败可以调整此值为resourceUrl试试
- // pic : '', // 分享的主题图片,会自动抓取
- // title : '', // 分享的标题
- // description : '' // 分享的详细描述
- // NOTE: 经过测试,直接使用 url 参数即可
- douban: {
- title: '豆瓣',
- url: 'http://www.douban.com/recommend/',
- icon: 'share-alt'
- },
- // http://www.douban.com/service/sharebutton
- // 450 * 330
- // http://www.douban.com/share/service?bm=1&image=&href=xxx&updated=&name=
- // href 链接
- // name 标题
- /* void (function() {
- var d = document, e = encodeURIComponent,
- s1 = window.getSelection, s2 = d.getSelection,
- s3 = d.selection, s = s1 ? s1()
- : s2 ? s2() : s3 ? s3.createRange().text : '',
- r = 'http://www.douban.com/recommend/?url=&title=&sel=&v=1&r=1'
- })();
- */
- // tsohu: '',
- // http://t.sohu.com/third/post.jsp?url=&title=&content=utf-8&pic=
- // print: '',
- mail: {
- title: '邮件分享',
- url: 'mailto:',
- icon: 'envelope-o'
- },
- sms: {
- title: '短信分享',
- url: 'sms:',
- icon: 'comment'
- }
- };
- Share.prototype.render = function() {
- var options = this.options;
- var snsData = [];
- var title = encodeURIComponent(doc.title);
- var link = encodeURIComponent(doc.location);
- var msgBody = '?body=' + title + link;
- options.sns.forEach(function(item, i) {
- if (Share.SNS[item]) {
- var tmp = Share.SNS[item];
- var shareUrl;
- tmp.id = item;
- if (item === 'mail') {
- shareUrl = msgBody + '&subject=' + options.desc;
- } else if (item === 'sms') {
- shareUrl = msgBody;
- } else {
- shareUrl = '?url=' + link + '&title=' + title;
- }
- tmp.shareUrl = tmp.url + shareUrl;
- snsData.push(tmp);
- }
- });
- return UI.template(options.tpl, $.extend({}, options, {sns: snsData}));
- };
- Share.prototype.init = function() {
- if (this.inited) {
- return;
- }
- var me = this;
- var shareItem = '[data-am-share-to]';
- $doc.ready($.proxy(function() {
- $('body').append(this.render()); // append share DOM to body
- this.$element = $('#' + this.options.id);
- this.$element.find('[data-am-share-close]').
- on('click.share.amui', function() {
- me.close();
- });
- }, this));
- $doc.on('click.share.amui', shareItem, $.proxy(function(e) {
- var $clicked = $(e.target);
- var $target = $clicked.is(shareItem) && $clicked ||
- $clicked.parent(shareItem);
- var sns = $target.attr('data-am-share-to');
- if (!(sns === 'mail' || sns === 'sms')) {
- e.preventDefault();
- this.shareTo(sns, this.setData(sns));
- }
- this.close();
- }, this));
- this.inited = true;
- };
- Share.prototype.open = function() {
- !this.inited && this.init();
- this.$element && this.$element.modal('open');
- this.$element.trigger('open.share.amui');
- this.active = true;
- };
- Share.prototype.close = function() {
- this.$element && this.$element.modal('close');
- this.$element.trigger('close.share.amui');
- this.active = false;
- };
- Share.prototype.toggle = function() {
- this.active ? this.close() : this.open();
- };
- Share.prototype.setData = function(sns) {
- if (!sns) {
- return;
- }
- var shareData = {
- url: doc.location,
- title: doc.title
- };
- var desc = this.options.desc;
- var imgSrc = this.pics || [];
- var qqReg = /^(qzone|qq|tqq)$/;
- if (qqReg.test(sns) && !imgSrc.length) {
- var allImages = doc.images;
- for (var i = 0; i < allImages.length && i < 10; i++) {
- !!allImages[i].src && imgSrc.push(encodeURIComponent(allImages[i].src));
- }
- this.pics = imgSrc; // 缓存图片
- }
- switch (sns) {
- case 'qzone':
- shareData.desc = desc;
- shareData.site = this.options.via;
- shareData.pics = imgSrc.join('|');
- // TODO: 抓取图片多张
- break;
- case 'qq':
- shareData.desc = desc;
- shareData.site = this.options.via;
- shareData.pics = imgSrc[0];
- // 抓取一张图片
- break;
- case 'tqq':
- // 抓取图片多张
- shareData.pic = imgSrc.join('|');
- break;
- }
- return shareData;
- };
- Share.prototype.shareTo = function(sns, data) {
- var snsInfo = Share.SNS[sns];
- if (!snsInfo) {
- return;
- }
- if (sns === 'wechat' || sns === 'weixin') {
- return this.wechatQr();
- }
- var query = [];
- for (var key in data) {
- if (data[key]) {
- // 避免 encode 图片分隔符 |
- query.push(key.toString() + '=' + ((key === 'pic' || key === 'pics') ?
- data[key] : encodeURIComponent(data[key])));
- }
- }
- window.open(snsInfo.url + '?' + query.join('&'));
- };
- Share.prototype.wechatQr = function() {
- if (!this.$wechatQr) {
- var qrId = UI.utils.generateGUID('am-share-wechat');
- var $qr = $('<div class="am-modal am-modal-no-btn am-share-wechat-qr">' +
- '<div class="am-modal-dialog"><div class="am-modal-hd">分享到微信 ' +
- '<a href="" class="am-close am-close-spin" ' +
- 'data-am-modal-close>×</a> </div>' +
- '<div class="am-modal-bd">' +
- '<div class="am-share-wx-qr"></div>' +
- '<div class="am-share-wechat-tip">' +
- '打开微信,点击底部的<em>发现</em>,<br/> ' +
- '使用<em>扫一扫</em>将网页分享至朋友圈</div></div></div></div>');
- $qr.attr('id', qrId);
- var qrNode = new QRCode({
- render: 'canvas',
- correctLevel: 0,
- text: doc.location.href,
- width: 180,
- height: 180,
- background: '#fff',
- foreground: '#000'
- });
- $qr.find('.am-share-wx-qr').html(qrNode);
- $qr.appendTo($('body'));
- this.$wechatQr = $('#' + qrId);
- }
- this.$wechatQr.modal('open');
- };
- var share = new Share();
- $doc.on('click.share.amui.data-api', '[data-am-toggle="share"]', function(e) {
- e.preventDefault();
- share.toggle();
- });
- module.exports = UI.share = share;
- /***/ },
- /* 26 */
- /***/ function(module, exports, __webpack_require__) {
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * @ver 1.1.0
- * @via https://github.com/aralejs/qrcode/blob/master/src/qrcode.js
- * @license http://aralejs.org/
- */
- var qrcodeAlgObjCache = [];
- /**
- * 二维码构造函数,主要用于绘制
- * @param {Object} opt 传递参数
- * @return {String} qrcode
- * @constructor
- */
- var QRCode = function(opt) {
- if (typeof opt === 'string') { // 只编码ASCII字符串
- opt = {
- text: opt
- };
- }
- // 设置默认参数
- this.options = $.extend({}, {
- text: "",
- render: "",
- width: 256,
- height: 256,
- correctLevel: 3,
- background: "#ffffff",
- foreground: "#000000"
- }, opt);
- // 使用QRCodeAlg创建二维码结构
- var qrCodeAlg = null;
- for (var i = 0, l = qrcodeAlgObjCache.length; i < l; i++) {
- if (qrcodeAlgObjCache[i].text == this.options.text && qrcodeAlgObjCache[i].text.correctLevel == this.options.correctLevel) {
- qrCodeAlg = qrcodeAlgObjCache[i].obj;
- break;
- }
- }
- if (i == l) {
- qrCodeAlg = new QRCodeAlg(this.options.text, this.options.correctLevel);
- qrcodeAlgObjCache.push({
- text: this.options.text,
- correctLevel: this.options.correctLevel,
- obj: qrCodeAlg
- });
- }
- if (this.options.render) {
- switch (this.options.render) {
- case "canvas":
- return this.createCanvas(qrCodeAlg);
- case "table":
- return this.createTable(qrCodeAlg);
- case "svg":
- return this.createSVG(qrCodeAlg);
- default:
- return this.createDefault(qrCodeAlg);
- }
- }
- return this.createDefault(qrCodeAlg);
- };
- /**
- * 使用Canvas来画二维码
- * @return {}
- */
- QRCode.prototype.createDefault = function(qrCodeAlg) {
- var canvas = document.createElement('canvas');
- if (canvas.getContext)
- return this.createCanvas(qrCodeAlg);
- if (!!document.createElementNS && !!document.createElementNS(SVG_NS, 'svg').createSVGRect)
- return this.createSVG(qrCodeAlg);
- return this.createTable(qrCodeAlg);
- };
- QRCode.prototype.createCanvas = function(qrCodeAlg) {
- //创建canvas节点
- var canvas = document.createElement('canvas');
- canvas.width = this.options.width;
- canvas.height = this.options.height;
- var ctx = canvas.getContext('2d');
- //计算每个点的长宽
- var tileW = (this.options.width / qrCodeAlg.getModuleCount()).toPrecision(4);
- var tileH = this.options.height / qrCodeAlg.getModuleCount().toPrecision(4);
- //绘制
- for (var row = 0; row < qrCodeAlg.getModuleCount(); row++) {
- for (var col = 0; col < qrCodeAlg.getModuleCount(); col++) {
- ctx.fillStyle = qrCodeAlg.modules[row][col] ? this.options.foreground : this.options.background;
- var w = (Math.ceil((col + 1) * tileW) - Math.floor(col * tileW));
- var h = (Math.ceil((row + 1) * tileW) - Math.floor(row * tileW));
- ctx.fillRect(Math.round(col * tileW), Math.round(row * tileH), w, h);
- }
- }
- //返回绘制的节点
- return canvas;
- };
- /**
- * 使用table来绘制二维码
- * @return {}
- */
- QRCode.prototype.createTable = function(qrCodeAlg) {
- //创建table节点
- var s = [];
- s.push('<table style="border:0px; margin:0px; padding:0px; border-collapse:collapse; background-color: ' +
- this.options.background +
- ';">');
- // 计算每个节点的长宽;取整,防止点之间出现分离
- var tileW = -1, tileH = -1, caculateW = -1, caculateH = -1;
- tileW = caculateW = Math.floor(this.options.width / qrCodeAlg.getModuleCount());
- tileH = caculateH = Math.floor(this.options.height / qrCodeAlg.getModuleCount());
- if (caculateW <= 0) {
- if (qrCodeAlg.getModuleCount() < 80) {
- tileW = 2;
- } else {
- tileW = 1;
- }
- }
- if (caculateH <= 0) {
- if (qrCodeAlg.getModuleCount() < 80) {
- tileH = 2;
- } else {
- tileH = 1;
- }
- }
- // 绘制二维码
- foreTd = '<td style="border:0px; margin:0px; padding:0px; width:' + tileW + 'px; background-color: ' + this.options.foreground + '"></td>',
- backTd = '<td style="border:0px; margin:0px; padding:0px; width:' + tileW + 'px; background-color: ' + this.options.background + '"></td>',
- l = qrCodeAlg.getModuleCount();
- for (var row = 0; row < l; row++) {
- s.push('<tr style="border:0px; margin:0px; padding:0px; height: ' + tileH + 'px">');
- for (var col = 0; col < l; col++) {
- s.push(qrCodeAlg.modules[row][col] ? foreTd : backTd);
- }
- s.push('</tr>');
- }
- s.push('</table>');
- var span = document.createElement("span");
- span.innerHTML = s.join('');
- return span.firstChild;
- };
- /**
- * 使用SVG开绘制二维码
- * @return {}
- */
- QRCode.prototype.createSVG = function(qrCodeAlg) {
- var x, dx, y, dy,
- moduleCount = qrCodeAlg.getModuleCount(),
- scale = this.options.height / this.options.width,
- svg = '<svg xmlns="http://www.w3.org/2000/svg" '
- + 'width="' + this.options.width + 'px" height="' + this.options.height + 'px" '
- + 'viewbox="0 0 ' + moduleCount * 10 + ' ' + moduleCount * 10 * scale + '">',
- rectHead = '<path ',
- foreRect = ' style="stroke-width:0.5;stroke:' + this.options.foreground
- + ';fill:' + this.options.foreground + ';"></path>',
- backRect = ' style="stroke-width:0.5;stroke:' + this.options.background
- + ';fill:' + this.options.background + ';"></path>';
- // draw in the svg
- for (var row = 0; row < moduleCount; row++) {
- for (var col = 0; col < moduleCount; col++) {
- x = col * 10;
- y = row * 10 * scale;
- dx = (col + 1) * 10;
- dy = (row + 1) * 10 * scale;
- svg += rectHead + 'd="M ' + x + ',' + y
- + ' L ' + dx + ',' + y
- + ' L ' + dx + ',' + dy
- + ' L ' + x + ',' + dy
- + ' Z"';
- svg += qrCodeAlg.modules[row][col] ? foreRect : backRect;
- }
- }
- svg += '</svg>';
- // return just built svg
- return $(svg)[0];
- };
- /**
- * 获取单个字符的utf8编码
- * unicode BMP平面约65535个字符
- * @param {num} code
- * return {array}
- */
- function unicodeFormat8(code) {
- // 1 byte
- if (code < 128) {
- return [code];
- // 2 bytes
- } else if (code < 2048) {
- c0 = 192 + (code >> 6);
- c1 = 128 + (code & 63);
- return [c0, c1];
- // 3 bytes
- } else {
- c0 = 224 + (code >> 12);
- c1 = 128 + (code >> 6 & 63);
- c2 = 128 + (code & 63);
- return [c0, c1, c2];
- }
- }
- /**
- * 获取字符串的utf8编码字节串
- * @param {string} string
- * @return {array}
- */
- function getUTF8Bytes(string) {
- var utf8codes = [];
- for (var i = 0; i < string.length; i++) {
- var code = string.charCodeAt(i);
- var utf8 = unicodeFormat8(code);
- for (var j = 0; j < utf8.length; j++) {
- utf8codes.push(utf8[j]);
- }
- }
- return utf8codes;
- }
- /**
- * 二维码算法实现
- * @param {string} data 要编码的信息字符串
- * @param {num} errorCorrectLevel 纠错等级
- */
- function QRCodeAlg(data, errorCorrectLevel) {
- this.typeNumber = -1; // 版本
- this.errorCorrectLevel = errorCorrectLevel;
- this.modules = null; // 二维矩阵,存放最终结果
- this.moduleCount = 0; // 矩阵大小
- this.dataCache = null; // 数据缓存
- this.rsBlocks = null; // 版本数据信息
- this.totalDataCount = -1; // 可使用的数据量
- this.data = data;
- this.utf8bytes = getUTF8Bytes(data);
- this.make();
- }
- QRCodeAlg.prototype = {
- constructor: QRCodeAlg,
- /**
- * 获取二维码矩阵大小
- * @return {num} 矩阵大小
- */
- getModuleCount: function() {
- return this.moduleCount;
- },
- /**
- * 编码
- */
- make: function() {
- this.getRightType();
- this.dataCache = this.createData();
- this.createQrcode();
- },
- /**
- * 设置二位矩阵功能图形
- * @param {bool} test 表示是否在寻找最好掩膜阶段
- * @param {num} maskPattern 掩膜的版本
- */
- makeImpl: function(maskPattern) {
- this.moduleCount = this.typeNumber * 4 + 17;
- this.modules = new Array(this.moduleCount);
- for (var row = 0; row < this.moduleCount; row++) {
- this.modules[row] = new Array(this.moduleCount);
- }
- this.setupPositionProbePattern(0, 0);
- this.setupPositionProbePattern(this.moduleCount - 7, 0);
- this.setupPositionProbePattern(0, this.moduleCount - 7);
- this.setupPositionAdjustPattern();
- this.setupTimingPattern();
- this.setupTypeInfo(true, maskPattern);
- if (this.typeNumber >= 7) {
- this.setupTypeNumber(true);
- }
- this.mapData(this.dataCache, maskPattern);
- },
- /**
- * 设置二维码的位置探测图形
- * @param {num} row 探测图形的中心横坐标
- * @param {num} col 探测图形的中心纵坐标
- */
- setupPositionProbePattern: function(row, col) {
- for (var r = -1; r <= 7; r++) {
- if (row + r <= -1 || this.moduleCount <= row + r) continue;
- for (var c = -1; c <= 7; c++) {
- if (col + c <= -1 || this.moduleCount <= col + c) continue;
- if ((0 <= r && r <= 6 && (c == 0 || c == 6)) || (0 <= c && c <= 6 && (r == 0 || r == 6)) || (2 <= r && r <= 4 && 2 <= c && c <= 4)) {
- this.modules[row + r][col + c] = true;
- } else {
- this.modules[row + r][col + c] = false;
- }
- }
- }
- },
- /**
- * 创建二维码
- * @return {[type]} [description]
- */
- createQrcode: function() {
- var minLostPoint = 0;
- var pattern = 0;
- var bestModules = null;
- for (var i = 0; i < 8; i++) {
- this.makeImpl(i);
- var lostPoint = QRUtil.getLostPoint(this);
- if (i == 0 || minLostPoint > lostPoint) {
- minLostPoint = lostPoint;
- pattern = i;
- bestModules = this.modules;
- }
- }
- this.modules = bestModules;
- this.setupTypeInfo(false, pattern);
- if (this.typeNumber >= 7) {
- this.setupTypeNumber(false);
- }
- },
- /**
- * 设置定位图形
- * @return {[type]} [description]
- */
- setupTimingPattern: function() {
- for (var r = 8; r < this.moduleCount - 8; r++) {
- if (this.modules[r][6] != null) {
- continue;
- }
- this.modules[r][6] = (r % 2 == 0);
- if (this.modules[6][r] != null) {
- continue;
- }
- this.modules[6][r] = (r % 2 == 0);
- }
- },
- /**
- * 设置矫正图形
- * @return {[type]} [description]
- */
- setupPositionAdjustPattern: function() {
- var pos = QRUtil.getPatternPosition(this.typeNumber);
- for (var i = 0; i < pos.length; i++) {
- for (var j = 0; j < pos.length; j++) {
- var row = pos[i];
- var col = pos[j];
- if (this.modules[row][col] != null) {
- continue;
- }
- for (var r = -2; r <= 2; r++) {
- for (var c = -2; c <= 2; c++) {
- if (r == -2 || r == 2 || c == -2 || c == 2 || (r == 0 && c == 0)) {
- this.modules[row + r][col + c] = true;
- } else {
- this.modules[row + r][col + c] = false;
- }
- }
- }
- }
- }
- },
- /**
- * 设置版本信息(7以上版本才有)
- * @param {bool} test 是否处于判断最佳掩膜阶段
- * @return {[type]} [description]
- */
- setupTypeNumber: function(test) {
- var bits = QRUtil.getBCHTypeNumber(this.typeNumber);
- for (var i = 0; i < 18; i++) {
- var mod = (!test && ((bits >> i) & 1) == 1);
- this.modules[Math.floor(i / 3)][i % 3 + this.moduleCount - 8 - 3] = mod;
- this.modules[i % 3 + this.moduleCount - 8 - 3][Math.floor(i / 3)] = mod;
- }
- },
- /**
- * 设置格式信息(纠错等级和掩膜版本)
- * @param {bool} test
- * @param {num} maskPattern 掩膜版本
- * @return {}
- */
- setupTypeInfo: function(test, maskPattern) {
- var data = (QRErrorCorrectLevel[this.errorCorrectLevel] << 3) | maskPattern;
- var bits = QRUtil.getBCHTypeInfo(data);
- // vertical
- for (var i = 0; i < 15; i++) {
- var mod = (!test && ((bits >> i) & 1) == 1);
- if (i < 6) {
- this.modules[i][8] = mod;
- } else if (i < 8) {
- this.modules[i + 1][8] = mod;
- } else {
- this.modules[this.moduleCount - 15 + i][8] = mod;
- }
- // horizontal
- var mod = (!test && ((bits >> i) & 1) == 1);
- if (i < 8) {
- this.modules[8][this.moduleCount - i - 1] = mod;
- } else if (i < 9) {
- this.modules[8][15 - i - 1 + 1] = mod;
- } else {
- this.modules[8][15 - i - 1] = mod;
- }
- }
- // fixed module
- this.modules[this.moduleCount - 8][8] = (!test);
- },
- /**
- * 数据编码
- * @return {[type]} [description]
- */
- createData: function() {
- var buffer = new QRBitBuffer();
- var lengthBits = this.typeNumber > 9 ? 16 : 8;
- buffer.put(4, 4); //添加模式
- buffer.put(this.utf8bytes.length, lengthBits);
- for (var i = 0, l = this.utf8bytes.length; i < l; i++) {
- buffer.put(this.utf8bytes[i], 8);
- }
- if (buffer.length + 4 <= this.totalDataCount * 8) {
- buffer.put(0, 4);
- }
- // padding
- while (buffer.length % 8 != 0) {
- buffer.putBit(false);
- }
- // padding
- while (true) {
- if (buffer.length >= this.totalDataCount * 8) {
- break;
- }
- buffer.put(QRCodeAlg.PAD0, 8);
- if (buffer.length >= this.totalDataCount * 8) {
- break;
- }
- buffer.put(QRCodeAlg.PAD1, 8);
- }
- return this.createBytes(buffer);
- },
- /**
- * 纠错码编码
- * @param {buffer} buffer 数据编码
- * @return {[type]}
- */
- createBytes: function(buffer) {
- var offset = 0;
- var maxDcCount = 0;
- var maxEcCount = 0;
- var length = this.rsBlock.length / 3;
- var rsBlocks = new Array();
- for (var i = 0; i < length; i++) {
- var count = this.rsBlock[i * 3 + 0];
- var totalCount = this.rsBlock[i * 3 + 1];
- var dataCount = this.rsBlock[i * 3 + 2];
- for (var j = 0; j < count; j++) {
- rsBlocks.push([dataCount, totalCount]);
- }
- }
- var dcdata = new Array(rsBlocks.length);
- var ecdata = new Array(rsBlocks.length);
- for (var r = 0; r < rsBlocks.length; r++) {
- var dcCount = rsBlocks[r][0];
- var ecCount = rsBlocks[r][1] - dcCount;
- maxDcCount = Math.max(maxDcCount, dcCount);
- maxEcCount = Math.max(maxEcCount, ecCount);
- dcdata[r] = new Array(dcCount);
- for (var i = 0; i < dcdata[r].length; i++) {
- dcdata[r][i] = 0xff & buffer.buffer[i + offset];
- }
- offset += dcCount;
- var rsPoly = QRUtil.getErrorCorrectPolynomial(ecCount);
- var rawPoly = new QRPolynomial(dcdata[r], rsPoly.getLength() - 1);
- var modPoly = rawPoly.mod(rsPoly);
- ecdata[r] = new Array(rsPoly.getLength() - 1);
- for (var i = 0; i < ecdata[r].length; i++) {
- var modIndex = i + modPoly.getLength() - ecdata[r].length;
- ecdata[r][i] = (modIndex >= 0) ? modPoly.get(modIndex) : 0;
- }
- }
- var data = new Array(this.totalDataCount);
- var index = 0;
- for (var i = 0; i < maxDcCount; i++) {
- for (var r = 0; r < rsBlocks.length; r++) {
- if (i < dcdata[r].length) {
- data[index++] = dcdata[r][i];
- }
- }
- }
- for (var i = 0; i < maxEcCount; i++) {
- for (var r = 0; r < rsBlocks.length; r++) {
- if (i < ecdata[r].length) {
- data[index++] = ecdata[r][i];
- }
- }
- }
- return data;
- },
- /**
- * 布置模块,构建最终信息
- * @param {} data
- * @param {} maskPattern
- * @return {}
- */
- mapData: function(data, maskPattern) {
- var inc = -1;
- var row = this.moduleCount - 1;
- var bitIndex = 7;
- var byteIndex = 0;
- for (var col = this.moduleCount - 1; col > 0; col -= 2) {
- if (col == 6) col--;
- while (true) {
- for (var c = 0; c < 2; c++) {
- if (this.modules[row][col - c] == null) {
- var dark = false;
- if (byteIndex < data.length) {
- dark = (((data[byteIndex] >>> bitIndex) & 1) == 1);
- }
- var mask = QRUtil.getMask(maskPattern, row, col - c);
- if (mask) {
- dark = !dark;
- }
- this.modules[row][col - c] = dark;
- bitIndex--;
- if (bitIndex == -1) {
- byteIndex++;
- bitIndex = 7;
- }
- }
- }
- row += inc;
- if (row < 0 || this.moduleCount <= row) {
- row -= inc;
- inc = -inc;
- break;
- }
- }
- }
- }
- };
- /**
- * 填充字段
- */
- QRCodeAlg.PAD0 = 0xEC;
- QRCodeAlg.PAD1 = 0x11;
- //---------------------------------------------------------------------
- // 纠错等级对应的编码
- //---------------------------------------------------------------------
- var QRErrorCorrectLevel = [1, 0, 3, 2];
- //---------------------------------------------------------------------
- // 掩膜版本
- //---------------------------------------------------------------------
- var QRMaskPattern = {
- PATTERN000: 0,
- PATTERN001: 1,
- PATTERN010: 2,
- PATTERN011: 3,
- PATTERN100: 4,
- PATTERN101: 5,
- PATTERN110: 6,
- PATTERN111: 7
- };
- //---------------------------------------------------------------------
- // 工具类
- //---------------------------------------------------------------------
- var QRUtil = {
- /*
- 每个版本矫正图形的位置
- */
- PATTERN_POSITION_TABLE: [
- [],
- [6, 18],
- [6, 22],
- [6, 26],
- [6, 30],
- [6, 34],
- [6, 22, 38],
- [6, 24, 42],
- [6, 26, 46],
- [6, 28, 50],
- [6, 30, 54],
- [6, 32, 58],
- [6, 34, 62],
- [6, 26, 46, 66],
- [6, 26, 48, 70],
- [6, 26, 50, 74],
- [6, 30, 54, 78],
- [6, 30, 56, 82],
- [6, 30, 58, 86],
- [6, 34, 62, 90],
- [6, 28, 50, 72, 94],
- [6, 26, 50, 74, 98],
- [6, 30, 54, 78, 102],
- [6, 28, 54, 80, 106],
- [6, 32, 58, 84, 110],
- [6, 30, 58, 86, 114],
- [6, 34, 62, 90, 118],
- [6, 26, 50, 74, 98, 122],
- [6, 30, 54, 78, 102, 126],
- [6, 26, 52, 78, 104, 130],
- [6, 30, 56, 82, 108, 134],
- [6, 34, 60, 86, 112, 138],
- [6, 30, 58, 86, 114, 142],
- [6, 34, 62, 90, 118, 146],
- [6, 30, 54, 78, 102, 126, 150],
- [6, 24, 50, 76, 102, 128, 154],
- [6, 28, 54, 80, 106, 132, 158],
- [6, 32, 58, 84, 110, 136, 162],
- [6, 26, 54, 82, 110, 138, 166],
- [6, 30, 58, 86, 114, 142, 170]
- ],
- G15: (1 << 10) | (1 << 8) | (1 << 5) | (1 << 4) | (1 << 2) | (1 << 1) | (1 << 0),
- G18: (1 << 12) | (1 << 11) | (1 << 10) | (1 << 9) | (1 << 8) | (1 << 5) | (1 << 2) | (1 << 0),
- G15_MASK: (1 << 14) | (1 << 12) | (1 << 10) | (1 << 4) | (1 << 1),
- /*
- BCH编码格式信息
- */
- getBCHTypeInfo: function(data) {
- var d = data << 10;
- while (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G15) >= 0) {
- d ^= (QRUtil.G15 << (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G15)));
- }
- return ((data << 10) | d) ^ QRUtil.G15_MASK;
- },
- /*
- BCH编码版本信息
- */
- getBCHTypeNumber: function(data) {
- var d = data << 12;
- while (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G18) >= 0) {
- d ^= (QRUtil.G18 << (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G18)));
- }
- return (data << 12) | d;
- },
- /*
- 获取BCH位信息
- */
- getBCHDigit: function(data) {
- var digit = 0;
- while (data != 0) {
- digit++;
- data >>>= 1;
- }
- return digit;
- },
- /*
- 获取版本对应的矫正图形位置
- */
- getPatternPosition: function(typeNumber) {
- return QRUtil.PATTERN_POSITION_TABLE[typeNumber - 1];
- },
- /*
- 掩膜算法
- */
- getMask: function(maskPattern, i, j) {
- switch (maskPattern) {
- case QRMaskPattern.PATTERN000:
- return (i + j) % 2 == 0;
- case QRMaskPattern.PATTERN001:
- return i % 2 == 0;
- case QRMaskPattern.PATTERN010:
- return j % 3 == 0;
- case QRMaskPattern.PATTERN011:
- return (i + j) % 3 == 0;
- case QRMaskPattern.PATTERN100:
- return (Math.floor(i / 2) + Math.floor(j / 3)) % 2 == 0;
- case QRMaskPattern.PATTERN101:
- return (i * j) % 2 + (i * j) % 3 == 0;
- case QRMaskPattern.PATTERN110:
- return ((i * j) % 2 + (i * j) % 3) % 2 == 0;
- case QRMaskPattern.PATTERN111:
- return ((i * j) % 3 + (i + j) % 2) % 2 == 0;
- default:
- throw new Error("bad maskPattern:" + maskPattern);
- }
- },
- /*
- 获取RS的纠错多项式
- */
- getErrorCorrectPolynomial: function(errorCorrectLength) {
- var a = new QRPolynomial([1], 0);
- for (var i = 0; i < errorCorrectLength; i++) {
- a = a.multiply(new QRPolynomial([1, QRMath.gexp(i)], 0));
- }
- return a;
- },
- /*
- 获取评价
- */
- getLostPoint: function(qrCode) {
- var moduleCount = qrCode.getModuleCount(),
- lostPoint = 0,
- darkCount = 0;
- for (var row = 0; row < moduleCount; row++) {
- var sameCount = 0;
- var head = qrCode.modules[row][0];
- for (var col = 0; col < moduleCount; col++) {
- var current = qrCode.modules[row][col];
- //level 3 评价
- if (col < moduleCount - 6) {
- if (current && !qrCode.modules[row][col + 1] && qrCode.modules[row][col + 2] && qrCode.modules[row][col + 3] && qrCode.modules[row][col + 4] && !qrCode.modules[row][col + 5] && qrCode.modules[row][col + 6]) {
- if (col < moduleCount - 10) {
- if (qrCode.modules[row][col + 7] && qrCode.modules[row][col + 8] && qrCode.modules[row][col + 9] && qrCode.modules[row][col + 10]) {
- lostPoint += 40;
- }
- } else if (col > 3) {
- if (qrCode.modules[row][col - 1] && qrCode.modules[row][col - 2] && qrCode.modules[row][col - 3] && qrCode.modules[row][col - 4]) {
- lostPoint += 40;
- }
- }
- }
- }
- //level 2 评价
- if ((row < moduleCount - 1) && (col < moduleCount - 1)) {
- var count = 0;
- if (current) count++;
- if (qrCode.modules[row + 1][col]) count++;
- if (qrCode.modules[row][col + 1]) count++;
- if (qrCode.modules[row + 1][col + 1]) count++;
- if (count == 0 || count == 4) {
- lostPoint += 3;
- }
- }
- //level 1 评价
- if (head ^ current) {
- sameCount++;
- } else {
- head = current;
- if (sameCount >= 5) {
- lostPoint += (3 + sameCount - 5);
- }
- sameCount = 1;
- }
- //level 4 评价
- if (current) {
- darkCount++;
- }
- }
- }
- for (var col = 0; col < moduleCount; col++) {
- var sameCount = 0;
- var head = qrCode.modules[0][col];
- for (var row = 0; row < moduleCount; row++) {
- var current = qrCode.modules[row][col];
- //level 3 评价
- if (row < moduleCount - 6) {
- if (current && !qrCode.modules[row + 1][col] && qrCode.modules[row + 2][col] && qrCode.modules[row + 3][col] && qrCode.modules[row + 4][col] && !qrCode.modules[row + 5][col] && qrCode.modules[row + 6][col]) {
- if (row < moduleCount - 10) {
- if (qrCode.modules[row + 7][col] && qrCode.modules[row + 8][col] && qrCode.modules[row + 9][col] && qrCode.modules[row + 10][col]) {
- lostPoint += 40;
- }
- } else if (row > 3) {
- if (qrCode.modules[row - 1][col] && qrCode.modules[row - 2][col] && qrCode.modules[row - 3][col] && qrCode.modules[row - 4][col]) {
- lostPoint += 40;
- }
- }
- }
- }
- //level 1 评价
- if (head ^ current) {
- sameCount++;
- } else {
- head = current;
- if (sameCount >= 5) {
- lostPoint += (3 + sameCount - 5);
- }
- sameCount = 1;
- }
- }
- }
- // LEVEL4
- var ratio = Math.abs(100 * darkCount / moduleCount / moduleCount - 50) / 5;
- lostPoint += ratio * 10;
- return lostPoint;
- }
- };
- //---------------------------------------------------------------------
- // QRMath使用的数学工具
- //---------------------------------------------------------------------
- var QRMath = {
- /*
- 将n转化为a^m
- */
- glog: function(n) {
- if (n < 1) {
- throw new Error("glog(" + n + ")");
- }
- return QRMath.LOG_TABLE[n];
- },
- /*
- 将a^m转化为n
- */
- gexp: function(n) {
- while (n < 0) {
- n += 255;
- }
- while (n >= 256) {
- n -= 255;
- }
- return QRMath.EXP_TABLE[n];
- },
- EXP_TABLE: new Array(256),
- LOG_TABLE: new Array(256)
- };
- for (var i = 0; i < 8; i++) {
- QRMath.EXP_TABLE[i] = 1 << i;
- }
- for (var i = 8; i < 256; i++) {
- QRMath.EXP_TABLE[i] = QRMath.EXP_TABLE[i - 4] ^ QRMath.EXP_TABLE[i - 5] ^ QRMath.EXP_TABLE[i - 6] ^ QRMath.EXP_TABLE[i - 8];
- }
- for (var i = 0; i < 255; i++) {
- QRMath.LOG_TABLE[QRMath.EXP_TABLE[i]] = i;
- }
- //---------------------------------------------------------------------
- // QRPolynomial 多项式
- //---------------------------------------------------------------------
- /**
- * 多项式类
- * @param {Array} num 系数
- * @param {num} shift a^shift
- */
- function QRPolynomial(num, shift) {
- if (num.length == undefined) {
- throw new Error(num.length + "/" + shift);
- }
- var offset = 0;
- while (offset < num.length && num[offset] == 0) {
- offset++;
- }
- this.num = new Array(num.length - offset + shift);
- for (var i = 0; i < num.length - offset; i++) {
- this.num[i] = num[i + offset];
- }
- }
- QRPolynomial.prototype = {
- get: function(index) {
- return this.num[index];
- },
- getLength: function() {
- return this.num.length;
- },
- /**
- * 多项式乘法
- * @param {QRPolynomial} e 被乘多项式
- * @return {[type]} [description]
- */
- multiply: function(e) {
- var num = new Array(this.getLength() + e.getLength() - 1);
- for (var i = 0; i < this.getLength(); i++) {
- for (var j = 0; j < e.getLength(); j++) {
- num[i + j] ^= QRMath.gexp(QRMath.glog(this.get(i)) + QRMath.glog(e.get(j)));
- }
- }
- return new QRPolynomial(num, 0);
- },
- /**
- * 多项式模运算
- * @param {QRPolynomial} e 模多项式
- * @return {}
- */
- mod: function(e) {
- var tl = this.getLength(),
- el = e.getLength();
- if (tl - el < 0) {
- return this;
- }
- var num = new Array(tl);
- for (var i = 0; i < tl; i++) {
- num[i] = this.get(i);
- }
- while (num.length >= el) {
- var ratio = QRMath.glog(num[0]) - QRMath.glog(e.get(0));
- for (var i = 0; i < e.getLength(); i++) {
- num[i] ^= QRMath.gexp(QRMath.glog(e.get(i)) + ratio);
- }
- while (num[0] == 0) {
- num.shift();
- }
- }
- return new QRPolynomial(num, 0);
- }
- };
- //---------------------------------------------------------------------
- // RS_BLOCK_TABLE
- //---------------------------------------------------------------------
- /*
- 二维码各个版本信息[块数, 每块中的数据块数, 每块中的信息块数]
- */
- var RS_BLOCK_TABLE = [
- // L
- // M
- // Q
- // H
- // 1
- [1, 26, 19],
- [1, 26, 16],
- [1, 26, 13],
- [1, 26, 9],
- // 2
- [1, 44, 34],
- [1, 44, 28],
- [1, 44, 22],
- [1, 44, 16],
- // 3
- [1, 70, 55],
- [1, 70, 44],
- [2, 35, 17],
- [2, 35, 13],
- // 4
- [1, 100, 80],
- [2, 50, 32],
- [2, 50, 24],
- [4, 25, 9],
- // 5
- [1, 134, 108],
- [2, 67, 43],
- [2, 33, 15, 2, 34, 16],
- [2, 33, 11, 2, 34, 12],
- // 6
- [2, 86, 68],
- [4, 43, 27],
- [4, 43, 19],
- [4, 43, 15],
- // 7
- [2, 98, 78],
- [4, 49, 31],
- [2, 32, 14, 4, 33, 15],
- [4, 39, 13, 1, 40, 14],
- // 8
- [2, 121, 97],
- [2, 60, 38, 2, 61, 39],
- [4, 40, 18, 2, 41, 19],
- [4, 40, 14, 2, 41, 15],
- // 9
- [2, 146, 116],
- [3, 58, 36, 2, 59, 37],
- [4, 36, 16, 4, 37, 17],
- [4, 36, 12, 4, 37, 13],
- // 10
- [2, 86, 68, 2, 87, 69],
- [4, 69, 43, 1, 70, 44],
- [6, 43, 19, 2, 44, 20],
- [6, 43, 15, 2, 44, 16],
- // 11
- [4, 101, 81],
- [1, 80, 50, 4, 81, 51],
- [4, 50, 22, 4, 51, 23],
- [3, 36, 12, 8, 37, 13],
- // 12
- [2, 116, 92, 2, 117, 93],
- [6, 58, 36, 2, 59, 37],
- [4, 46, 20, 6, 47, 21],
- [7, 42, 14, 4, 43, 15],
- // 13
- [4, 133, 107],
- [8, 59, 37, 1, 60, 38],
- [8, 44, 20, 4, 45, 21],
- [12, 33, 11, 4, 34, 12],
- // 14
- [3, 145, 115, 1, 146, 116],
- [4, 64, 40, 5, 65, 41],
- [11, 36, 16, 5, 37, 17],
- [11, 36, 12, 5, 37, 13],
- // 15
- [5, 109, 87, 1, 110, 88],
- [5, 65, 41, 5, 66, 42],
- [5, 54, 24, 7, 55, 25],
- [11, 36, 12],
- // 16
- [5, 122, 98, 1, 123, 99],
- [7, 73, 45, 3, 74, 46],
- [15, 43, 19, 2, 44, 20],
- [3, 45, 15, 13, 46, 16],
- // 17
- [1, 135, 107, 5, 136, 108],
- [10, 74, 46, 1, 75, 47],
- [1, 50, 22, 15, 51, 23],
- [2, 42, 14, 17, 43, 15],
- // 18
- [5, 150, 120, 1, 151, 121],
- [9, 69, 43, 4, 70, 44],
- [17, 50, 22, 1, 51, 23],
- [2, 42, 14, 19, 43, 15],
- // 19
- [3, 141, 113, 4, 142, 114],
- [3, 70, 44, 11, 71, 45],
- [17, 47, 21, 4, 48, 22],
- [9, 39, 13, 16, 40, 14],
- // 20
- [3, 135, 107, 5, 136, 108],
- [3, 67, 41, 13, 68, 42],
- [15, 54, 24, 5, 55, 25],
- [15, 43, 15, 10, 44, 16],
- // 21
- [4, 144, 116, 4, 145, 117],
- [17, 68, 42],
- [17, 50, 22, 6, 51, 23],
- [19, 46, 16, 6, 47, 17],
- // 22
- [2, 139, 111, 7, 140, 112],
- [17, 74, 46],
- [7, 54, 24, 16, 55, 25],
- [34, 37, 13],
- // 23
- [4, 151, 121, 5, 152, 122],
- [4, 75, 47, 14, 76, 48],
- [11, 54, 24, 14, 55, 25],
- [16, 45, 15, 14, 46, 16],
- // 24
- [6, 147, 117, 4, 148, 118],
- [6, 73, 45, 14, 74, 46],
- [11, 54, 24, 16, 55, 25],
- [30, 46, 16, 2, 47, 17],
- // 25
- [8, 132, 106, 4, 133, 107],
- [8, 75, 47, 13, 76, 48],
- [7, 54, 24, 22, 55, 25],
- [22, 45, 15, 13, 46, 16],
- // 26
- [10, 142, 114, 2, 143, 115],
- [19, 74, 46, 4, 75, 47],
- [28, 50, 22, 6, 51, 23],
- [33, 46, 16, 4, 47, 17],
- // 27
- [8, 152, 122, 4, 153, 123],
- [22, 73, 45, 3, 74, 46],
- [8, 53, 23, 26, 54, 24],
- [12, 45, 15, 28, 46, 16],
- // 28
- [3, 147, 117, 10, 148, 118],
- [3, 73, 45, 23, 74, 46],
- [4, 54, 24, 31, 55, 25],
- [11, 45, 15, 31, 46, 16],
- // 29
- [7, 146, 116, 7, 147, 117],
- [21, 73, 45, 7, 74, 46],
- [1, 53, 23, 37, 54, 24],
- [19, 45, 15, 26, 46, 16],
- // 30
- [5, 145, 115, 10, 146, 116],
- [19, 75, 47, 10, 76, 48],
- [15, 54, 24, 25, 55, 25],
- [23, 45, 15, 25, 46, 16],
- // 31
- [13, 145, 115, 3, 146, 116],
- [2, 74, 46, 29, 75, 47],
- [42, 54, 24, 1, 55, 25],
- [23, 45, 15, 28, 46, 16],
- // 32
- [17, 145, 115],
- [10, 74, 46, 23, 75, 47],
- [10, 54, 24, 35, 55, 25],
- [19, 45, 15, 35, 46, 16],
- // 33
- [17, 145, 115, 1, 146, 116],
- [14, 74, 46, 21, 75, 47],
- [29, 54, 24, 19, 55, 25],
- [11, 45, 15, 46, 46, 16],
- // 34
- [13, 145, 115, 6, 146, 116],
- [14, 74, 46, 23, 75, 47],
- [44, 54, 24, 7, 55, 25],
- [59, 46, 16, 1, 47, 17],
- // 35
- [12, 151, 121, 7, 152, 122],
- [12, 75, 47, 26, 76, 48],
- [39, 54, 24, 14, 55, 25],
- [22, 45, 15, 41, 46, 16],
- // 36
- [6, 151, 121, 14, 152, 122],
- [6, 75, 47, 34, 76, 48],
- [46, 54, 24, 10, 55, 25],
- [2, 45, 15, 64, 46, 16],
- // 37
- [17, 152, 122, 4, 153, 123],
- [29, 74, 46, 14, 75, 47],
- [49, 54, 24, 10, 55, 25],
- [24, 45, 15, 46, 46, 16],
- // 38
- [4, 152, 122, 18, 153, 123],
- [13, 74, 46, 32, 75, 47],
- [48, 54, 24, 14, 55, 25],
- [42, 45, 15, 32, 46, 16],
- // 39
- [20, 147, 117, 4, 148, 118],
- [40, 75, 47, 7, 76, 48],
- [43, 54, 24, 22, 55, 25],
- [10, 45, 15, 67, 46, 16],
- // 40
- [19, 148, 118, 6, 149, 119],
- [18, 75, 47, 31, 76, 48],
- [34, 54, 24, 34, 55, 25],
- [20, 45, 15, 61, 46, 16]
- ];
- /**
- * 根据数据获取对应版本
- * @return {[type]} [description]
- */
- QRCodeAlg.prototype.getRightType = function() {
- for (var typeNumber = 1; typeNumber < 41; typeNumber++) {
- var rsBlock = RS_BLOCK_TABLE[(typeNumber - 1) * 4 + this.errorCorrectLevel];
- if (rsBlock == undefined) {
- throw new Error("bad rs block @ typeNumber:" + typeNumber + "/errorCorrectLevel:" + this.errorCorrectLevel);
- }
- var length = rsBlock.length / 3;
- var totalDataCount = 0;
- for (var i = 0; i < length; i++) {
- var count = rsBlock[i * 3 + 0];
- var dataCount = rsBlock[i * 3 + 2];
- totalDataCount += dataCount * count;
- }
- var lengthBytes = typeNumber > 9 ? 2 : 1;
- if (this.utf8bytes.length + lengthBytes < totalDataCount || typeNumber == 40) {
- this.typeNumber = typeNumber;
- this.rsBlock = rsBlock;
- this.totalDataCount = totalDataCount;
- break;
- }
- }
- };
- //---------------------------------------------------------------------
- // QRBitBuffer
- //---------------------------------------------------------------------
- function QRBitBuffer() {
- this.buffer = new Array();
- this.length = 0;
- }
- QRBitBuffer.prototype = {
- get: function(index) {
- var bufIndex = Math.floor(index / 8);
- return ((this.buffer[bufIndex] >>> (7 - index % 8)) & 1);
- },
- put: function(num, length) {
- for (var i = 0; i < length; i++) {
- this.putBit(((num >>> (length - i - 1)) & 1));
- }
- },
- putBit: function(bit) {
- var bufIndex = Math.floor(this.length / 8);
- if (this.buffer.length <= bufIndex) {
- this.buffer.push(0);
- }
- if (bit) {
- this.buffer[bufIndex] |= (0x80 >>> (this.length % 8));
- }
- this.length++;
- }
- };
- /**
- * 获取单个字符的utf8编码
- * unicode BMP平面约65535个字符
- * @param {num} code
- * return {array}
- */
- function unicodeFormat8(code) {
- // 1 byte
- if (code < 128) {
- return [code];
- // 2 bytes
- } else if (code < 2048) {
- c0 = 192 + (code >> 6);
- c1 = 128 + (code & 63);
- return [c0, c1];
- // 3 bytes
- } else {
- c0 = 224 + (code >> 12);
- c1 = 128 + (code >> 6 & 63);
- c2 = 128 + (code & 63);
- return [c0, c1, c2];
- }
- }
- /**
- * 获取字符串的utf8编码字节串
- * @param {string} string
- * @return {array}
- */
- function getUTF8Bytes(string) {
- var utf8codes = [];
- for (var i = 0; i < string.length; i++) {
- var code = string.charCodeAt(i);
- var utf8 = unicodeFormat8(code);
- for (var j = 0; j < utf8.length; j++) {
- utf8codes.push(utf8[j]);
- }
- }
- return utf8codes;
- }
- /**
- * 二维码算法实现
- * @param {string} data 要编码的信息字符串
- * @param {num} errorCorrectLevel 纠错等级
- */
- function QRCodeAlg(data, errorCorrectLevel) {
- this.typeNumber = -1; //版本
- this.errorCorrectLevel = errorCorrectLevel;
- this.modules = null; //二维矩阵,存放最终结果
- this.moduleCount = 0; //矩阵大小
- this.dataCache = null; //数据缓存
- this.rsBlocks = null; //版本数据信息
- this.totalDataCount = -1; //可使用的数据量
- this.data = data;
- this.utf8bytes = getUTF8Bytes(data);
- this.make();
- }
- QRCodeAlg.prototype = {
- constructor: QRCodeAlg,
- /**
- * 获取二维码矩阵大小
- * @return {num} 矩阵大小
- */
- getModuleCount: function() {
- return this.moduleCount;
- },
- /**
- * 编码
- */
- make: function() {
- this.getRightType();
- this.dataCache = this.createData();
- this.createQrcode();
- },
- /**
- * 设置二位矩阵功能图形
- * @param {bool} test 表示是否在寻找最好掩膜阶段
- * @param {num} maskPattern 掩膜的版本
- */
- makeImpl: function(maskPattern) {
- this.moduleCount = this.typeNumber * 4 + 17;
- this.modules = new Array(this.moduleCount);
- for (var row = 0; row < this.moduleCount; row++) {
- this.modules[row] = new Array(this.moduleCount);
- }
- this.setupPositionProbePattern(0, 0);
- this.setupPositionProbePattern(this.moduleCount - 7, 0);
- this.setupPositionProbePattern(0, this.moduleCount - 7);
- this.setupPositionAdjustPattern();
- this.setupTimingPattern();
- this.setupTypeInfo(true, maskPattern);
- if (this.typeNumber >= 7) {
- this.setupTypeNumber(true);
- }
- this.mapData(this.dataCache, maskPattern);
- },
- /**
- * 设置二维码的位置探测图形
- * @param {num} row 探测图形的中心横坐标
- * @param {num} col 探测图形的中心纵坐标
- */
- setupPositionProbePattern: function(row, col) {
- for (var r = -1; r <= 7; r++) {
- if (row + r <= -1 || this.moduleCount <= row + r) continue;
- for (var c = -1; c <= 7; c++) {
- if (col + c <= -1 || this.moduleCount <= col + c) continue;
- if ((0 <= r && r <= 6 && (c == 0 || c == 6)) || (0 <= c && c <= 6 && (r == 0 || r == 6)) || (2 <= r && r <= 4 && 2 <= c && c <= 4)) {
- this.modules[row + r][col + c] = true;
- } else {
- this.modules[row + r][col + c] = false;
- }
- }
- }
- },
- /**
- * 创建二维码
- * @return {[type]} [description]
- */
- createQrcode: function() {
- var minLostPoint = 0;
- var pattern = 0;
- var bestModules = null;
- for (var i = 0; i < 8; i++) {
- this.makeImpl(i);
- var lostPoint = QRUtil.getLostPoint(this);
- if (i == 0 || minLostPoint > lostPoint) {
- minLostPoint = lostPoint;
- pattern = i;
- bestModules = this.modules;
- }
- }
- this.modules = bestModules;
- this.setupTypeInfo(false, pattern);
- if (this.typeNumber >= 7) {
- this.setupTypeNumber(false);
- }
- },
- /**
- * 设置定位图形
- * @return {[type]} [description]
- */
- setupTimingPattern: function() {
- for (var r = 8; r < this.moduleCount - 8; r++) {
- if (this.modules[r][6] != null) {
- continue;
- }
- this.modules[r][6] = (r % 2 == 0);
- if (this.modules[6][r] != null) {
- continue;
- }
- this.modules[6][r] = (r % 2 == 0);
- }
- },
- /**
- * 设置矫正图形
- * @return {[type]} [description]
- */
- setupPositionAdjustPattern: function() {
- var pos = QRUtil.getPatternPosition(this.typeNumber);
- for (var i = 0; i < pos.length; i++) {
- for (var j = 0; j < pos.length; j++) {
- var row = pos[i];
- var col = pos[j];
- if (this.modules[row][col] != null) {
- continue;
- }
- for (var r = -2; r <= 2; r++) {
- for (var c = -2; c <= 2; c++) {
- if (r == -2 || r == 2 || c == -2 || c == 2 || (r == 0 && c == 0)) {
- this.modules[row + r][col + c] = true;
- } else {
- this.modules[row + r][col + c] = false;
- }
- }
- }
- }
- }
- },
- /**
- * 设置版本信息(7以上版本才有)
- * @param {bool} test 是否处于判断最佳掩膜阶段
- * @return {[type]} [description]
- */
- setupTypeNumber: function(test) {
- var bits = QRUtil.getBCHTypeNumber(this.typeNumber);
- for (var i = 0; i < 18; i++) {
- var mod = (!test && ((bits >> i) & 1) == 1);
- this.modules[Math.floor(i / 3)][i % 3 + this.moduleCount - 8 - 3] = mod;
- this.modules[i % 3 + this.moduleCount - 8 - 3][Math.floor(i / 3)] = mod;
- }
- },
- /**
- * 设置格式信息(纠错等级和掩膜版本)
- * @param {bool} test
- * @param {num} maskPattern 掩膜版本
- * @return {}
- */
- setupTypeInfo: function(test, maskPattern) {
- var data = (QRErrorCorrectLevel[this.errorCorrectLevel] << 3) | maskPattern;
- var bits = QRUtil.getBCHTypeInfo(data);
- // vertical
- for (var i = 0; i < 15; i++) {
- var mod = (!test && ((bits >> i) & 1) == 1);
- if (i < 6) {
- this.modules[i][8] = mod;
- } else if (i < 8) {
- this.modules[i + 1][8] = mod;
- } else {
- this.modules[this.moduleCount - 15 + i][8] = mod;
- }
- // horizontal
- var mod = (!test && ((bits >> i) & 1) == 1);
- if (i < 8) {
- this.modules[8][this.moduleCount - i - 1] = mod;
- } else if (i < 9) {
- this.modules[8][15 - i - 1 + 1] = mod;
- } else {
- this.modules[8][15 - i - 1] = mod;
- }
- }
- // fixed module
- this.modules[this.moduleCount - 8][8] = (!test);
- },
- /**
- * 数据编码
- * @return {[type]} [description]
- */
- createData: function() {
- var buffer = new QRBitBuffer();
- var lengthBits = this.typeNumber > 9 ? 16 : 8;
- buffer.put(4, 4); //添加模式
- buffer.put(this.utf8bytes.length, lengthBits);
- for (var i = 0, l = this.utf8bytes.length; i < l; i++) {
- buffer.put(this.utf8bytes[i], 8);
- }
- if (buffer.length + 4 <= this.totalDataCount * 8) {
- buffer.put(0, 4);
- }
- // padding
- while (buffer.length % 8 != 0) {
- buffer.putBit(false);
- }
- // padding
- while (true) {
- if (buffer.length >= this.totalDataCount * 8) {
- break;
- }
- buffer.put(QRCodeAlg.PAD0, 8);
- if (buffer.length >= this.totalDataCount * 8) {
- break;
- }
- buffer.put(QRCodeAlg.PAD1, 8);
- }
- return this.createBytes(buffer);
- },
- /**
- * 纠错码编码
- * @param {buffer} buffer 数据编码
- * @return {[type]}
- */
- createBytes: function(buffer) {
- var offset = 0;
- var maxDcCount = 0;
- var maxEcCount = 0;
- var length = this.rsBlock.length / 3;
- var rsBlocks = new Array();
- for (var i = 0; i < length; i++) {
- var count = this.rsBlock[i * 3 + 0];
- var totalCount = this.rsBlock[i * 3 + 1];
- var dataCount = this.rsBlock[i * 3 + 2];
- for (var j = 0; j < count; j++) {
- rsBlocks.push([dataCount, totalCount]);
- }
- }
- var dcdata = new Array(rsBlocks.length);
- var ecdata = new Array(rsBlocks.length);
- for (var r = 0; r < rsBlocks.length; r++) {
- var dcCount = rsBlocks[r][0];
- var ecCount = rsBlocks[r][1] - dcCount;
- maxDcCount = Math.max(maxDcCount, dcCount);
- maxEcCount = Math.max(maxEcCount, ecCount);
- dcdata[r] = new Array(dcCount);
- for (var i = 0; i < dcdata[r].length; i++) {
- dcdata[r][i] = 0xff & buffer.buffer[i + offset];
- }
- offset += dcCount;
- var rsPoly = QRUtil.getErrorCorrectPolynomial(ecCount);
- var rawPoly = new QRPolynomial(dcdata[r], rsPoly.getLength() - 1);
- var modPoly = rawPoly.mod(rsPoly);
- ecdata[r] = new Array(rsPoly.getLength() - 1);
- for (var i = 0; i < ecdata[r].length; i++) {
- var modIndex = i + modPoly.getLength() - ecdata[r].length;
- ecdata[r][i] = (modIndex >= 0) ? modPoly.get(modIndex) : 0;
- }
- }
- var data = new Array(this.totalDataCount);
- var index = 0;
- for (var i = 0; i < maxDcCount; i++) {
- for (var r = 0; r < rsBlocks.length; r++) {
- if (i < dcdata[r].length) {
- data[index++] = dcdata[r][i];
- }
- }
- }
- for (var i = 0; i < maxEcCount; i++) {
- for (var r = 0; r < rsBlocks.length; r++) {
- if (i < ecdata[r].length) {
- data[index++] = ecdata[r][i];
- }
- }
- }
- return data;
- },
- /**
- * 布置模块,构建最终信息
- * @param {} data
- * @param {} maskPattern
- * @return {}
- */
- mapData: function(data, maskPattern) {
- var inc = -1;
- var row = this.moduleCount - 1;
- var bitIndex = 7;
- var byteIndex = 0;
- for (var col = this.moduleCount - 1; col > 0; col -= 2) {
- if (col == 6) col--;
- while (true) {
- for (var c = 0; c < 2; c++) {
- if (this.modules[row][col - c] == null) {
- var dark = false;
- if (byteIndex < data.length) {
- dark = (((data[byteIndex] >>> bitIndex) & 1) == 1);
- }
- var mask = QRUtil.getMask(maskPattern, row, col - c);
- if (mask) {
- dark = !dark;
- }
- this.modules[row][col - c] = dark;
- bitIndex--;
- if (bitIndex == -1) {
- byteIndex++;
- bitIndex = 7;
- }
- }
- }
- row += inc;
- if (row < 0 || this.moduleCount <= row) {
- row -= inc;
- inc = -inc;
- break;
- }
- }
- }
- }
- };
- /**
- * 填充字段
- */
- QRCodeAlg.PAD0 = 0xEC;
- QRCodeAlg.PAD1 = 0x11;
- //---------------------------------------------------------------------
- // 纠错等级对应的编码
- //---------------------------------------------------------------------
- var QRErrorCorrectLevel = [1, 0, 3, 2];
- //---------------------------------------------------------------------
- // 掩膜版本
- //---------------------------------------------------------------------
- var QRMaskPattern = {
- PATTERN000: 0,
- PATTERN001: 1,
- PATTERN010: 2,
- PATTERN011: 3,
- PATTERN100: 4,
- PATTERN101: 5,
- PATTERN110: 6,
- PATTERN111: 7
- };
- //---------------------------------------------------------------------
- // 工具类
- //---------------------------------------------------------------------
- var QRUtil = {
- /*
- 每个版本矫正图形的位置
- */
- PATTERN_POSITION_TABLE: [
- [],
- [6, 18],
- [6, 22],
- [6, 26],
- [6, 30],
- [6, 34],
- [6, 22, 38],
- [6, 24, 42],
- [6, 26, 46],
- [6, 28, 50],
- [6, 30, 54],
- [6, 32, 58],
- [6, 34, 62],
- [6, 26, 46, 66],
- [6, 26, 48, 70],
- [6, 26, 50, 74],
- [6, 30, 54, 78],
- [6, 30, 56, 82],
- [6, 30, 58, 86],
- [6, 34, 62, 90],
- [6, 28, 50, 72, 94],
- [6, 26, 50, 74, 98],
- [6, 30, 54, 78, 102],
- [6, 28, 54, 80, 106],
- [6, 32, 58, 84, 110],
- [6, 30, 58, 86, 114],
- [6, 34, 62, 90, 118],
- [6, 26, 50, 74, 98, 122],
- [6, 30, 54, 78, 102, 126],
- [6, 26, 52, 78, 104, 130],
- [6, 30, 56, 82, 108, 134],
- [6, 34, 60, 86, 112, 138],
- [6, 30, 58, 86, 114, 142],
- [6, 34, 62, 90, 118, 146],
- [6, 30, 54, 78, 102, 126, 150],
- [6, 24, 50, 76, 102, 128, 154],
- [6, 28, 54, 80, 106, 132, 158],
- [6, 32, 58, 84, 110, 136, 162],
- [6, 26, 54, 82, 110, 138, 166],
- [6, 30, 58, 86, 114, 142, 170]
- ],
- G15: (1 << 10) | (1 << 8) | (1 << 5) | (1 << 4) | (1 << 2) | (1 << 1) | (1 << 0),
- G18: (1 << 12) | (1 << 11) | (1 << 10) | (1 << 9) | (1 << 8) | (1 << 5) | (1 << 2) | (1 << 0),
- G15_MASK: (1 << 14) | (1 << 12) | (1 << 10) | (1 << 4) | (1 << 1),
- /*
- BCH编码格式信息
- */
- getBCHTypeInfo: function(data) {
- var d = data << 10;
- while (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G15) >= 0) {
- d ^= (QRUtil.G15 << (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G15)));
- }
- return ((data << 10) | d) ^ QRUtil.G15_MASK;
- },
- /*
- BCH编码版本信息
- */
- getBCHTypeNumber: function(data) {
- var d = data << 12;
- while (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G18) >= 0) {
- d ^= (QRUtil.G18 << (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G18)));
- }
- return (data << 12) | d;
- },
- /*
- 获取BCH位信息
- */
- getBCHDigit: function(data) {
- var digit = 0;
- while (data != 0) {
- digit++;
- data >>>= 1;
- }
- return digit;
- },
- /*
- 获取版本对应的矫正图形位置
- */
- getPatternPosition: function(typeNumber) {
- return QRUtil.PATTERN_POSITION_TABLE[typeNumber - 1];
- },
- /*
- 掩膜算法
- */
- getMask: function(maskPattern, i, j) {
- switch (maskPattern) {
- case QRMaskPattern.PATTERN000:
- return (i + j) % 2 == 0;
- case QRMaskPattern.PATTERN001:
- return i % 2 == 0;
- case QRMaskPattern.PATTERN010:
- return j % 3 == 0;
- case QRMaskPattern.PATTERN011:
- return (i + j) % 3 == 0;
- case QRMaskPattern.PATTERN100:
- return (Math.floor(i / 2) + Math.floor(j / 3)) % 2 == 0;
- case QRMaskPattern.PATTERN101:
- return (i * j) % 2 + (i * j) % 3 == 0;
- case QRMaskPattern.PATTERN110:
- return ((i * j) % 2 + (i * j) % 3) % 2 == 0;
- case QRMaskPattern.PATTERN111:
- return ((i * j) % 3 + (i + j) % 2) % 2 == 0;
- default:
- throw new Error("bad maskPattern:" + maskPattern);
- }
- },
- /*
- 获取RS的纠错多项式
- */
- getErrorCorrectPolynomial: function(errorCorrectLength) {
- var a = new QRPolynomial([1], 0);
- for (var i = 0; i < errorCorrectLength; i++) {
- a = a.multiply(new QRPolynomial([1, QRMath.gexp(i)], 0));
- }
- return a;
- },
- /*
- 获取评价
- */
- getLostPoint: function(qrCode) {
- var moduleCount = qrCode.getModuleCount(),
- lostPoint = 0,
- darkCount = 0;
- for (var row = 0; row < moduleCount; row++) {
- var sameCount = 0;
- var head = qrCode.modules[row][0];
- for (var col = 0; col < moduleCount; col++) {
- var current = qrCode.modules[row][col];
- //level 3 评价
- if (col < moduleCount - 6) {
- if (current && !qrCode.modules[row][col + 1] && qrCode.modules[row][col + 2] && qrCode.modules[row][col + 3] && qrCode.modules[row][col + 4] && !qrCode.modules[row][col + 5] && qrCode.modules[row][col + 6]) {
- if (col < moduleCount - 10) {
- if (qrCode.modules[row][col + 7] && qrCode.modules[row][col + 8] && qrCode.modules[row][col + 9] && qrCode.modules[row][col + 10]) {
- lostPoint += 40;
- }
- } else if (col > 3) {
- if (qrCode.modules[row][col - 1] && qrCode.modules[row][col - 2] && qrCode.modules[row][col - 3] && qrCode.modules[row][col - 4]) {
- lostPoint += 40;
- }
- }
- }
- }
- //level 2 评价
- if ((row < moduleCount - 1) && (col < moduleCount - 1)) {
- var count = 0;
- if (current) count++;
- if (qrCode.modules[row + 1][col]) count++;
- if (qrCode.modules[row][col + 1]) count++;
- if (qrCode.modules[row + 1][col + 1]) count++;
- if (count == 0 || count == 4) {
- lostPoint += 3;
- }
- }
- //level 1 评价
- if (head ^ current) {
- sameCount++;
- } else {
- head = current;
- if (sameCount >= 5) {
- lostPoint += (3 + sameCount - 5);
- }
- sameCount = 1;
- }
- //level 4 评价
- if (current) {
- darkCount++;
- }
- }
- }
- for (var col = 0; col < moduleCount; col++) {
- var sameCount = 0;
- var head = qrCode.modules[0][col];
- for (var row = 0; row < moduleCount; row++) {
- var current = qrCode.modules[row][col];
- //level 3 评价
- if (row < moduleCount - 6) {
- if (current && !qrCode.modules[row + 1][col] && qrCode.modules[row + 2][col] && qrCode.modules[row + 3][col] && qrCode.modules[row + 4][col] && !qrCode.modules[row + 5][col] && qrCode.modules[row + 6][col]) {
- if (row < moduleCount - 10) {
- if (qrCode.modules[row + 7][col] && qrCode.modules[row + 8][col] && qrCode.modules[row + 9][col] && qrCode.modules[row + 10][col]) {
- lostPoint += 40;
- }
- } else if (row > 3) {
- if (qrCode.modules[row - 1][col] && qrCode.modules[row - 2][col] && qrCode.modules[row - 3][col] && qrCode.modules[row - 4][col]) {
- lostPoint += 40;
- }
- }
- }
- }
- //level 1 评价
- if (head ^ current) {
- sameCount++;
- } else {
- head = current;
- if (sameCount >= 5) {
- lostPoint += (3 + sameCount - 5);
- }
- sameCount = 1;
- }
- }
- }
- // LEVEL4
- var ratio = Math.abs(100 * darkCount / moduleCount / moduleCount - 50) / 5;
- lostPoint += ratio * 10;
- return lostPoint;
- }
- };
- //---------------------------------------------------------------------
- // QRMath使用的数学工具
- //---------------------------------------------------------------------
- var QRMath = {
- /*
- 将n转化为a^m
- */
- glog: function(n) {
- if (n < 1) {
- throw new Error("glog(" + n + ")");
- }
- return QRMath.LOG_TABLE[n];
- },
- /*
- 将a^m转化为n
- */
- gexp: function(n) {
- while (n < 0) {
- n += 255;
- }
- while (n >= 256) {
- n -= 255;
- }
- return QRMath.EXP_TABLE[n];
- },
- EXP_TABLE: new Array(256),
- LOG_TABLE: new Array(256)
- };
- for (var i = 0; i < 8; i++) {
- QRMath.EXP_TABLE[i] = 1 << i;
- }
- for (var i = 8; i < 256; i++) {
- QRMath.EXP_TABLE[i] = QRMath.EXP_TABLE[i - 4] ^ QRMath.EXP_TABLE[i - 5] ^ QRMath.EXP_TABLE[i - 6] ^ QRMath.EXP_TABLE[i - 8];
- }
- for (var i = 0; i < 255; i++) {
- QRMath.LOG_TABLE[QRMath.EXP_TABLE[i]] = i;
- }
- //---------------------------------------------------------------------
- // QRPolynomial 多项式
- //---------------------------------------------------------------------
- /**
- * 多项式类
- * @param {Array} num 系数
- * @param {num} shift a^shift
- */
- function QRPolynomial(num, shift) {
- if (num.length == undefined) {
- throw new Error(num.length + "/" + shift);
- }
- var offset = 0;
- while (offset < num.length && num[offset] == 0) {
- offset++;
- }
- this.num = new Array(num.length - offset + shift);
- for (var i = 0; i < num.length - offset; i++) {
- this.num[i] = num[i + offset];
- }
- }
- QRPolynomial.prototype = {
- get: function(index) {
- return this.num[index];
- },
- getLength: function() {
- return this.num.length;
- },
- /**
- * 多项式乘法
- * @param {QRPolynomial} e 被乘多项式
- * @return {[type]} [description]
- */
- multiply: function(e) {
- var num = new Array(this.getLength() + e.getLength() - 1);
- for (var i = 0; i < this.getLength(); i++) {
- for (var j = 0; j < e.getLength(); j++) {
- num[i + j] ^= QRMath.gexp(QRMath.glog(this.get(i)) + QRMath.glog(e.get(j)));
- }
- }
- return new QRPolynomial(num, 0);
- },
- /**
- * 多项式模运算
- * @param {QRPolynomial} e 模多项式
- * @return {}
- */
- mod: function(e) {
- var tl = this.getLength(),
- el = e.getLength();
- if (tl - el < 0) {
- return this;
- }
- var num = new Array(tl);
- for (var i = 0; i < tl; i++) {
- num[i] = this.get(i);
- }
- while (num.length >= el) {
- var ratio = QRMath.glog(num[0]) - QRMath.glog(e.get(0));
- for (var i = 0; i < e.getLength(); i++) {
- num[i] ^= QRMath.gexp(QRMath.glog(e.get(i)) + ratio);
- }
- while (num[0] == 0) {
- num.shift();
- }
- }
- return new QRPolynomial(num, 0);
- }
- };
- //---------------------------------------------------------------------
- // RS_BLOCK_TABLE
- //---------------------------------------------------------------------
- /*
- 二维码各个版本信息[块数, 每块中的数据块数, 每块中的信息块数]
- */
- RS_BLOCK_TABLE = [
- // L
- // M
- // Q
- // H
- // 1
- [1, 26, 19],
- [1, 26, 16],
- [1, 26, 13],
- [1, 26, 9],
- // 2
- [1, 44, 34],
- [1, 44, 28],
- [1, 44, 22],
- [1, 44, 16],
- // 3
- [1, 70, 55],
- [1, 70, 44],
- [2, 35, 17],
- [2, 35, 13],
- // 4
- [1, 100, 80],
- [2, 50, 32],
- [2, 50, 24],
- [4, 25, 9],
- // 5
- [1, 134, 108],
- [2, 67, 43],
- [2, 33, 15, 2, 34, 16],
- [2, 33, 11, 2, 34, 12],
- // 6
- [2, 86, 68],
- [4, 43, 27],
- [4, 43, 19],
- [4, 43, 15],
- // 7
- [2, 98, 78],
- [4, 49, 31],
- [2, 32, 14, 4, 33, 15],
- [4, 39, 13, 1, 40, 14],
- // 8
- [2, 121, 97],
- [2, 60, 38, 2, 61, 39],
- [4, 40, 18, 2, 41, 19],
- [4, 40, 14, 2, 41, 15],
- // 9
- [2, 146, 116],
- [3, 58, 36, 2, 59, 37],
- [4, 36, 16, 4, 37, 17],
- [4, 36, 12, 4, 37, 13],
- // 10
- [2, 86, 68, 2, 87, 69],
- [4, 69, 43, 1, 70, 44],
- [6, 43, 19, 2, 44, 20],
- [6, 43, 15, 2, 44, 16],
- // 11
- [4, 101, 81],
- [1, 80, 50, 4, 81, 51],
- [4, 50, 22, 4, 51, 23],
- [3, 36, 12, 8, 37, 13],
- // 12
- [2, 116, 92, 2, 117, 93],
- [6, 58, 36, 2, 59, 37],
- [4, 46, 20, 6, 47, 21],
- [7, 42, 14, 4, 43, 15],
- // 13
- [4, 133, 107],
- [8, 59, 37, 1, 60, 38],
- [8, 44, 20, 4, 45, 21],
- [12, 33, 11, 4, 34, 12],
- // 14
- [3, 145, 115, 1, 146, 116],
- [4, 64, 40, 5, 65, 41],
- [11, 36, 16, 5, 37, 17],
- [11, 36, 12, 5, 37, 13],
- // 15
- [5, 109, 87, 1, 110, 88],
- [5, 65, 41, 5, 66, 42],
- [5, 54, 24, 7, 55, 25],
- [11, 36, 12],
- // 16
- [5, 122, 98, 1, 123, 99],
- [7, 73, 45, 3, 74, 46],
- [15, 43, 19, 2, 44, 20],
- [3, 45, 15, 13, 46, 16],
- // 17
- [1, 135, 107, 5, 136, 108],
- [10, 74, 46, 1, 75, 47],
- [1, 50, 22, 15, 51, 23],
- [2, 42, 14, 17, 43, 15],
- // 18
- [5, 150, 120, 1, 151, 121],
- [9, 69, 43, 4, 70, 44],
- [17, 50, 22, 1, 51, 23],
- [2, 42, 14, 19, 43, 15],
- // 19
- [3, 141, 113, 4, 142, 114],
- [3, 70, 44, 11, 71, 45],
- [17, 47, 21, 4, 48, 22],
- [9, 39, 13, 16, 40, 14],
- // 20
- [3, 135, 107, 5, 136, 108],
- [3, 67, 41, 13, 68, 42],
- [15, 54, 24, 5, 55, 25],
- [15, 43, 15, 10, 44, 16],
- // 21
- [4, 144, 116, 4, 145, 117],
- [17, 68, 42],
- [17, 50, 22, 6, 51, 23],
- [19, 46, 16, 6, 47, 17],
- // 22
- [2, 139, 111, 7, 140, 112],
- [17, 74, 46],
- [7, 54, 24, 16, 55, 25],
- [34, 37, 13],
- // 23
- [4, 151, 121, 5, 152, 122],
- [4, 75, 47, 14, 76, 48],
- [11, 54, 24, 14, 55, 25],
- [16, 45, 15, 14, 46, 16],
- // 24
- [6, 147, 117, 4, 148, 118],
- [6, 73, 45, 14, 74, 46],
- [11, 54, 24, 16, 55, 25],
- [30, 46, 16, 2, 47, 17],
- // 25
- [8, 132, 106, 4, 133, 107],
- [8, 75, 47, 13, 76, 48],
- [7, 54, 24, 22, 55, 25],
- [22, 45, 15, 13, 46, 16],
- // 26
- [10, 142, 114, 2, 143, 115],
- [19, 74, 46, 4, 75, 47],
- [28, 50, 22, 6, 51, 23],
- [33, 46, 16, 4, 47, 17],
- // 27
- [8, 152, 122, 4, 153, 123],
- [22, 73, 45, 3, 74, 46],
- [8, 53, 23, 26, 54, 24],
- [12, 45, 15, 28, 46, 16],
- // 28
- [3, 147, 117, 10, 148, 118],
- [3, 73, 45, 23, 74, 46],
- [4, 54, 24, 31, 55, 25],
- [11, 45, 15, 31, 46, 16],
- // 29
- [7, 146, 116, 7, 147, 117],
- [21, 73, 45, 7, 74, 46],
- [1, 53, 23, 37, 54, 24],
- [19, 45, 15, 26, 46, 16],
- // 30
- [5, 145, 115, 10, 146, 116],
- [19, 75, 47, 10, 76, 48],
- [15, 54, 24, 25, 55, 25],
- [23, 45, 15, 25, 46, 16],
- // 31
- [13, 145, 115, 3, 146, 116],
- [2, 74, 46, 29, 75, 47],
- [42, 54, 24, 1, 55, 25],
- [23, 45, 15, 28, 46, 16],
- // 32
- [17, 145, 115],
- [10, 74, 46, 23, 75, 47],
- [10, 54, 24, 35, 55, 25],
- [19, 45, 15, 35, 46, 16],
- // 33
- [17, 145, 115, 1, 146, 116],
- [14, 74, 46, 21, 75, 47],
- [29, 54, 24, 19, 55, 25],
- [11, 45, 15, 46, 46, 16],
- // 34
- [13, 145, 115, 6, 146, 116],
- [14, 74, 46, 23, 75, 47],
- [44, 54, 24, 7, 55, 25],
- [59, 46, 16, 1, 47, 17],
- // 35
- [12, 151, 121, 7, 152, 122],
- [12, 75, 47, 26, 76, 48],
- [39, 54, 24, 14, 55, 25],
- [22, 45, 15, 41, 46, 16],
- // 36
- [6, 151, 121, 14, 152, 122],
- [6, 75, 47, 34, 76, 48],
- [46, 54, 24, 10, 55, 25],
- [2, 45, 15, 64, 46, 16],
- // 37
- [17, 152, 122, 4, 153, 123],
- [29, 74, 46, 14, 75, 47],
- [49, 54, 24, 10, 55, 25],
- [24, 45, 15, 46, 46, 16],
- // 38
- [4, 152, 122, 18, 153, 123],
- [13, 74, 46, 32, 75, 47],
- [48, 54, 24, 14, 55, 25],
- [42, 45, 15, 32, 46, 16],
- // 39
- [20, 147, 117, 4, 148, 118],
- [40, 75, 47, 7, 76, 48],
- [43, 54, 24, 22, 55, 25],
- [10, 45, 15, 67, 46, 16],
- // 40
- [19, 148, 118, 6, 149, 119],
- [18, 75, 47, 31, 76, 48],
- [34, 54, 24, 34, 55, 25],
- [20, 45, 15, 61, 46, 16]
- ];
- /**
- * 根据数据获取对应版本
- * @return {[type]} [description]
- */
- QRCodeAlg.prototype.getRightType = function() {
- for (var typeNumber = 1; typeNumber < 41; typeNumber++) {
- var rsBlock = RS_BLOCK_TABLE[(typeNumber - 1) * 4 + this.errorCorrectLevel];
- if (rsBlock == undefined) {
- throw new Error('bad rs block @ typeNumber:' + typeNumber + '/errorCorrectLevel:' + this.errorCorrectLevel);
- }
- var length = rsBlock.length / 3;
- var totalDataCount = 0;
- for (var i = 0; i < length; i++) {
- var count = rsBlock[i * 3 + 0];
- var dataCount = rsBlock[i * 3 + 2];
- totalDataCount += dataCount * count;
- }
- var lengthBytes = typeNumber > 9 ? 2 : 1;
- if (this.utf8bytes.length + lengthBytes < totalDataCount || typeNumber == 40) {
- this.typeNumber = typeNumber;
- this.rsBlock = rsBlock;
- this.totalDataCount = totalDataCount;
- break;
- }
- }
- };
- // ---------------------------------------------------------------------
- // QRBitBuffer
- // ---------------------------------------------------------------------
- function QRBitBuffer() {
- this.buffer = new Array();
- this.length = 0;
- }
- QRBitBuffer.prototype = {
- get: function(index) {
- var bufIndex = Math.floor(index / 8);
- return ((this.buffer[bufIndex] >>> (7 - index % 8)) & 1);
- },
- put: function(num, length) {
- for (var i = 0; i < length; i++) {
- this.putBit(((num >>> (length - i - 1)) & 1));
- }
- },
- putBit: function(bit) {
- var bufIndex = Math.floor(this.length / 8);
- if (this.buffer.length <= bufIndex) {
- this.buffer.push(0);
- }
- if (bit) {
- this.buffer[bufIndex] |= (0x80 >>> (this.length % 8));
- }
- this.length++;
- }
- };
- $.fn.qrcode = function QRCodePlugin(option) {
- return this.each(function() {
- $(this).append(new QRCode(option));
- });
- };
- module.exports = UI.qrcode = QRCode;
- /***/ },
- /* 27 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * @via https://github.com/uikit/uikit/blob/master/src/js/addons/sticky.js
- * @license https://github.com/uikit/uikit/blob/master/LICENSE.md
- */
- // Sticky Class
- var Sticky = function(element, options) {
- var _this = this;
- this.options = $.extend({}, Sticky.DEFAULTS, options);
- this.$element = $(element);
- this.sticked = null;
- this.inited = null;
- this.$holder = undefined;
- this.$window = $(window).
- on('scroll.sticky.amui',
- UI.utils.debounce($.proxy(this.checkPosition, this), 10)).
- on('resize.sticky.amui orientationchange.sticky.amui',
- UI.utils.debounce(function() {
- _this.reset(true, function() {
- _this.checkPosition();
- });
- }, 50)).
- on('load.sticky.amui', $.proxy(this.checkPosition, this));
- // the `.offset()` is diff between jQuery & Zepto.js
- // jQuery: return `top` and `left`
- // Zepto.js: return `top`, `left`, `width`, `height`
- this.offset = this.$element.offset();
- this.init();
- };
- Sticky.DEFAULTS = {
- top: 0,
- bottom: 0,
- animation: '',
- className: {
- sticky: 'am-sticky',
- resetting: 'am-sticky-resetting',
- stickyBtm: 'am-sticky-bottom',
- animationRev: 'am-animation-reverse'
- }
- };
- Sticky.prototype.init = function() {
- var result = this.check();
- if (!result) {
- return false;
- }
- var $element = this.$element;
- var $elementMargin = '';
- $.each($element.css(
- ['marginTop', 'marginRight', 'marginBottom', 'marginLeft']),
- function(name, value) {
- return $elementMargin += ' ' + value;
- });
- var $holder = $('<div class="am-sticky-placeholder"></div>').css({
- height: $element.css('position') !== 'absolute' ?
- $element.outerHeight() : '',
- float: $element.css('float') != 'none' ? $element.css('float') : '',
- margin: $elementMargin
- });
- this.$holder = $element.css('margin', 0).wrap($holder).parent();
- this.inited = 1;
- return true;
- };
- Sticky.prototype.reset = function(force, cb) {
- var options = this.options;
- var $element = this.$element;
- var animation = (options.animation) ?
- ' am-animation-' + options.animation : '';
- var complete = function() {
- $element.css({position: '', top: '', width: '', left: '', margin: 0});
- $element.removeClass([
- animation,
- options.className.animationRev,
- options.className.sticky,
- options.className.resetting
- ].join(' '));
- this.animating = false;
- this.sticked = false;
- this.offset = $element.offset();
- cb && cb();
- }.bind(this);
- $element.addClass(options.className.resetting);
- if (!force && options.animation && UI.support.animation) {
- this.animating = true;
- $element.removeClass(animation).one(UI.support.animation.end, function() {
- complete();
- }).width(); // force redraw
- $element.addClass(animation + ' ' + options.className.animationRev);
- } else {
- complete();
- }
- };
- Sticky.prototype.check = function() {
- if (!this.$element.is(':visible')) {
- return false;
- }
- var media = this.options.media;
- if (media) {
- switch (typeof(media)) {
- case 'number':
- if (window.innerWidth < media) {
- return false;
- }
- break;
- case 'string':
- if (window.matchMedia && !window.matchMedia(media).matches) {
- return false;
- }
- break;
- }
- }
- return true;
- };
- Sticky.prototype.checkPosition = function() {
- if (!this.inited) {
- var initialized = this.init();
- if (!initialized) {
- return;
- }
- }
- var options = this.options;
- var scrollTop = this.$window.scrollTop();
- var offsetTop = options.top;
- var offsetBottom = options.bottom;
- var $element = this.$element;
- var animation = (options.animation) ?
- ' am-animation-' + options.animation : '';
- var className = [options.className.sticky, animation].join(' ');
- if (typeof offsetBottom == 'function') {
- offsetBottom = offsetBottom(this.$element);
- }
- var checkResult = (scrollTop > this.$holder.offset().top);
- if (!this.sticked && checkResult) {
- $element.addClass(className);
- } else if (this.sticked && !checkResult) {
- this.reset();
- }
- this.$holder.css({
- height: $element.is(':visible') && $element.css('position') !== 'absolute' ?
- $element.outerHeight() : ''
- });
- if (checkResult) {
- $element.css({
- top: offsetTop,
- left: this.$holder.offset().left,
- width: this.$holder.width()
- });
- /*
- if (offsetBottom) {
- // (底部边距 + 元素高度 > 窗口高度) 时定位到底部
- if ((offsetBottom + this.offset.height > $(window).height()) &&
- (scrollTop + $(window).height() >= scrollHeight - offsetBottom)) {
- $element.addClass(options.className.stickyBtm).
- css({top: $(window).height() - offsetBottom - this.offset.height});
- } else {
- $element.removeClass(options.className.stickyBtm).css({top: offsetTop});
- }
- }
- */
- }
- this.sticked = checkResult;
- };
- // Sticky Plugin
- UI.plugin('sticky', Sticky);
- // Init code
- $(window).on('load', function() {
- $('[data-am-sticky]').sticky();
- });
- module.exports = Sticky;
- /***/ },
- /* 28 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var Hammer = __webpack_require__(3);
- var supportTransition = UI.support.transition;
- var animation = UI.support.animation;
- /**
- * @via https://github.com/twbs/bootstrap/blob/master/js/tab.js
- * @copyright 2011-2014 Twitter, Inc.
- * @license MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
- */
- /**
- * Tabs
- * @param {HTMLElement} element
- * @param {Object} options
- * @constructor
- */
- var Tabs = function(element, options) {
- this.$element = $(element);
- this.options = $.extend({}, Tabs.DEFAULTS, options || {});
- this.transitioning = this.activeIndex = null;
- this.refresh();
- this.init();
- };
- Tabs.DEFAULTS = {
- selector: {
- nav: '> .am-tabs-nav',
- content: '> .am-tabs-bd',
- panel: '> .am-tab-panel'
- },
- activeClass: 'am-active'
- };
- Tabs.prototype.refresh = function() {
- var selector = this.options.selector;
- this.$tabNav = this.$element.find(selector.nav);
- this.$navs = this.$tabNav.find('a');
- this.$content = this.$element.find(selector.content);
- this.$tabPanels = this.$content.find(selector.panel);
- var $active = this.$tabNav.find('> .' + this.options.activeClass);
- // Activate the first Tab when no active Tab or multiple active Tabs
- if ($active.length !== 1) {
- this.open(0);
- } else {
- this.activeIndex = this.$navs.index($active.children('a'));
- }
- };
- Tabs.prototype.init = function() {
- var _this = this;
- var options = this.options;
- this.$element.on('click.tabs.amui', options.selector.nav + ' a', function(e) {
- e.preventDefault();
- _this.open($(this));
- });
- // TODO: nested Tabs touch events
- if (!options.noSwipe) {
- if (!this.$content.length) {
- return this;
- }
- var hammer = new Hammer.Manager(this.$content[0]);
- var swipe = new Hammer.Swipe({
- direction: Hammer.DIRECTION_HORIZONTAL
- // threshold: 40
- });
- hammer.add(swipe);
- hammer.on('swipeleft', UI.utils.debounce(function(e) {
- e.preventDefault();
- _this.goTo('next');
- }, 100));
- hammer.on('swiperight', UI.utils.debounce(function(e) {
- e.preventDefault();
- _this.goTo('prev');
- }, 100));
- this._hammer = hammer;
- }
- };
- /**
- * Open $nav tab
- * @param {jQuery|HTMLElement|Number} $nav
- * @returns {Tabs}
- */
- Tabs.prototype.open = function($nav) {
- var activeClass = this.options.activeClass;
- var activeIndex = typeof $nav === 'number' ? $nav : this.$navs.index($($nav));
- $nav = typeof $nav === 'number' ? this.$navs.eq(activeIndex) : $($nav);
- if (!$nav ||
- !$nav.length ||
- this.transitioning ||
- $nav.parent('li').hasClass(activeClass)) {
- return;
- }
- var $tabNav = this.$tabNav;
- var href = $nav.attr('href');
- var regexHash = /^#.+$/;
- var $target = regexHash.test(href) && this.$content.find(href) ||
- this.$tabPanels.eq(activeIndex);
- var previous = $tabNav.find('.' + activeClass + ' a')[0];
- var e = $.Event('open.tabs.amui', {
- relatedTarget: previous
- });
- $nav.trigger(e);
- if (e.isDefaultPrevented()) {
- return;
- }
- // activate Tab nav
- this.activate($nav.closest('li'), $tabNav);
- // activate Tab content
- this.activate($target, this.$content, function() {
- $nav.trigger({
- type: 'opened.tabs.amui',
- relatedTarget: previous
- });
- });
- this.activeIndex = activeIndex;
- };
- Tabs.prototype.activate = function($element, $container, callback) {
- this.transitioning = true;
- var activeClass = this.options.activeClass;
- var $active = $container.find('> .' + activeClass);
- var transition = callback && supportTransition && !!$active.length;
- $active.removeClass(activeClass + ' am-in');
- $element.addClass(activeClass);
- if (transition) {
- $element.redraw(); // reflow for transition
- $element.addClass('am-in');
- } else {
- $element.removeClass('am-fade');
- }
- var complete = $.proxy(function complete() {
- callback && callback();
- this.transitioning = false;
- }, this);
- transition && !this.$content.is('.am-tabs-bd-ofv') ?
- $active.one(supportTransition.end, complete) : complete();
- };
- /**
- * Go to `next` or `prev` tab
- * @param {String} direction - `next` or `prev`
- */
- Tabs.prototype.goTo = function(direction) {
- var navIndex = this.activeIndex;
- var isNext = direction === 'next';
- var spring = isNext ? 'am-animation-right-spring' :
- 'am-animation-left-spring';
- if ((isNext && navIndex + 1 >= this.$navs.length) || // last one
- (!isNext && navIndex === 0)) { // first one
- var $panel = this.$tabPanels.eq(navIndex);
- animation && $panel.addClass(spring).on(animation.end, function() {
- $panel.removeClass(spring);
- });
- } else {
- this.open(isNext ? navIndex + 1 : navIndex - 1);
- }
- };
- Tabs.prototype.destroy = function() {
- this.$element.off('.tabs.amui');
- Hammer.off(this.$content[0], 'swipeleft swiperight');
- this._hammer && this._hammer.destroy();
- $.removeData(this.$element, 'amui.tabs');
- };
- // Plugin
- function Plugin(option) {
- var args = Array.prototype.slice.call(arguments, 1);
- var methodReturn;
- this.each(function() {
- var $this = $(this);
- var $tabs = $this.is('.am-tabs') && $this || $this.closest('.am-tabs');
- var data = $tabs.data('amui.tabs');
- var options = $.extend({}, UI.utils.parseOptions($this.data('amTabs')),
- $.isPlainObject(option) && option);
- if (!data) {
- $tabs.data('amui.tabs', (data = new Tabs($tabs[0], options)));
- }
- if (typeof option === 'string') {
- if (option === 'open' && $this.is('.am-tabs-nav a')) {
- data.open($this);
- } else {
- methodReturn = typeof data[option] === 'function' ?
- data[option].apply(data, args) : data[option];
- }
- }
- });
- return methodReturn === undefined ? this : methodReturn;
- }
- $.fn.tabs = Plugin;
- // Init code
- UI.ready(function(context) {
- $('[data-am-tabs]', context).tabs();
- });
- $(document).on('click.tabs.amui.data-api', '[data-am-tabs] .am-tabs-nav a',
- function(e) {
- e.preventDefault();
- Plugin.call($(this), 'open');
- });
- module.exports = UI.tabs = Tabs;
- // TODO: 1. Ajax 支持
- // 2. touch 事件处理逻辑优化
- /***/ },
- /* 29 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- /**
- * UCheck
- * @via https://github.com/designmodo/Flat-UI/blob/8ef98df23ba7f5033e596a9bd05b53b535a9fe99/js/radiocheck.js
- * @license CC BY 3.0 & MIT
- * @param {HTMLElement} element
- * @param {object} options
- * @constructor
- */
- var UCheck = function(element, options) {
- this.options = $.extend({}, UCheck.DEFAULTS, options);
- // this.options = $.extend({}, UCheck.DEFAULTS, this.$element.data(), options);
- this.$element = $(element);
- this.init();
- };
- UCheck.DEFAULTS = {
- checkboxClass: 'am-ucheck-checkbox',
- radioClass: 'am-ucheck-radio',
- checkboxTpl: '<span class="am-ucheck-icons">' +
- '<i class="am-icon-unchecked"></i><i class="am-icon-checked"></i></span>',
- radioTpl: '<span class="am-ucheck-icons">' +
- '<i class="am-icon-unchecked"></i><i class="am-icon-checked"></i></span>'
- };
- UCheck.prototype.init = function() {
- var $element = this.$element;
- var element = $element[0];
- var options = this.options;
- if (element.type === 'checkbox') {
- $element.addClass(options.checkboxClass)
- .after(options.checkboxTpl);
- } else if (element.type === 'radio') {
- $element.addClass(options.radioClass)
- .after(options.radioTpl);
- }
- };
- UCheck.prototype.check = function() {
- this.$element
- .prop('checked', true)
- .trigger('change.ucheck.amui')
- .trigger('checked.ucheck.amui');
- },
- UCheck.prototype.uncheck = function() {
- this.$element
- .prop('checked', false)
- // trigger `change` event for form validation, etc.
- // @see https://forum.jquery.com/topic/should-chk-prop-checked-true-trigger-change-event
- .trigger('change')
- .trigger('unchecked.ucheck.amui');
- },
- UCheck.prototype.toggle = function() {
- this.$element.
- prop('checked', function(i, value) {
- return !value;
- })
- .trigger('change.ucheck.amui')
- .trigger('toggled.ucheck.amui');
- },
- UCheck.prototype.disable = function() {
- this.$element
- .prop('disabled', true)
- .trigger('change.ucheck.amui')
- .trigger('disabled.ucheck.amui');
- },
- UCheck.prototype.enable = function() {
- this.$element.prop('disabled', false);
- this.$element.trigger('change.ucheck.amui').trigger('enabled.ucheck.amui');
- },
- UCheck.prototype.destroy = function() {
- this.$element
- .removeData('amui.ucheck')
- .removeClass(this.options.checkboxClass + ' ' + this.options.radioClass)
- .next('.am-ucheck-icons')
- .remove()
- .end()
- .trigger('destroyed.ucheck.amui');
- };
- UI.plugin('uCheck', UCheck, {
- after: function() {
- // Adding 'am-nohover' class for touch devices
- if (UI.support.touch) {
- this.parent().hover(function() {
- $(this).addClass('am-nohover');
- }, function() {
- $(this).removeClass('am-nohover');
- });
- }
- }
- });
- UI.ready(function(context) {
- $('[data-am-ucheck]', context).uCheck();
- });
- module.exports = UCheck;
- // TODO: 与表单验证结合使用的情况
- /***/ },
- /* 30 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var Validator = function(element, options) {
- this.options = $.extend({}, Validator.DEFAULTS, options);
- this.options.patterns = $.extend({}, Validator.patterns,
- this.options.patterns);
- var locales = this.options.locales;
- !Validator.validationMessages[locales] && (this.options.locales = 'zh_CN');
- this.$element = $(element);
- this.init();
- };
- Validator.DEFAULTS = {
- debug: false,
- locales: 'zh_CN',
- H5validation: false,
- H5inputType: ['email', 'url', 'number'],
- patterns: {},
- patternClassPrefix: 'js-pattern-',
- activeClass: 'am-active',
- inValidClass: 'am-field-error',
- validClass: 'am-field-valid',
- validateOnSubmit: true,
- alwaysRevalidate: false,
- // Elements to validate with allValid (only validating visible elements)
- // :input: selects all input, textarea, select and button elements.
- // @since 2.5: move `:visible` to `ignore` option (became to `:hidden`)
- allFields: ':input:not(:submit, :button, :disabled, .am-novalidate)',
- // ignored elements
- // @since 2.5
- ignore: ':hidden:not([data-am-selected], .am-validate)',
- // Custom events
- customEvents: 'validate',
- // Keyboard events
- keyboardFields: ':input:not(:submit, :button, :disabled, .am-novalidate)',
- keyboardEvents: 'focusout, change', // keyup, focusin
- // bind `keyup` event to active field
- activeKeyup: false,
- textareaMaxlenthKeyup: true,
- // Mouse events
- pointerFields: 'input[type="range"]:not(:disabled, .am-novalidate), ' +
- 'input[type="radio"]:not(:disabled, .am-novalidate), ' +
- 'input[type="checkbox"]:not(:disabled, .am-novalidate), ' +
- 'select:not(:disabled, .am-novalidate), ' +
- 'option:not(:disabled, .am-novalidate)',
- pointerEvents: 'click',
- onValid: function(validity) {
- },
- onInValid: function(validity) {
- },
- markValid: function(validity) {
- // this is Validator instance
- var options = this.options;
- var $field = $(validity.field);
- var $parent = $field.closest('.am-form-group');
- $field.addClass(options.validClass).removeClass(options.inValidClass);
- $parent.addClass('am-form-success').removeClass('am-form-error');
- options.onValid.call(this, validity);
- },
- markInValid: function(validity) {
- var options = this.options;
- var $field = $(validity.field);
- var $parent = $field.closest('.am-form-group');
- $field.addClass(options.inValidClass + ' ' + options.activeClass).
- removeClass(options.validClass);
- $parent.addClass('am-form-error').removeClass('am-form-success');
- options.onInValid.call(this, validity);
- },
- validate: function(validity) {
- // return validity;
- },
- submit: null
- };
- Validator.VERSION = '2.7.2';
- /* jshint -W101 */
- Validator.patterns = {
- email: /^((([a-zA-Z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-zA-Z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-zA-Z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-zA-Z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-zA-Z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-zA-Z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-zA-Z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-zA-Z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?$/,
- url: /^(https?|ftp):\/\/(((([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:)*@)?(((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5]))|((([a-zA-Z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-zA-Z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-zA-Z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-zA-Z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-zA-Z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-zA-Z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?)(:\d*)?)(\/((([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)+(\/(([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)*)*)?)?(\?((([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|[\uE000-\uF8FF]|\/|\?)*)?(\#((([a-zA-Z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|\/|\?)*)?$/,
- // Number, including positive, negative, and floating decimal
- number: /^-?(?:\d+|\d{1,3}(?:,\d{3})+)?(?:\.\d+)?$/,
- dateISO: /^\d{4}[\/\-]\d{1,2}[\/\-]\d{1,2}$/,
- integer: /^-?\d+$/
- };
- /* jshint +W101 */
- Validator.validationMessages = {
- zh_CN: {
- valueMissing: '请填写(选择)此字段',
- customError: {
- tooShort: '至少填写 %s 个字符',
- checkedOverflow: '至多选择 %s 项',
- checkedUnderflow: '至少选择 %s 项'
- },
- patternMismatch: '请按照要求的格式填写',
- rangeOverflow: '请填写小于等于 %s 的值',
- rangeUnderflow: '请填写大于等于 %s 的值',
- stepMismatch: '',
- tooLong: '至多填写 %s 个字符',
- typeMismatch: '请按照要求的类型填写'
- }
- };
- Validator.ERROR_MAP = {
- tooShort: 'minlength',
- checkedOverflow: 'maxchecked',
- checkedUnderflow: 'minchecked',
- rangeOverflow: 'max',
- rangeUnderflow: 'min',
- tooLong: 'maxlength'
- };
- // TODO: 考虑表单元素不是 form 子元素的情形
- // TODO: change/click/focusout 同时触发时处理重复
- // TODO: 显示提示信息
- Validator.prototype.init = function() {
- var _this = this;
- var $element = this.$element;
- var options = this.options;
- // using H5 form validation if option set and supported
- if (options.H5validation && UI.support.formValidation) {
- return false;
- }
- // disable HTML5 form validation
- $element.attr('novalidate', 'novalidate');
- function regexToPattern(regex) {
- var pattern = regex.toString();
- return pattern.substring(1, pattern.length - 1);
- }
- // add pattern to H5 input type
- $.each(options.H5inputType, function(i, type) {
- var $field = $element.find('input[type=' + type + ']');
- if (!$field.attr('pattern') &&
- !$field.is('[class*=' + options.patternClassPrefix + ']')) {
- $field.attr('pattern', regexToPattern(options.patterns[type]));
- }
- });
- // add pattern to .js-pattern-xx
- $.each(options.patterns, function(key, value) {
- var $field = $element.find('.' + options.patternClassPrefix + key);
- !$field.attr('pattern') && $field.attr('pattern', regexToPattern(value));
- });
- $element.on('submit.validator.amui', function(e) {
- // user custom submit handler
- if (typeof options.submit === 'function') {
- return options.submit.call(_this, e);
- }
- if (options.validateOnSubmit) {
- var formValidity = _this.isFormValid();
- // sync validate, return result
- if ($.type(formValidity) === 'boolean') {
- return formValidity;
- }
- if ($element.data('amui.checked')) {
- return true;
- } else {
- $.when(formValidity).then(function() {
- // done, submit form
- $element.data('amui.checked', true).submit();
- }, function() {
- // fail
- $element.data('amui.checked', false).
- find('.' + options.inValidClass).eq(0).focus();
- });
- return false;
- }
- }
- });
- function bindEvents(fields, eventFlags, debounce) {
- var events = eventFlags.split(',');
- var validate = function(e) {
- // console.log(e.type);
- _this.validate(this);
- };
- if (debounce) {
- validate = UI.utils.debounce(validate, debounce);
- }
- $.each(events, function(i, event) {
- $element.on(event + '.validator.amui', fields, validate);
- });
- }
- bindEvents(':input', options.customEvents);
- bindEvents(options.keyboardFields, options.keyboardEvents);
- bindEvents(options.pointerFields, options.pointerEvents);
- if (options.textareaMaxlenthKeyup) {
- bindEvents('textarea[maxlength]', 'keyup', 50);
- }
- if (options.activeKeyup) {
- bindEvents('.am-active', 'keyup', 50);
- }
- /*if (options.errorMessage === 'tooltip') {
- this.$tooltip = $('<div></div>', {
- 'class': 'am-validator-message',
- id: UI.utils.generateGUID('am-validator-message')
- });
- $(document.body).append(this.$tooltip);
- }*/
- };
- Validator.prototype.isValid = function(field) {
- var $field = $(field);
- var options = this.options;
- // valid field not has been validated
- if ($field.data('validity') === undefined || options.alwaysRevalidate) {
- this.validate(field);
- }
- return $field.data('validity') && $field.data('validity').valid;
- };
- Validator.prototype.validate = function(field) {
- var _this = this;
- var $element = this.$element;
- var options = this.options;
- var $field = $(field);
- // Validate equal, e.g. confirm password
- var equalTo = $field.data('equalTo');
- if (equalTo) {
- $field.attr('pattern', '^' + $element.find(equalTo).val() + '$');
- }
- var pattern = $field.attr('pattern') || false;
- var re = new RegExp(pattern);
- var $radioGroup = null;
- var $checkboxGroup = null;
- // if checkbox, return `:chcked` length
- // NOTE: checkbox and radio should have name attribute
- var value = ($field.is('[type=checkbox]')) ?
- ($checkboxGroup = $element.find('input[name="' + field.name + '"]')).
- filter(':checked').length : ($field.is('[type=radio]') ?
- ($radioGroup = this.$element.find('input[name="' + field.name + '"]')).
- filter(':checked').length > 0 : $field.val());
- // if checkbox, valid the first input of checkbox group
- $field = ($checkboxGroup && $checkboxGroup.length) ?
- $checkboxGroup.first() : $field;
- var required = ($field.attr('required') !== undefined) &&
- ($field.attr('required') !== 'false');
- var maxLength = parseInt($field.attr('maxlength'), 10);
- var minLength = parseInt($field.attr('minlength'), 10);
- var min = Number($field.attr('min'));
- var max = Number($field.attr('max'));
- var validity = this.createValidity({field: $field[0], valid: true});
- // Debug
- if (options.debug && window.console) {
- console.log('Validate: value -> [' + value + ', regex -> [' + re +
- '], required -> ' + required);
- console.log('Regex test: ' + re.test(value) + ', Pattern: ' + pattern);
- }
- // check value length
- if (!isNaN(maxLength) && value.length > maxLength) {
- validity.valid = false;
- validity.tooLong = true;
- }
- if (!isNaN(minLength) && value.length < minLength) {
- validity.valid = false;
- validity.customError = 'tooShort';
- }
- // check minimum and maximum
- // https://developer.mozilla.org/en-US/docs/Web/HTML/Element/Input
- // TODO: 日期验证最小值和最大值 min/max
- if (!isNaN(min) && Number(value) < min) {
- validity.valid = false;
- validity.rangeUnderflow = true;
- }
- if (!isNaN(max) && Number(value) > max) {
- validity.valid = false;
- validity.rangeOverflow = true;
- }
- // check required
- if (required && !value) {
- validity.valid = false;
- validity.valueMissing = true;
- } else if (($checkboxGroup || $field.is('select[multiple="multiple"]')) &&
- value) {
- // check checkboxes / multiple select with `minchecked`/`maxchecked` attr
- // var $multipleField = $checkboxGroup ? $checkboxGroup.first() : $field;
- // if is select[multiple="multiple"], return selected length
- value = $checkboxGroup ? value : value.length;
- // at least checked
- var minChecked = parseInt($field.attr('minchecked'), 10);
- // at most checked
- var maxChecked = parseInt($field.attr('maxchecked'), 10);
- if (!isNaN(minChecked) && value < minChecked) {
- // console.log('At least [%d] items checked!', maxChecked);
- validity.valid = false;
- validity.customError = 'checkedUnderflow';
- }
- if (!isNaN(maxChecked) && value > maxChecked) {
- // console.log('At most [%d] items checked!', maxChecked);
- validity.valid = false;
- validity.customError = 'checkedOverflow';
- }
- } else if (pattern && !re.test(value) && value) { // check pattern
- validity.valid = false;
- validity.patternMismatch = true;
- }
- var validateComplete = function(validity) {
- this.markField(validity);
- var event = $.Event('validated.field.validator.amui');
- event.validity = validity;
- $field.trigger(event).data('validity', validity);
- // validate the radios/checkboxes with the same name
- var $fields = $radioGroup || $checkboxGroup;
- if ($fields) {
- $fields.not($field).data('validity', validity).each(function() {
- validity.field = this;
- _this.markField(validity);
- });
- }
- return validity;
- };
- // Run custom validate
- // NOTE: async custom validate should return Deferred project
- var customValidate;
- (typeof options.validate === 'function') &&
- (customValidate = options.validate.call(this, validity));
- // Deferred
- if (customValidate) {
- var dfd = new $.Deferred();
- $field.data('amui.dfdValidity', dfd.promise());
- return $.when(customValidate).always(function(validity) {
- dfd[validity.valid ? 'resolve' : 'reject'](validity);
- validateComplete.call(_this, validity);
- });
- }
- validateComplete.call(this, validity);
- };
- Validator.prototype.markField = function(validity) {
- var options = this.options;
- var flag = 'mark' + (validity.valid ? '' : 'In') + 'Valid';
- options[flag] && options[flag].call(this, validity);
- };
- // check all fields in the form are valid
- Validator.prototype.validateForm = function() {
- var _this = this;
- var $element = this.$element;
- var options = this.options;
- var $allFields = $element.find(options.allFields).not(options.ignore);
- var radioNames = [];
- var valid = true;
- var formValidity = [];
- var $inValidFields = $([]);
- var promises = [];
- // for async validate
- var async = false;
- $element.trigger('validate.form.validator.amui');
- // Filter radio with the same name and keep only one,
- // since they will be checked as a group by validate()
- var $filteredFields = $allFields.filter(function(index) {
- var name;
- if (this.tagName === 'INPUT' && this.type === 'radio') {
- name = this.name;
- if (radioNames[name] === true) {
- return false;
- }
- radioNames[name] = true;
- }
- return true;
- });
- $filteredFields.each(function() {
- var $this = $(this);
- var fieldValid = _this.isValid(this);
- var fieldValidity = $this.data('validity');
- valid = !!fieldValid && valid;
- formValidity.push(fieldValidity);
- if (!fieldValid) {
- $inValidFields = $inValidFields.add($(this), $element);
- }
- // async validity
- var promise = $this.data('amui.dfdValidity');
- if (promise) {
- promises.push(promise);
- async = true;
- } else {
- // convert sync validity to Promise
- var dfd = new $.Deferred();
- promises.push(dfd.promise());
- dfd[fieldValid ? 'resolve' : 'reject'](fieldValidity);
- }
- });
- // NOTE: If there are async validity, the valid may be not exact result.
- var validity = {
- valid: valid,
- $invalidFields: $inValidFields,
- validity: formValidity,
- promises: promises,
- async: async
- };
- $element.trigger('validated.form.validator.amui', validity);
- return validity;
- };
- Validator.prototype.isFormValid = function() {
- var _this = this;
- var formValidity = this.validateForm();
- var triggerValid = function(type) {
- _this.$element.trigger(type + '.validator.amui');
- };
- if (formValidity.async) {
- var masterDfd = new $.Deferred();
- $.when.apply(null, formValidity.promises).then(function() {
- masterDfd.resolve();
- triggerValid('valid');
- }, function() {
- masterDfd.reject();
- triggerValid('invalid');
- });
- return masterDfd.promise();
- } else {
- if (!formValidity.valid) {
- var $first = formValidity.$invalidFields.first();
- // Selected plugin support
- // @since 2.5
- if ($first.is('[data-am-selected]')) {
- $first = $first.next('.am-selected').find('.am-selected-btn');
- }
- $first.focus();
- triggerValid('invalid');
- return false;
- }
- triggerValid('valid');
- return true;
- }
- };
- // customErrors:
- // 1. tooShort
- // 2. checkedOverflow
- // 3. checkedUnderflow
- Validator.prototype.createValidity = function(validity) {
- return $.extend({
- customError: validity.customError || false,
- patternMismatch: validity.patternMismatch || false,
- rangeOverflow: validity.rangeOverflow || false, // higher than maximum
- rangeUnderflow: validity.rangeUnderflow || false, // lower than minimum
- stepMismatch: validity.stepMismatch || false,
- tooLong: validity.tooLong || false,
- // value is not in the correct syntax
- typeMismatch: validity.typeMismatch || false,
- valid: validity.valid || true,
- // Returns true if the element has no value but is a required field
- valueMissing: validity.valueMissing || false
- }, validity);
- };
- Validator.prototype.getValidationMessage = function(validity) {
- var messages = Validator.validationMessages[this.options.locales];
- var error;
- var message;
- var placeholder = '%s';
- var $field = $(validity.field);
- if ($field.is('[type="checkbox"]') || $field.is('[type="radio"]')) {
- $field = this.$element.find('[name=' + $field.attr('name') + ']').first();
- }
- // get error name
- $.each(validity, function(key, val) {
- // skip `field` and `valid`
- if (key === 'field' || key === 'valid') {
- return key;
- }
- // Amaze UI custom error type
- if (key === 'customError' && val) {
- error = val;
- messages = messages.customError;
- return false;
- }
- // W3C specs error type
- if (val === true) {
- error = key;
- return false;
- }
- });
- message = messages[error] || undefined;
- if (message && Validator.ERROR_MAP[error]) {
- message = message.replace(placeholder,
- $field.attr(Validator.ERROR_MAP[error]) || '规定的');
- }
- return message;
- };
- // remove valid mark
- Validator.prototype.removeMark = function() {
- this.$element
- .find('.am-form-success, .am-form-error, .' + this.options.inValidClass +
- ', .' + this.options.validClass)
- .removeClass([
- 'am-form-success',
- 'am-form-error',
- this.options.inValidClass,
- this.options.validClass
- ].join(' '));
- };
- // @since 2.5
- Validator.prototype.destroy = function() {
- this.removeMark();
- // Remove data
- // - Validator.prototype.init -> $element.data('amui.checked')
- // - Validator.prototype.validateForm
- // - Validator.prototype.isValid
- this.$element.removeData('amui.validator amui.checked')
- .off('.validator.amui')
- .find(this.options.allFields).removeData('validity amui.dfdValidity');
- };
- UI.plugin('validator', Validator);
- // init code
- UI.ready(function(context) {
- $('[data-am-validator]', context).validator();
- });
- module.exports = Validator;
- /***/ },
- /* 31 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- var cookie = {
- get: function(name) {
- var cookieName = encodeURIComponent(name) + '=';
- var cookieStart = document.cookie.indexOf(cookieName);
- var cookieValue = null;
- var cookieEnd;
- if (cookieStart > -1) {
- cookieEnd = document.cookie.indexOf(';', cookieStart);
- if (cookieEnd == -1) {
- cookieEnd = document.cookie.length;
- }
- cookieValue = decodeURIComponent(document.cookie.substring(cookieStart +
- cookieName.length, cookieEnd));
- }
- return cookieValue;
- },
- set: function(name, value, expires, path, domain, secure) {
- var cookieText = encodeURIComponent(name) + '=' +
- encodeURIComponent(value);
- if (expires instanceof Date) {
- cookieText += '; expires=' + expires.toUTCString();
- }
- if (path) {
- cookieText += '; path=' + path;
- }
- if (domain) {
- cookieText += '; domain=' + domain;
- }
- if (secure) {
- cookieText += '; secure';
- }
- document.cookie = cookieText;
- },
- unset: function(name, path, domain, secure) {
- this.set(name, '', new Date(0), path, domain, secure);
- }
- };
- UI.utils = UI.utils || {};
- module.exports = UI.utils.cookie = cookie;
- /***/ },
- /* 32 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- /**
- * @see https://github.com/sindresorhus/screenfull.js
- * @license MIT © Sindre Sorhus
- */
- var UI = __webpack_require__(2);
- var screenfull = (function() {
- var keyboardAllowed = typeof Element !== 'undefined' &&
- 'ALLOW_KEYBOARD_INPUT' in Element;
- var fn = (function() {
- var val;
- var valLength;
- var fnMap = [
- [
- 'requestFullscreen',
- 'exitFullscreen',
- 'fullscreenElement',
- 'fullscreenEnabled',
- 'fullscreenchange',
- 'fullscreenerror'
- ],
- // new WebKit
- [
- 'webkitRequestFullscreen',
- 'webkitExitFullscreen',
- 'webkitFullscreenElement',
- 'webkitFullscreenEnabled',
- 'webkitfullscreenchange',
- 'webkitfullscreenerror'
- ],
- // old WebKit (Safari 5.1)
- [
- 'webkitRequestFullScreen',
- 'webkitCancelFullScreen',
- 'webkitCurrentFullScreenElement',
- 'webkitCancelFullScreen',
- 'webkitfullscreenchange',
- 'webkitfullscreenerror'
- ],
- [
- 'mozRequestFullScreen',
- 'mozCancelFullScreen',
- 'mozFullScreenElement',
- 'mozFullScreenEnabled',
- 'mozfullscreenchange',
- 'mozfullscreenerror'
- ],
- [
- 'msRequestFullscreen',
- 'msExitFullscreen',
- 'msFullscreenElement',
- 'msFullscreenEnabled',
- 'MSFullscreenChange',
- 'MSFullscreenError'
- ]
- ];
- var i = 0;
- var l = fnMap.length;
- var ret = {};
- for (; i < l; i++) {
- val = fnMap[i];
- if (val && val[1] in document) {
- for (i = 0, valLength = val.length; i < valLength; i++) {
- ret[fnMap[0][i]] = val[i];
- }
- return ret;
- }
- }
- return false;
- })();
- var screenfull = {
- request: function(elem) {
- var request = fn.requestFullscreen;
- elem = elem || document.documentElement;
- // Work around Safari 5.1 bug: reports support for
- // keyboard in fullscreen even though it doesn't.
- // Browser sniffing, since the alternative with
- // setTimeout is even worse.
- if (/5\.1[\.\d]* Safari/.test(navigator.userAgent)) {
- elem[request]();
- } else {
- elem[request](keyboardAllowed && Element.ALLOW_KEYBOARD_INPUT);
- }
- },
- exit: function() {
- document[fn.exitFullscreen]();
- },
- toggle: function(elem) {
- if (this.isFullscreen) {
- this.exit();
- } else {
- this.request(elem);
- }
- },
- raw: fn
- };
- if (!fn) {
- return false;
- }
- Object.defineProperties(screenfull, {
- isFullscreen: {
- get: function() {
- return !!document[fn.fullscreenElement];
- }
- },
- element: {
- enumerable: true,
- get: function() {
- return document[fn.fullscreenElement];
- }
- },
- enabled: {
- enumerable: true,
- get: function() {
- // Coerce to boolean in case of old WebKit
- return !!document[fn.fullscreenEnabled];
- }
- }
- });
- screenfull.VERSION = '3.0.0';
- return screenfull;
- })();
- module.exports = UI.fullscreen = screenfull;
- /***/ },
- /* 33 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- UI.support.geolocation = window.navigator && window.navigator.geolocation;
- var geo = UI.support.geolocation;
- var Geolocation = function(options) {
- this.options = options || {};
- };
- Geolocation.MESSAGES = {
- unsupportedBrowser: 'Browser does not support location services',
- permissionDenied: 'You have rejected access to your location',
- positionUnavailable: 'Unable to determine your location',
- timeout: 'Service timeout has been reached'
- };
- Geolocation.ERROR_CODE = {
- 0: 'unsupportedBrowser',
- 1: 'permissionDenied',
- 2: 'positionUnavailable',
- 3: 'timeout'
- };
- Geolocation.prototype.get = function(options) {
- var _this = this;
- options = $.extend({}, this.options, options);
- var deferred = new $.Deferred();
- if (geo) {
- this.watchID = geo.getCurrentPosition(function(position) {
- deferred.resolve.call(_this, position);
- }, function(error) {
- deferred.reject(Geolocation.MESSAGES[Geolocation.ERROR_CODE[error.code]]);
- }, options);
- } else {
- deferred.reject(Geolocation.MESSAGES.unsupportedBrowser);
- }
- return deferred.promise();
- };
- Geolocation.prototype.watch = function(options) {
- if (!geo) {
- return;
- }
- options = $.extend({}, this.options, options);
- if (!$.isFunction(options.done)) {
- return;
- }
- this.clearWatch();
- var fail = $.isFunction(options.fail) ? options.fail : null;
- this.watchID = geo.watchPosition(options.done, fail, options);
- return this.watchID;
- };
- Geolocation.prototype.clearWatch = function() {
- if (!geo || !this.watchID) {
- return;
- }
- geo.clearWatch(this.watchID);
- this.watchID = null;
- };
- module.exports = UI.Geolocation = Geolocation;
- /***/ },
- /* 34 */
- /***/ function(module, exports, __webpack_require__) {
- /* WEBPACK VAR INJECTION */(function(global) {'use strict';
- var UI = __webpack_require__(2);
- /**
- * store.js
- * @see https://github.com/marcuswestin/store.js
- * @license https://github.com/marcuswestin/store.js/blob/master/LICENSE
- */
- var store = {};
- var win = (typeof window != 'undefined' ? window : global);
- var localStorageName = 'localStorage';
- var storage;
- store.disabled = false;
- store.version = '1.3.20';
- store.set = function(key, value) {
- };
- store.get = function(key, defaultVal) {
- };
- store.has = function(key) {
- return store.get(key) !== undefined;
- };
- store.remove = function(key) {
- };
- store.clear = function() {
- };
- store.transact = function(key, defaultVal, transactionFn) {
- if (transactionFn == null) {
- transactionFn = defaultVal;
- defaultVal = null;
- }
- if (defaultVal == null) {
- defaultVal = {};
- }
- var val = store.get(key, defaultVal);
- transactionFn(val);
- store.set(key, val);
- };
- store.getAll = function() {
- };
- store.forEach = function() {
- };
- store.serialize = function(value) {
- return JSON.stringify(value);
- };
- store.deserialize = function(value) {
- if (typeof value != 'string') {
- return undefined;
- }
- try {
- return JSON.parse(value);
- } catch (e) {
- return value || undefined;
- }
- };
- // Functions to encapsulate questionable FireFox 3.6.13 behavior
- // when about.config::dom.storage.enabled === false
- // See https://github.com/marcuswestin/store.js/issues#issue/13
- function isLocalStorageNameSupported() {
- try {
- return (localStorageName in win && win[localStorageName]);
- } catch (err) {
- return false;
- }
- }
- if (isLocalStorageNameSupported()) {
- storage = win[localStorageName];
- store.set = function(key, val) {
- if (val === undefined) {
- return store.remove(key);
- }
- storage.setItem(key, store.serialize(val));
- return val;
- };
- store.get = function(key, defaultVal) {
- var val = store.deserialize(storage.getItem(key));
- return (val === undefined ? defaultVal : val);
- };
- store.remove = function(key) {
- storage.removeItem(key);
- };
- store.clear = function() {
- storage.clear();
- };
- store.getAll = function() {
- var ret = {};
- store.forEach(function(key, val) {
- ret[key] = val;
- });
- return ret;
- };
- store.forEach = function(callback) {
- for (var i = 0; i < storage.length; i++) {
- var key = storage.key(i);
- callback(key, store.get(key));
- }
- };
- }
- try {
- var testKey = '__storejs__';
- store.set(testKey, testKey);
- if (store.get(testKey) != testKey) {
- store.disabled = true;
- }
- store.remove(testKey);
- } catch (e) {
- store.disabled = true;
- }
- store.enabled = !store.disabled;
- module.exports = UI.store = store;
- /* WEBPACK VAR INJECTION */}.call(exports, (function() { return this; }())))
- /***/ },
- /* 35 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(7);
- function accordionInit() {
- var $accordion = $('[data-am-widget="accordion"]');
- var selector = {
- item: '.am-accordion-item',
- title: '.am-accordion-title',
- body: '.am-accordion-bd',
- disabled: '.am-disabled'
- };
- $accordion.each(function(i, item) {
- var options = UI.utils.parseOptions($(item).attr('data-am-accordion'));
- var $title = $(item).find(selector.title);
- $title.on('click.accordion.amui', function() {
- var $collapse = $(this).next(selector.body);
- var $parent = $(this).parent(selector.item);
- var data = $collapse.data('amui.collapse');
- if ($parent.is(selector.disabled)) {
- return;
- }
- $parent.toggleClass('am-active');
- if (!data) {
- $collapse.collapse();
- } else {
- $collapse.collapse('toggle');
- }
- !options.multiple &&
- $(item).children('.am-active').
- not($parent).not(selector.disabled).removeClass('am-active').
- find(selector.body + '.am-in').collapse('close');
- });
- });
- }
- // Init on DOM ready
- $(accordionInit);
- module.exports = UI.accordion = {
- VERSION: '2.1.0',
- init: accordionInit
- };
- /***/ },
- /* 36 */
- /***/ function(module, exports) {
- 'use strict';
- module.exports = {
- VERSION: '2.0.1'
- };
- /***/ },
- /* 37 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- function duoshuoInit() {
- var $dsThread = $('.ds-thread');
- var dsShortName = $dsThread.parent('[data-am-widget="duoshuo"]').
- attr('data-ds-short-name');
- var dsSrc = (document.location.protocol == 'https:' ? 'https:' : 'http:') +
- '//static.duoshuo.com/embed.js';
- if (!$dsThread.length || !dsShortName) {
- return;
- }
- window.duoshuoQuery = {
- short_name: dsShortName
- };
- // 已经有多说脚本
- if ($('script[src="' + dsSrc + '"]').length) {
- return;
- }
- var $dsJS = $('<script>', {
- async: true,
- type: 'text/javascript',
- src: dsSrc,
- charset: 'utf-8'
- });
- $('body').append($dsJS);
- }
- $(window).on('load', duoshuoInit);
- module.exports = UI.duoshuo = {
- VERSION: '2.0.1',
- init: duoshuoInit
- };
- /***/ },
- /* 38 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(20);
- /**
- * Is Images zoomable
- * @return {Boolean}
- */
- $.isImgZoomAble = function(element) {
- var t = new Image();
- t.src = element.src;
- var zoomAble = ($(element).width() < t.width);
- if (zoomAble) {
- $(element).closest('.am-figure').addClass('am-figure-zoomable');
- }
- return zoomAble;
- };
- function figureInit() {
- $('.am-figure').each(function(i, item) {
- var options = UI.utils.parseOptions($(item).attr('data-am-figure'));
- var $item = $(item);
- var data;
- if (options.pureview) {
- if (options.pureview === 'auto') {
- var zoomAble = $.isImgZoomAble($item.find('img')[0]);
- zoomAble && $item.pureview();
- } else {
- $item.addClass('am-figure-zoomable').pureview();
- }
- }
- data = $item.data('amui.pureview');
- if (data) {
- $item.on('click', ':not(img)', function() {
- data.open(0);
- });
- }
- });
- }
- $(window).on('load', figureInit);
- module.exports = UI.figure = {
- VERSION: '2.0.3',
- init: figureInit
- };
- /***/ },
- /* 39 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(15);
- var addToHS = __webpack_require__(4);
- var cookie = __webpack_require__(31);
- function footerInit() {
- // modal mode
- $('.am-footer-ysp').on('click', function() {
- $('#am-footer-modal').modal();
- });
- var options = UI.utils.parseOptions($('.am-footer').data('amFooter'));
- options.addToHS && addToHS();
- // switch mode
- // switch to desktop
- $('[data-rel="desktop"]').on('click', function(e) {
- e.preventDefault();
- if (window.AMPlatform) { // front end
- window.AMPlatform.util.goDesktop();
- } else { // back end
- cookie.set('allmobilize', 'desktop', '', '/');
- window.location = window.location;
- }
- });
- }
- $(footerInit);
- module.exports = UI.footer = {
- VERSION: '3.1.2',
- init: footerInit
- };
- /***/ },
- /* 40 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(20);
- function galleryInit() {
- var $gallery = $('[data-am-widget="gallery"]');
- $gallery.each(function() {
- var options = UI.utils.parseOptions($(this).attr('data-am-gallery'));
- if (options.pureview) {
- (typeof options.pureview === 'object') ?
- $(this).pureview(options.pureview) : $(this).pureview();
- }
- });
- }
- $(galleryInit);
- module.exports = UI.gallery = {
- VERSION: '3.0.0',
- init: galleryInit
- };
- /***/ },
- /* 41 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(23);
- function goTopInit() {
- var $goTop = $('[data-am-widget="gotop"]');
- var $fixed = $goTop.filter('.am-gotop-fixed');
- var $win = $(window);
- if ($goTop.data('init')) {
- return;
- }
- $goTop.find('a').on('click', function(e) {
- e.preventDefault();
- $win.smoothScroll();
- });
- function checkPosition() {
- $fixed[($win.scrollTop() > 50 ? 'add' : 'remove') + 'Class']('am-active');
- }
- checkPosition();
- $win.on('scroll.gotop.amui', UI.utils.debounce(checkPosition, 100));
- $goTop.data('init', true);
- }
- $(goTopInit);
- module.exports = UI.gotop = {
- VERSION: '4.0.2',
- init: goTopInit
- };
- /***/ },
- /* 42 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- function headerInit() {
- $('[data-am-widget="header"]').each(function() {
- if ($(this).hasClass('am-header-fixed')) {
- $('body').addClass('am-with-fixed-header');
- return false;
- }
- });
- }
- $(headerInit);
- module.exports = UI.header = {
- VERSION: '2.0.0',
- init: headerInit
- };
- /***/ },
- /* 43 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- module.exports = UI.intro = {
- VERSION: '4.0.2',
- init: function() {}
- };
- /***/ },
- /* 44 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- module.exports = UI.listNews = {
- VERSION: '4.0.0',
- init: function() {}
- };
- /***/ },
- /* 45 */
- /***/ function(module, exports, __webpack_require__) {
- /* jshint strict: false, maxlen: 200 */
- /* global BMap */
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- function addMapApi(callback) {
- var $mapApi0 = $('<script />', {
- id: 'am-map-api-0'
- });
- $('body').append($mapApi0);
- $mapApi0.on('load', function() {
- console.log('load');
- var $mapApi1 = $('<script/>', {
- id: 'am-map-api-1'
- });
- $('body').append($mapApi1);
- $mapApi1.on('load', function() {
- var script = document.createElement('script');
- script.textContent = '(' + callback.toString() + ')();';
- $('body')[0].appendChild(script);
- }).attr('src', 'http://api.map.baidu.com/getscript' +
- '?type=quick&file=feature' +
- '&ak=WVAXZ05oyNRXS5egLImmentg&t=20140109092002');
- }).attr('src', 'http://api.map.baidu.com/getscript' +
- '?type=quick&file=api&ak=WVAXZ05oyNRXS5egLImmentg&t=20140109092002');
- // jQuery 中 `load` 事件触发有问题,动态设置 src 属性才会触发 `load` 事件
- // $mapApi0 = $('<script />', {src: 'xxx'}); 这样的写法在 Zepto.js 中则没有问题
- }
- function addBdMap() {
- // 如果使用 $ 选择符,minify 以后会报错: $ is undefined
- // 即使传入 $ 也无效,改为使用原生方法
- // 这个函数作为 callback 会插入到 body 以后才执行,应该是 $ 引用错误导致
- var content = document.querySelector('.am-map');
- var defaultLng = 116.331398; // 经度默认值
- var defaultLat = 39.897445; // 纬度默认值
- var name = content.getAttribute('data-name');
- var address = content.getAttribute('data-address');
- var lng = content.getAttribute('data-longitude') || defaultLng;
- var lat = content.getAttribute('data-latitude') || defaultLat;
- var setZoom = content.getAttribute('data-setZoom') || 17;
- var icon = content.getAttribute('data-icon');
- var map = new BMap.Map('bd-map');
- // 实例化一个地理坐标点
- var point = new BMap.Point(lng, lat);
- // 设初始化地图, options: 3-18
- map.centerAndZoom(point, setZoom);
- // 添加地图缩放控件
- if (content.getAttribute('data-zoomControl')) {
- map.addControl(new BMap.ZoomControl());
- }
- // 添加比例尺控件
- if (content.getAttribute('data-scaleControl')) {
- map.addControl(new BMap.ScaleControl());
- }
- // 创建标准与自定义 icon
- var marker = new BMap.Marker(point);
- if (icon) {
- marker.setIcon(new BMap.Icon(icon, new BMap.Size(40, 40)));
- }
- var opts = {
- width: 200, // 信息窗口宽度
- // height: 'auto', // 信息窗口高度
- title: name // 信息窗口标题
- };
- // 创建信息窗口对象
- var infoWindow = new BMap.InfoWindow('地址:' + address, opts);
- // 创建地址解析器实例
- var myGeo = new BMap.Geocoder();
- // 判断有没有使用经纬度
- if (lng == defaultLng && lat == defaultLat) {
- // 使用地址反解析来设置地图
- // 将地址解析结果显示在地图上,并调整地图视野
- myGeo.getPoint(address, function(point) {
- if (point) {
- map.centerAndZoom(point, setZoom);
- marker.setPosition(point);
- map.addOverlay(marker);
- map.openInfoWindow(infoWindow, point); // 开启信息窗口
- }
- }, '');
- } else {
- // 使用经纬度来设置地图
- myGeo.getLocation(point, function(result) {
- map.centerAndZoom(point, setZoom);
- marker.setPosition(point);
- map.addOverlay(marker);
- if (address) {
- map.openInfoWindow(infoWindow, point); // 开启信息窗口
- } else {
- map.openInfoWindow(new BMap.InfoWindow(address, opts), point); // 开启信息窗口
- }
- });
- }
- }
- var mapInit = function() {
- $('.am-map').length && addMapApi(addBdMap);
- };
- $(mapInit);
- module.exports = UI.map = {
- VERSION: '2.0.2',
- init: mapInit
- };
- /***/ },
- /* 46 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- function mechatInit() {
- if (!$('#mechat').length) {
- return;
- }
- var $mechat = $('[data-am-widget="mechat"]');
- var unitid = $mechat.data('am-mechat-unitid');
- var $mechatData = $('<script>', {
- charset: 'utf-8',
- src: 'http://mechatim.com/js/unit/button.js?id=' + unitid
- });
- $('body').append($mechatData);
- }
- // Lazy load
- $(window).on('load', mechatInit);
- module.exports = UI.mechat = {
- VERSION: '2.0.1',
- init: mechatInit
- };
- /***/ },
- /* 47 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var IScroll = __webpack_require__(14);
- __webpack_require__(16);
- __webpack_require__(7);
- var menuInit = function() {
- var $menus = $('[data-am-widget="menu"]');
- $menus.find('.am-menu-nav .am-parent > a').on('click', function(e) {
- e.preventDefault();
- var $clicked = $(this);
- var $parent = $clicked.parent();
- var $subMenu = $clicked.next('.am-menu-sub');
- $parent.toggleClass('am-open');
- $subMenu.collapse('toggle');
- $parent.siblings('.am-parent').removeClass('am-open')
- .children('.am-menu-sub.am-in').collapse('close');
- });
- // Dropdown/slideDown menu
- $menus.
- filter('[data-am-menu-collapse]').
- find('> .am-menu-toggle').
- on('click', function(e) {
- e.preventDefault();
- var $this = $(this);
- var $nav = $this.next('.am-menu-nav');
- $this.toggleClass('am-active');
- $nav.collapse('toggle');
- });
- // OffCanvas menu
- $menus.
- filter('[data-am-menu-offcanvas]').
- find('> .am-menu-toggle').
- on('click', function(e) {
- e.preventDefault();
- var $this = $(this);
- var $nav = $this.next('.am-offcanvas');
- $this.toggleClass('am-active');
- $nav.offCanvas('open');
- });
- // Close offCanvas when link clicked
- var autoCloseOffCanvas = '.am-offcanvas[data-dismiss-on="click"]';
- var $autoCloseOffCanvas = $(autoCloseOffCanvas);
- $autoCloseOffCanvas.find('a').not('.am-parent>a').on('click', function(e) {
- $(this).parents(autoCloseOffCanvas).offCanvas('close');
- });
- // one theme
- $menus.filter('.am-menu-one').each(function(index) {
- var $this = $(this);
- var $wrap = $('<div class="am-menu-nav-sub-wrap"></div>');
- var allWidth = 0;
- var $nav = $this.find('.am-menu-nav');
- var $navTopItem = $nav.children('li');
- var prevIndex;
- $navTopItem.filter('.am-parent').each(function(index) {
- $(this).attr('data-rel', '#am-menu-sub-' + index);
- $(this).
- find('.am-menu-sub').
- attr('id', 'am-menu-sub-' + index).
- appendTo($wrap);
- });
- $this.append($wrap);
- $nav.wrap('<div class="am-menu-nav-wrap" id="am-menu-' + index + '">');
- // $navTopItem.eq(0).addClass('am-active');
- // 计算出所有 li 宽度
- $navTopItem.each(function(i) {
- allWidth += parseFloat($(this).css('width'));
- });
- $nav.width(allWidth);
- var menuScroll = new IScroll('#am-menu-' + index, {
- eventPassthrough: true,
- scrollX: true,
- scrollY: false,
- preventDefault: false
- });
- $navTopItem.on('click', function() {
- var $clicked = $(this);
- $clicked.addClass('am-active').siblings().removeClass('am-active');
- $wrap.find('.am-menu-sub.am-in').collapse('close');
- if ($clicked.is('.am-parent')) {
- !$clicked.hasClass('.am-open') &&
- $wrap.find($clicked.attr('data-rel')).collapse('open');
- } else {
- $clicked.siblings().removeClass('am-open');
- }
- // 第一次调用,没有prevIndex
- if (prevIndex === undefined) {
- prevIndex = $(this).index() ? 0 : 1;
- }
- // 判断方向
- var dir = $(this).index() > prevIndex;
- var target = $(this)[dir ? 'next' : 'prev']();
- // 点击的按钮,显示一半
- var offset = target.offset() || $(this).offset();
- var within = $this.offset();
- // 父类左边距
- var listOffset;
- var parentLeft = parseInt($this.css('padding-left'));
- if (dir ? offset.left + offset.width > within.left + within.width :
- offset.left < within.left) {
- listOffset = $nav.offset();
- menuScroll.scrollTo(dir ?
- within.width - offset.left + listOffset.left -
- offset.width - parentLeft :
- listOffset.left - offset.left, 0, 400);
- }
- prevIndex = $(this).index();
- });
- $this.on('touchmove', function(event) {
- event.preventDefault();
- });
- });
- };
- $(menuInit);
- module.exports = UI.menu = {
- VERSION: '4.0.3',
- init: menuInit
- };
- /***/ },
- /* 48 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var share = __webpack_require__(25);
- var QRCode = __webpack_require__(26);
- __webpack_require__(15);
- function navbarInit() {
- var $navBar = $('[data-am-widget="navbar"]');
- if (!$navBar.length) {
- return;
- }
- var $win = $(window);
- var $body = $('body');
- var $navBarNav = $navBar.find('.am-navbar-nav');
- var $navItems = $navBar.find('li');
- var navItemsCounter = $navItems.length;
- var configItems = $navBarNav.attr('class') &&
- parseInt($navBarNav.attr('class').
- match(/am-avg-sm-(\d+)/)[1]) || 3;
- var navMinWidth = 60; // 每个 li 最小宽度
- var offsetWidth = 16;
- var $share = $navItems.filter('[data-am-navbar-share]');
- var $qrcode = $navItems.filter('[data-am-navbar-qrcode]');
- var activeStatus = 'am-active';
- var $moreActions = $('<ul class="am-navbar-actions"></ul>', {
- id: UI.utils.generateGUID('am-navbar-actions')
- });
- var $moreLink = $('<li class="am-navbar-labels am-navbar-more">' +
- '<a href="javascript: void(0);">' +
- '<span class="am-icon-angle-up"></span>' +
- '<span class="am-navbar-label">更多</span></a></li>');
- // 如果有 Fix 的工具栏则设置 body 的 padding-bottom
- if ($navBar.css('position') == 'fixed') {
- $body.addClass('am-with-fixed-navbar');
- }
- if ($qrcode.length) {
- var qrId = 'am-navbar-qrcode';
- $qrModal = $('#' + qrId);
- if (!$qrModal.length) {
- var qrImg = $qrcode.attr('data-am-navbar-qrcode');
- var $qrModal = $('<div class="am-modal am-modal-no-btn" id="">' +
- '<div class="am-modal-dialog">' +
- '<div class="am-modal-bd"></div></div>' +
- '</div>', {
- id: qrId
- });
- var $qrContainer = $qrModal.find('.am-modal-bd');
- // 判断上传自定义的二维码没有,否则生成二维码
- if (qrImg) {
- $qrContainer.html('<img src="' + qrImg + '"/>');
- } else {
- var qrnode = new QRCode({
- render: 'canvas',
- correctLevel: 0,
- text: window.location.href,
- width: 200,
- height: 200,
- background: '#fff',
- foreground: '#000'
- });
- $qrContainer.html(qrnode);
- }
- $body.append($qrModal);
- }
- $qrcode.on('click', function(e) {
- e.preventDefault();
- $qrModal.modal();
- });
- }
- if (navItemsCounter > configItems && navItemsCounter > calcSuiteItems()) {
- initActions();
- }
- // console.log('NavItems: %d, config: %d, best: %d',
- // navItemsCounter, configItems, calcSuiteItems());
- function initActions() {
- $navBarNav.append($moreLink);
- $navBarNav.
- find('li').
- not('.am-navbar-more').
- slice(calcSuiteItems() - 1).
- appendTo($moreActions);
- // Append more actions
- $navBar.append($moreActions);
- }
- function checkNavBarItems() {
- if (calcSuiteItems() >= navItemsCounter) {
- // 显示所有链接,隐藏 more
- $moreLink.hide();
- $moreActions.find('li').insertBefore($moreLink);
- return;
- }
- !$navBar.find('.am-navbar-actions').length && initActions();
- $moreLink.show();
- if ($navBarNav.find('li').length < calcSuiteItems()) {
- $moreActions.find('li').
- slice(0, calcSuiteItems() - $navBarNav.find('li').length).
- insertBefore($moreLink);
- } else if ($navBarNav.find('li').length > calcSuiteItems()) {
- if ($moreActions.find('li').length) {
- $navBarNav.find('li').not($moreLink).slice(calcSuiteItems() - 1).
- insertBefore($moreActions.find('li').first());
- } else {
- $navBarNav.find('li').not($moreLink).slice(calcSuiteItems() - 1).
- appendTo($moreActions);
- }
- }
- }
- /**
- * 计算最适合显示的条目个数
- * @returns {number}
- */
- function calcSuiteItems() {
- return Math.floor(($win.width() - offsetWidth) / navMinWidth);
- }
- $navBar.on('click.navbar.amui', '.am-navbar-more', function(e) {
- e.preventDefault();
- $moreLink[$moreActions.hasClass(activeStatus) ?
- 'removeClass' : 'addClass'](activeStatus);
- $moreActions.toggleClass(activeStatus);
- });
- if ($share.length) {
- $share.on('click.navbar.amui', function(e) {
- e.preventDefault();
- share.toggle();
- });
- }
- $win.on('resize.navbar.amui orientationchange.navbar.amui',
- UI.utils.debounce(checkNavBarItems, 150));
- }
- // DOMContent ready
- $(navbarInit);
- module.exports = UI.navbar = {
- VERSION: '2.0.2',
- init: navbarInit
- };
- /***/ },
- /* 49 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- module.exports = UI.pagination = {
- VERSION: '3.0.1'
- };
- /***/ },
- /* 50 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var IScroll = __webpack_require__(14);
- __webpack_require__(20);
- /**
- * 表格滚动
- * @param {number} index ID 标识,多个 paragraph 里面多个 table
- */
- $.fn.scrollTable = function(index) {
- var $this = $(this);
- var $parent;
- $this.wrap('<div class="am-paragraph-table-container" ' +
- 'id="am-paragraph-table-' + index + '">' +
- '<div class="am-paragraph-table-scroller"></div></div>');
- $parent = $this.parent();
- $parent.width($this.width());
- $parent.height($this.height());
- new IScroll('#am-paragraph-table-' + index, {
- eventPassthrough: true,
- scrollX: true,
- scrollY: false,
- preventDefault: false
- });
- };
- function paragraphInit() {
- var $paragraph = $('[data-am-widget="paragraph"]');
- $paragraph.each(function(index) {
- var $this = $(this);
- var options = UI.utils.parseOptions($this.attr('data-am-paragraph'));
- var $index = index;
- if (options.pureview) {
- $this.pureview();
- }
- if (options.tableScrollable) {
- $this.find('table').each(function(index) {
- if ($(this).width() > $(window).width()) {
- $(this).scrollTable($index + '-' + index);
- }
- });
- }
- });
- }
- $(window).on('load', paragraphInit);
- module.exports = UI.paragraph = {
- VERSION: '2.0.1',
- init: paragraphInit
- };
- /***/ },
- /* 51 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(11);
- function sliderInit() {
- var $sliders = $('[data-am-widget="slider"]');
- $sliders.not('.am-slider-manual').each(function(i, item) {
- var options = UI.utils.parseOptions($(item).attr('data-am-slider'));
- $(item).flexslider(options);
- });
- }
- $(sliderInit);
- module.exports = UI.slider = {
- VERSION: '3.0.1',
- init: sliderInit
- };
- /***/ },
- /* 52 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- __webpack_require__(28);
- function tabsInit() {
- $('[data-am-widget="tabs"]').each(function() {
- var options = $(this).data('amTabsNoswipe') ? {noSwipe: 1} : {};
- $(this).tabs(options);
- });
- }
- $(tabsInit);
- module.exports = UI.tab = {
- VERSION: '4.0.1',
- init: tabsInit
- };
- /***/ },
- /* 53 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var UI = __webpack_require__(2);
- module.exports = UI.titlebar = {
- VERSION: '4.0.1'
- };
- /***/ },
- /* 54 */
- /***/ function(module, exports, __webpack_require__) {
- 'use strict';
- var $ = __webpack_require__(1);
- var UI = __webpack_require__(2);
- var isWeChat = window.navigator.userAgent.indexOf('MicroMessenger') > -1;
- /* global wx,alert */
- function appendWeChatSDK(callback) {
- var $weChatSDK = $('<script/>', {
- id: 'wechat-sdk'
- });
- $('body').append($weChatSDK);
- $weChatSDK.on('load', function() {
- callback && callback();
- }).attr('src', 'http://res.wx.qq.com/open/js/jweixin-1.0.0.js');
- }
- function payHandler() {
- var $paymentBtn = $('[data-am-widget="wechatpay"]');
- if (!isWeChat) {
- $paymentBtn.hide();
- return false;
- }
- $paymentBtn.on('click', '.am-wechatpay-btn', function(e) {
- e.preventDefault();
- var options = UI.utils.parseOptions($(this).parent().data('wechatPay'));
- // console.log(options);
- // alert('pay button clicked');
- if (!window.wx) {
- alert('没有微信 JS SDK');
- return;
- }
- wx.checkJsApi({
- jsApiList: ['chooseWXPay'],
- success: function(res) {
- if (res.checkResult.chooseWXPay) {
- wx.chooseWXPay(options);
- } else {
- alert('微信版本不支持支付接口或没有开启!');
- }
- },
- fail: function() {
- alert('调用 checkJsApi 接口时发生错误!');
- }
- });
- });
- }
- var payInit = payHandler;
- // Init on DOM ready
- $(payInit);
- module.exports = UI.pay = {
- VERSION: '1.0.0',
- init: payInit
- };
- /***/ }
- /******/ ])
- });
- ;
|