1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690769176927693769476957696769776987699770077017702770377047705770677077708770977107711771277137714771577167717771877197720772177227723772477257726772777287729773077317732773377347735773677377738773977407741774277437744774577467747774877497750775177527753775477557756775777587759776077617762776377647765776677677768776977707771777277737774777577767777777877797780778177827783778477857786778777887789779077917792779377947795779677977798779978007801780278037804780578067807780878097810781178127813781478157816781778187819782078217822782378247825782678277828782978307831783278337834783578367837783878397840784178427843784478457846784778487849785078517852785378547855785678577858785978607861786278637864786578667867786878697870787178727873787478757876787778787879788078817882788378847885788678877888788978907891789278937894789578967897789878997900790179027903790479057906790779087909791079117912791379147915791679177918791979207921792279237924792579267927792879297930793179327933793479357936793779387939794079417942794379447945794679477948794979507951795279537954795579567957795879597960796179627963796479657966796779687969797079717972797379747975797679777978797979807981798279837984798579867987798879897990799179927993799479957996799779987999800080018002800380048005800680078008800980108011801280138014801580168017801880198020802180228023802480258026802780288029803080318032803380348035803680378038803980408041804280438044804580468047804880498050805180528053805480558056805780588059806080618062806380648065806680678068806980708071807280738074807580768077807880798080808180828083808480858086808780888089809080918092809380948095809680978098809981008101810281038104810581068107810881098110811181128113811481158116811781188119812081218122812381248125812681278128812981308131813281338134813581368137813881398140814181428143814481458146814781488149815081518152815381548155815681578158815981608161816281638164816581668167816881698170817181728173817481758176817781788179818081818182818381848185818681878188818981908191819281938194819581968197819881998200820182028203820482058206820782088209821082118212821382148215821682178218821982208221822282238224822582268227822882298230823182328233823482358236823782388239824082418242824382448245824682478248824982508251825282538254825582568257825882598260826182628263826482658266826782688269827082718272827382748275827682778278827982808281828282838284828582868287828882898290829182928293829482958296829782988299830083018302830383048305830683078308830983108311831283138314831583168317831883198320832183228323832483258326832783288329833083318332833383348335833683378338833983408341834283438344834583468347834883498350835183528353835483558356835783588359836083618362836383648365836683678368836983708371837283738374837583768377837883798380838183828383838483858386838783888389839083918392839383948395839683978398839984008401840284038404840584068407840884098410841184128413841484158416841784188419842084218422842384248425842684278428842984308431843284338434843584368437843884398440844184428443844484458446844784488449845084518452845384548455845684578458845984608461846284638464846584668467846884698470847184728473847484758476847784788479848084818482848384848485848684878488848984908491849284938494849584968497849884998500850185028503850485058506850785088509851085118512851385148515851685178518851985208521852285238524852585268527852885298530853185328533853485358536853785388539854085418542854385448545854685478548854985508551855285538554855585568557855885598560856185628563856485658566856785688569857085718572857385748575857685778578857985808581858285838584858585868587858885898590859185928593859485958596859785988599860086018602860386048605860686078608860986108611861286138614861586168617861886198620862186228623862486258626862786288629863086318632863386348635863686378638863986408641864286438644864586468647864886498650865186528653865486558656865786588659866086618662866386648665866686678668866986708671867286738674867586768677867886798680868186828683868486858686868786888689869086918692869386948695869686978698869987008701870287038704870587068707870887098710871187128713871487158716871787188719872087218722872387248725872687278728872987308731873287338734873587368737873887398740874187428743874487458746874787488749875087518752875387548755875687578758875987608761876287638764876587668767876887698770877187728773877487758776877787788779878087818782878387848785878687878788878987908791879287938794879587968797879887998800880188028803880488058806880788088809881088118812881388148815881688178818881988208821882288238824882588268827882888298830883188328833883488358836883788388839884088418842884388448845884688478848884988508851885288538854885588568857885888598860886188628863886488658866886788688869887088718872887388748875887688778878887988808881888288838884888588868887888888898890889188928893889488958896889788988899890089018902890389048905890689078908890989108911891289138914891589168917891889198920892189228923892489258926892789288929893089318932893389348935893689378938893989408941894289438944894589468947894889498950895189528953895489558956895789588959896089618962896389648965896689678968896989708971897289738974897589768977897889798980898189828983898489858986898789888989899089918992899389948995899689978998899990009001900290039004900590069007900890099010901190129013901490159016901790189019902090219022902390249025902690279028902990309031903290339034903590369037903890399040904190429043904490459046904790489049905090519052905390549055905690579058905990609061906290639064906590669067906890699070907190729073907490759076907790789079908090819082908390849085908690879088908990909091909290939094909590969097909890999100910191029103910491059106910791089109911091119112911391149115911691179118911991209121912291239124912591269127912891299130913191329133913491359136913791389139914091419142914391449145914691479148914991509151915291539154915591569157915891599160916191629163916491659166916791689169917091719172917391749175917691779178917991809181918291839184918591869187918891899190919191929193919491959196919791989199920092019202920392049205920692079208920992109211921292139214921592169217921892199220922192229223922492259226922792289229923092319232923392349235923692379238923992409241924292439244924592469247924892499250925192529253925492559256925792589259926092619262926392649265926692679268926992709271927292739274927592769277927892799280928192829283928492859286928792889289929092919292929392949295929692979298929993009301930293039304930593069307930893099310931193129313931493159316931793189319932093219322932393249325932693279328932993309331933293339334933593369337933893399340934193429343934493459346934793489349935093519352935393549355935693579358935993609361936293639364936593669367936893699370937193729373937493759376937793789379938093819382938393849385938693879388938993909391939293939394939593969397939893999400940194029403940494059406940794089409941094119412941394149415941694179418941994209421942294239424942594269427942894299430943194329433943494359436943794389439944094419442944394449445944694479448944994509451945294539454945594569457945894599460946194629463946494659466946794689469947094719472947394749475947694779478947994809481948294839484948594869487948894899490949194929493949494959496949794989499950095019502950395049505950695079508950995109511951295139514951595169517951895199520952195229523952495259526952795289529953095319532953395349535953695379538953995409541954295439544954595469547954895499550955195529553955495559556955795589559956095619562956395649565956695679568956995709571957295739574957595769577957895799580958195829583958495859586958795889589959095919592959395949595959695979598959996009601960296039604960596069607960896099610961196129613961496159616961796189619962096219622962396249625962696279628962996309631963296339634963596369637963896399640964196429643964496459646964796489649965096519652965396549655965696579658965996609661966296639664966596669667966896699670967196729673967496759676967796789679968096819682968396849685968696879688968996909691969296939694969596969697969896999700970197029703970497059706970797089709971097119712971397149715971697179718971997209721972297239724972597269727972897299730973197329733973497359736973797389739974097419742974397449745974697479748974997509751975297539754975597569757975897599760976197629763976497659766976797689769977097719772977397749775977697779778977997809781978297839784978597869787978897899790979197929793979497959796979797989799980098019802980398049805980698079808980998109811981298139814981598169817981898199820982198229823982498259826982798289829983098319832983398349835983698379838983998409841984298439844984598469847984898499850985198529853985498559856985798589859986098619862986398649865986698679868986998709871987298739874987598769877987898799880988198829883988498859886988798889889989098919892989398949895989698979898989999009901990299039904990599069907990899099910991199129913991499159916991799189919992099219922992399249925992699279928992999309931993299339934993599369937993899399940994199429943994499459946994799489949995099519952995399549955995699579958995999609961996299639964996599669967996899699970997199729973997499759976997799789979998099819982998399849985998699879988998999909991999299939994999599969997999899991000010001100021000310004100051000610007100081000910010100111001210013100141001510016100171001810019100201002110022100231002410025100261002710028100291003010031100321003310034100351003610037100381003910040100411004210043100441004510046100471004810049100501005110052100531005410055100561005710058100591006010061100621006310064100651006610067100681006910070100711007210073100741007510076100771007810079100801008110082100831008410085100861008710088100891009010091100921009310094100951009610097100981009910100101011010210103101041010510106101071010810109101101011110112101131011410115101161011710118101191012010121101221012310124101251012610127101281012910130101311013210133101341013510136101371013810139101401014110142101431014410145101461014710148101491015010151101521015310154101551015610157101581015910160101611016210163101641016510166101671016810169101701017110172101731017410175101761017710178101791018010181101821018310184101851018610187101881018910190101911019210193101941019510196101971019810199102001020110202102031020410205102061020710208102091021010211102121021310214102151021610217102181021910220102211022210223102241022510226102271022810229102301023110232102331023410235102361023710238102391024010241102421024310244102451024610247102481024910250102511025210253102541025510256102571025810259102601026110262102631026410265102661026710268102691027010271102721027310274102751027610277102781027910280102811028210283102841028510286102871028810289102901029110292102931029410295102961029710298102991030010301103021030310304103051030610307103081030910310103111031210313103141031510316103171031810319103201032110322103231032410325103261032710328103291033010331103321033310334103351033610337103381033910340103411034210343103441034510346103471034810349103501035110352103531035410355103561035710358103591036010361103621036310364103651036610367103681036910370103711037210373103741037510376103771037810379103801038110382103831038410385103861038710388103891039010391103921039310394103951039610397103981039910400104011040210403104041040510406104071040810409104101041110412104131041410415104161041710418104191042010421104221042310424104251042610427104281042910430104311043210433104341043510436104371043810439104401044110442104431044410445104461044710448104491045010451104521045310454104551045610457104581045910460104611046210463104641046510466104671046810469104701047110472104731047410475104761047710478104791048010481104821048310484104851048610487104881048910490104911049210493104941049510496104971049810499105001050110502105031050410505105061050710508105091051010511105121051310514105151051610517105181051910520105211052210523105241052510526105271052810529105301053110532105331053410535105361053710538105391054010541105421054310544105451054610547105481054910550105511055210553105541055510556105571055810559105601056110562105631056410565105661056710568105691057010571105721057310574105751057610577105781057910580105811058210583105841058510586105871058810589105901059110592105931059410595105961059710598105991060010601106021060310604106051060610607106081060910610106111061210613106141061510616106171061810619106201062110622106231062410625106261062710628106291063010631106321063310634106351063610637106381063910640106411064210643106441064510646106471064810649106501065110652106531065410655106561065710658106591066010661106621066310664106651066610667106681066910670106711067210673106741067510676106771067810679106801068110682106831068410685106861068710688106891069010691106921069310694106951069610697106981069910700107011070210703107041070510706107071070810709107101071110712107131071410715107161071710718107191072010721107221072310724107251072610727107281072910730107311073210733107341073510736107371073810739107401074110742107431074410745107461074710748107491075010751107521075310754107551075610757107581075910760107611076210763107641076510766107671076810769107701077110772107731077410775107761077710778107791078010781107821078310784107851078610787107881078910790107911079210793107941079510796107971079810799108001080110802108031080410805108061080710808108091081010811108121081310814108151081610817108181081910820108211082210823108241082510826108271082810829108301083110832108331083410835108361083710838108391084010841108421084310844108451084610847108481084910850108511085210853108541085510856108571085810859108601086110862108631086410865108661086710868108691087010871108721087310874108751087610877108781087910880108811088210883108841088510886108871088810889108901089110892108931089410895108961089710898108991090010901109021090310904109051090610907109081090910910109111091210913109141091510916109171091810919109201092110922109231092410925109261092710928109291093010931109321093310934109351093610937109381093910940109411094210943109441094510946109471094810949109501095110952109531095410955109561095710958109591096010961109621096310964109651096610967109681096910970109711097210973109741097510976109771097810979109801098110982109831098410985109861098710988109891099010991109921099310994109951099610997109981099911000110011100211003110041100511006110071100811009110101101111012110131101411015110161101711018110191102011021110221102311024110251102611027110281102911030110311103211033110341103511036110371103811039110401104111042110431104411045110461104711048110491105011051110521105311054110551105611057110581105911060110611106211063110641106511066110671106811069110701107111072110731107411075110761107711078110791108011081110821108311084110851108611087110881108911090110911109211093110941109511096110971109811099111001110111102111031110411105111061110711108111091111011111111121111311114111151111611117111181111911120111211112211123111241112511126111271112811129111301113111132111331113411135111361113711138111391114011141111421114311144111451114611147111481114911150111511115211153111541115511156111571115811159111601116111162111631116411165111661116711168111691117011171111721117311174111751117611177111781117911180111811118211183111841118511186111871118811189111901119111192111931119411195111961119711198111991120011201112021120311204112051120611207112081120911210112111121211213112141121511216112171121811219112201122111222112231122411225112261122711228112291123011231112321123311234112351123611237112381123911240112411124211243112441124511246112471124811249112501125111252112531125411255112561125711258112591126011261112621126311264112651126611267112681126911270112711127211273112741127511276112771127811279112801128111282112831128411285112861128711288112891129011291112921129311294112951129611297112981129911300113011130211303113041130511306113071130811309113101131111312113131131411315113161131711318113191132011321113221132311324113251132611327113281132911330113311133211333113341133511336113371133811339113401134111342113431134411345113461134711348113491135011351113521135311354113551135611357113581135911360113611136211363113641136511366113671136811369113701137111372113731137411375113761137711378113791138011381113821138311384113851138611387113881138911390113911139211393113941139511396113971139811399114001140111402114031140411405114061140711408114091141011411114121141311414114151141611417114181141911420114211142211423114241142511426114271142811429114301143111432114331143411435114361143711438114391144011441114421144311444114451144611447114481144911450114511145211453114541145511456114571145811459114601146111462114631146411465114661146711468114691147011471114721147311474114751147611477114781147911480114811148211483114841148511486114871148811489114901149111492114931149411495114961149711498114991150011501115021150311504115051150611507115081150911510115111151211513115141151511516115171151811519115201152111522115231152411525115261152711528115291153011531115321153311534115351153611537115381153911540115411154211543115441154511546115471154811549115501155111552115531155411555115561155711558115591156011561115621156311564115651156611567115681156911570115711157211573115741157511576115771157811579115801158111582115831158411585115861158711588115891159011591115921159311594115951159611597115981159911600116011160211603116041160511606116071160811609116101161111612116131161411615116161161711618116191162011621116221162311624116251162611627116281162911630116311163211633116341163511636116371163811639116401164111642116431164411645116461164711648116491165011651116521165311654116551165611657116581165911660116611166211663116641166511666116671166811669116701167111672116731167411675116761167711678116791168011681116821168311684116851168611687116881168911690116911169211693116941169511696116971169811699117001170111702117031170411705117061170711708117091171011711117121171311714117151171611717117181171911720117211172211723117241172511726117271172811729117301173111732117331173411735117361173711738117391174011741117421174311744117451174611747117481174911750117511175211753117541175511756117571175811759117601176111762117631176411765117661176711768117691177011771117721177311774117751177611777117781177911780117811178211783117841178511786117871178811789117901179111792117931179411795117961179711798117991180011801118021180311804118051180611807118081180911810118111181211813118141181511816118171181811819118201182111822118231182411825118261182711828118291183011831118321183311834118351183611837118381183911840118411184211843118441184511846118471184811849118501185111852118531185411855118561185711858118591186011861118621186311864118651186611867118681186911870118711187211873118741187511876118771187811879118801188111882118831188411885118861188711888118891189011891118921189311894118951189611897118981189911900119011190211903119041190511906119071190811909119101191111912119131191411915119161191711918119191192011921119221192311924119251192611927119281192911930119311193211933119341193511936119371193811939119401194111942119431194411945119461194711948119491195011951119521195311954119551195611957119581195911960119611196211963119641196511966119671196811969119701197111972119731197411975119761197711978119791198011981119821198311984119851198611987119881198911990119911199211993119941199511996119971199811999120001200112002120031200412005120061200712008120091201012011120121201312014120151201612017120181201912020120211202212023120241202512026120271202812029120301203112032120331203412035120361203712038120391204012041120421204312044120451204612047120481204912050120511205212053120541205512056120571205812059120601206112062120631206412065120661206712068120691207012071120721207312074120751207612077120781207912080120811208212083120841208512086120871208812089120901209112092120931209412095120961209712098120991210012101121021210312104121051210612107121081210912110121111211212113121141211512116121171211812119121201212112122121231212412125121261212712128121291213012131121321213312134121351213612137121381213912140121411214212143121441214512146121471214812149121501215112152121531215412155121561215712158121591216012161121621216312164121651216612167121681216912170121711217212173121741217512176121771217812179121801218112182121831218412185121861218712188121891219012191121921219312194121951219612197121981219912200122011220212203122041220512206122071220812209122101221112212122131221412215122161221712218122191222012221122221222312224122251222612227122281222912230122311223212233122341223512236122371223812239122401224112242122431224412245122461224712248122491225012251122521225312254122551225612257122581225912260122611226212263122641226512266122671226812269122701227112272122731227412275122761227712278122791228012281122821228312284122851228612287122881228912290122911229212293122941229512296122971229812299123001230112302123031230412305123061230712308123091231012311123121231312314123151231612317123181231912320123211232212323123241232512326123271232812329123301233112332123331233412335123361233712338123391234012341123421234312344123451234612347123481234912350123511235212353123541235512356123571235812359123601236112362123631236412365123661236712368123691237012371123721237312374123751237612377123781237912380123811238212383123841238512386123871238812389123901239112392123931239412395123961239712398123991240012401124021240312404124051240612407124081240912410124111241212413124141241512416124171241812419124201242112422124231242412425124261242712428124291243012431124321243312434124351243612437124381243912440124411244212443124441244512446124471244812449124501245112452124531245412455124561245712458124591246012461124621246312464124651246612467124681246912470124711247212473124741247512476124771247812479124801248112482124831248412485124861248712488124891249012491124921249312494124951249612497124981249912500125011250212503125041250512506125071250812509125101251112512125131251412515125161251712518125191252012521125221252312524125251252612527125281252912530125311253212533125341253512536125371253812539125401254112542125431254412545125461254712548125491255012551125521255312554125551255612557125581255912560125611256212563125641256512566125671256812569125701257112572125731257412575125761257712578125791258012581125821258312584125851258612587125881258912590125911259212593125941259512596125971259812599126001260112602126031260412605126061260712608126091261012611126121261312614126151261612617126181261912620126211262212623126241262512626126271262812629126301263112632126331263412635126361263712638126391264012641126421264312644126451264612647126481264912650126511265212653126541265512656126571265812659126601266112662126631266412665126661266712668126691267012671126721267312674126751267612677126781267912680126811268212683126841268512686126871268812689126901269112692126931269412695126961269712698126991270012701127021270312704127051270612707127081270912710127111271212713127141271512716127171271812719127201272112722127231272412725127261272712728127291273012731127321273312734127351273612737127381273912740127411274212743127441274512746127471274812749127501275112752127531275412755127561275712758127591276012761127621276312764127651276612767127681276912770127711277212773127741277512776127771277812779127801278112782127831278412785127861278712788127891279012791127921279312794127951279612797127981279912800128011280212803128041280512806128071280812809128101281112812128131281412815128161281712818128191282012821128221282312824128251282612827128281282912830128311283212833128341283512836128371283812839128401284112842128431284412845128461284712848128491285012851128521285312854128551285612857128581285912860128611286212863128641286512866128671286812869128701287112872128731287412875128761287712878128791288012881128821288312884128851288612887128881288912890128911289212893128941289512896128971289812899129001290112902129031290412905129061290712908129091291012911129121291312914129151291612917129181291912920129211292212923129241292512926129271292812929129301293112932129331293412935129361293712938129391294012941129421294312944129451294612947129481294912950129511295212953129541295512956129571295812959129601296112962129631296412965129661296712968129691297012971129721297312974129751297612977129781297912980129811298212983129841298512986129871298812989129901299112992129931299412995129961299712998129991300013001130021300313004130051300613007130081300913010130111301213013130141301513016130171301813019130201302113022130231302413025130261302713028130291303013031130321303313034130351303613037130381303913040130411304213043130441304513046130471304813049130501305113052130531305413055130561305713058130591306013061130621306313064130651306613067130681306913070130711307213073130741307513076130771307813079130801308113082130831308413085130861308713088130891309013091130921309313094130951309613097130981309913100131011310213103131041310513106131071310813109131101311113112131131311413115131161311713118131191312013121131221312313124131251312613127131281312913130131311313213133131341313513136131371313813139131401314113142131431314413145131461314713148131491315013151131521315313154131551315613157131581315913160131611316213163131641316513166131671316813169131701317113172131731317413175131761317713178131791318013181131821318313184131851318613187131881318913190131911319213193131941319513196131971319813199132001320113202132031320413205132061320713208132091321013211132121321313214132151321613217132181321913220132211322213223132241322513226132271322813229132301323113232132331323413235132361323713238132391324013241132421324313244132451324613247132481324913250132511325213253132541325513256132571325813259132601326113262132631326413265132661326713268132691327013271132721327313274132751327613277132781327913280132811328213283132841328513286132871328813289132901329113292132931329413295132961329713298132991330013301133021330313304133051330613307133081330913310133111331213313133141331513316133171331813319133201332113322133231332413325133261332713328133291333013331133321333313334133351333613337133381333913340133411334213343133441334513346133471334813349133501335113352133531335413355133561335713358133591336013361133621336313364133651336613367133681336913370133711337213373133741337513376133771337813379133801338113382133831338413385133861338713388133891339013391133921339313394133951339613397133981339913400134011340213403134041340513406134071340813409134101341113412134131341413415134161341713418134191342013421134221342313424134251342613427134281342913430134311343213433134341343513436134371343813439134401344113442134431344413445134461344713448134491345013451134521345313454134551345613457134581345913460134611346213463134641346513466134671346813469134701347113472134731347413475134761347713478134791348013481134821348313484134851348613487134881348913490134911349213493134941349513496134971349813499135001350113502135031350413505135061350713508135091351013511135121351313514135151351613517135181351913520135211352213523135241352513526135271352813529135301353113532135331353413535135361353713538135391354013541135421354313544135451354613547135481354913550135511355213553135541355513556135571355813559135601356113562135631356413565135661356713568135691357013571135721357313574135751357613577135781357913580135811358213583135841358513586135871358813589135901359113592135931359413595135961359713598135991360013601136021360313604136051360613607136081360913610136111361213613136141361513616136171361813619136201362113622136231362413625136261362713628136291363013631136321363313634136351363613637136381363913640136411364213643136441364513646136471364813649136501365113652136531365413655136561365713658136591366013661136621366313664136651366613667136681366913670136711367213673136741367513676136771367813679136801368113682136831368413685136861368713688136891369013691136921369313694136951369613697136981369913700137011370213703137041370513706137071370813709137101371113712137131371413715137161371713718137191372013721137221372313724137251372613727137281372913730137311373213733137341373513736137371373813739137401374113742137431374413745137461374713748137491375013751137521375313754137551375613757137581375913760137611376213763137641376513766137671376813769137701377113772137731377413775137761377713778137791378013781137821378313784137851378613787137881378913790137911379213793137941379513796137971379813799138001380113802138031380413805138061380713808138091381013811138121381313814138151381613817138181381913820138211382213823138241382513826138271382813829138301383113832138331383413835138361383713838138391384013841138421384313844138451384613847138481384913850138511385213853138541385513856138571385813859138601386113862138631386413865138661386713868138691387013871138721387313874138751387613877138781387913880138811388213883138841388513886138871388813889138901389113892138931389413895138961389713898138991390013901139021390313904139051390613907139081390913910139111391213913139141391513916139171391813919139201392113922139231392413925139261392713928139291393013931139321393313934139351393613937139381393913940139411394213943139441394513946139471394813949139501395113952139531395413955139561395713958139591396013961139621396313964139651396613967139681396913970139711397213973139741397513976139771397813979139801398113982139831398413985139861398713988139891399013991139921399313994139951399613997139981399914000140011400214003140041400514006140071400814009140101401114012140131401414015140161401714018140191402014021140221402314024140251402614027140281402914030140311403214033140341403514036140371403814039140401404114042140431404414045140461404714048140491405014051140521405314054140551405614057140581405914060140611406214063140641406514066140671406814069140701407114072140731407414075140761407714078140791408014081140821408314084140851408614087140881408914090140911409214093140941409514096140971409814099141001410114102141031410414105141061410714108141091411014111141121411314114141151411614117141181411914120141211412214123141241412514126141271412814129141301413114132141331413414135141361413714138141391414014141141421414314144141451414614147141481414914150141511415214153141541415514156141571415814159141601416114162141631416414165141661416714168141691417014171141721417314174141751417614177141781417914180141811418214183141841418514186141871418814189141901419114192141931419414195141961419714198141991420014201142021420314204142051420614207142081420914210142111421214213142141421514216142171421814219142201422114222142231422414225142261422714228142291423014231142321423314234142351423614237142381423914240142411424214243142441424514246142471424814249142501425114252142531425414255142561425714258142591426014261142621426314264142651426614267142681426914270142711427214273142741427514276142771427814279142801428114282142831428414285142861428714288142891429014291142921429314294142951429614297142981429914300143011430214303143041430514306143071430814309143101431114312143131431414315143161431714318143191432014321143221432314324143251432614327143281432914330143311433214333143341433514336143371433814339143401434114342143431434414345143461434714348143491435014351143521435314354143551435614357143581435914360143611436214363143641436514366143671436814369143701437114372143731437414375143761437714378143791438014381143821438314384143851438614387143881438914390143911439214393143941439514396143971439814399144001440114402144031440414405144061440714408144091441014411144121441314414144151441614417144181441914420144211442214423144241442514426144271442814429144301443114432144331443414435144361443714438144391444014441144421444314444144451444614447144481444914450144511445214453144541445514456144571445814459144601446114462144631446414465144661446714468144691447014471144721447314474144751447614477144781447914480144811448214483144841448514486144871448814489144901449114492144931449414495144961449714498144991450014501145021450314504145051450614507145081450914510145111451214513145141451514516145171451814519145201452114522145231452414525145261452714528145291453014531145321453314534145351453614537145381453914540145411454214543145441454514546145471454814549145501455114552145531455414555145561455714558145591456014561145621456314564145651456614567145681456914570145711457214573145741457514576145771457814579145801458114582145831458414585145861458714588145891459014591145921459314594145951459614597145981459914600146011460214603146041460514606146071460814609146101461114612146131461414615146161461714618146191462014621146221462314624146251462614627146281462914630146311463214633146341463514636146371463814639146401464114642146431464414645146461464714648146491465014651146521465314654146551465614657146581465914660146611466214663146641466514666146671466814669146701467114672146731467414675146761467714678146791468014681146821468314684146851468614687146881468914690146911469214693146941469514696146971469814699147001470114702147031470414705147061470714708147091471014711147121471314714147151471614717147181471914720147211472214723147241472514726147271472814729147301473114732147331473414735147361473714738147391474014741147421474314744147451474614747147481474914750147511475214753147541475514756147571475814759147601476114762147631476414765147661476714768147691477014771147721477314774147751477614777147781477914780147811478214783147841478514786147871478814789147901479114792147931479414795147961479714798147991480014801148021480314804148051480614807148081480914810148111481214813148141481514816148171481814819148201482114822148231482414825148261482714828148291483014831148321483314834148351483614837148381483914840148411484214843148441484514846148471484814849148501485114852148531485414855148561485714858148591486014861148621486314864148651486614867148681486914870148711487214873148741487514876148771487814879148801488114882148831488414885148861488714888148891489014891148921489314894148951489614897148981489914900149011490214903149041490514906149071490814909149101491114912149131491414915149161491714918149191492014921149221492314924149251492614927149281492914930149311493214933149341493514936149371493814939149401494114942149431494414945149461494714948149491495014951149521495314954149551495614957149581495914960149611496214963149641496514966149671496814969149701497114972149731497414975149761497714978149791498014981149821498314984149851498614987149881498914990149911499214993149941499514996149971499814999150001500115002150031500415005150061500715008150091501015011150121501315014150151501615017150181501915020150211502215023150241502515026150271502815029150301503115032150331503415035150361503715038150391504015041150421504315044150451504615047150481504915050150511505215053150541505515056150571505815059150601506115062150631506415065150661506715068150691507015071150721507315074150751507615077150781507915080150811508215083150841508515086150871508815089150901509115092150931509415095150961509715098150991510015101151021510315104151051510615107151081510915110151111511215113151141511515116151171511815119151201512115122151231512415125151261512715128151291513015131151321513315134151351513615137151381513915140151411514215143151441514515146151471514815149151501515115152151531515415155151561515715158151591516015161151621516315164151651516615167151681516915170151711517215173151741517515176151771517815179151801518115182151831518415185151861518715188151891519015191151921519315194151951519615197151981519915200152011520215203152041520515206152071520815209152101521115212152131521415215152161521715218152191522015221152221522315224152251522615227152281522915230152311523215233152341523515236152371523815239152401524115242152431524415245152461524715248152491525015251152521525315254152551525615257152581525915260152611526215263152641526515266152671526815269152701527115272152731527415275152761527715278152791528015281152821528315284152851528615287152881528915290152911529215293152941529515296152971529815299153001530115302153031530415305153061530715308153091531015311153121531315314153151531615317153181531915320153211532215323153241532515326153271532815329153301533115332153331533415335153361533715338153391534015341153421534315344153451534615347153481534915350153511535215353153541535515356153571535815359153601536115362153631536415365153661536715368153691537015371153721537315374153751537615377153781537915380153811538215383153841538515386153871538815389153901539115392153931539415395153961539715398153991540015401154021540315404154051540615407154081540915410154111541215413154141541515416154171541815419154201542115422154231542415425154261542715428154291543015431154321543315434154351543615437154381543915440154411544215443154441544515446154471544815449154501545115452154531545415455154561545715458154591546015461154621546315464154651546615467154681546915470154711547215473154741547515476154771547815479154801548115482154831548415485154861548715488154891549015491154921549315494154951549615497154981549915500155011550215503155041550515506155071550815509155101551115512155131551415515155161551715518155191552015521155221552315524155251552615527155281552915530155311553215533155341553515536155371553815539155401554115542155431554415545155461554715548155491555015551155521555315554155551555615557155581555915560155611556215563155641556515566155671556815569155701557115572155731557415575155761557715578155791558015581155821558315584155851558615587155881558915590155911559215593155941559515596155971559815599156001560115602156031560415605156061560715608156091561015611156121561315614156151561615617156181561915620156211562215623156241562515626156271562815629156301563115632156331563415635156361563715638156391564015641156421564315644156451564615647156481564915650156511565215653156541565515656156571565815659156601566115662156631566415665156661566715668156691567015671156721567315674156751567615677156781567915680156811568215683156841568515686156871568815689156901569115692156931569415695156961569715698156991570015701157021570315704157051570615707157081570915710157111571215713157141571515716157171571815719157201572115722157231572415725157261572715728157291573015731157321573315734157351573615737157381573915740157411574215743157441574515746157471574815749157501575115752157531575415755157561575715758157591576015761157621576315764157651576615767157681576915770157711577215773157741577515776157771577815779157801578115782157831578415785157861578715788157891579015791157921579315794157951579615797157981579915800158011580215803158041580515806158071580815809158101581115812158131581415815158161581715818158191582015821158221582315824158251582615827158281582915830158311583215833158341583515836158371583815839158401584115842158431584415845158461584715848158491585015851158521585315854158551585615857158581585915860158611586215863158641586515866158671586815869158701587115872158731587415875158761587715878158791588015881158821588315884158851588615887158881588915890158911589215893158941589515896158971589815899159001590115902159031590415905159061590715908159091591015911159121591315914159151591615917159181591915920159211592215923159241592515926159271592815929159301593115932159331593415935159361593715938159391594015941159421594315944159451594615947159481594915950159511595215953159541595515956159571595815959159601596115962159631596415965159661596715968159691597015971159721597315974159751597615977159781597915980159811598215983159841598515986159871598815989159901599115992159931599415995159961599715998159991600016001160021600316004160051600616007160081600916010160111601216013160141601516016160171601816019160201602116022160231602416025160261602716028160291603016031160321603316034160351603616037160381603916040160411604216043160441604516046160471604816049160501605116052160531605416055160561605716058160591606016061160621606316064160651606616067160681606916070160711607216073160741607516076160771607816079160801608116082160831608416085160861608716088160891609016091160921609316094160951609616097160981609916100161011610216103161041610516106161071610816109161101611116112161131611416115161161611716118161191612016121161221612316124161251612616127161281612916130161311613216133161341613516136161371613816139161401614116142161431614416145161461614716148161491615016151161521615316154161551615616157161581615916160161611616216163161641616516166161671616816169161701617116172161731617416175161761617716178161791618016181161821618316184161851618616187161881618916190161911619216193161941619516196161971619816199162001620116202162031620416205162061620716208162091621016211162121621316214162151621616217162181621916220162211622216223162241622516226162271622816229162301623116232162331623416235162361623716238162391624016241162421624316244162451624616247162481624916250162511625216253162541625516256162571625816259162601626116262162631626416265162661626716268162691627016271162721627316274162751627616277162781627916280162811628216283162841628516286162871628816289162901629116292162931629416295162961629716298162991630016301163021630316304163051630616307163081630916310163111631216313163141631516316163171631816319163201632116322163231632416325163261632716328163291633016331163321633316334163351633616337163381633916340163411634216343163441634516346163471634816349163501635116352163531635416355163561635716358163591636016361163621636316364163651636616367163681636916370163711637216373163741637516376163771637816379163801638116382163831638416385163861638716388163891639016391163921639316394163951639616397163981639916400164011640216403164041640516406164071640816409164101641116412164131641416415164161641716418164191642016421164221642316424164251642616427164281642916430164311643216433164341643516436164371643816439164401644116442164431644416445 |
- /*! Amaze UI v2.5.2 | by Amaze UI Team | (c) 2016 AllMobilize, Inc. | Licensed under MIT | 2016-01-26T11:06:52+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';
- /* jshint -W040 */
- 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.5.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};
- })();
- /* jshint -W069 */
- 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);
- // 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.
- * @desc Executes a function when it stops being invoked for n seconds
- * @via _.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());
- };
- /* jshint -W054 */
- 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;
- };
- /* jshint +W054 */
- 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() {
- /* jshint unused:false */
- var redraw = this.offsetHeight;
- });
- };
- /* jshint unused:true */
- $.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);
- }
- };
- /**
- * https://github.com/cho45/micro-template.js
- * (c) cho45 http://cho45.github.com/mit-license
- */
- /* jshint -W109 */
- 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;
- 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, '');
- /* jshint -W054 */
- 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];
- };
- /* jshint +W109 */
- /* jshint +W054 */
- 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() {
- UI.DOMObserve(this, options, callback);
- });
- };
- 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 = $('body');
- UI.DOMReady = true;
- // Run default init
- $.each(UI.DOMWatchers, function(i, watcher) {
- watcher(document);
- });
- // watches DOM
- UI.DOMObserve('[data-am-observe]');
- $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.4 - 2014-09-28
- * http://hammerjs.github.io/
- *
- * Copyright (c) 2014 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);
- }
- }
- }
- /**
- * extend object.
- * means that properties in dest will be overwritten by the ones in src.
- * @param {Object} dest
- * @param {Object} src
- * @param {Boolean} [merge]
- * @returns {Object} dest
- */
- 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;
- }
- /**
- * 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
- */
- function merge(dest, src) {
- return extend(dest, src, true);
- }
- /**
- * 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) {
- extend(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;
- return (doc.defaultView || doc.parentWindow);
- }
- 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);
- input.scale = firstMultiple ? getScale(firstMultiple.pointers, pointers) : 1;
- input.rotation = firstMultiple ? getRotation(firstMultiple.pointers, pointers) : 0;
- 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 = last.deltaX - input.deltaX;
- var deltaY = last.deltaY - input.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.allow = true; // used by Input.TouchMouse to disable mouse 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, and mouse events are allowed (see the TouchMouse input)
- if (!this.pressed || !this.allow) {
- 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) {
- 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
- */
- 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);
- }
- 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);
- // when we're in a touch event, so block all upcoming mouse events
- // most mobile browser also emit mouseevents, right after touchstart
- if (isTouch) {
- this.mouse.allow = false;
- } else if (isMouse && !this.mouse.allow) {
- return;
- }
- // reset the allowMouse when we're done
- if (inputEvent & (INPUT_END | INPUT_CANCEL)) {
- this.mouse.allow = true;
- }
- this.callback(manager, inputEvent, inputData);
- },
- /**
- * remove the event listeners
- */
- destroy: function destroy() {
- this.touch.destroy();
- this.mouse.destroy();
- }
- });
- 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';
- /**
- * 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[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) {
- // not needed with native support for the touchAction property
- if (NATIVE_TOUCH_ACTION) {
- return;
- }
- 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);
- var hasPanY = inStr(actions, TOUCH_ACTION_PAN_Y);
- var hasPanX = inStr(actions, TOUCH_ACTION_PAN_X);
- 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);
- // pan-x and pan-y can be combined
- if (hasPanX && hasPanY) {
- return TOUCH_ACTION_PAN_X + ' ' + TOUCH_ACTION_PAN_Y;
- }
- // 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;
- }
- /**
- * 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.id = uniqueId();
- this.manager = null;
- this.options = merge(options || {}, this.defaults);
- // 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) {
- extend(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(withState) {
- self.manager.emit(self.options.event + (withState ? stateStr(state) : ''), input);
- }
- // 'panstart' and 'panmove'
- if (state < STATE_ENDED) {
- emit(true);
- }
- emit(); // simple 'eventName' events
- // panend and pancancel
- if (state >= STATE_ENDED) {
- emit(true);
- }
- },
- /**
- * 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 = extend({}, 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) {
- this.manager.emit(this.options.event + direction, input);
- }
- 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) {
- this._super.emit.call(this, input);
- if (input.scale !== 1) {
- var inOut = input.scale < 1 ? 'in' : 'out';
- this.manager.emit(this.options.event + inOut, 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: 500, // minimal time of the pointer to be pressed
- threshold: 5 // 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.65,
- 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.velocity;
- } else if (direction & DIRECTION_HORIZONTAL) {
- velocity = input.velocityX;
- } else if (direction & DIRECTION_VERTICAL) {
- velocity = input.velocityY;
- }
- return this._super.attrTest.call(this, input) &&
- direction & input.direction &&
- input.distance > this.options.threshold &&
- abs(velocity) > this.options.velocity && input.eventType & INPUT_END;
- },
- emit: function(input) {
- var direction = directionStr(input.direction);
- 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: 2, // 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 an 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.4';
- /**
- * 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) {
- options = options || {};
- this.options = merge(options, Hammer.defaults);
- this.options.inputTarget = this.options.inputTarget || element;
- this.handlers = {};
- this.session = {};
- this.recognizers = [];
- this.element = element;
- this.input = createInputInstance(this);
- this.touchAction = new TouchAction(this, this.options.touchAction);
- toggleCssProps(this, true);
- each(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) {
- extend(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;
- }
- var recognizers = this.recognizers;
- recognizer = this.get(recognizer);
- recognizers.splice(inArray(recognizers, recognizer), 1);
- this.touchAction.update();
- return this;
- },
- /**
- * bind event
- * @param {String} events
- * @param {Function} handler
- * @returns {EventEmitter} this
- */
- on: function(events, handler) {
- 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) {
- var handlers = this.handlers;
- each(splitStr(events), function(event) {
- if (!handler) {
- delete handlers[event];
- } else {
- 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;
- each(manager.options.cssProps, function(value, name) {
- element.style[prefixed(element.style, name)] = add ? value : '';
- });
- }
- /**
- * 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);
- }
- extend(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,
- 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);
- /* jshint -W101, -W106 */
- /**
- * 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',
- 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"]');
- if ($parent.length) {
- var $input = this.$element.find('input');
- if ($input.prop('type') == 'radio') {
- if ($input.prop('checked') && $element.hasClass('am-active')) {
- changed = false;
- } else {
- $parent.find('.am-active').removeClass('am-active');
- }
- }
- if (changed) {
- $input.prop('checked',
- !$element.hasClass('am-active')).trigger('change');
- }
- }
- if (changed) {
- $element.toggleClass('am-active');
- if (!$element.hasClass('am-active')) {
- $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();
- });
- 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();
- 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() {
- if (!this.$dropdown.length) {
- return;
- }
- var $dropdown = this.$dropdown;
- 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 226: add `<i></i>`
- // - namespace
- // - Init code
- // TODO: start after x ms when pause on actions
- /*
- * jQuery FlexSlider v2.4.0
- * Copyright 2012 WooThemes
- * Contributing Author: Tyler Smith
- */
- // 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 = (touch) ? "touchend" : "click",
- 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 = {},
- focused = true;
- // 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);
- // 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);
- item = (slider.vars.controlNav === "thumbnails") ? '<img src="' + slide.attr('data-thumb') + '"/>' : '<a>' + 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>' + 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>');
- // CONTROLSCONTAINER:
- if (slider.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></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,
- scrolling = false,
- localX = 0,
- localY = 0,
- accDx = 0;
- if (!msGesture) {
- el.addEventListener('touchstart', onTouchStart, false);
- function onTouchStart(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);
- }
- }
- function onTouchMove(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");
- }
- }
- }
- function onTouchEnd(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;
- }
- } 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).height()}, dur) : $obj.height(slider.slides.eq(slider.animatingTo).height());
- }
- },
- 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,
- "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.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.pagingCount = slider.count;
- slider.last = slider.count - 1;
- }
- slider.computedW = slider.itemW - slider.boxPadding;
- };
- 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.
- 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 === true) || $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 = {};
- var queue = [];
- var draining = false;
- var currentQueue;
- var queueIndex = -1;
- function cleanUpNextTick() {
- draining = false;
- if (currentQueue.length) {
- queue = currentQueue.concat(queue);
- } else {
- queueIndex = -1;
- }
- if (queue.length) {
- drainQueue();
- }
- }
- function drainQueue() {
- if (draining) {
- return;
- }
- var timeout = setTimeout(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;
- clearTimeout(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) {
- setTimeout(drainQueue, 0);
- }
- };
- // 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);
- /* jshint unused: false */
- /* jshint -W101, -W116, -W109 */
- /*! iScroll v5.1.3
- * (c) 2008-2014 Matteo Spinelli
- * http://cubiq.org/license
- */
- var rAF = window.requestAnimationFrame ||
- window.webkitRequestAnimationFrame ||
- window.mozRequestAnimationFrame ||
- window.oRequestAnimationFrame ||
- window.msRequestAnimationFrame ||
- function(callback) {
- window.setTimeout(callback, 1000 / 60);
- };
- 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(9).toUpperCase() + pointerEvent.substr(10) :
- 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)
- me.isBadAndroid = /Android /.test(window.navigator.appVersion) && !(/Chrome\/\d/.test(window.navigator.appVersion));
- 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)) {
- ev = document.createEvent('MouseEvents');
- ev.initMouseEvent('click', true, true, e.view, 1,
- target.screenX, target.screenY, target.clientX, target.clientY,
- e.ctrlKey, e.altKey, e.shiftKey, e.metaKey,
- 0, 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
- 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
- };
- 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';
- }
- // 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.1.3',
- _init: function() {
- this._initEvents();
- // INSERT POINT: _init
- },
- destroy: function() {
- this._initEvents(true);
- 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) {
- if (e.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._transitionTime();
- this.startTime = utils.getTime();
- if (this.options.useTransition && this.isInTransition) {
- 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;
- if (!time || (this.options.useTransition && easing.style)) {
- 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) {
- time = time || 0;
- this.scrollerStyle[utils.style.transitionDuration] = time + 'ms';
- if (!time && utils.isBadAndroid) {
- this.scrollerStyle[utils.style.transitionDuration] = '0.001s';
- }
- // 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 (!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) {
- width = parseInt(width, 10);
- style.width = width + 'px';
- style.marginLeft = -parseInt(width / 2) + 'px';
- }
- if (height) {
- height = parseInt(height, 10);
- // style.height = height + 'px';
- style.marginTop = -parseInt(height / 2) + 'px';
- // the background color is styled to $dialog
- // so the height should set to $dialog
- this.$dialog.css({height: height + 'px'});
- } else {
- style.marginTop = -parseInt($element.height() / 2, 10) + 'px';
- }
- $element.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 options = this.options;
- var _this = this;
- var $element = this.$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) {
- this.dimmer.$element.on('click.dimmer.modal.amui', function(e) {
- _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();
- }
- });
- $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);
- /**
- * @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,
- animationInterval: 5,
- maxZoom: 5,
- minZoom: 0.5,
- lockDragAxis: false,
- use2d: false,
- 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, this.options.animationInterval, 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,
- this.options.animationInterval,
- 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,
- this.options.animationInterval,
- updateProgress,
- this.swing
- );
- },
- /**
- * Updates the aspect ratio
- */
- updateAspectRatio: function() {
- // this.setContainerY(this.getContainerX() / this.getAspectRatio());
- // @modified
- this.setContainerY()
- },
- /**
- * 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 interval
- * @param framefn
- * @param timefn
- * @param callback
- */
- animate: function(duration, interval, 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();
- setTimeout(renderFrame, interval);
- }
- }).bind(this);
- this.inAnimation = true;
- 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;
- // @modified
- return window.innerWidth
- },
- getContainerY: function() {
- // return this.container[0].offsetHeight;
- // @modified
- return window.innerHeight
- },
- setContainerY: function(y) {
- // return this.container.height(y);
- // @modified
- var t = window.innerHeight;
- return this.el.css({height: t}), this.container.height(t);
- },
- /**
- * 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-scrollspy-nav]', 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});
- if (this.$element[0].disabled) {
- this.$selector.addClass(options.disabledClass);
- }
- 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
- });
- // 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 {
- if (isChecked) {
- return false;
- }
- this.dropdown.close();
- this.$shadowOptions.not($item).removeClass(options.selectedClass);
- }
- $item.toggleClass(options.selectedClass);
- this.syncData(item);
- };
- /**
- * syncData
- * @desc 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,
- attributes: 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 ? $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.ucheck.amui')
- .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.5.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';
- 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 = '2.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__) {
- 'use strict';
- var UI = __webpack_require__(2);
- /**
- * store.js
- * @via https://github.com/marcuswestin/store.js
- * @license https://github.com/marcuswestin/store.js/blob/master/LICENSE
- */
- var store = {};
- var win = window;
- var localStorageName = 'localStorage';
- var storage;
- store.disabled = false;
- store.version = '1.3.17';
- 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;
- /***/ },
- /* 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
- };
- /***/ }
- /******/ ])
- });
- ;
|