seed.js 190 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121
  1. /*
  2. Copyright 2013, KISSY v1.40
  3. MIT Licensed
  4. build time: Nov 21 23:47
  5. */
  6. /**
  7. * @ignore
  8. * A seed where KISSY grows up from, KISS Yeah !
  9. * @author https://github.com/kissyteam/kissy/contributors
  10. */
  11. /**
  12. * The KISSY global namespace object. you can use
  13. *
  14. *
  15. * KISSY.each/mix
  16. *
  17. * to do basic operation. or
  18. *
  19. *
  20. * KISSY.use('overlay,node', function(S, Overlay, Node){
  21. * //
  22. * });
  23. *
  24. * to do complex task with modules.
  25. * @singleton
  26. * @class KISSY
  27. */
  28. var KISSY = (function (undefined) {
  29. var host = this,
  30. S,
  31. guid = 0,
  32. EMPTY = '';
  33. var loggerLevel = {
  34. 'debug': 10,
  35. 'info': 20,
  36. 'warn': 30,
  37. 'error': 40
  38. };
  39. S = {
  40. /**
  41. * The build time of the library.
  42. * NOTICE: '20131121234724' will replace with current timestamp when compressing.
  43. * @private
  44. * @type {String}
  45. */
  46. __BUILD_TIME: '20131121234724',
  47. /**
  48. * KISSY Environment.
  49. * @private
  50. * @type {Object}
  51. */
  52. Env: {
  53. host: host
  54. },
  55. /**
  56. * KISSY Config.
  57. * If load kissy.js, Config.debug defaults to true.
  58. * Else If load kissy-min.js, Config.debug defaults to false.
  59. * @private
  60. * @property {Object} Config
  61. * @property {Boolean} Config.debug
  62. * @member KISSY
  63. */
  64. Config: {
  65. debug: '@DEBUG@',
  66. fns: {}
  67. },
  68. /**
  69. * The version of the library.
  70. * NOTICE: '1.40' will replace with current version when compressing.
  71. * @type {String}
  72. */
  73. version: '1.40',
  74. /**
  75. * set KISSY configuration
  76. * @param {Object|String} configName Config object or config key.
  77. * @param {String} configName.base KISSY 's base path. Default: get from kissy(-min).js or seed(-min).js
  78. * @param {String} configName.tag KISSY 's timestamp for native module. Default: KISSY 's build time.
  79. * @param {Boolean} configName.debug whether to enable debug mod.
  80. * @param {Boolean} configName.combine whether to enable combo.
  81. * @param {Object} configName.logger logger config
  82. * @param {Object[]} configName.logger.excludes exclude configs
  83. * @param {Object} configName.logger.excludes.0 a single exclude config
  84. * @param {RegExp} configName.logger.excludes.0.logger matched logger will be excluded from logging
  85. * @param {KISSY.Logger.Level} configName.logger.excludes.0.minLevel minimum logger level
  86. * @param {KISSY.Logger.Level} configName.logger.excludes.0.maxLevel maximum logger level
  87. * @param {Object[]} configName.logger.includes include configs
  88. * @param {Object} configName.logger.includes.0 a single include config
  89. * @param {RegExp} configName.logger.includes.0.logger matched logger will be included from logging
  90. * @param {KISSY.Logger.Level} configName.logger.excludes.0.minLevel minimum logger level
  91. * @param {KISSY.Logger.Level} configName.logger.excludes.0.maxLevel maximum logger level
  92. * @param {Object} configName.packages Packages definition with package name as the key.
  93. * @param {String} configName.packages.base Package base path.
  94. * @param {String} configName.packages.tag Timestamp for this package's module file.
  95. * @param {String} configName.packages.debug Whether force debug mode for current package.
  96. * @param {String} configName.packages.combine Whether allow combine for current package modules.
  97. * @param {String} [configName.packages.ignorePackageNameInUri=false] whether remove packageName from module request uri,
  98. * can only be used in production mode.
  99. * @param {Array[]} configName.map file map File url map configs.
  100. * @param {Array[]} configName.map.0 A single map rule.
  101. * @param {RegExp} configName.map.0.0 A regular expression to match url.
  102. * @param {String|Function} configName.map.0.1 Replacement for String.replace.
  103. * @param [configValue] config value.
  104. *
  105. * for example:
  106. * @example
  107. * KISSY.config({
  108. * combine: true,
  109. * base: '',
  110. * packages: {
  111. * 'gallery': {
  112. * base: 'http://a.tbcdn.cn/s/kissy/gallery/'
  113. * }
  114. * },
  115. * modules: {
  116. * 'gallery/x/y': {
  117. * requires: ['gallery/x/z']
  118. * }
  119. * }
  120. * });
  121. */
  122. config: function (configName, configValue) {
  123. var cfg,
  124. r,
  125. self = this,
  126. fn,
  127. Config = S.Config,
  128. configFns = Config.fns;
  129. if (S.isObject(configName)) {
  130. S.each(configName, function (configValue, p) {
  131. fn = configFns[p];
  132. if (fn) {
  133. fn.call(self, configValue);
  134. } else {
  135. Config[p] = configValue;
  136. }
  137. });
  138. } else {
  139. cfg = configFns[configName];
  140. if (configValue === undefined) {
  141. if (cfg) {
  142. r = cfg.call(self);
  143. } else {
  144. r = Config[configName];
  145. }
  146. } else {
  147. if (cfg) {
  148. r = cfg.call(self, configValue);
  149. } else {
  150. Config[configName] = configValue;
  151. }
  152. }
  153. }
  154. return r;
  155. },
  156. /**
  157. * Prints debug info.
  158. * @param msg {String} the message to log.
  159. * @param {String} [cat] the log category for the message. Default
  160. * categories are 'info', 'warn', 'error', 'time' etc.
  161. * @param {String} [logger] the logger of the the message (opt)
  162. */
  163. log: function (msg, cat, logger) {
  164. if ('@DEBUG@') {
  165. var matched = 1;
  166. if (logger) {
  167. var loggerCfg = S.Config.logger || {},
  168. list, i, l, level, minLevel, maxLevel, reg;
  169. cat = cat || 'debug';
  170. level = loggerLevel[cat] || loggerLevel['debug'];
  171. if (list = loggerCfg.includes) {
  172. matched = 0;
  173. for (i = 0; i < list.length; i++) {
  174. l = list[i];
  175. reg = l.logger;
  176. maxLevel = loggerLevel[l.maxLevel] || loggerLevel['error'];
  177. minLevel = loggerLevel[l.minLevel] || loggerLevel['debug'];
  178. if (minLevel <= level && maxLevel >= level && logger.match(reg)) {
  179. matched = 1;
  180. break;
  181. }
  182. }
  183. } else if (list = loggerCfg.excludes) {
  184. matched = 1;
  185. for (i = 0; i < list.length; i++) {
  186. l = list[i];
  187. reg = l.logger;
  188. maxLevel = loggerLevel[l.maxLevel] || loggerLevel['error'];
  189. minLevel = loggerLevel[l.minLevel] || loggerLevel['debug'];
  190. if (minLevel <= level && maxLevel >= level && logger.match(reg)) {
  191. matched = 0;
  192. break;
  193. }
  194. }
  195. }
  196. if (matched) {
  197. msg = logger + ': ' + msg;
  198. }
  199. }
  200. if (host['console'] !== undefined && console.log && matched) {
  201. console[cat && console[cat] ? cat : 'log'](msg);
  202. return msg;
  203. }
  204. }
  205. return undefined;
  206. },
  207. /**
  208. * get log instance for specified logger
  209. * @param {String} logger logger name
  210. * @returns {KISSY.Logger} log instance
  211. */
  212. 'getLogger': function (logger) {
  213. return getLogger(logger);
  214. },
  215. /**
  216. * Throws error message.
  217. */
  218. error: function (msg) {
  219. if ('@DEBUG@') {
  220. // with stack info!
  221. throw msg instanceof Error ? msg : new Error(msg);
  222. }
  223. },
  224. /*
  225. * Generate a global unique id.
  226. * @param {String} [pre] guid prefix
  227. * @return {String} the guid
  228. */
  229. guid: function (pre) {
  230. return (pre || EMPTY) + guid++;
  231. }
  232. };
  233. if ('@DEBUG@') {
  234. S.Config.logger = {
  235. excludes: [
  236. {
  237. logger: /^s\/.*/,
  238. maxLevel: 'info',
  239. minLevel: 'debug'
  240. }
  241. ]
  242. };
  243. /**
  244. * Log class for specified logger
  245. * @class KISSY.Logger
  246. * @private
  247. */
  248. function Logger() {
  249. }
  250. /**
  251. * print debug log
  252. * @method
  253. * @param {String} str log str
  254. */
  255. Logger.prototype.debug = function (str) {
  256. };
  257. /**
  258. * print info log
  259. * @method
  260. * @param {String} str log str
  261. */
  262. Logger.prototype.info = function (str) {
  263. };
  264. /**
  265. * print warn log
  266. * @method
  267. * @param {String} str log str
  268. */
  269. Logger.prototype.warn = function (str) {
  270. };
  271. /**
  272. * print error log
  273. * @method
  274. * @param {String} str log str
  275. */
  276. Logger.prototype.error = function (str) {
  277. };
  278. }
  279. function getLogger(logger) {
  280. var obj = {};
  281. S.each(loggerLevel, function (_, cat) {
  282. obj[cat] = function (msg) {
  283. return S.log(msg, cat, logger);
  284. };
  285. });
  286. return obj;
  287. }
  288. /**
  289. * Logger level enum
  290. * @enum {String} KISSY.Logger.Level
  291. */
  292. S.Logger = /**@type Function
  293. @ignore */{};
  294. S.Logger.Level = {
  295. /**
  296. * debug level
  297. */
  298. 'DEBUG': 'debug',
  299. /**
  300. * info level
  301. */
  302. INFO: 'info',
  303. /**
  304. * warn level
  305. */
  306. WARN: 'warn',
  307. /**
  308. * error level
  309. */
  310. ERROR: 'error'
  311. };
  312. return S;
  313. })();/**
  314. * @ignore
  315. * object utilities of lang
  316. * @author yiminghe@gmail.com
  317. *
  318. */
  319. (function (S, undefined) {
  320. var MIX_CIRCULAR_DETECTION = '__MIX_CIRCULAR',
  321. STAMP_MARKER = '__~ks_stamped',
  322. host = this,
  323. TRUE = true,
  324. EMPTY = '',
  325. Obj = Object,
  326. ObjectCreate = Obj.create,
  327. // error in native ie678, not in simulated ie9
  328. hasEnumBug = !({toString: 1}['propertyIsEnumerable']('toString')),
  329. enumProperties = [
  330. 'constructor',
  331. 'hasOwnProperty',
  332. 'isPrototypeOf',
  333. 'propertyIsEnumerable',
  334. 'toString',
  335. 'toLocaleString',
  336. 'valueOf'
  337. ];
  338. mix(S, {
  339. /**
  340. * stamp a object by guid
  341. * @param {Object} o object needed to be stamped
  342. * @param {Boolean} [readOnly] while set marker on o if marker does not exist
  343. * @param {String} [marker] the marker will be set on Object
  344. * @return {String} guid associated with this object
  345. * @member KISSY
  346. */
  347. stamp: function (o, readOnly, marker) {
  348. marker = marker || STAMP_MARKER;
  349. var guid = o[marker];
  350. if (guid) {
  351. return guid;
  352. } else if (!readOnly) {
  353. try {
  354. guid = o[marker] = S.guid(marker);
  355. }
  356. catch (e) {
  357. guid = undefined;
  358. }
  359. }
  360. return guid;
  361. },
  362. /**
  363. * Get all the property names of o as array
  364. * @param {Object} o
  365. * @return {Array}
  366. * @member KISSY
  367. */
  368. keys: Obj.keys || function (o) {
  369. var result = [], p, i;
  370. for (p in o) {
  371. // S.keys(new XX())
  372. if (o.hasOwnProperty(p)) {
  373. result.push(p);
  374. }
  375. }
  376. if (hasEnumBug) {
  377. for (i = enumProperties.length - 1; i >= 0; i--) {
  378. p = enumProperties[i];
  379. if (o.hasOwnProperty(p)) {
  380. result.push(p);
  381. }
  382. }
  383. }
  384. return result;
  385. },
  386. /**
  387. * Copies all the properties of s to r.
  388. * @method
  389. * @param {Object} r the augmented object
  390. * @param {Object} s the object need to augment
  391. * @param {Boolean|Object} [ov=TRUE] whether overwrite existing property or config.
  392. * @param {Boolean} [ov.overwrite=TRUE] whether overwrite existing property.
  393. * @param {String[]|Function} [ov.whitelist] array of white-list properties
  394. * @param {Boolean}[ov.deep=false] whether recursive mix if encounter object.
  395. * @param {String[]|Function} [wl] array of white-list properties
  396. * @param [deep=false] {Boolean} whether recursive mix if encounter object.
  397. * @return {Object} the augmented object
  398. * @member KISSY
  399. *
  400. * for example:
  401. * @example
  402. * var t = {};
  403. * S.mix({x: {y: 2, z: 4}}, {x: {y: 3, a: t}}, {deep: TRUE}) => {x: {y: 3, z: 4, a: {}}}, a !== t
  404. * S.mix({x: {y: 2, z: 4}}, {x: {y: 3, a: t}}, {deep: TRUE, overwrite: false}) => {x: {y: 2, z: 4, a: {}}}, a !== t
  405. * S.mix({x: {y: 2, z: 4}}, {x: {y: 3, a: t}}, 1) => {x: {y: 3, a: t}}
  406. */
  407. mix: function (r, s, ov, wl, deep) {
  408. if (typeof ov === 'object') {
  409. wl = /**
  410. @ignore
  411. @type {String[]|Function}
  412. */ov['whitelist'];
  413. deep = ov['deep'];
  414. ov = ov['overwrite'];
  415. }
  416. if (wl && (typeof wl !== 'function')) {
  417. var originalWl = wl;
  418. wl = function (name, val) {
  419. return S.inArray(name, originalWl) ? val : undefined;
  420. };
  421. }
  422. if (ov === undefined) {
  423. ov = TRUE;
  424. }
  425. var cache = [],
  426. c,
  427. i = 0;
  428. mixInternal(r, s, ov, wl, deep, cache);
  429. while (c = cache[i++]) {
  430. delete c[MIX_CIRCULAR_DETECTION];
  431. }
  432. return r;
  433. },
  434. /**
  435. * Returns a new object containing all of the properties of
  436. * all the supplied objects. The properties from later objects
  437. * will overwrite those in earlier objects. Passing in a
  438. * single object will create a shallow copy of it.
  439. * @param {...Object} var_args objects need to be merged
  440. * @return {Object} the new merged object
  441. * @member KISSY
  442. */
  443. merge: function (var_args) {
  444. var_args = S.makeArray(arguments);
  445. var o = {},
  446. i,
  447. l = var_args.length;
  448. for (i = 0; i < l; i++) {
  449. S.mix(o, var_args[i]);
  450. }
  451. return o;
  452. },
  453. /**
  454. * Applies prototype properties from the supplier to the receiver.
  455. * @param {Object} r received object
  456. * @param {...Object} var_args object need to augment
  457. * {Boolean} [ov=TRUE] whether overwrite existing property
  458. * {String[]} [wl] array of white-list properties
  459. * @return {Object} the augmented object
  460. * @member KISSY
  461. */
  462. augment: function (r, var_args) {
  463. var args = S.makeArray(arguments),
  464. len = args.length - 2,
  465. i = 1,
  466. proto,
  467. arg,
  468. ov = args[len],
  469. wl = args[len + 1];
  470. if (!S.isArray(wl)) {
  471. ov = wl;
  472. wl = undefined;
  473. len++;
  474. }
  475. if (typeof ov !== 'boolean') {
  476. ov = undefined;
  477. len++;
  478. }
  479. for (; i < len; i++) {
  480. arg = args[i];
  481. if (proto = arg.prototype) {
  482. arg = S.mix({}, proto, true, removeConstructor);
  483. }
  484. S.mix(r.prototype, arg, ov, wl);
  485. }
  486. return r;
  487. },
  488. /**
  489. * Utility to set up the prototype, constructor and superclass properties to
  490. * support an inheritance strategy that can chain constructors and methods.
  491. * Static members will not be inherited.
  492. * @param r {Function} the object to modify
  493. * @param s {Function} the object to inherit
  494. * @param {Object} [px] prototype properties to add/override
  495. * @param {Object} [sx] static properties to add/override
  496. * @return r {Object}
  497. * @member KISSY
  498. */
  499. extend: function (r, s, px, sx) {
  500. if (!s || !r) {
  501. return r;
  502. }
  503. var sp = s.prototype,
  504. rp;
  505. // in case parent does not set constructor
  506. // eg: parent.prototype={};
  507. sp.constructor = s;
  508. // add prototype chain
  509. rp = createObject(sp, r);
  510. r.prototype = S.mix(rp, r.prototype);
  511. r.superclass = sp;
  512. // add prototype overrides
  513. if (px) {
  514. S.mix(rp, px);
  515. }
  516. // add object overrides
  517. if (sx) {
  518. S.mix(r, sx);
  519. }
  520. return r;
  521. },
  522. /**
  523. * Returns the namespace specified and creates it if it doesn't exist. Be careful
  524. * when naming packages. Reserved words may work in some browsers and not others.
  525. *
  526. * for example:
  527. * @example
  528. * S.namespace('KISSY.app'); // returns KISSY.app
  529. * S.namespace('app.Shop'); // returns KISSY.app.Shop
  530. * S.namespace('TB.app.Shop', TRUE); // returns TB.app.Shop
  531. *
  532. * @return {Object} A reference to the last namespace object created
  533. * @member KISSY
  534. */
  535. namespace: function () {
  536. var args = S.makeArray(arguments),
  537. l = args.length,
  538. o = null, i, j, p,
  539. global = (args[l - 1] === TRUE && l--);
  540. for (i = 0; i < l; i++) {
  541. p = (EMPTY + args[i]).split('.');
  542. o = global ? host : this;
  543. for (j = (host[p[0]] === o) ? 1 : 0; j < p.length; ++j) {
  544. o = o[p[j]] = o[p[j]] || { };
  545. }
  546. }
  547. return o;
  548. }
  549. });
  550. function Empty() {
  551. }
  552. function createObject(proto, constructor) {
  553. var newProto;
  554. if (ObjectCreate) {
  555. newProto = ObjectCreate(proto);
  556. } else {
  557. Empty.prototype = proto;
  558. newProto = new Empty();
  559. }
  560. newProto.constructor = constructor;
  561. return newProto;
  562. }
  563. function mix(r, s) {
  564. for (var i in s) {
  565. r[i] = s[i];
  566. }
  567. }
  568. function mixInternal(r, s, ov, wl, deep, cache) {
  569. if (!s || !r) {
  570. return r;
  571. }
  572. var i, p, keys, len;
  573. // 记录循环标志
  574. s[MIX_CIRCULAR_DETECTION] = r;
  575. // 记录被记录了循环标志的对像
  576. cache.push(s);
  577. // mix all properties
  578. keys = S.keys(s);
  579. len = keys.length;
  580. for (i = 0; i < len; i++) {
  581. p = keys[i];
  582. if (p != MIX_CIRCULAR_DETECTION) {
  583. // no hasOwnProperty judge!
  584. _mix(p, r, s, ov, wl, deep, cache);
  585. }
  586. }
  587. return r;
  588. }
  589. function removeConstructor(k, v) {
  590. return k == 'constructor' ? undefined : v;
  591. }
  592. function _mix(p, r, s, ov, wl, deep, cache) {
  593. // 要求覆盖
  594. // 或者目的不存在
  595. // 或者深度mix
  596. if (ov || !(p in r) || deep) {
  597. var target = r[p],
  598. src = s[p];
  599. // prevent never-end loop
  600. if (target === src) {
  601. // S.mix({},{x:undefined})
  602. if (target === undefined) {
  603. r[p] = target;
  604. }
  605. return;
  606. }
  607. if (wl) {
  608. src = wl.call(s, p, src);
  609. }
  610. // 来源是数组和对象,并且要求深度 mix
  611. if (deep && src && (S.isArray(src) || S.isPlainObject(src))) {
  612. if (src[MIX_CIRCULAR_DETECTION]) {
  613. r[p] = src[MIX_CIRCULAR_DETECTION];
  614. } else {
  615. // 目标值为对象或数组,直接 mix
  616. // 否则 新建一个和源值类型一样的空数组/对象,递归 mix
  617. var clone = target && (S.isArray(target) || S.isPlainObject(target)) ?
  618. target :
  619. (S.isArray(src) ? [] : {});
  620. r[p] = clone;
  621. mixInternal(clone, src, ov, wl, TRUE, cache);
  622. }
  623. } else if (src !== undefined && (ov || !(p in r))) {
  624. r[p] = src;
  625. }
  626. }
  627. }
  628. })(KISSY);/**
  629. * @ignore
  630. * array utilities of lang
  631. * @author yiminghe@gmail.com
  632. *
  633. */
  634. (function (S, undefined) {
  635. var TRUE = true,
  636. AP = Array.prototype,
  637. indexOf = AP.indexOf,
  638. lastIndexOf = AP.lastIndexOf,
  639. filter = AP.filter,
  640. every = AP.every,
  641. some = AP.some,
  642. map = AP.map,
  643. FALSE = false;
  644. S.mix(S, {
  645. /**
  646. * Executes the supplied function on each item in the array.
  647. * @param object {Object} the object to iterate
  648. * @param fn {Function} the function to execute on each item. The function
  649. * receives three arguments: the value, the index, the full array.
  650. * @param {Object} [context]
  651. * @member KISSY
  652. */
  653. each: function (object, fn, context) {
  654. if (object) {
  655. var key,
  656. val,
  657. keys,
  658. i = 0,
  659. length = object && object.length,
  660. // do not use typeof obj == 'function': bug in phantomjs
  661. isObj = length === undefined || S.type(object) == 'function';
  662. context = context || null;
  663. if (isObj) {
  664. keys = S.keys(object);
  665. for (; i < keys.length; i++) {
  666. key = keys[i];
  667. // can not use hasOwnProperty
  668. if (fn.call(context, object[key], key, object) === FALSE) {
  669. break;
  670. }
  671. }
  672. } else {
  673. for (val = object[0];
  674. i < length; val = object[++i]) {
  675. if (fn.call(context, val, i, object) === FALSE) {
  676. break;
  677. }
  678. }
  679. }
  680. }
  681. return object;
  682. },
  683. /**
  684. * Search for a specified value within an array.
  685. * @param item individual item to be searched
  686. * @method
  687. * @member KISSY
  688. * @param {Array} arr the array of items where item will be search
  689. * @return {number} item's index in array
  690. */
  691. indexOf: indexOf ?
  692. function (item, arr) {
  693. return indexOf.call(arr, item);
  694. } :
  695. function (item, arr) {
  696. for (var i = 0, len = arr.length; i < len; ++i) {
  697. if (arr[i] === item) {
  698. return i;
  699. }
  700. }
  701. return -1;
  702. },
  703. /**
  704. * Returns the index of the last item in the array
  705. * that contains the specified value, -1 if the
  706. * value isn't found.
  707. * @method
  708. * @param item individual item to be searched
  709. * @param {Array} arr the array of items where item will be search
  710. * @return {number} item's last index in array
  711. * @member KISSY
  712. */
  713. lastIndexOf: (lastIndexOf) ?
  714. function (item, arr) {
  715. return lastIndexOf.call(arr, item);
  716. } :
  717. function (item, arr) {
  718. for (var i = arr.length - 1; i >= 0; i--) {
  719. if (arr[i] === item) {
  720. break;
  721. }
  722. }
  723. return i;
  724. },
  725. /**
  726. * Returns a copy of the array with the duplicate entries removed
  727. * @param a {Array} the array to find the subset of unique for
  728. * @param [override] {Boolean} if override is TRUE, S.unique([a, b, a]) => [b, a].
  729. * if override is FALSE, S.unique([a, b, a]) => [a, b]
  730. * @return {Array} a copy of the array with duplicate entries removed
  731. * @member KISSY
  732. */
  733. unique: function (a, override) {
  734. var b = a.slice();
  735. if (override) {
  736. b.reverse();
  737. }
  738. var i = 0,
  739. n,
  740. item;
  741. while (i < b.length) {
  742. item = b[i];
  743. while ((n = S.lastIndexOf(item, b)) !== i) {
  744. b.splice(n, 1);
  745. }
  746. i += 1;
  747. }
  748. if (override) {
  749. b.reverse();
  750. }
  751. return b;
  752. },
  753. /**
  754. * Search for a specified value index within an array.
  755. * @param item individual item to be searched
  756. * @param {Array} arr the array of items where item will be search
  757. * @return {Boolean} the item exists in arr
  758. * @member KISSY
  759. */
  760. inArray: function (item, arr) {
  761. return S.indexOf(item, arr) > -1;
  762. },
  763. /**
  764. * Executes the supplied function on each item in the array.
  765. * Returns a new array containing the items that the supplied
  766. * function returned TRUE for.
  767. * @member KISSY
  768. * @method
  769. * @param arr {Array} the array to iterate
  770. * @param fn {Function} the function to execute on each item
  771. * @param [context] {Object} optional context object
  772. * @return {Array} The items on which the supplied function returned TRUE.
  773. * If no items matched an empty array is returned.
  774. */
  775. filter: filter ?
  776. function (arr, fn, context) {
  777. return filter.call(arr, fn, context || this);
  778. } :
  779. function (arr, fn, context) {
  780. var ret = [];
  781. S.each(arr, function (item, i, arr) {
  782. if (fn.call(context || this, item, i, arr)) {
  783. ret.push(item);
  784. }
  785. });
  786. return ret;
  787. },
  788. /**
  789. * Executes the supplied function on each item in the array.
  790. * Returns a new array containing the items that the supplied
  791. * function returned for.
  792. * @method
  793. * @param arr {Array} the array to iterate
  794. * @param fn {Function} the function to execute on each item
  795. * @param [context] {Object} optional context object
  796. * refer: https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Array/map
  797. * @return {Array} The items on which the supplied function returned
  798. * @member KISSY
  799. */
  800. map: map ?
  801. function (arr, fn, context) {
  802. return map.call(arr, fn, context || this);
  803. } :
  804. function (arr, fn, context) {
  805. var len = arr.length,
  806. res = new Array(len);
  807. for (var i = 0; i < len; i++) {
  808. var el = typeof arr == 'string' ? arr.charAt(i) : arr[i];
  809. if (el
  810. ||
  811. //ie<9 in invalid when typeof arr == string
  812. i in arr) {
  813. res[i] = fn.call(context || this, el, i, arr);
  814. }
  815. }
  816. return res;
  817. },
  818. /**
  819. * Executes the supplied function on each item in the array.
  820. * Returns a value which is accumulation of the value that the supplied
  821. * function returned.
  822. *
  823. * @param arr {Array} the array to iterate
  824. * @param callback {Function} the function to execute on each item
  825. * @param initialValue {number} optional context object
  826. * refer: https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/array/reduce
  827. * @return {Array} The items on which the supplied function returned
  828. * @member KISSY
  829. */
  830. reduce: /*
  831. NaN ?
  832. reduce ? function(arr, callback, initialValue) {
  833. return arr.reduce(callback, initialValue);
  834. } : */function (arr, callback, initialValue) {
  835. var len = arr.length;
  836. if (typeof callback !== 'function') {
  837. throw new TypeError('callback is not function!');
  838. }
  839. // no value to return if no initial value and an empty array
  840. if (len === 0 && arguments.length == 2) {
  841. throw new TypeError('arguments invalid');
  842. }
  843. var k = 0;
  844. var accumulator;
  845. if (arguments.length >= 3) {
  846. accumulator = arguments[2];
  847. }
  848. else {
  849. do {
  850. if (k in arr) {
  851. accumulator = arr[k++];
  852. break;
  853. }
  854. // if array contains no values, no initial value to return
  855. k += 1;
  856. if (k >= len) {
  857. throw new TypeError();
  858. }
  859. }
  860. while (TRUE);
  861. }
  862. while (k < len) {
  863. if (k in arr) {
  864. accumulator = callback.call(undefined, accumulator, arr[k], k, arr);
  865. }
  866. k++;
  867. }
  868. return accumulator;
  869. },
  870. /**
  871. * Tests whether all elements in the array pass the test implemented by the provided function.
  872. * @method
  873. * @param arr {Array} the array to iterate
  874. * @param callback {Function} the function to execute on each item
  875. * @param [context] {Object} optional context object
  876. * @member KISSY
  877. * @return {Boolean} whether all elements in the array pass the test implemented by the provided function.
  878. */
  879. every: every ?
  880. function (arr, fn, context) {
  881. return every.call(arr, fn, context || this);
  882. } :
  883. function (arr, fn, context) {
  884. var len = arr && arr.length || 0;
  885. for (var i = 0; i < len; i++) {
  886. if (i in arr && !fn.call(context, arr[i], i, arr)) {
  887. return FALSE;
  888. }
  889. }
  890. return TRUE;
  891. },
  892. /**
  893. * Tests whether some element in the array passes the test implemented by the provided function.
  894. * @method
  895. * @param arr {Array} the array to iterate
  896. * @param callback {Function} the function to execute on each item
  897. * @param [context] {Object} optional context object
  898. * @member KISSY
  899. * @return {Boolean} whether some element in the array passes the test implemented by the provided function.
  900. */
  901. some: some ?
  902. function (arr, fn, context) {
  903. return some.call(arr, fn, context || this);
  904. } :
  905. function (arr, fn, context) {
  906. var len = arr && arr.length || 0;
  907. for (var i = 0; i < len; i++) {
  908. if (i in arr && fn.call(context, arr[i], i, arr)) {
  909. return TRUE;
  910. }
  911. }
  912. return FALSE;
  913. },
  914. /**
  915. * Converts object to a TRUE array.
  916. * @param o {object|Array} array like object or array
  917. * @return {Array} native Array
  918. * @member KISSY
  919. */
  920. makeArray: function (o) {
  921. if (o == null) {
  922. return [];
  923. }
  924. if (S.isArray(o)) {
  925. return o;
  926. }
  927. var lengthType = typeof o.length,
  928. oType = typeof o;
  929. // The strings and functions also have 'length'
  930. if (lengthType != 'number' ||
  931. // form.elements in ie78 has nodeName 'form'
  932. // then caution select
  933. // o.nodeName
  934. // window
  935. o.alert ||
  936. oType == 'string' ||
  937. // https://github.com/ariya/phantomjs/issues/11478
  938. (oType == 'function' && !( 'item' in o && lengthType == 'number'))) {
  939. return [o];
  940. }
  941. var ret = [];
  942. for (var i = 0, l = o.length; i < l; i++) {
  943. ret[i] = o[i];
  944. }
  945. return ret;
  946. }
  947. });
  948. })(KISSY);/**
  949. * @ignore
  950. * escape of lang
  951. * @author yiminghe@gmail.com
  952. *
  953. */
  954. (function (S, undefined) {
  955. // IE doesn't include non-breaking-space (0xa0) in their \s character
  956. // class (as required by section 7.2 of the ECMAScript spec), we explicitly
  957. // include it in the regexp to enforce consistent cross-browser behavior.
  958. var SEP = '&',
  959. EMPTY = '',
  960. EQ = '=',
  961. logger= S.getLogger('s/lang'),
  962. TRUE = true,
  963. // FALSE = false,
  964. HEX_BASE = 16,
  965. // http://www.owasp.org/index.php/XSS_(Cross_Site_Scripting)_Prevention_Cheat_Sheet
  966. // http://wonko.com/post/html-escaping
  967. htmlEntities = {
  968. '&amp;': '&',
  969. '&gt;': '>',
  970. '&lt;': '<',
  971. '&#x60;': '`',
  972. '&#x2F;': '/',
  973. '&quot;': '"',
  974. '&#x27;': "'"
  975. },
  976. reverseEntities = {},
  977. escapeReg,
  978. unEscapeReg,
  979. // - # $ ^ * ( ) + [ ] { } | \ , . ?
  980. escapeRegExp = /[\-#$\^*()+\[\]{}|\\,.?\s]/g;
  981. (function () {
  982. for (var k in htmlEntities) {
  983. reverseEntities[htmlEntities[k]] = k;
  984. }
  985. })();
  986. function isValidParamValue(val) {
  987. var t = typeof val;
  988. // If the type of val is null, undefined, number, string, boolean, return TRUE.
  989. return val == null || (t !== 'object' && t !== 'function');
  990. }
  991. function getEscapeReg() {
  992. if (escapeReg) {
  993. return escapeReg
  994. }
  995. var str = EMPTY;
  996. S.each(htmlEntities, function (entity) {
  997. str += entity + '|';
  998. });
  999. str = str.slice(0, -1);
  1000. return escapeReg = new RegExp(str, 'g');
  1001. }
  1002. function getUnEscapeReg() {
  1003. if (unEscapeReg) {
  1004. return unEscapeReg
  1005. }
  1006. var str = EMPTY;
  1007. S.each(reverseEntities, function (entity) {
  1008. str += entity + '|';
  1009. });
  1010. str += '&#(\\d{1,5});';
  1011. return unEscapeReg = new RegExp(str, 'g');
  1012. }
  1013. S.mix(S, {
  1014. /**
  1015. * Call encodeURIComponent to encode a url component
  1016. * @param {String} s part of url to be encoded.
  1017. * @return {String} encoded url part string.
  1018. * @member KISSY
  1019. */
  1020. urlEncode: function (s) {
  1021. return encodeURIComponent(String(s));
  1022. },
  1023. /**
  1024. * Call decodeURIComponent to decode a url component
  1025. * and replace '+' with space.
  1026. * @param {String} s part of url to be decoded.
  1027. * @return {String} decoded url part string.
  1028. * @member KISSY
  1029. */
  1030. urlDecode: function (s) {
  1031. return decodeURIComponent(s.replace(/\+/g, ' '));
  1032. },
  1033. /**
  1034. * frequently used in taobao cookie about nick
  1035. * @member KISSY
  1036. * @return {String} un-unicode string.
  1037. */
  1038. fromUnicode: function (str) {
  1039. return str.replace(/\\u([a-f\d]{4})/ig, function (m, u) {
  1040. return String.fromCharCode(parseInt(u, HEX_BASE));
  1041. });
  1042. },
  1043. /**
  1044. * get escaped string from html.
  1045. * only escape
  1046. * & > < ` / " '
  1047. * refer:
  1048. *
  1049. * [http://yiminghe.javaeye.com/blog/788929](http://yiminghe.javaeye.com/blog/788929)
  1050. *
  1051. * [http://wonko.com/post/html-escaping](http://wonko.com/post/html-escaping)
  1052. * @param str {string} text2html show
  1053. * @member KISSY
  1054. * @return {String} escaped html
  1055. */
  1056. escapeHtml: function (str) {
  1057. return (str + '').replace(getEscapeReg(), function (m) {
  1058. return reverseEntities[m];
  1059. });
  1060. },
  1061. /**
  1062. * get escaped regexp string for construct regexp.
  1063. * @param str
  1064. * @member KISSY
  1065. * @return {String} escaped regexp
  1066. */
  1067. escapeRegExp: function (str) {
  1068. return str.replace(escapeRegExp, '\\$&');
  1069. },
  1070. /**
  1071. * un-escape html to string.
  1072. * only unescape
  1073. * &amp; &lt; &gt; &#x60; &#x2F; &quot; &#x27; &#\d{1,5}
  1074. * @param str {string} html2text
  1075. * @member KISSY
  1076. * @return {String} un-escaped html
  1077. */
  1078. unEscapeHtml: function (str) {
  1079. return str.replace(getUnEscapeReg(), function (m, n) {
  1080. return htmlEntities[m] || String.fromCharCode(+n);
  1081. });
  1082. },
  1083. /**
  1084. * Creates a serialized string of an array or object.
  1085. *
  1086. * for example:
  1087. * @example
  1088. * {foo: 1, bar: 2} // -> 'foo=1&bar=2'
  1089. * {foo: 1, bar: [2, 3]} // -> 'foo=1&bar=2&bar=3'
  1090. * {foo: '', bar: 2} // -> 'foo=&bar=2'
  1091. * {foo: undefined, bar: 2} // -> 'foo=undefined&bar=2'
  1092. * {foo: TRUE, bar: 2} // -> 'foo=TRUE&bar=2'
  1093. *
  1094. * @param {Object} o json data
  1095. * @param {String} [sep='&'] separator between each pair of data
  1096. * @param {String} [eq='='] separator between key and value of data
  1097. * @param {Boolean} [serializeArray=true] whether add '[]' to array key of data
  1098. * @return {String}
  1099. * @member KISSY
  1100. */
  1101. param: function (o, sep, eq, serializeArray) {
  1102. sep = sep || SEP;
  1103. eq = eq || EQ;
  1104. if (serializeArray === undefined) {
  1105. serializeArray = TRUE;
  1106. }
  1107. var buf = [], key, i, v, len, val,
  1108. encode = S.urlEncode;
  1109. for (key in o) {
  1110. val = o[key];
  1111. key = encode(key);
  1112. // val is valid non-array value
  1113. if (isValidParamValue(val)) {
  1114. buf.push(key);
  1115. if (val !== undefined) {
  1116. buf.push(eq, encode(val + EMPTY));
  1117. }
  1118. buf.push(sep);
  1119. }
  1120. // val is not empty array
  1121. else if (S.isArray(val) && val.length) {
  1122. for (i = 0, len = val.length; i < len; ++i) {
  1123. v = val[i];
  1124. if (isValidParamValue(v)) {
  1125. buf.push(key, (serializeArray ? encode('[]') : EMPTY));
  1126. if (v !== undefined) {
  1127. buf.push(eq, encode(v + EMPTY));
  1128. }
  1129. buf.push(sep);
  1130. }
  1131. }
  1132. }
  1133. // ignore other cases, including empty array, Function, RegExp, Date etc.
  1134. }
  1135. buf.pop();
  1136. return buf.join(EMPTY);
  1137. },
  1138. /**
  1139. * Parses a URI-like query string and returns an object composed of parameter/value pairs.
  1140. *
  1141. * for example:
  1142. * @example
  1143. * 'section=blog&id=45' // -> {section: 'blog', id: '45'}
  1144. * 'section=blog&tag=js&tag=doc' // -> {section: 'blog', tag: ['js', 'doc']}
  1145. * 'tag=ruby%20on%20rails' // -> {tag: 'ruby on rails'}
  1146. * 'id=45&raw' // -> {id: '45', raw: ''}
  1147. * @param {String} str param string
  1148. * @param {String} [sep='&'] separator between each pair of data
  1149. * @param {String} [eq='='] separator between key and value of data
  1150. * @return {Object} json data
  1151. * @member KISSY
  1152. */
  1153. unparam: function (str, sep, eq) {
  1154. if (typeof str != 'string' || !(str = S.trim(str))) {
  1155. return {};
  1156. }
  1157. sep = sep || SEP;
  1158. eq = eq || EQ;
  1159. var ret = {},
  1160. eqIndex,
  1161. decode = S.urlDecode,
  1162. pairs = str.split(sep),
  1163. key, val,
  1164. i = 0, len = pairs.length;
  1165. for (; i < len; ++i) {
  1166. eqIndex = pairs[i].indexOf(eq);
  1167. if (eqIndex == -1) {
  1168. key = decode(pairs[i]);
  1169. val = undefined;
  1170. } else {
  1171. // remember to decode key!
  1172. key = decode(pairs[i].substring(0, eqIndex));
  1173. val = pairs[i].substring(eqIndex + 1);
  1174. try {
  1175. val = decode(val);
  1176. } catch (e) {
  1177. logger.error('decodeURIComponent error : ' + val);
  1178. logger.error(e);
  1179. }
  1180. if (S.endsWith(key, '[]')) {
  1181. key = key.substring(0, key.length - 2);
  1182. }
  1183. }
  1184. if (key in ret) {
  1185. if (S.isArray(ret[key])) {
  1186. ret[key].push(val);
  1187. } else {
  1188. ret[key] = [ret[key], val];
  1189. }
  1190. } else {
  1191. ret[key] = val;
  1192. }
  1193. }
  1194. return ret;
  1195. }
  1196. });
  1197. S.escapeHTML = S.escapeHtml;
  1198. S.unEscapeHTML = S.unEscapeHtml;
  1199. })(KISSY);/**
  1200. * @ignore
  1201. * function utilities of lang
  1202. * @author yiminghe@gmail.com
  1203. *
  1204. */
  1205. (function (S, undefined) {
  1206. // ios Function.prototype.bind === undefined
  1207. function bindFn(r, fn, obj) {
  1208. var slice = [].slice,
  1209. args = slice.call(arguments, 3),
  1210. fNOP = function () {
  1211. },
  1212. bound = function () {
  1213. var inArgs = slice.call(arguments);
  1214. return fn.apply(
  1215. this instanceof fNOP ? this : obj,
  1216. (r ? inArgs.concat(args) : args.concat(inArgs))
  1217. );
  1218. };
  1219. fNOP.prototype = fn.prototype;
  1220. bound.prototype = new fNOP();
  1221. return bound;
  1222. }
  1223. S.mix(S, {
  1224. /**
  1225. * empty function
  1226. * @member KISSY
  1227. */
  1228. noop: function () {
  1229. },
  1230. /**
  1231. * Creates a new function that, when called, itself calls this function in the context of the provided this value,
  1232. * with a given sequence of arguments preceding any provided when the new function was called.
  1233. * refer: https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Function/bind
  1234. * @param {Function} fn internal called function
  1235. * @param {Object} obj context in which fn runs
  1236. * @param {*...} var_args extra arguments
  1237. * @member KISSY
  1238. * @return {Function} new function with context and arguments
  1239. */
  1240. bind: bindFn(0, bindFn, null, 0),
  1241. /**
  1242. * Creates a new function that, when called, itself calls this function in the context of the provided this value,
  1243. * with a given sequence of arguments preceding any provided when the new function was called.
  1244. * refer: https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Function/bind
  1245. * @param {Function} fn internal called function
  1246. * @param {Object} obj context in which fn runs
  1247. * @param {*...} var_args extra arguments
  1248. * @member KISSY
  1249. * @return {Function} new function with context and arguments
  1250. */
  1251. rbind: bindFn(0, bindFn, null, 1),
  1252. /**
  1253. * Executes the supplied function in the context of the supplied
  1254. * object 'when' milliseconds later. Executes the function a
  1255. * single time unless periodic is set to true.
  1256. *
  1257. * @param fn {Function|String} the function to execute or the name of the method in
  1258. * the 'o' object to execute.
  1259. *
  1260. * @param [when=0] {Number} the number of milliseconds to wait until the fn is executed.
  1261. *
  1262. * @param {Boolean} [periodic] if true, executes continuously at supplied interval
  1263. * until canceled.
  1264. *
  1265. * @param {Object} [context] the context object.
  1266. *
  1267. * @param [data] that is provided to the function. This accepts either a single
  1268. * item or an array. If an array is provided, the function is executed with
  1269. * one parameter for each array item. If you need to pass a single array
  1270. * parameter, it needs to be wrapped in an array.
  1271. *
  1272. * @return {Object} a timer object. Call the cancel() method on this object to stop
  1273. * the timer.
  1274. *
  1275. * @member KISSY
  1276. */
  1277. later: function (fn, when, periodic, context, data) {
  1278. when = when || 0;
  1279. var m = fn,
  1280. d = S.makeArray(data),
  1281. f,
  1282. r;
  1283. if (typeof fn == 'string') {
  1284. m = context[fn];
  1285. }
  1286. if (!m) {
  1287. S.error('method undefined');
  1288. }
  1289. f = function () {
  1290. m.apply(context, d);
  1291. };
  1292. r = (periodic) ? setInterval(f, when) : setTimeout(f, when);
  1293. return {
  1294. id: r,
  1295. interval: periodic,
  1296. cancel: function () {
  1297. if (this.interval) {
  1298. clearInterval(r);
  1299. } else {
  1300. clearTimeout(r);
  1301. }
  1302. }
  1303. };
  1304. },
  1305. /**
  1306. * Throttles a call to a method based on the time between calls.
  1307. * @param {Function} fn The function call to throttle.
  1308. * @param {Object} [context] context fn to run
  1309. * @param {Number} [ms] The number of milliseconds to throttle the method call.
  1310. * Passing a -1 will disable the throttle. Defaults to 150.
  1311. * @return {Function} Returns a wrapped function that calls fn throttled.
  1312. * @member KISSY
  1313. */
  1314. throttle: function (fn, ms, context) {
  1315. ms = ms || 150;
  1316. if (ms === -1) {
  1317. return (function () {
  1318. fn.apply(context || this, arguments);
  1319. });
  1320. }
  1321. var last = S.now();
  1322. return (function () {
  1323. var now = S.now();
  1324. if (now - last > ms) {
  1325. last = now;
  1326. fn.apply(context || this, arguments);
  1327. }
  1328. });
  1329. },
  1330. /**
  1331. * buffers a call between a fixed time
  1332. * @param {Function} fn
  1333. * @param {Number} ms
  1334. * @param {Object} [context]
  1335. * @return {Function} Returns a wrapped function that calls fn buffered.
  1336. * @member KISSY
  1337. */
  1338. buffer: function (fn, ms, context) {
  1339. ms = ms || 150;
  1340. if (ms === -1) {
  1341. return function () {
  1342. fn.apply(context || this, arguments);
  1343. };
  1344. }
  1345. var bufferTimer = null;
  1346. function f() {
  1347. f.stop();
  1348. bufferTimer = S.later(fn, ms, 0, context || this, arguments);
  1349. }
  1350. f.stop = function () {
  1351. if (bufferTimer) {
  1352. bufferTimer.cancel();
  1353. bufferTimer = 0;
  1354. }
  1355. };
  1356. return f;
  1357. }
  1358. });
  1359. })(KISSY);/**
  1360. * @ignore
  1361. * lang
  1362. * @author yiminghe@gmail.com, lifesinger@gmail.com
  1363. *
  1364. */
  1365. (function (S, undefined) {
  1366. var TRUE = true,
  1367. FALSE = false,
  1368. CLONE_MARKER = '__~ks_cloned',
  1369. COMPARE_MARKER = '__~ks_compared';
  1370. S.mix(S, {
  1371. /**
  1372. * Checks to see whether two object are equals.
  1373. * @param a 比较目标1
  1374. * @param b 比较目标2
  1375. * @param [mismatchKeys] internal usage
  1376. * @param [mismatchValues] internal usage
  1377. * @return {Boolean} a.equals(b)
  1378. * @member KISSY
  1379. */
  1380. equals: function (a, b, /*internal use*/mismatchKeys, /*internal use*/mismatchValues) {
  1381. // inspired by jasmine
  1382. mismatchKeys = mismatchKeys || [];
  1383. mismatchValues = mismatchValues || [];
  1384. if (a === b) {
  1385. return TRUE;
  1386. }
  1387. if (a === undefined || a === null || b === undefined || b === null) {
  1388. // need type coercion
  1389. return a == null && b == null;
  1390. }
  1391. if (a instanceof Date && b instanceof Date) {
  1392. return a.getTime() == b.getTime();
  1393. }
  1394. if (typeof a == 'string' && typeof b == 'string') {
  1395. return (a == b);
  1396. }
  1397. if (typeof a==='number' && typeof b==='number') {
  1398. return (a == b);
  1399. }
  1400. if (typeof a === 'object' && typeof b === 'object') {
  1401. return compareObjects(a, b, mismatchKeys, mismatchValues);
  1402. }
  1403. // Straight check
  1404. return (a === b);
  1405. },
  1406. /**
  1407. * Creates a deep copy of a plain object or array. Others are returned untouched.
  1408. * @param input
  1409. * @member KISSY
  1410. * @param {Function} [filter] filter function
  1411. * @return {Object} the new cloned object
  1412. * refer: http://www.w3.org/TR/html5/common-dom-interfaces.html#safe-passing-of-structured-data
  1413. */
  1414. clone: function (input, filter) {
  1415. // 稍微改改就和规范一样了 :)
  1416. // Let memory be an association list of pairs of objects,
  1417. // initially empty. This is used to handle duplicate references.
  1418. // In each pair of objects, one is called the source object
  1419. // and the other the destination object.
  1420. var memory = {},
  1421. ret = cloneInternal(input, filter, memory);
  1422. S.each(memory, function (v) {
  1423. // 清理在源对象上做的标记
  1424. v = v.input;
  1425. if (v[CLONE_MARKER]) {
  1426. try {
  1427. delete v[CLONE_MARKER];
  1428. } catch (e) {
  1429. v[CLONE_MARKER] = undefined;
  1430. }
  1431. }
  1432. });
  1433. memory = null;
  1434. return ret;
  1435. },
  1436. /**
  1437. * Gets current date in milliseconds.
  1438. * @method
  1439. * refer: https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Date/now
  1440. * http://j-query.blogspot.com/2011/02/timing-ecmascript-5-datenow-function.html
  1441. * http://kangax.github.com/es5-compat-table/
  1442. * @member KISSY
  1443. * @return {Number} current time
  1444. */
  1445. now: Date.now || function () {
  1446. return +new Date();
  1447. }
  1448. });
  1449. function cloneInternal(input, f, memory) {
  1450. var destination = input,
  1451. isArray,
  1452. isPlainObject,
  1453. k,
  1454. stamp;
  1455. if (!input) {
  1456. return destination;
  1457. }
  1458. // If input is the source object of a pair of objects in memory,
  1459. // then return the destination object in that pair of objects .
  1460. // and abort these steps.
  1461. if (input[CLONE_MARKER]) {
  1462. // 对应的克隆后对象
  1463. return memory[input[CLONE_MARKER]].destination;
  1464. } else if (typeof input === 'object') {
  1465. // 引用类型要先记录
  1466. var constructor = input.constructor;
  1467. if (S.inArray(constructor, [Boolean, String, Number, Date, RegExp])) {
  1468. destination = new constructor(input.valueOf());
  1469. }
  1470. // ImageData , File, Blob , FileList .. etc
  1471. else if (isArray = S.isArray(input)) {
  1472. destination = f ? S.filter(input, f) : input.concat();
  1473. } else if (isPlainObject = S.isPlainObject(input)) {
  1474. destination = {};
  1475. }
  1476. // Add a mapping from input (the source object)
  1477. // to output (the destination object) to memory.
  1478. // 做标记
  1479. input[CLONE_MARKER] = (stamp = S.guid('c'));
  1480. // 存储源对象以及克隆后的对象
  1481. memory[stamp] = {destination: destination, input: input};
  1482. }
  1483. // If input is an Array object or an Object object,
  1484. // then, for each enumerable property in input,
  1485. // add a new property to output having the same name,
  1486. // and having a value created from invoking the internal structured cloning algorithm recursively
  1487. // with the value of the property as the 'input' argument and memory as the 'memory' argument.
  1488. // The order of the properties in the input and output objects must be the same.
  1489. // clone it
  1490. if (isArray) {
  1491. for (var i = 0; i < destination.length; i++) {
  1492. destination[i] = cloneInternal(destination[i], f, memory);
  1493. }
  1494. } else if (isPlainObject) {
  1495. for (k in input) {
  1496. if (k !== CLONE_MARKER &&
  1497. (!f || (f.call(input, input[k], k, input) !== FALSE))) {
  1498. destination[k] = cloneInternal(input[k], f, memory);
  1499. }
  1500. }
  1501. }
  1502. return destination;
  1503. }
  1504. function compareObjects(a, b, mismatchKeys, mismatchValues) {
  1505. // 两个比较过了,无需再比较,防止循环比较
  1506. if (a[COMPARE_MARKER] === b && b[COMPARE_MARKER] === a) {
  1507. return TRUE;
  1508. }
  1509. a[COMPARE_MARKER] = b;
  1510. b[COMPARE_MARKER] = a;
  1511. var hasKey = function (obj, keyName) {
  1512. return (obj !== null && obj !== undefined) && obj[keyName] !== undefined;
  1513. };
  1514. for (var property in b) {
  1515. if (!hasKey(a, property) && hasKey(b, property)) {
  1516. mismatchKeys.push("expected has key '" + property + "', but missing from actual.");
  1517. }
  1518. }
  1519. for (property in a) {
  1520. if (!hasKey(b, property) && hasKey(a, property)) {
  1521. mismatchKeys.push("expected missing key '" + property + "', but present in actual.");
  1522. }
  1523. }
  1524. for (property in b) {
  1525. if (property == COMPARE_MARKER) {
  1526. continue;
  1527. }
  1528. if (!S.equals(a[property], b[property], mismatchKeys, mismatchValues)) {
  1529. mismatchValues.push("'" + property + "' was '" + (b[property] ? (b[property].toString()) : b[property])
  1530. + "' in expected, but was '" +
  1531. (a[property] ? (a[property].toString()) : a[property]) + "' in actual.");
  1532. }
  1533. }
  1534. if (S.isArray(a) && S.isArray(b) && a.length != b.length) {
  1535. mismatchValues.push('arrays were not the same length');
  1536. }
  1537. delete a[COMPARE_MARKER];
  1538. delete b[COMPARE_MARKER];
  1539. return (mismatchKeys.length === 0 && mismatchValues.length === 0);
  1540. }
  1541. })(KISSY);
  1542. /**
  1543. * @ignore
  1544. * string utilities of lang
  1545. * @author yiminghe@gmail.com
  1546. *
  1547. */
  1548. (function (S, undefined) {
  1549. // IE doesn't include non-breaking-space (0xa0) in their \s character
  1550. // class (as required by section 7.2 of the ECMAScript spec), we explicitly
  1551. // include it in the regexp to enforce consistent cross-browser behavior.
  1552. var RE_TRIM = /^[\s\xa0]+|[\s\xa0]+$/g,
  1553. trim = String.prototype.trim,
  1554. SUBSTITUTE_REG = /\\?\{([^{}]+)\}/g,
  1555. EMPTY = '';
  1556. S.mix(S, {
  1557. /**
  1558. * Removes the whitespace from the beginning and end of a string.
  1559. * @method
  1560. * @member KISSY
  1561. */
  1562. trim: trim ?
  1563. function (str) {
  1564. return str == null ? EMPTY : trim.call(str);
  1565. } :
  1566. function (str) {
  1567. return str == null ? EMPTY : (str + '').replace(RE_TRIM, EMPTY);
  1568. },
  1569. /**
  1570. * Substitutes keywords in a string using an object/array.
  1571. * Removes undefined keywords and ignores escaped keywords.
  1572. * @param {String} str template string
  1573. * @param {Object} o json data
  1574. * @member KISSY
  1575. * @param {RegExp} [regexp] to match a piece of template string
  1576. */
  1577. substitute: function (str, o, regexp) {
  1578. if (typeof str != 'string' || !o) {
  1579. return str;
  1580. }
  1581. return str.replace(regexp || SUBSTITUTE_REG, function (match, name) {
  1582. if (match.charAt(0) === '\\') {
  1583. return match.slice(1);
  1584. }
  1585. return (o[name] === undefined) ? EMPTY : o[name];
  1586. });
  1587. },
  1588. /** uppercase first character.
  1589. * @member KISSY
  1590. * @param s
  1591. * @return {String}
  1592. */
  1593. ucfirst: function (s) {
  1594. s += '';
  1595. return s.charAt(0).toUpperCase() + s.substring(1);
  1596. },
  1597. /**
  1598. * test whether a string start with a specified substring
  1599. * @param {String} str the whole string
  1600. * @param {String} prefix a specified substring
  1601. * @return {Boolean} whether str start with prefix
  1602. * @member KISSY
  1603. */
  1604. startsWith: function (str, prefix) {
  1605. return str.lastIndexOf(prefix, 0) === 0;
  1606. },
  1607. /**
  1608. * test whether a string end with a specified substring
  1609. * @param {String} str the whole string
  1610. * @param {String} suffix a specified substring
  1611. * @return {Boolean} whether str end with suffix
  1612. * @member KISSY
  1613. */
  1614. endsWith: function (str, suffix) {
  1615. var ind = str.length - suffix.length;
  1616. return ind >= 0 && str.indexOf(suffix, ind) == ind;
  1617. }
  1618. });
  1619. })(KISSY);/**
  1620. * @ignore
  1621. * type judgement
  1622. * @author yiminghe@gmail.com, lifesinger@gmail.com
  1623. *
  1624. */
  1625. (function (S, undefined) {
  1626. // [[Class]] -> type pairs
  1627. var class2type = {},
  1628. FALSE = false,
  1629. noop = S.noop,
  1630. OP = Object.prototype,
  1631. toString = OP.toString;
  1632. function hasOwnProperty(o, p) {
  1633. return OP.hasOwnProperty.call(o, p);
  1634. }
  1635. S.mix(S, {
  1636. /**
  1637. * Determine the internal JavaScript [[Class]] of an object.
  1638. * @member KISSY
  1639. */
  1640. type: function (o) {
  1641. return o == null ?
  1642. String(o) :
  1643. class2type[toString.call(o)] || 'object';
  1644. },
  1645. /**
  1646. * whether o === null
  1647. * @param o
  1648. * @member KISSY
  1649. */
  1650. isNull: function (o) {
  1651. return o === null;
  1652. },
  1653. /**
  1654. * whether o === undefined
  1655. * @param o
  1656. * @member KISSY
  1657. */
  1658. isUndefined: function (o) {
  1659. return o === undefined;
  1660. },
  1661. /**
  1662. * Checks to see if an object is empty.
  1663. * @member KISSY
  1664. */
  1665. isEmptyObject: function (o) {
  1666. for (var p in o) {
  1667. if (p !== undefined) {
  1668. return FALSE;
  1669. }
  1670. }
  1671. return true;
  1672. },
  1673. /**
  1674. * Checks to see if an object is a plain object (created using '{}'
  1675. * or 'new Object()' but not 'new FunctionClass()').
  1676. * @member KISSY
  1677. */
  1678. isPlainObject: function (obj) {
  1679. // Must be an Object.
  1680. // Because of IE, we also have to check the presence of the constructor property.
  1681. // Make sure that Dom nodes and window objects don't pass through, as well
  1682. if (!obj || S.type(obj) !== "object" || obj.nodeType || obj.window == obj) {
  1683. return FALSE;
  1684. }
  1685. var key, objConstructor;
  1686. try {
  1687. // Not own constructor property must be Object
  1688. if ((objConstructor = obj.constructor) && !hasOwnProperty(obj, "constructor") && !hasOwnProperty(objConstructor.prototype, "isPrototypeOf")) {
  1689. return FALSE;
  1690. }
  1691. } catch (e) {
  1692. // IE8,9 Will throw exceptions on certain host objects
  1693. return FALSE;
  1694. }
  1695. // Own properties are enumerated firstly, so to speed up,
  1696. // if last one is own, then all properties are own.
  1697. for (key in obj) {
  1698. }
  1699. return key === undefined || hasOwnProperty(obj, key);
  1700. }
  1701. });
  1702. if ('@DEBUG@') {
  1703. S.mix(S, {
  1704. /**
  1705. * test whether o is boolean
  1706. * @method
  1707. * @param o
  1708. * @return {Boolean}
  1709. * @member KISSY
  1710. */
  1711. isBoolean: noop,
  1712. /**
  1713. * test whether o is number
  1714. * @method
  1715. * @param o
  1716. * @return {Boolean}
  1717. * @member KISSY
  1718. */
  1719. isNumber: noop,
  1720. /**
  1721. * test whether o is String
  1722. * @method
  1723. * @param o
  1724. * @return {Boolean}
  1725. * @member KISSY
  1726. */
  1727. isString: noop,
  1728. /**
  1729. * test whether o is function
  1730. * @method
  1731. * @param o
  1732. * @return {Boolean}
  1733. * @member KISSY
  1734. */
  1735. isFunction: noop,
  1736. /**
  1737. * test whether o is Array
  1738. * @method
  1739. * @param o
  1740. * @return {Boolean}
  1741. * @member KISSY
  1742. */
  1743. isArray: noop,
  1744. /**
  1745. * test whether o is Date
  1746. * @method
  1747. * @param o
  1748. * @return {Boolean}
  1749. * @member KISSY
  1750. */
  1751. isDate: noop,
  1752. /**
  1753. * test whether o is RegExp
  1754. * @method
  1755. * @param o
  1756. * @return {Boolean}
  1757. * @member KISSY
  1758. */
  1759. isRegExp: noop,
  1760. /**
  1761. * test whether o is Object
  1762. * @method
  1763. * @param o
  1764. * @return {Boolean}
  1765. * @member KISSY
  1766. */
  1767. isObject: noop
  1768. });
  1769. }
  1770. S.each('Boolean Number String Function Date RegExp Object Array'.split(' '), function (name, lc) {
  1771. // populate the class2type map
  1772. class2type['[object ' + name + ']'] = (lc = name.toLowerCase());
  1773. // add isBoolean/isNumber/...
  1774. S['is' + name] = function (o) {
  1775. return S.type(o) == lc;
  1776. }
  1777. });
  1778. S.isArray = Array.isArray || S.isArray;
  1779. })(KISSY);/*
  1780. setImmediate polyfill inspired by Q
  1781. @author yiminghe@gmail.com
  1782. */
  1783. (function (S) {
  1784. var queue = [];
  1785. var flushing = 0;
  1786. function flush() {
  1787. var i = 0, item;
  1788. while (item = queue[i++]) {
  1789. try {
  1790. item();
  1791. } catch (e) {
  1792. setTimeout(function () {
  1793. throw e;
  1794. }, 0);
  1795. }
  1796. }
  1797. if (i > 1) {
  1798. queue = [];
  1799. }
  1800. flushing = 0;
  1801. }
  1802. /*
  1803. setImmediate for loader and promise
  1804. @param {Function} fn async function to call
  1805. @private
  1806. */
  1807. S.setImmediate = function (fn) {
  1808. queue.push(fn);
  1809. if (!flushing) {
  1810. flushing = 1;
  1811. requestFlush();
  1812. }
  1813. };
  1814. var requestFlush;
  1815. if (typeof setImmediate === "function") {
  1816. requestFlush = function () {
  1817. setImmediate(flush);
  1818. };
  1819. } else if (typeof process !== 'undefined' && typeof process.nextTick == 'function') {
  1820. requestFlush = function () {
  1821. process.nextTick(flush);
  1822. };
  1823. } else if (typeof MessageChannel !== "undefined") {
  1824. // modern browsers
  1825. // http://msdn.microsoft.com/en-us/library/windows/apps/hh441303.aspx
  1826. var channel = new MessageChannel();
  1827. // At least Safari Version 6.0.5 (8536.30.1) intermittently cannot create
  1828. // working message ports the first time a page loads.
  1829. channel.port1.onmessage = function () {
  1830. requestFlush = realRequestFlush;
  1831. channel.port1.onmessage = flush;
  1832. flush();
  1833. };
  1834. var realRequestFlush = function () {
  1835. // Opera requires us to provide a message payload, regardless of
  1836. // whether we use it.
  1837. channel.port2.postMessage(0);
  1838. };
  1839. requestFlush = function () {
  1840. setTimeout(flush, 0);
  1841. realRequestFlush();
  1842. };
  1843. } else {
  1844. // old browsers
  1845. requestFlush = function () {
  1846. setTimeout(flush, 0);
  1847. };
  1848. }
  1849. })(KISSY);/**
  1850. * @ignore
  1851. * Port Node Utils For KISSY.
  1852. * Note: Only posix mode.
  1853. * @author yiminghe@gmail.com
  1854. */
  1855. (function (S) {
  1856. // [root, dir, basename, ext]
  1857. var splitPathRe = /^(\/?)([\s\S]+\/(?!$)|\/)?((?:\.{1,2}$|[\s\S]+?)?(\.[^.\/]*)?)$/;
  1858. // Remove .. and . in path array
  1859. function normalizeArray(parts, allowAboveRoot) {
  1860. // level above root
  1861. var up = 0,
  1862. i = parts.length - 1,
  1863. // splice costs a lot in ie
  1864. // use new array instead
  1865. newParts = [],
  1866. last;
  1867. for (; i >= 0; i--) {
  1868. last = parts[i];
  1869. if (last == '.') {
  1870. } else if (last === '..') {
  1871. up++;
  1872. } else if (up) {
  1873. up--;
  1874. } else {
  1875. newParts[newParts.length] = last;
  1876. }
  1877. }
  1878. // if allow above root, has to add ..
  1879. if (allowAboveRoot) {
  1880. for (; up--; up) {
  1881. newParts[newParts.length] = '..';
  1882. }
  1883. }
  1884. newParts = newParts.reverse();
  1885. return newParts;
  1886. }
  1887. /**
  1888. * Path Utils For KISSY.
  1889. * @class KISSY.Path
  1890. * @singleton
  1891. */
  1892. var Path = S.Path = {
  1893. /**
  1894. * resolve([from ...], to)
  1895. * @return {String} Resolved path.
  1896. */
  1897. resolve: function () {
  1898. var resolvedPath = '',
  1899. resolvedPathStr,
  1900. i,
  1901. args = (arguments),
  1902. path,
  1903. absolute = 0;
  1904. for (i = args.length - 1; i >= 0 && !absolute; i--) {
  1905. path = args[i];
  1906. if (typeof path != 'string' || !path) {
  1907. continue;
  1908. }
  1909. resolvedPath = path + '/' + resolvedPath;
  1910. absolute = path.charAt(0) == '/';
  1911. }
  1912. resolvedPathStr = normalizeArray(S.filter(resolvedPath.split('/'), function (p) {
  1913. return !!p;
  1914. }), !absolute).join('/');
  1915. return ((absolute ? '/' : '') + resolvedPathStr) || '.';
  1916. },
  1917. /**
  1918. * normalize .. and . in path
  1919. * @param {String} path Path tobe normalized
  1920. *
  1921. *
  1922. * 'x/y/../z' => 'x/z'
  1923. * 'x/y/z/../' => 'x/y/'
  1924. *
  1925. * @return {String}
  1926. */
  1927. normalize: function (path) {
  1928. var absolute = path.charAt(0) == '/',
  1929. trailingSlash = path.slice(-1) == '/';
  1930. path = normalizeArray(S.filter(path.split('/'), function (p) {
  1931. return !!p;
  1932. }), !absolute).join('/');
  1933. if (!path && !absolute) {
  1934. path = '.';
  1935. }
  1936. if (path && trailingSlash) {
  1937. path += '/';
  1938. }
  1939. return (absolute ? '/' : '') + path;
  1940. },
  1941. /**
  1942. * join([path ...]) and normalize
  1943. * @return {String}
  1944. */
  1945. join: function () {
  1946. var args = S.makeArray(arguments);
  1947. return Path.normalize(S.filter(args,function (p) {
  1948. return p && (typeof p == 'string');
  1949. }).join('/'));
  1950. },
  1951. /**
  1952. * Get string which is to relative to from
  1953. * @param {String} from
  1954. * @param {String} to
  1955. *
  1956. *
  1957. * relative('x/','x/y/z') => 'y/z'
  1958. * relative('x/t/z','x/') => '../../'
  1959. *
  1960. * @return {String}
  1961. */
  1962. relative: function (from, to) {
  1963. from = Path.normalize(from);
  1964. to = Path.normalize(to);
  1965. var fromParts = S.filter(from.split('/'), function (p) {
  1966. return !!p;
  1967. }),
  1968. path = [],
  1969. sameIndex,
  1970. sameIndex2,
  1971. toParts = S.filter(to.split('/'), function (p) {
  1972. return !!p;
  1973. }), commonLength = Math.min(fromParts.length, toParts.length);
  1974. for (sameIndex = 0; sameIndex < commonLength; sameIndex++) {
  1975. if (fromParts[sameIndex] != toParts[sameIndex]) {
  1976. break;
  1977. }
  1978. }
  1979. sameIndex2 = sameIndex;
  1980. while (sameIndex < fromParts.length) {
  1981. path.push('..');
  1982. sameIndex++;
  1983. }
  1984. path = path.concat(toParts.slice(sameIndex2));
  1985. path = path.join('/');
  1986. return /**@type String @ignore*/path;
  1987. },
  1988. /**
  1989. * Get base name of path
  1990. * @param {String} path
  1991. * @param {String} [ext] ext to be stripped from result returned.
  1992. * @return {String}
  1993. */
  1994. basename: function (path, ext) {
  1995. var result = path.match(splitPathRe) || [],
  1996. basename;
  1997. basename = result[3] || '';
  1998. if (ext && basename && basename.slice(-1 * ext.length) == ext) {
  1999. basename = basename.slice(0, -1 * ext.length);
  2000. }
  2001. return basename;
  2002. },
  2003. /**
  2004. * Get dirname of path
  2005. * @param {String} path
  2006. * @return {String}
  2007. */
  2008. dirname: function (path) {
  2009. var result = path.match(splitPathRe) || [],
  2010. root = result[1] || '',
  2011. dir = result[2] || '';
  2012. if (!root && !dir) {
  2013. // No dirname
  2014. return '.';
  2015. }
  2016. if (dir) {
  2017. // It has a dirname, strip trailing slash
  2018. dir = dir.substring(0, dir.length - 1);
  2019. }
  2020. return root + dir;
  2021. },
  2022. /**
  2023. * Get extension name of file in path
  2024. * @param {String} path
  2025. * @return {String}
  2026. */
  2027. extname: function (path) {
  2028. return (path.match(splitPathRe) || [])[4] || '';
  2029. }
  2030. };
  2031. })(KISSY);
  2032. /*
  2033. Refer
  2034. - https://github.com/joyent/node/blob/master/lib/path.js
  2035. *//**
  2036. * @ignore
  2037. * Uri class for KISSY.
  2038. * @author yiminghe@gmail.com
  2039. */
  2040. (function (S, undefined) {
  2041. var reDisallowedInSchemeOrUserInfo = /[#\/\?@]/g,
  2042. reDisallowedInPathName = /[#\?]/g,
  2043. logger= S.getLogger('s/uri'),
  2044. // ?? combo of taobao
  2045. reDisallowedInQuery = /[#@]/g,
  2046. reDisallowedInFragment = /#/g,
  2047. URI_SPLIT_REG = new RegExp(
  2048. '^' +
  2049. /*
  2050. Scheme names consist of a sequence of characters beginning with a
  2051. letter and followed by any combination of letters, digits, plus
  2052. ('+'), period ('.'), or hyphen ('-').
  2053. */
  2054. '(?:([\\w\\d+.-]+):)?' + // scheme
  2055. '(?://' +
  2056. /*
  2057. The authority component is preceded by a double slash ('//') and is
  2058. terminated by the next slash ('/'), question mark ('?'), or number
  2059. sign ('#') character, or by the end of the URI.
  2060. */
  2061. '(?:([^/?#@]*)@)?' + // userInfo
  2062. '(' +
  2063. '[\\w\\d\\-\\u0100-\\uffff.+%]*' +
  2064. '|' +
  2065. // ipv6
  2066. '\\[[^\\]]+\\]' +
  2067. ')' + // hostname - restrict to letters,
  2068. // digits, dashes, dots, percent
  2069. // escapes, and unicode characters.
  2070. '(?::([0-9]+))?' + // port
  2071. ')?' +
  2072. /*
  2073. The path is terminated
  2074. by the first question mark ('?') or number sign ('#') character, or
  2075. by the end of the URI.
  2076. */
  2077. '([^?#]+)?' + // path. hierarchical part
  2078. /*
  2079. The query component is indicated by the first question
  2080. mark ('?') character and terminated by a number sign ('#') character
  2081. or by the end of the URI.
  2082. */
  2083. '(?:\\?([^#]*))?' + // query. non-hierarchical data
  2084. /*
  2085. The fragment identifier component of a URI allows indirect
  2086. identification of a secondary resource by reference to a primary
  2087. resource and additional identifying information.
  2088. A
  2089. fragment identifier component is indicated by the presence of a
  2090. number sign ('#') character and terminated by the end of the URI.
  2091. */
  2092. '(?:#(.*))?' + // fragment
  2093. '$'),
  2094. Path = S.Path,
  2095. REG_INFO = {
  2096. scheme: 1,
  2097. userInfo: 2,
  2098. hostname: 3,
  2099. port: 4,
  2100. path: 5,
  2101. query: 6,
  2102. fragment: 7
  2103. };
  2104. function parseQuery(self) {
  2105. if (!self._queryMap) {
  2106. self._queryMap = S.unparam(self._query);
  2107. }
  2108. }
  2109. /**
  2110. * @class KISSY.Uri.Query
  2111. * Query data structure.
  2112. * @param {String} [query] encoded query string(without question mask).
  2113. */
  2114. function Query(query) {
  2115. this._query = query || '';
  2116. }
  2117. Query.prototype = {
  2118. constructor: Query,
  2119. /**
  2120. * Cloned new instance.
  2121. * @return {KISSY.Uri.Query}
  2122. */
  2123. clone: function () {
  2124. return new Query(this.toString());
  2125. },
  2126. /**
  2127. * reset to a new query string
  2128. * @param {String} query
  2129. * @chainable
  2130. */
  2131. reset: function (query) {
  2132. var self = this;
  2133. self._query = query || '';
  2134. self._queryMap = null;
  2135. return self;
  2136. },
  2137. /**
  2138. * Parameter count.
  2139. * @return {Number}
  2140. */
  2141. count: function () {
  2142. var self = this,
  2143. count = 0,
  2144. _queryMap,
  2145. k;
  2146. parseQuery(self);
  2147. _queryMap = self._queryMap;
  2148. for (k in _queryMap) {
  2149. if (S.isArray(_queryMap[k])) {
  2150. count += _queryMap[k].length;
  2151. } else {
  2152. count++;
  2153. }
  2154. }
  2155. return count;
  2156. },
  2157. /**
  2158. * judge whether has query parameter
  2159. * @param {String} [key]
  2160. */
  2161. has: function (key) {
  2162. var self = this, _queryMap;
  2163. parseQuery(self);
  2164. _queryMap = self._queryMap;
  2165. if (key) {
  2166. return key in _queryMap;
  2167. } else {
  2168. return !S.isEmptyObject(_queryMap);
  2169. }
  2170. },
  2171. /**
  2172. * Return parameter value corresponding to current key
  2173. * @param {String} [key]
  2174. */
  2175. get: function (key) {
  2176. var self = this, _queryMap;
  2177. parseQuery(self);
  2178. _queryMap = self._queryMap;
  2179. if (key) {
  2180. return _queryMap[key];
  2181. } else {
  2182. return _queryMap;
  2183. }
  2184. },
  2185. /**
  2186. * Parameter names.
  2187. * @return {String[]}
  2188. */
  2189. keys: function () {
  2190. var self = this;
  2191. parseQuery(self);
  2192. return S.keys(self._queryMap);
  2193. },
  2194. /**
  2195. * Set parameter value corresponding to current key
  2196. * @param {String} key
  2197. * @param value
  2198. * @chainable
  2199. */
  2200. set: function (key, value) {
  2201. var self = this, _queryMap;
  2202. parseQuery(self);
  2203. _queryMap = self._queryMap;
  2204. if (typeof key == 'string') {
  2205. self._queryMap[key] = value;
  2206. } else {
  2207. if (key instanceof Query) {
  2208. key = key.get();
  2209. }
  2210. S.each(key, function (v, k) {
  2211. _queryMap[k] = v;
  2212. });
  2213. }
  2214. return self;
  2215. },
  2216. /**
  2217. * Remove parameter with specified name.
  2218. * @param {String} key
  2219. * @chainable
  2220. */
  2221. remove: function (key) {
  2222. var self = this;
  2223. parseQuery(self);
  2224. if (key) {
  2225. delete self._queryMap[key];
  2226. } else {
  2227. self._queryMap = {};
  2228. }
  2229. return self;
  2230. },
  2231. /**
  2232. * Add parameter value corresponding to current key
  2233. * @param {String} key
  2234. * @param value
  2235. * @chainable
  2236. */
  2237. add: function (key, value) {
  2238. var self = this,
  2239. _queryMap,
  2240. currentValue;
  2241. if (typeof key == 'string') {
  2242. parseQuery(self);
  2243. _queryMap = self._queryMap;
  2244. currentValue = _queryMap[key];
  2245. if (currentValue === undefined) {
  2246. currentValue = value;
  2247. } else {
  2248. currentValue = [].concat(currentValue).concat(value);
  2249. }
  2250. _queryMap[key] = currentValue;
  2251. } else {
  2252. if (key instanceof Query) {
  2253. key = key.get();
  2254. }
  2255. for (var k in key) {
  2256. self.add(k, key[k]);
  2257. }
  2258. }
  2259. return self;
  2260. },
  2261. /**
  2262. * Serialize query to string.
  2263. * @param {Boolean} [serializeArray=true]
  2264. * whether append [] to key name when value 's type is array
  2265. */
  2266. toString: function (serializeArray) {
  2267. var self = this;
  2268. parseQuery(self);
  2269. return S.param(self._queryMap, undefined, undefined, serializeArray);
  2270. }
  2271. };
  2272. function padding2(str) {
  2273. return str.length == 1 ? '0' + str : str;
  2274. }
  2275. function equalsIgnoreCase(str1, str2) {
  2276. return str1.toLowerCase() == str2.toLowerCase();
  2277. }
  2278. // www.ta#bao.com // => www.ta.com/#bao.com
  2279. // www.ta%23bao.com
  2280. // Percent-Encoding
  2281. function encodeSpecialChars(str, specialCharsReg) {
  2282. // encodeURI( ) is intended to encode complete URIs,
  2283. // the following ASCII punctuation characters,
  2284. // which have special meaning in URIs, are not escaped either:
  2285. // ; / ? : @ & = + $ , #
  2286. return encodeURI(str).replace(specialCharsReg, function (m) {
  2287. return '%' + padding2(m.charCodeAt(0).toString(16));
  2288. });
  2289. }
  2290. /**
  2291. * @class KISSY.Uri
  2292. * Uri class for KISSY.
  2293. * Most of its interfaces are same with window.location.
  2294. * @param {String|KISSY.Uri} [uriStr] Encoded uri string.
  2295. */
  2296. function Uri(uriStr) {
  2297. if (uriStr instanceof Uri) {
  2298. return uriStr['clone']();
  2299. }
  2300. var components, self = this;
  2301. S.mix(self,
  2302. {
  2303. /**
  2304. * scheme such as 'http:'. aka protocol without colon
  2305. * @type {String}
  2306. */
  2307. scheme: '',
  2308. /**
  2309. * User credentials such as 'yiminghe:gmail'
  2310. * @type {String}
  2311. */
  2312. userInfo: '',
  2313. /**
  2314. * hostname such as 'docs.kissyui.com'. aka domain
  2315. * @type {String}
  2316. */
  2317. hostname: '',
  2318. /**
  2319. * Port such as '8080'
  2320. * @type {String}
  2321. */
  2322. port: '',
  2323. /**
  2324. * path such as '/index.htm'. aka pathname
  2325. * @type {String}
  2326. */
  2327. path: '',
  2328. /**
  2329. * Query object for search string. aka search
  2330. * @type {KISSY.Uri.Query}
  2331. */
  2332. query: '',
  2333. /**
  2334. * fragment such as '#!/test/2'. aka hash
  2335. */
  2336. fragment: ''
  2337. });
  2338. components = Uri.getComponents(uriStr);
  2339. S.each(components, function (v, key) {
  2340. v = v || '';
  2341. if (key == 'query') {
  2342. // need encoded content
  2343. self.query = new Query(v);
  2344. } else {
  2345. // https://github.com/kissyteam/kissy/issues/298
  2346. try {
  2347. v = S.urlDecode(v);
  2348. } catch (e) {
  2349. logger.error(e + 'urlDecode error : ' + v);
  2350. }
  2351. // need to decode to get data structure in memory
  2352. self[key] = v;
  2353. }
  2354. });
  2355. return self;
  2356. }
  2357. Uri.prototype = {
  2358. constructor: Uri,
  2359. /**
  2360. * Return a cloned new instance.
  2361. * @return {KISSY.Uri}
  2362. */
  2363. clone: function () {
  2364. var uri = new Uri(), self = this;
  2365. S.each(REG_INFO, function (index, key) {
  2366. uri[key] = self[key];
  2367. });
  2368. uri.query = uri.query.clone();
  2369. return uri;
  2370. },
  2371. /**
  2372. * The reference resolution algorithm.rfc 5.2
  2373. * return a resolved uri corresponding to current uri
  2374. * @param {KISSY.Uri|String} relativeUri
  2375. *
  2376. * for example:
  2377. * @example
  2378. * this: 'http://y/yy/z.com?t=1#v=2'
  2379. * 'https:/y/' => 'https:/y/'
  2380. * '//foo' => 'http://foo'
  2381. * 'foo' => 'http://y/yy/foo'
  2382. * '/foo' => 'http://y/foo'
  2383. * '?foo' => 'http://y/yy/z.com?foo'
  2384. * '#foo' => http://y/yy/z.com?t=1#foo'
  2385. *
  2386. * @return {KISSY.Uri}
  2387. */
  2388. resolve: function (relativeUri) {
  2389. if (typeof relativeUri == 'string') {
  2390. relativeUri = new Uri(relativeUri);
  2391. }
  2392. var self = this,
  2393. override = 0,
  2394. lastSlashIndex,
  2395. order = ['scheme', 'userInfo', 'hostname', 'port', 'path', 'query', 'fragment'],
  2396. target = self.clone();
  2397. S.each(order, function (o) {
  2398. if (o == 'path') {
  2399. // relativeUri does not set for scheme/userInfo/hostname/port
  2400. if (override) {
  2401. target[o] = relativeUri[o];
  2402. } else {
  2403. var path = relativeUri['path'];
  2404. if (path) {
  2405. // force to override target 's query with relative
  2406. override = 1;
  2407. if (!S.startsWith(path, '/')) {
  2408. if (target.hostname && !target.path) {
  2409. // RFC 3986, section 5.2.3, case 1
  2410. path = '/' + path;
  2411. } else if (target.path) {
  2412. // RFC 3986, section 5.2.3, case 2
  2413. lastSlashIndex = target.path.lastIndexOf('/');
  2414. if (lastSlashIndex != -1) {
  2415. path = target.path.slice(0, lastSlashIndex + 1) + path;
  2416. }
  2417. }
  2418. }
  2419. // remove .. / . as part of the resolution process
  2420. target.path = Path.normalize(path);
  2421. }
  2422. }
  2423. } else if (o == 'query') {
  2424. if (override || relativeUri['query'].toString()) {
  2425. target.query = relativeUri['query'].clone();
  2426. override = 1;
  2427. }
  2428. } else if (override || relativeUri[o]) {
  2429. target[o] = relativeUri[o];
  2430. override = 1;
  2431. }
  2432. });
  2433. return target;
  2434. },
  2435. /**
  2436. * Get scheme part
  2437. */
  2438. getScheme: function () {
  2439. return this.scheme;
  2440. },
  2441. /**
  2442. * Set scheme part
  2443. * @param {String} scheme
  2444. * @chainable
  2445. */
  2446. setScheme: function (scheme) {
  2447. this.scheme = scheme;
  2448. return this;
  2449. },
  2450. /**
  2451. * Return hostname
  2452. * @return {String}
  2453. */
  2454. getHostname: function () {
  2455. return this.hostname;
  2456. },
  2457. /**
  2458. * Set hostname
  2459. * @param {String} hostname
  2460. * @chainable
  2461. */
  2462. setHostname: function (hostname) {
  2463. this.hostname = hostname;
  2464. return this;
  2465. },
  2466. /**
  2467. * Set user info
  2468. * @param {String} userInfo
  2469. * @chainable
  2470. */
  2471. 'setUserInfo': function (userInfo) {
  2472. this.userInfo = userInfo;
  2473. return this;
  2474. },
  2475. /**
  2476. * Get user info
  2477. * @return {String}
  2478. */
  2479. getUserInfo: function () {
  2480. return this.userInfo;
  2481. },
  2482. /**
  2483. * Set port
  2484. * @param {String} port
  2485. * @chainable
  2486. */
  2487. 'setPort': function (port) {
  2488. this.port = port;
  2489. return this;
  2490. },
  2491. /**
  2492. * Get port
  2493. * @return {String}
  2494. */
  2495. 'getPort': function () {
  2496. return this.port;
  2497. },
  2498. /**
  2499. * Set path
  2500. * @param {string} path
  2501. * @chainable
  2502. */
  2503. setPath: function (path) {
  2504. this.path = path;
  2505. return this;
  2506. },
  2507. /**
  2508. * Get path
  2509. * @return {String}
  2510. */
  2511. getPath: function () {
  2512. return this.path;
  2513. },
  2514. /**
  2515. * Set query
  2516. * @param {String|KISSY.Uri.Query} query
  2517. * @chainable
  2518. */
  2519. 'setQuery': function (query) {
  2520. if (typeof query == 'string') {
  2521. if (S.startsWith(query, '?')) {
  2522. query = query.slice(1);
  2523. }
  2524. query = new Query(encodeSpecialChars(query, reDisallowedInQuery));
  2525. }
  2526. this.query = query;
  2527. return this;
  2528. },
  2529. /**
  2530. * Get query
  2531. * @return {KISSY.Uri.Query}
  2532. */
  2533. getQuery: function () {
  2534. return this.query;
  2535. },
  2536. /**
  2537. * Get fragment
  2538. * @return {String}
  2539. */
  2540. getFragment: function () {
  2541. return this.fragment;
  2542. },
  2543. /**
  2544. * Set fragment
  2545. * @param {String} fragment
  2546. * @chainable
  2547. */
  2548. 'setFragment': function (fragment) {
  2549. var self = this;
  2550. if (S.startsWith(fragment, '#')) {
  2551. fragment = fragment.slice(1);
  2552. }
  2553. self.fragment = fragment;
  2554. return self;
  2555. },
  2556. /**
  2557. * Judge whether two uri has same domain.
  2558. * @param {KISSY.Uri} other
  2559. * @return {Boolean}
  2560. */
  2561. isSameOriginAs: function (other) {
  2562. var self = this;
  2563. // port and hostname has to be same
  2564. return equalsIgnoreCase(self.hostname, other['hostname']) &&
  2565. equalsIgnoreCase(self.scheme, other['scheme']) &&
  2566. equalsIgnoreCase(self.port, other['port']);
  2567. },
  2568. /**
  2569. * Serialize to string.
  2570. * See rfc 5.3 Component Recomposition.
  2571. * But kissy does not differentiate between undefined and empty.
  2572. * @param {Boolean} [serializeArray=true]
  2573. * whether append [] to key name when value 's type is array
  2574. * @return {String}
  2575. */
  2576. toString: function (serializeArray) {
  2577. var out = [],
  2578. self = this,
  2579. scheme,
  2580. hostname,
  2581. path,
  2582. port,
  2583. fragment,
  2584. query,
  2585. userInfo;
  2586. if (scheme = self.scheme) {
  2587. out.push(encodeSpecialChars(scheme, reDisallowedInSchemeOrUserInfo));
  2588. out.push(':');
  2589. }
  2590. if (hostname = self.hostname) {
  2591. out.push('//');
  2592. if (userInfo = self.userInfo) {
  2593. out.push(encodeSpecialChars(userInfo, reDisallowedInSchemeOrUserInfo));
  2594. out.push('@');
  2595. }
  2596. out.push(encodeURIComponent(hostname));
  2597. if (port = self.port) {
  2598. out.push(':');
  2599. out.push(port);
  2600. }
  2601. }
  2602. if (path = self.path) {
  2603. if (hostname && !S.startsWith(path, '/')) {
  2604. path = '/' + path;
  2605. }
  2606. path = Path.normalize(path);
  2607. out.push(encodeSpecialChars(path, reDisallowedInPathName));
  2608. }
  2609. if (query = ( self.query.toString.call(self.query, serializeArray))) {
  2610. out.push('?');
  2611. out.push(query);
  2612. }
  2613. if (fragment = self.fragment) {
  2614. out.push('#');
  2615. out.push(encodeSpecialChars(fragment, reDisallowedInFragment))
  2616. }
  2617. return out.join('');
  2618. }
  2619. };
  2620. Uri.Query = Query;
  2621. Uri.getComponents = function (url) {
  2622. url = url || "";
  2623. var m = url.match(URI_SPLIT_REG) || [],
  2624. ret = {};
  2625. S.each(REG_INFO, function (index, key) {
  2626. ret[key] = m[index];
  2627. });
  2628. return ret;
  2629. };
  2630. S.Uri = Uri;
  2631. })(KISSY);
  2632. /*
  2633. Refer
  2634. - application/x-www-form-urlencoded
  2635. - http://www.ietf.org/rfc/rfc3986.txt
  2636. - http://en.wikipedia.org/wiki/URI_scheme
  2637. - http://unixpapa.com/js/querystring.html
  2638. - http://code.stephenmorley.org/javascript/parsing-query-strings-for-get-data/
  2639. - same origin: http://tools.ietf.org/html/rfc6454
  2640. *//**
  2641. * @ignore
  2642. * ua
  2643. */
  2644. (function (S, undefined) {
  2645. var win = S.Env.host,
  2646. doc = win.document,
  2647. navigator = win.navigator,
  2648. ua = navigator && navigator.userAgent || "";
  2649. function numberify(s) {
  2650. var c = 0;
  2651. // convert '1.2.3.4' to 1.234
  2652. return parseFloat(s.replace(/\./g, function () {
  2653. return (c++ === 0) ? '.' : '';
  2654. }));
  2655. }
  2656. function setTridentVersion(ua, UA) {
  2657. var core, m;
  2658. UA[core = 'trident'] = 0.1; // Trident detected, look for revision
  2659. // Get the Trident's accurate version
  2660. if ((m = ua.match(/Trident\/([\d.]*)/)) && m[1]) {
  2661. UA[core] = numberify(m[1]);
  2662. }
  2663. UA.core = core;
  2664. }
  2665. function getIEVersion(ua) {
  2666. var m, v;
  2667. if ((m = ua.match(/MSIE ([^;]*)|Trident.*; rv(?:\s|:)?([0-9.]+)/)) &&
  2668. (v = (m[1] || m[2]))) {
  2669. return numberify(v);
  2670. }
  2671. return 0;
  2672. }
  2673. function getDescriptorFromUserAgent(ua) {
  2674. var EMPTY = '',
  2675. os,
  2676. core = EMPTY,
  2677. shell = EMPTY, m,
  2678. IE_DETECT_RANGE = [6, 9],
  2679. ieVersion,
  2680. v,
  2681. end,
  2682. VERSION_PLACEHOLDER = '{{version}}',
  2683. IE_DETECT_TPL = '<!--[if IE ' + VERSION_PLACEHOLDER + ']><' + 's></s><![endif]-->',
  2684. div = doc && doc.createElement('div'),
  2685. s = [];
  2686. /**
  2687. * KISSY UA
  2688. * @class KISSY.UA
  2689. * @singleton
  2690. */
  2691. var UA = {
  2692. /**
  2693. * webkit version
  2694. * @type undefined|Number
  2695. * @member KISSY.UA
  2696. */
  2697. webkit: undefined,
  2698. /**
  2699. * trident version
  2700. * @type undefined|Number
  2701. * @member KISSY.UA
  2702. */
  2703. trident: undefined,
  2704. /**
  2705. * gecko version
  2706. * @type undefined|Number
  2707. * @member KISSY.UA
  2708. */
  2709. gecko: undefined,
  2710. /**
  2711. * presto version
  2712. * @type undefined|Number
  2713. * @member KISSY.UA
  2714. */
  2715. presto: undefined,
  2716. /**
  2717. * chrome version
  2718. * @type undefined|Number
  2719. * @member KISSY.UA
  2720. */
  2721. chrome: undefined,
  2722. /**
  2723. * safari version
  2724. * @type undefined|Number
  2725. * @member KISSY.UA
  2726. */
  2727. safari: undefined,
  2728. /**
  2729. * firefox version
  2730. * @type undefined|Number
  2731. * @member KISSY.UA
  2732. */
  2733. firefox: undefined,
  2734. /**
  2735. * ie version
  2736. * @type undefined|Number
  2737. * @member KISSY.UA
  2738. */
  2739. ie: undefined,
  2740. /**
  2741. * opera version
  2742. * @type undefined|Number
  2743. * @member KISSY.UA
  2744. */
  2745. opera: undefined,
  2746. /**
  2747. * mobile browser. apple, android.
  2748. * @type String
  2749. * @member KISSY.UA
  2750. */
  2751. mobile: undefined,
  2752. /**
  2753. * browser render engine name. webkit, trident
  2754. * @type String
  2755. * @member KISSY.UA
  2756. */
  2757. core: undefined,
  2758. /**
  2759. * browser shell name. ie, chrome, firefox
  2760. * @type String
  2761. * @member KISSY.UA
  2762. */
  2763. shell: undefined,
  2764. /**
  2765. * PhantomJS version number
  2766. * @type undefined|Number
  2767. * @member KISSY.UA
  2768. */
  2769. phantomjs: undefined,
  2770. /**
  2771. * operating system. android, ios, linux, windows
  2772. * @type string
  2773. * @member KISSY.UA
  2774. */
  2775. os: undefined,
  2776. /**
  2777. * ipad ios version
  2778. * @type Number
  2779. * @member KISSY.UA
  2780. */
  2781. ipad: undefined,
  2782. /**
  2783. * iphone ios version
  2784. * @type Number
  2785. * @member KISSY.UA
  2786. */
  2787. iphone: undefined,
  2788. /**
  2789. * ipod ios
  2790. * @type Number
  2791. * @member KISSY.UA
  2792. */
  2793. ipod: undefined,
  2794. /**
  2795. * ios version
  2796. * @type Number
  2797. * @member KISSY.UA
  2798. */
  2799. ios: undefined,
  2800. /**
  2801. * android version
  2802. * @type Number
  2803. * @member KISSY.UA
  2804. */
  2805. android: undefined,
  2806. /**
  2807. * nodejs version
  2808. * @type Number
  2809. * @member KISSY.UA
  2810. */
  2811. nodejs: undefined
  2812. };
  2813. // ejecta
  2814. if (div && div.getElementsByTagName) {
  2815. // try to use IE-Conditional-Comment detect IE more accurately
  2816. // IE10 doesn't support this method, @ref: http://blogs.msdn.com/b/ie/archive/2011/07/06/html5-parsing-in-ie10.aspx
  2817. div.innerHTML = IE_DETECT_TPL.replace(VERSION_PLACEHOLDER, '');
  2818. s = div.getElementsByTagName('s');
  2819. }
  2820. if (s.length > 0) {
  2821. setTridentVersion(ua, UA);
  2822. // Detect the accurate version
  2823. // 注意:
  2824. // UA.shell = ie, 表示外壳是 ie
  2825. // 但 UA.ie = 7, 并不代表外壳是 ie7, 还有可能是 ie8 的兼容模式
  2826. // 对于 ie8 的兼容模式,还要通过 documentMode 去判断。但此处不能让 UA.ie = 8, 否则
  2827. // 很多脚本判断会失误。因为 ie8 的兼容模式表现行为和 ie7 相同,而不是和 ie8 相同
  2828. for (v = IE_DETECT_RANGE[0], end = IE_DETECT_RANGE[1]; v <= end; v++) {
  2829. div.innerHTML = IE_DETECT_TPL.replace(VERSION_PLACEHOLDER, v);
  2830. if (s.length > 0) {
  2831. UA[shell = 'ie'] = v;
  2832. break;
  2833. }
  2834. }
  2835. // https://github.com/kissyteam/kissy/issues/321
  2836. // win8 embed app
  2837. if (!UA.ie && (ieVersion = getIEVersion(ua))) {
  2838. UA[shell = 'ie'] = ieVersion;
  2839. }
  2840. } else {
  2841. // WebKit
  2842. if ((m = ua.match(/AppleWebKit\/([\d.]*)/)) && m[1]) {
  2843. UA[core = 'webkit'] = numberify(m[1]);
  2844. if ((m = ua.match(/OPR\/(\d+\.\d+)/)) && m[1]) {
  2845. UA[shell = 'opera'] = numberify(m[1]);
  2846. }
  2847. // Chrome
  2848. else if ((m = ua.match(/Chrome\/([\d.]*)/)) && m[1]) {
  2849. UA[shell = 'chrome'] = numberify(m[1]);
  2850. }
  2851. // Safari
  2852. else if ((m = ua.match(/\/([\d.]*) Safari/)) && m[1]) {
  2853. UA[shell = 'safari'] = numberify(m[1]);
  2854. }
  2855. // Apple Mobile
  2856. if (/ Mobile\//.test(ua) && ua.match(/iPad|iPod|iPhone/)) {
  2857. UA.mobile = 'apple'; // iPad, iPhone or iPod Touch
  2858. m = ua.match(/OS ([^\s]*)/);
  2859. if (m && m[1]) {
  2860. UA.ios = numberify(m[1].replace('_', '.'));
  2861. }
  2862. os = 'ios';
  2863. m = ua.match(/iPad|iPod|iPhone/);
  2864. if (m && m[0]) {
  2865. UA[m[0].toLowerCase()] = UA.ios;
  2866. }
  2867. } else if (/ Android/i.test(ua)) {
  2868. if (/Mobile/.test(ua)) {
  2869. os = UA.mobile = 'android';
  2870. }
  2871. m = ua.match(/Android ([^\s]*);/);
  2872. if (m && m[1]) {
  2873. UA.android = numberify(m[1]);
  2874. }
  2875. }
  2876. // Other WebKit Mobile Browsers
  2877. else if ((m = ua.match(/NokiaN[^\/]*|Android \d\.\d|webOS\/\d\.\d/))) {
  2878. UA.mobile = m[0].toLowerCase(); // Nokia N-series, Android, webOS, ex: NokiaN95
  2879. }
  2880. if ((m = ua.match(/PhantomJS\/([^\s]*)/)) && m[1]) {
  2881. UA.phantomjs = numberify(m[1]);
  2882. }
  2883. }
  2884. // NOT WebKit
  2885. else {
  2886. // Presto
  2887. // ref: http://www.useragentstring.com/pages/useragentstring.php
  2888. if ((m = ua.match(/Presto\/([\d.]*)/)) && m[1]) {
  2889. UA[core = 'presto'] = numberify(m[1]);
  2890. // Opera
  2891. if ((m = ua.match(/Opera\/([\d.]*)/)) && m[1]) {
  2892. UA[shell = 'opera'] = numberify(m[1]); // Opera detected, look for revision
  2893. if ((m = ua.match(/Opera\/.* Version\/([\d.]*)/)) && m[1]) {
  2894. UA[shell] = numberify(m[1]);
  2895. }
  2896. // Opera Mini
  2897. if ((m = ua.match(/Opera Mini[^;]*/)) && m) {
  2898. UA.mobile = m[0].toLowerCase(); // ex: Opera Mini/2.0.4509/1316
  2899. }
  2900. // Opera Mobile
  2901. // ex: Opera/9.80 (Windows NT 6.1; Opera Mobi/49; U; en) Presto/2.4.18 Version/10.00
  2902. // issue: 由于 Opera Mobile 有 Version/ 字段,可能会与 Opera 混淆,同时对于 Opera Mobile 的版本号也比较混乱
  2903. else if ((m = ua.match(/Opera Mobi[^;]*/)) && m) {
  2904. UA.mobile = m[0];
  2905. }
  2906. }
  2907. // NOT WebKit or Presto
  2908. } else {
  2909. // MSIE
  2910. // 由于最开始已经使用了 IE 条件注释判断,因此落到这里的唯一可能性只有 IE10+
  2911. // and analysis tools in nodejs
  2912. if (ieVersion = getIEVersion(ua)) {
  2913. UA[shell = 'ie'] = ieVersion;
  2914. setTridentVersion(ua, UA);
  2915. // NOT WebKit, Presto or IE
  2916. } else {
  2917. // Gecko
  2918. if ((m = ua.match(/Gecko/))) {
  2919. UA[core = 'gecko'] = 0.1; // Gecko detected, look for revision
  2920. if ((m = ua.match(/rv:([\d.]*)/)) && m[1]) {
  2921. UA[core] = numberify(m[1]);
  2922. if (/Mobile|Tablet/.test(ua)) {
  2923. UA.mobile = "firefox";
  2924. }
  2925. }
  2926. // Firefox
  2927. if ((m = ua.match(/Firefox\/([\d.]*)/)) && m[1]) {
  2928. UA[shell = 'firefox'] = numberify(m[1]);
  2929. }
  2930. }
  2931. }
  2932. }
  2933. }
  2934. }
  2935. if (!os) {
  2936. if ((/windows|win32/i).test(ua)) {
  2937. os = 'windows';
  2938. } else if ((/macintosh|mac_powerpc/i).test(ua)) {
  2939. os = 'macintosh';
  2940. } else if ((/linux/i).test(ua)) {
  2941. os = 'linux';
  2942. } else if ((/rhino/i).test(ua)) {
  2943. os = 'rhino';
  2944. }
  2945. }
  2946. UA.os = os;
  2947. UA.core = UA.core || core;
  2948. UA.shell = shell;
  2949. return UA;
  2950. }
  2951. var UA = KISSY.UA = getDescriptorFromUserAgent(ua);
  2952. // nodejs
  2953. if (typeof process === 'object') {
  2954. var versions, nodeVersion;
  2955. if ((versions = process.versions) && (nodeVersion = versions.node)) {
  2956. UA.os = process.platform;
  2957. UA.nodejs = numberify(nodeVersion);
  2958. }
  2959. }
  2960. // use by analysis tools in nodejs
  2961. UA.getDescriptorFromUserAgent = getDescriptorFromUserAgent;
  2962. var browsers = [
  2963. // browser core type
  2964. 'webkit',
  2965. 'trident',
  2966. 'gecko',
  2967. 'presto',
  2968. // browser type
  2969. 'chrome',
  2970. 'safari',
  2971. 'firefox',
  2972. 'ie',
  2973. 'opera'
  2974. ],
  2975. documentElement = doc && doc.documentElement,
  2976. className = '';
  2977. if (documentElement) {
  2978. S.each(browsers, function (key) {
  2979. var v = UA[key];
  2980. if (v) {
  2981. className += ' ks-' + key + (parseInt(v) + '');
  2982. className += ' ks-' + key;
  2983. }
  2984. });
  2985. if (S.trim(className)) {
  2986. documentElement.className = S.trim(documentElement.className + className);
  2987. }
  2988. }
  2989. })(KISSY);
  2990. /*
  2991. NOTES:
  2992. 2013.07.08 yiminghe@gmail.com
  2993. - support ie11 and opera(using blink)
  2994. 2013.01.17 yiminghe@gmail.com
  2995. - expose getDescriptorFromUserAgent for analysis tool in nodejs
  2996. 2012.11.27 yiminghe@gmail.com
  2997. - moved to seed for conditional loading and better code share
  2998. 2012.11.21 yiminghe@gmail.com
  2999. - touch and os support
  3000. 2011.11.08 gonghaocn@gmail.com
  3001. - ie < 10 使用条件注释判断内核,更精确
  3002. 2010.03
  3003. - jQuery, YUI 等类库都推荐用特性探测替代浏览器嗅探。特性探测的好处是能自动适应未来设备和未知设备,比如
  3004. if(document.addEventListener) 假设 IE9 支持标准事件,则代码不用修改,就自适应了“未来浏览器”。
  3005. 对于未知浏览器也是如此。但是,这并不意味着浏览器嗅探就得彻底抛弃。当代码很明确就是针对已知特定浏览器的,
  3006. 同时并非是某个特性探测可以解决时,用浏览器嗅探反而能带来代码的简洁,同时也也不会有什么后患。总之,一切
  3007. 皆权衡。
  3008. - UA.ie && UA.ie < 8 并不意味着浏览器就不是 IE8, 有可能是 IE8 的兼容模式。进一步的判断需要使用 documentMode.
  3009. */
  3010. /**
  3011. * @ignore
  3012. * detect if current browser supports various features.
  3013. * @author yiminghe@gmail.com
  3014. */
  3015. (function (S, undefined) {
  3016. var Env = S.Env,
  3017. win = Env.host,
  3018. UA = S.UA,
  3019. VENDORS = [
  3020. '',
  3021. 'Webkit',
  3022. 'Moz',
  3023. 'O',
  3024. // ms is special .... !
  3025. 'ms'
  3026. ],
  3027. // nodejs
  3028. doc = win.document || {},
  3029. documentMode = doc.documentMode,
  3030. isMsPointerSupported,
  3031. transitionProperty,
  3032. transformProperty,
  3033. transitionPrefix,
  3034. transformPrefix,
  3035. documentElement = doc.documentElement,
  3036. documentElementStyle,
  3037. isClassListSupportedState = true,
  3038. isQuerySelectorSupportedState = false,
  3039. // phantomjs issue: http://code.google.com/p/phantomjs/issues/detail?id=375
  3040. isTouchEventSupportedState = ('ontouchstart' in doc) && !(UA.phantomjs),
  3041. ie = documentMode || UA.ie;
  3042. if (documentElement) {
  3043. if (documentElement.querySelector &&
  3044. // broken ie8
  3045. ie != 8) {
  3046. isQuerySelectorSupportedState = true;
  3047. }
  3048. documentElementStyle = documentElement.style;
  3049. S.each(VENDORS, function (val) {
  3050. var transition = val ? val + 'Transition' : 'transition',
  3051. transform = val ? val + 'Transform' : 'transform';
  3052. if (transitionPrefix === undefined &&
  3053. transition in documentElementStyle) {
  3054. transitionPrefix = val;
  3055. transitionProperty = transition;
  3056. }
  3057. if (transformPrefix === undefined &&
  3058. transform in documentElementStyle) {
  3059. transformPrefix = val;
  3060. transformProperty = transform;
  3061. }
  3062. });
  3063. isClassListSupportedState = 'classList' in documentElement;
  3064. isMsPointerSupported = "msPointerEnabled" in (win.navigator || {});
  3065. }
  3066. /**
  3067. * browser features detection
  3068. * @class KISSY.Features
  3069. * @private
  3070. * @singleton
  3071. */
  3072. S.Features = {
  3073. // http://blogs.msdn.com/b/ie/archive/2011/09/20/touch-input-for-ie10-and-metro-style-apps.aspx
  3074. /**
  3075. * whether support microsoft pointer event.
  3076. * @type {Boolean}
  3077. */
  3078. isMsPointerSupported: function () {
  3079. return isMsPointerSupported;
  3080. },
  3081. /**
  3082. * whether support touch event.
  3083. * @return {Boolean}
  3084. */
  3085. isTouchEventSupported: function () {
  3086. return isTouchEventSupportedState;
  3087. },
  3088. /**
  3089. * whether support device motion event
  3090. * @returns {boolean}
  3091. */
  3092. isDeviceMotionSupported: function () {
  3093. return !!win['DeviceMotionEvent'];
  3094. },
  3095. /**
  3096. * whether support hashchange event
  3097. * @returns {boolean}
  3098. */
  3099. 'isHashChangeSupported': function () {
  3100. // ie8 支持 hashchange
  3101. // 但 ie8 以上切换浏览器模式到 ie7(兼容模式),
  3102. // 会导致 'onhashchange' in window === true,但是不触发事件
  3103. return ( 'onhashchange' in win) && (!ie || ie > 7);
  3104. },
  3105. /**
  3106. * whether support css transition
  3107. * @returns {boolean}
  3108. */
  3109. 'isTransitionSupported': function () {
  3110. return transitionPrefix !== undefined;
  3111. },
  3112. /**
  3113. * whether support css transform
  3114. * @returns {boolean}
  3115. */
  3116. 'isTransformSupported': function () {
  3117. return transformPrefix !== undefined;
  3118. },
  3119. /**
  3120. * whether support class list api
  3121. * @returns {boolean}
  3122. */
  3123. 'isClassListSupported': function () {
  3124. return isClassListSupportedState
  3125. },
  3126. /**
  3127. * whether support querySelectorAll
  3128. * @returns {boolean}
  3129. */
  3130. 'isQuerySelectorSupported': function () {
  3131. // force to use js selector engine
  3132. return !S.config('dom/selector') &&
  3133. isQuerySelectorSupportedState;
  3134. },
  3135. /**
  3136. * whether is ie and ie version is less than specified version
  3137. * @param {Number} v specified ie version to be compared
  3138. * @returns {boolean}
  3139. */
  3140. 'isIELessThan': function (v) {
  3141. return !!(ie && ie < v);
  3142. },
  3143. /**
  3144. * get css transition browser prefix if support css transition
  3145. * @returns {String}
  3146. */
  3147. 'getTransitionPrefix': function () {
  3148. return transitionPrefix;
  3149. },
  3150. /**
  3151. * get css transform browser prefix if support css transform
  3152. * @returns {String}
  3153. */
  3154. 'getTransformPrefix': function () {
  3155. return transformPrefix;
  3156. },
  3157. /**
  3158. * get css transition property if support css transition
  3159. * @returns {String}
  3160. */
  3161. 'getTransitionProperty': function () {
  3162. return transitionProperty;
  3163. },
  3164. /**
  3165. * get css transform property if support css transform
  3166. * @returns {String}
  3167. */
  3168. 'getTransformProperty': function () {
  3169. return transformProperty;
  3170. }
  3171. };
  3172. })(KISSY);/**
  3173. * @ignore
  3174. * setup data structure for kissy loader
  3175. * @author yiminghe@gmail.com
  3176. */
  3177. (function (S) {
  3178. var Loader = S.Loader = {};
  3179. /**
  3180. * Loader Status Enum
  3181. * @enum {Number} KISSY.Loader.Status
  3182. */
  3183. Loader.Status = {
  3184. /** init */
  3185. 'INIT': 0,
  3186. /** loading */
  3187. 'LOADING': 1,
  3188. /** loaded */
  3189. 'LOADED': 2,
  3190. /** error */
  3191. 'ERROR': 3,
  3192. /** attached */
  3193. 'ATTACHED': 4
  3194. };
  3195. })(KISSY);/**
  3196. * @ignore
  3197. * Utils for kissy loader
  3198. * @author yiminghe@gmail.com
  3199. */
  3200. (function (S) {
  3201. var Loader = S.Loader,
  3202. Path = S.Path,
  3203. logger = S.getLogger('s/loader'),
  3204. host = S.Env.host,
  3205. startsWith = S.startsWith,
  3206. data = Loader.Status,
  3207. ATTACHED = data.ATTACHED,
  3208. LOADED = data.LOADED,
  3209. ERROR = data.ERROR,
  3210. /**
  3211. * @class KISSY.Loader.Utils
  3212. * Utils for KISSY Loader
  3213. * @singleton
  3214. * @private
  3215. */
  3216. Utils = Loader.Utils = {},
  3217. doc = host.document;
  3218. // http://wiki.commonjs.org/wiki/Packages/Mappings/A
  3219. // 如果模块名以 / 结尾,自动加 index
  3220. function indexMap(s) {
  3221. if (typeof s == 'string') {
  3222. return indexMapStr(s);
  3223. } else {
  3224. var ret = [],
  3225. i = 0,
  3226. l = s.length;
  3227. for (; i < l; i++) {
  3228. ret[i] = indexMapStr(s[i]);
  3229. }
  3230. return ret;
  3231. }
  3232. }
  3233. function indexMapStr(s) {
  3234. // 'x/' 'x/y/z/'
  3235. if (s.charAt(s.length - 1) == '/') {
  3236. s += 'index';
  3237. }
  3238. return s;
  3239. }
  3240. function pluginAlias(runtime, name) {
  3241. var index = name.indexOf('!');
  3242. if (index != -1) {
  3243. var pluginName = name.substring(0, index);
  3244. name = name.substring(index + 1);
  3245. S.use(pluginName, {
  3246. sync: true,
  3247. success: function (S, Plugin) {
  3248. if (Plugin.alias) {
  3249. //noinspection JSReferencingMutableVariableFromClosure
  3250. name = Plugin.alias(runtime, name, pluginName);
  3251. }
  3252. }
  3253. });
  3254. }
  3255. return name;
  3256. }
  3257. S.mix(Utils, {
  3258. /**
  3259. * get document head
  3260. * @return {HTMLElement}
  3261. */
  3262. docHead: function () {
  3263. return doc.getElementsByTagName('head')[0] || doc.documentElement;
  3264. },
  3265. /**
  3266. * Get absolute path of dep module.similar to {@link KISSY.Path#resolve}
  3267. * @param {String} moduleName current module 's name
  3268. * @param {String|String[]} depName dependency module 's name
  3269. * @return {String|String[]} normalized dependency module 's name
  3270. */
  3271. normalDepModuleName: function (moduleName, depName) {
  3272. var i = 0, l;
  3273. if (!depName) {
  3274. return depName;
  3275. }
  3276. if (typeof depName == 'string') {
  3277. if (startsWith(depName, '../') || startsWith(depName, './')) {
  3278. // x/y/z -> x/y/
  3279. return Path.resolve(Path.dirname(moduleName), depName);
  3280. }
  3281. return Path.normalize(depName);
  3282. }
  3283. for (l = depName.length; i < l; i++) {
  3284. depName[i] = Utils.normalDepModuleName(moduleName, depName[i]);
  3285. }
  3286. return depName;
  3287. },
  3288. /**
  3289. * create modules info
  3290. * @param runtime Module container, such as KISSY
  3291. * @param {String[]} modNames to be created module names
  3292. */
  3293. createModulesInfo: function (runtime, modNames) {
  3294. S.each(modNames, function (m) {
  3295. Utils.createModuleInfo(runtime, m);
  3296. });
  3297. },
  3298. /**
  3299. * create single module info
  3300. * @param runtime Module container, such as KISSY
  3301. * @param {String} modName to be created module name
  3302. * @param {Object} [cfg] module config
  3303. * @return {KISSY.Loader.Module}
  3304. */
  3305. createModuleInfo: function (runtime, modName, cfg) {
  3306. modName = indexMapStr(modName);
  3307. var mods = runtime.Env.mods,
  3308. mod = mods[modName];
  3309. if (mod) {
  3310. return mod;
  3311. }
  3312. // 防止 cfg 里有 tag,构建 fullpath 需要
  3313. mods[modName] = mod = new Loader.Module(S.mix({
  3314. name: modName,
  3315. runtime: runtime
  3316. }, cfg));
  3317. return mod;
  3318. },
  3319. /**
  3320. * Whether modNames is attached.
  3321. * @param runtime Module container, such as KISSY
  3322. * @param modNames
  3323. * @return {Boolean}
  3324. */
  3325. 'isAttached': function (runtime, modNames) {
  3326. return isStatus(runtime, modNames, ATTACHED);
  3327. },
  3328. /**
  3329. * Get module values
  3330. * @param runtime Module container, such as KISSY
  3331. * @param {String[]} modNames module names
  3332. * @return {Array} module values
  3333. */
  3334. getModules: function (runtime, modNames) {
  3335. var mods = [runtime], mod,
  3336. unalias,
  3337. allOk,
  3338. m,
  3339. runtimeMods = runtime.Env.mods;
  3340. S.each(modNames, function (modName) {
  3341. mod = runtimeMods[modName];
  3342. if (!mod || mod.getType() != 'css') {
  3343. unalias = Utils.unalias(runtime, modName);
  3344. allOk = S.reduce(unalias, function (a, n) {
  3345. m = runtimeMods[n];
  3346. return a && m && m.status == ATTACHED;
  3347. }, true);
  3348. if (allOk) {
  3349. mods.push(runtimeMods[unalias[0]].value);
  3350. } else {
  3351. mods.push(null);
  3352. }
  3353. }
  3354. });
  3355. return mods;
  3356. },
  3357. /**
  3358. * attach modules and their dependency modules recursively
  3359. * @param {String[]} modNames module names
  3360. * @param runtime Module container, such as KISSY
  3361. * @param {String[]} [stack] stack for detecting circular dependency
  3362. * @param {Array} [errorList] errors when attach mods
  3363. * @param {Object} [cache] cached modules to avoid duplicate check
  3364. * @returns whether success attach all modules
  3365. */
  3366. attachModsRecursively: function (modNames, runtime, stack, errorList, cache) {
  3367. // for debug. prevent circular dependency
  3368. stack = stack || [];
  3369. // for efficiency. avoid duplicate non-attach check
  3370. cache = cache || {};
  3371. var i,
  3372. s = 1,
  3373. l = modNames.length,
  3374. stackDepth = stack.length;
  3375. for (i = 0; i < l; i++) {
  3376. s = s && Utils.attachModRecursively(modNames[i], runtime, stack, errorList, cache);
  3377. stack.length = stackDepth;
  3378. }
  3379. return s;
  3380. },
  3381. /**
  3382. * attach module and its dependency modules recursively
  3383. * @param {String} modName module name
  3384. * @param runtime Module container, such as KISSY
  3385. * @param {String[]} [stack] stack for detecting circular dependency
  3386. * @param {Array} [errorList] errors when attach mods
  3387. * @param {Object} [cache] cached modules to avoid duplicate check
  3388. * @returns whether success attach all modules
  3389. */
  3390. attachModRecursively: function (modName, runtime, stack, errorList, cache) {
  3391. var mods = runtime.Env.mods,
  3392. status,
  3393. m = mods[modName];
  3394. if (modName in cache) {
  3395. return cache[modName];
  3396. }
  3397. if (!m) {
  3398. return cache[modName] = 0;
  3399. }
  3400. status = m.status;
  3401. if (status == ATTACHED) {
  3402. return cache[modName] = 1;
  3403. }
  3404. if (status == ERROR) {
  3405. errorList.push(m);
  3406. }
  3407. if (status != LOADED) {
  3408. return cache[modName] = 0;
  3409. }
  3410. if ('@DEBUG@') {
  3411. if (S.inArray(modName, stack)) {
  3412. stack.push(modName);
  3413. S.error('find cyclic dependency between mods: ' + stack);
  3414. return cache[modName] = 0;
  3415. }
  3416. stack.push(modName);
  3417. }
  3418. if (Utils.attachModsRecursively(m.getNormalizedRequires(),
  3419. runtime, stack, errorList, cache)) {
  3420. Utils.attachMod(runtime, m);
  3421. return cache[modName] = 1;
  3422. }
  3423. return cache[modName] = 0;
  3424. },
  3425. /**
  3426. * Attach specified mod.
  3427. * @param runtime Module container, such as KISSY
  3428. * @param {KISSY.Loader.Module} mod module instance
  3429. */
  3430. attachMod: function (runtime, mod) {
  3431. if (mod.status != LOADED) {
  3432. return;
  3433. }
  3434. var fn = mod.fn;
  3435. if (typeof fn === 'function') {
  3436. // 需要解开 index,相对路径
  3437. // 但是需要保留 alias,防止值不对应
  3438. mod.value = fn.apply(mod, Utils.getModules(runtime, mod.getRequiresWithAlias()));
  3439. } else {
  3440. mod.value = fn;
  3441. }
  3442. mod.status = ATTACHED;
  3443. },
  3444. /**
  3445. * Get mod names as array.
  3446. * @param {String|String[]} modNames module names array or module names string separated by ','
  3447. * @return {String[]}
  3448. */
  3449. getModNamesAsArray: function (modNames) {
  3450. if (typeof modNames == 'string') {
  3451. modNames = modNames.replace(/\s+/g, '').split(',');
  3452. }
  3453. return modNames;
  3454. },
  3455. /**
  3456. * normalize module names
  3457. * 1. add index : / => /index
  3458. * 2. unalias : core => dom,event,ua
  3459. * 3. relative to absolute : ./x => y/x
  3460. * @param {KISSY} runtime Global KISSY instance
  3461. * @param {String|String[]} modNames Array of module names
  3462. * or module names string separated by comma
  3463. * @param {String} [refModName]
  3464. * @return {String[]} normalized module names
  3465. */
  3466. normalizeModNames: function (runtime, modNames, refModName) {
  3467. return Utils.unalias(runtime,
  3468. Utils.normalizeModNamesWithAlias(runtime, modNames, refModName));
  3469. },
  3470. /**
  3471. * unalias module name.
  3472. * @param runtime Module container, such as KISSY
  3473. * @param {String} names moduleNames
  3474. * @return {String[]} unaliased module names
  3475. */
  3476. unalias: function (runtime, names) {
  3477. var ret = [].concat(names),
  3478. i,
  3479. m,
  3480. alias,
  3481. ok = 0,
  3482. j,
  3483. mods = runtime['Env'].mods;
  3484. while (!ok) {
  3485. ok = 1;
  3486. for (i = ret.length - 1; i >= 0; i--) {
  3487. if ((m = mods[ret[i]]) && (alias = m.alias)) {
  3488. ok = 0;
  3489. for (j = alias.length - 1; j >= 0; j--) {
  3490. if (!alias[j]) {
  3491. alias.splice(j, 1);
  3492. }
  3493. }
  3494. ret.splice.apply(ret, [i, 1].concat(indexMap(alias)));
  3495. }
  3496. }
  3497. }
  3498. return ret;
  3499. },
  3500. /**
  3501. * normalize module names with alias
  3502. * @param runtime Module container, such as KISSY
  3503. * @param {String[]} modNames module names
  3504. * @param [refModName] module to be referred if module name path is relative
  3505. * @return {String[]} normalize module names with alias
  3506. */
  3507. normalizeModNamesWithAlias: function (runtime, modNames, refModName) {
  3508. var ret = [], i, l;
  3509. if (modNames) {
  3510. // 1. index map
  3511. for (i = 0, l = modNames.length; i < l; i++) {
  3512. // conditional loader
  3513. // requires:[window.localStorage?"local-storage":""]
  3514. if (modNames[i]) {
  3515. ret.push(pluginAlias(runtime, indexMap(modNames[i])));
  3516. }
  3517. }
  3518. }
  3519. // 2. relative to absolute (optional)
  3520. if (refModName) {
  3521. ret = Utils.normalDepModuleName(refModName, ret);
  3522. }
  3523. return ret;
  3524. },
  3525. /**
  3526. * register module with factory
  3527. * @param runtime Module container, such as KISSY
  3528. * @param {String} name module name
  3529. * @param {Function|*} fn module's factory or value
  3530. * @param [config] module config, such as dependency
  3531. */
  3532. registerModule: function (runtime, name, fn, config) {
  3533. name = indexMapStr(name);
  3534. var mods = runtime.Env.mods,
  3535. mod = mods[name];
  3536. if (mod && mod.fn) {
  3537. S.log(name + ' is defined more than once', 'warn');
  3538. return;
  3539. }
  3540. // 没有 use,静态载入的 add 可能执行
  3541. Utils.createModuleInfo(runtime, name);
  3542. mod = mods[name];
  3543. // 注意:通过 S.add(name[, fn[, config]]) 注册的代码,无论是页面中的代码,
  3544. // 还是 js 文件里的代码,add 执行时,都意味着该模块已经 LOADED
  3545. S.mix(mod, {
  3546. name: name,
  3547. status: LOADED,
  3548. fn: fn
  3549. });
  3550. S.mix(mod, config);
  3551. },
  3552. /**
  3553. * Get mapped path.
  3554. * @param runtime Module container, such as KISSY
  3555. * @param {String} path module path
  3556. * @param [rules] map rules
  3557. * @return {String} mapped path
  3558. */
  3559. getMappedPath: function (runtime, path, rules) {
  3560. var mappedRules = rules ||
  3561. runtime.Config.mappedRules ||
  3562. [],
  3563. i,
  3564. m,
  3565. rule;
  3566. for (i = 0; i < mappedRules.length; i++) {
  3567. rule = mappedRules[i];
  3568. if (m = path.match(rule[0])) {
  3569. return path.replace(rule[0], rule[1]);
  3570. }
  3571. }
  3572. return path;
  3573. },
  3574. /**
  3575. * Returns hash code of a string djb2 algorithm
  3576. * @param {String} str
  3577. * @returns {String} hash code
  3578. */
  3579. getHash:function (str) {
  3580. var hash = 5381,
  3581. i;
  3582. for (i = str.length; --i > -1;) {
  3583. hash = ((hash << 5) + hash) + str.charCodeAt(i);
  3584. /* hash * 33 + char */
  3585. }
  3586. return hash + '';
  3587. }
  3588. });
  3589. function isStatus(runtime, modNames, status) {
  3590. var mods = runtime.Env.mods,
  3591. mod,
  3592. i;
  3593. modNames = S.makeArray(modNames);
  3594. for (i = 0; i < modNames.length; i++) {
  3595. mod = mods[modNames[i]];
  3596. if (!mod || mod.status !== status) {
  3597. return 0;
  3598. }
  3599. }
  3600. return 1;
  3601. }
  3602. })(KISSY);/**
  3603. * @ignore
  3604. * setup data structure for kissy loader
  3605. * @author yiminghe@gmail.com
  3606. */
  3607. (function (S) {
  3608. var Loader = S.Loader,
  3609. Path = S.Path,
  3610. IGNORE_PACKAGE_NAME_IN_URI = 'ignorePackageNameInUri',
  3611. Utils = Loader.Utils;
  3612. function forwardSystemPackage(self, property) {
  3613. return property in self ?
  3614. self[property] :
  3615. self.runtime.Config[property];
  3616. }
  3617. /**
  3618. * @class KISSY.Loader.Package
  3619. * @private
  3620. * This class should not be instantiated manually.
  3621. */
  3622. function Package(cfg) {
  3623. S.mix(this, cfg);
  3624. }
  3625. S.augment(Package, {
  3626. reset: function (cfg) {
  3627. S.mix(this, cfg);
  3628. },
  3629. /**
  3630. * Tag for package.
  3631. * tag can not contain ".", eg: Math.random() !
  3632. * @return {String}
  3633. */
  3634. getTag: function () {
  3635. return forwardSystemPackage(this, 'tag');
  3636. },
  3637. /**
  3638. * Get package name.
  3639. * @return {String}
  3640. */
  3641. getName: function () {
  3642. return this.name;
  3643. },
  3644. /**
  3645. * Get package base.
  3646. * @return {String}
  3647. */
  3648. 'getBase': function () {
  3649. return forwardSystemPackage(this, 'base');
  3650. },
  3651. getPrefixUriForCombo: function () {
  3652. var self = this,
  3653. packageName = self.getName();
  3654. return self.getBase() + (
  3655. packageName && !self.isIgnorePackageNameInUri() ?
  3656. (packageName + '/') :
  3657. ''
  3658. );
  3659. },
  3660. /**
  3661. * get package uri
  3662. */
  3663. getPackageUri: function () {
  3664. var self = this;
  3665. if (self.packageUri) {
  3666. return self.packageUri;
  3667. }
  3668. return self.packageUri = new S.Uri(this.getPrefixUriForCombo());
  3669. },
  3670. /**
  3671. * Get package baseUri
  3672. * @return {KISSY.Uri}
  3673. */
  3674. getBaseUri: function () {
  3675. return forwardSystemPackage(this, 'baseUri');
  3676. },
  3677. /**
  3678. * Whether is debug for this package.
  3679. * @return {Boolean}
  3680. */
  3681. isDebug: function () {
  3682. return forwardSystemPackage(this, 'debug');
  3683. },
  3684. /**
  3685. * whether request mod file without insert package name into package base
  3686. * @return {Boolean}
  3687. */
  3688. isIgnorePackageNameInUri: function () {
  3689. return forwardSystemPackage(this, IGNORE_PACKAGE_NAME_IN_URI);
  3690. },
  3691. /**
  3692. * Get charset for package.
  3693. * @return {String}
  3694. */
  3695. getCharset: function () {
  3696. return forwardSystemPackage(this, 'charset');
  3697. },
  3698. /**
  3699. * Whether modules are combined for this package.
  3700. * @return {Boolean}
  3701. */
  3702. isCombine: function () {
  3703. return forwardSystemPackage(this, 'combine');
  3704. },
  3705. /**
  3706. * Get package group (for combo).
  3707. * @returns {String}
  3708. */
  3709. getGroup: function () {
  3710. return forwardSystemPackage(this, 'group');
  3711. }
  3712. });
  3713. Loader.Package = Package;
  3714. /**
  3715. * @class KISSY.Loader.Module
  3716. * @private
  3717. * This class should not be instantiated manually.
  3718. */
  3719. function Module(cfg) {
  3720. this.status = Loader.Status.INIT;
  3721. S.mix(this, cfg);
  3722. this.waitedCallbacks = [];
  3723. }
  3724. S.augment(Module, {
  3725. wait: function (callback) {
  3726. this.waitedCallbacks.push(callback);
  3727. },
  3728. notifyAll: function () {
  3729. var callback;
  3730. var len = this.waitedCallbacks.length,
  3731. i = 0;
  3732. for (; i < len; i++) {
  3733. callback = this.waitedCallbacks[i];
  3734. try {
  3735. callback(this);
  3736. } catch (e) {
  3737. S.log(e.stack || e, 'error');
  3738. setTimeout(function () {
  3739. throw e;
  3740. }, 0);
  3741. }
  3742. }
  3743. this.waitedCallbacks = [];
  3744. },
  3745. /**
  3746. * Set the value of current module
  3747. * @param v value to be set
  3748. */
  3749. 'setValue': function (v) {
  3750. this.value = v;
  3751. },
  3752. /**
  3753. * Get the type if current Module
  3754. * @return {String} css or js
  3755. */
  3756. getType: function () {
  3757. var self = this,
  3758. v = self.type;
  3759. if (!v) {
  3760. if (Path.extname(self.name).toLowerCase() == '.css') {
  3761. v = 'css';
  3762. } else {
  3763. v = 'js';
  3764. }
  3765. self.type = v;
  3766. }
  3767. return v;
  3768. },
  3769. /**
  3770. * Get the fullpath uri of current module if load dynamically
  3771. * @return {KISSY.Uri}
  3772. */
  3773. getFullPathUri: function () {
  3774. var self = this,
  3775. t,
  3776. fullPathUri,
  3777. packageBaseUri,
  3778. packageInfo,
  3779. packageName,
  3780. path;
  3781. if (!self.fullPathUri) {
  3782. if (self.fullpath) {
  3783. fullPathUri = new S.Uri(self.fullpath);
  3784. } else {
  3785. packageInfo = self.getPackage();
  3786. packageBaseUri = packageInfo.getBaseUri();
  3787. path = self.getPath();
  3788. // #262
  3789. if (packageInfo.isIgnorePackageNameInUri() &&
  3790. // native mod does not allow ignore package name
  3791. (packageName = packageInfo.getName())) {
  3792. path = Path.relative(packageName, path);
  3793. }
  3794. fullPathUri = packageBaseUri.resolve(path);
  3795. if (t = self.getTag()) {
  3796. t += '.' + self.getType();
  3797. fullPathUri.query.set('t', t);
  3798. }
  3799. }
  3800. self.fullPathUri = fullPathUri;
  3801. }
  3802. return self.fullPathUri;
  3803. },
  3804. /**
  3805. * Get the fullpath of current module if load dynamically
  3806. * @return {String}
  3807. */
  3808. getFullPath: function () {
  3809. var self = this,
  3810. fullPathUri;
  3811. if (!self.fullpath) {
  3812. fullPathUri = self.getFullPathUri();
  3813. self.fullpath = Utils.getMappedPath(self.runtime, fullPathUri.toString());
  3814. }
  3815. return self.fullpath;
  3816. },
  3817. /**
  3818. * Get the path (without package base)
  3819. * @return {String}
  3820. */
  3821. getPath: function () {
  3822. var self = this;
  3823. return self.path ||
  3824. (self.path = defaultComponentJsName(self))
  3825. },
  3826. /**
  3827. * Get the value of current module
  3828. * @return {*}
  3829. */
  3830. getValue: function () {
  3831. return this.value;
  3832. },
  3833. /**
  3834. * Get the name of current module
  3835. * @return {String}
  3836. */
  3837. getName: function () {
  3838. return this.name;
  3839. },
  3840. /**
  3841. * Get the package which current module belongs to.
  3842. * @return {KISSY.Loader.Package}
  3843. */
  3844. getPackage: function () {
  3845. var self = this;
  3846. return self.packageInfo ||
  3847. (self.packageInfo = getPackage(self.runtime, self.name));
  3848. },
  3849. /**
  3850. * Get the tag of current module.
  3851. * tag can not contain ".", eg: Math.random() !
  3852. * @return {String}
  3853. */
  3854. getTag: function () {
  3855. var self = this;
  3856. return self.tag || self.getPackage().getTag();
  3857. },
  3858. /**
  3859. * Get the charset of current module
  3860. * @return {String}
  3861. */
  3862. getCharset: function () {
  3863. var self = this;
  3864. return self.charset || self.getPackage().getCharset();
  3865. },
  3866. /**
  3867. * Get module objects required by this module
  3868. * @return {KISSY.Loader.Module[]}
  3869. */
  3870. 'getRequiredMods': function () {
  3871. var self = this,
  3872. runtime = self.runtime;
  3873. return S.map(self.getNormalizedRequires(), function (r) {
  3874. return Utils.createModuleInfo(runtime, r);
  3875. });
  3876. },
  3877. getRequiresWithAlias: function () {
  3878. var self = this,
  3879. requiresWithAlias = self.requiresWithAlias,
  3880. requires = self.requires;
  3881. if (!requires || requires.length == 0) {
  3882. return requires || [];
  3883. } else if (!requiresWithAlias) {
  3884. self.requiresWithAlias = requiresWithAlias =
  3885. Utils.normalizeModNamesWithAlias(self.runtime, requires, self.name);
  3886. }
  3887. return requiresWithAlias;
  3888. },
  3889. /**
  3890. * Get module names required by this module
  3891. * @return {KISSY.Loader.Module[]}
  3892. */
  3893. getNormalizedRequires: function () {
  3894. var self = this,
  3895. normalizedRequires,
  3896. normalizedRequiresStatus = self.normalizedRequiresStatus,
  3897. status = self.status,
  3898. requires = self.requires;
  3899. if (!requires || requires.length == 0) {
  3900. return requires || [];
  3901. } else if ((normalizedRequires = self.normalizedRequires) &&
  3902. // 事先声明的依赖不能当做 loaded 状态下真正的依赖
  3903. (normalizedRequiresStatus == status)) {
  3904. return normalizedRequires;
  3905. } else {
  3906. self.normalizedRequiresStatus = status;
  3907. return self.normalizedRequires =
  3908. Utils.normalizeModNames(self.runtime, requires, self.name);
  3909. }
  3910. }
  3911. });
  3912. Loader.Module = Module;
  3913. function defaultComponentJsName(m) {
  3914. var name = m.name,
  3915. extname = '.' + m.getType(),
  3916. min = '-min';
  3917. name = Path.join(Path.dirname(name), Path.basename(name, extname));
  3918. if (m.getPackage().isDebug()) {
  3919. min = '';
  3920. }
  3921. return name + min + extname;
  3922. }
  3923. var systemPackage = new Package({
  3924. name: '',
  3925. runtime: S
  3926. });
  3927. function getPackage(self, modName) {
  3928. var packages = self.config('packages'),
  3929. modNameSlash = modName + '/',
  3930. pName = '',
  3931. p;
  3932. for (p in packages) {
  3933. if (S.startsWith(modNameSlash, p + '/') && p.length > pName.length) {
  3934. pName = p;
  3935. }
  3936. }
  3937. return packages[pName] || systemPackage;
  3938. }
  3939. })(KISSY);/**
  3940. * @ignore
  3941. * script/css load across browser
  3942. * @author yiminghe@gmail.com
  3943. */
  3944. (function (S) {
  3945. var CSS_POLL_INTERVAL = 30,
  3946. UA = S.UA,
  3947. logger = S.getLogger('s/loader/getScript'),
  3948. Utils = S.Loader.Utils,
  3949. // central poll for link node
  3950. timer = 0,
  3951. monitors = {
  3952. // node.id:{callback:callback,node:node}
  3953. };
  3954. function startCssTimer() {
  3955. if (!timer) {
  3956. logger.debug('start css poll timer');
  3957. cssPoll();
  3958. }
  3959. }
  3960. function isCssLoaded(node, url) {
  3961. var loaded = 0;
  3962. if (UA.webkit) {
  3963. // http://www.w3.org/TR/Dom-Level-2-Style/stylesheets.html
  3964. if (node['sheet']) {
  3965. logger.debug('webkit css poll loaded: ' + url);
  3966. loaded = 1;
  3967. }
  3968. } else if (node['sheet']) {
  3969. try {
  3970. var cssRules = node['sheet'].cssRules;
  3971. if (cssRules) {
  3972. logger.debug('same domain css poll loaded: ' + url);
  3973. loaded = 1;
  3974. }
  3975. } catch (ex) {
  3976. var exName = ex.name;
  3977. logger.debug('css poll exception: ' + exName + ' ' + ex.code + ' ' + url);
  3978. // http://www.w3.org/TR/dom/#dom-domexception-code
  3979. if (// exName == 'SecurityError' ||
  3980. // for old firefox
  3981. exName == 'NS_ERROR_DOM_SECURITY_ERR') {
  3982. logger.debug('css poll exception: ' + exName + 'loaded : ' + url);
  3983. loaded = 1;
  3984. }
  3985. }
  3986. }
  3987. return loaded;
  3988. }
  3989. // single thread is ok
  3990. function cssPoll() {
  3991. for (var url in monitors) {
  3992. var callbackObj = monitors[url],
  3993. node = callbackObj.node;
  3994. if (isCssLoaded(node, url)) {
  3995. if (callbackObj.callback) {
  3996. callbackObj.callback.call(node);
  3997. }
  3998. delete monitors[url];
  3999. }
  4000. }
  4001. if (S.isEmptyObject(monitors)) {
  4002. logger.debug('clear css poll timer');
  4003. timer = 0;
  4004. } else {
  4005. timer = setTimeout(cssPoll, CSS_POLL_INTERVAL);
  4006. }
  4007. }
  4008. // refer : http://lifesinger.org/lab/2011/load-js-css/css-preload.html
  4009. // 暂时不考虑如何判断失败,如 404 等
  4010. Utils.pollCss = function (node, callback) {
  4011. var href = node.href,
  4012. arr;
  4013. arr = monitors[href] = {};
  4014. arr.node = node;
  4015. arr.callback = callback;
  4016. startCssTimer();
  4017. };
  4018. Utils.isCssLoaded = isCssLoaded;
  4019. })(KISSY);
  4020. /*
  4021. References:
  4022. - http://unixpapa.com/js/dyna.html
  4023. - http://www.blaze.io/technical/ies-premature-execution-problem/
  4024. `onload` event is supported in WebKit since 535.23
  4025. - https://bugs.webkit.org/show_activity.cgi?id=38995
  4026. `onload/onerror` event is supported since Firefox 9.0
  4027. - https://bugzilla.mozilla.org/show_bug.cgi?id=185236
  4028. - https://developer.mozilla.org/en/HTML/Element/link#Stylesheet_load_events
  4029. monitor css onload across browsers.issue about 404 failure.
  4030. - firefox not ok(4 is wrong):
  4031. - http://yearofmoo.com/2011/03/cross-browser-stylesheet-preloading/
  4032. - all is ok
  4033. - http://lifesinger.org/lab/2011/load-js-css/css-preload.html
  4034. - others
  4035. - http://www.zachleat.com/web/load-css-dynamically/
  4036. *//**
  4037. * @ignore
  4038. * getScript support for css and js callback after load
  4039. * @author yiminghe@gmail.com
  4040. */
  4041. (function (S) {
  4042. var MILLISECONDS_OF_SECOND = 1000,
  4043. doc = S.Env.host.document,
  4044. Utils = S.Loader.Utils,
  4045. Path = S.Path,
  4046. jsCssCallbacks = {},
  4047. headNode,
  4048. UA = S.UA;
  4049. /**
  4050. * Load a javascript/css file from the server using a GET HTTP request,
  4051. * then execute it.
  4052. *
  4053. * for example:
  4054. * @example
  4055. * getScript(url, success, charset);
  4056. * // or
  4057. * getScript(url, {
  4058. * charset: string
  4059. * success: fn,
  4060. * error: fn,
  4061. * timeout: number
  4062. * });
  4063. *
  4064. * Note 404/500 status in ie<9 will trigger success callback.
  4065. * If you want a jsonp operation, please use {@link KISSY.IO} instead.
  4066. *
  4067. * @param {String} url resource's url
  4068. * @param {Function|Object} [success] success callback or config
  4069. * @param {Function} [success.success] success callback
  4070. * @param {Function} [success.error] error callback
  4071. * @param {Number} [success.timeout] timeout (s)
  4072. * @param {String} [success.charset] charset of current resource
  4073. * @param {String} [charset] charset of current resource
  4074. * @return {HTMLElement} script/style node
  4075. * @member KISSY
  4076. */
  4077. S.getScript = function (url, success, charset) {
  4078. // can not use KISSY.Uri, url can not be encoded for some url
  4079. // eg: /??dom.js,event.js , ? , should not be encoded
  4080. var config = success,
  4081. css = 0,
  4082. error,
  4083. timeout,
  4084. attrs,
  4085. callbacks,
  4086. timer;
  4087. if (S.startsWith(Path.extname(url).toLowerCase(), '.css')) {
  4088. css = 1;
  4089. }
  4090. if (S.isPlainObject(config)) {
  4091. success = config['success'];
  4092. error = config['error'];
  4093. timeout = config['timeout'];
  4094. charset = config['charset'];
  4095. attrs = config['attrs'];
  4096. }
  4097. callbacks = jsCssCallbacks[url] = jsCssCallbacks[url] || [];
  4098. callbacks.push([success, error]);
  4099. if (callbacks.length > 1) {
  4100. return callbacks.node;
  4101. }
  4102. var node = doc.createElement(css ? 'link' : 'script'),
  4103. clearTimer = function () {
  4104. if (timer) {
  4105. timer.cancel();
  4106. timer = undefined;
  4107. }
  4108. };
  4109. if (attrs) {
  4110. S.each(attrs, function (v, n) {
  4111. node.setAttribute(n, v);
  4112. });
  4113. }
  4114. if (charset) {
  4115. node.charset = charset;
  4116. }
  4117. if (css) {
  4118. node.href = url;
  4119. node.rel = 'stylesheet';
  4120. } else {
  4121. node.src = url;
  4122. node.async = true;
  4123. }
  4124. callbacks.node = node;
  4125. var end = function (error) {
  4126. var index = error,
  4127. fn;
  4128. clearTimer();
  4129. S.each(jsCssCallbacks[url], function (callback) {
  4130. if (fn = callback[index]) {
  4131. fn.call(node);
  4132. }
  4133. });
  4134. delete jsCssCallbacks[url];
  4135. };
  4136. var useNative = 'onload' in node;
  4137. // onload for webkit 535.23 Firefox 9.0
  4138. // https://bugs.webkit.org/show_activity.cgi?id=38995
  4139. // https://bugzilla.mozilla.org/show_bug.cgi?id=185236
  4140. // https://developer.mozilla.org/en/HTML/Element/link#Stylesheet_load_events
  4141. // phantomjs 1.7 == webkit 534.34
  4142. var forceCssPoll = S.Config['forceCssPoll'] || (UA.webkit && UA.webkit < 536);
  4143. if (css && forceCssPoll && useNative) {
  4144. useNative = false;
  4145. }
  4146. function onload() {
  4147. var readyState = node.readyState;
  4148. if (!readyState ||
  4149. readyState == "loaded" ||
  4150. readyState == "complete") {
  4151. node.onreadystatechange = node.onload = null;
  4152. end(0);
  4153. }
  4154. }
  4155. //标准浏览器 css and all script
  4156. if (useNative) {
  4157. node.onload = onload;
  4158. node.onerror = function () {
  4159. node.onerror = null;
  4160. end(1);
  4161. };
  4162. }
  4163. // old chrome/firefox for css
  4164. else if (css) {
  4165. Utils.pollCss(node, function () {
  4166. end(0);
  4167. });
  4168. } else {
  4169. node.onreadystatechange = onload;
  4170. }
  4171. if (timeout) {
  4172. timer = S.later(function () {
  4173. end(1);
  4174. }, timeout * MILLISECONDS_OF_SECOND);
  4175. }
  4176. if (!headNode) {
  4177. headNode = Utils.docHead();
  4178. }
  4179. if (css) {
  4180. // css order matters
  4181. // so can not use css in head
  4182. headNode.appendChild(node);
  4183. } else {
  4184. // can use js in head
  4185. headNode.insertBefore(node, headNode.firstChild);
  4186. }
  4187. return node;
  4188. };
  4189. })(KISSY);
  4190. /*
  4191. yiminghe@gmail.com refactor@2012-03-29
  4192. - 考虑连续重复请求单个 script 的情况,内部排队
  4193. yiminghe@gmail.com 2012-03-13
  4194. - getScript
  4195. - 404 in ie<9 trigger success , others trigger error
  4196. - syntax error in all trigger success
  4197. *//**
  4198. * @ignore
  4199. * Declare config info for KISSY.
  4200. * @author yiminghe@gmail.com
  4201. */
  4202. (function (S, undefined) {
  4203. var Loader = S.Loader,
  4204. Utils = Loader.Utils,
  4205. host = S.Env.host,
  4206. location = host.location,
  4207. simulatedLocation,
  4208. locationHref,
  4209. configFns = S.Config.fns;
  4210. if (!S.UA.nodejs && location && (locationHref = location.href)) {
  4211. simulatedLocation = new S.Uri(locationHref)
  4212. }
  4213. S.Config.loadModsFn = function (rs, config) {
  4214. S.getScript(rs.fullpath, config);
  4215. };
  4216. configFns.map = function (rules) {
  4217. var Config = this.Config;
  4218. if (rules === false) {
  4219. return Config.mappedRules = [];
  4220. }
  4221. return Config.mappedRules = (Config.mappedRules || []).concat(rules || []);
  4222. };
  4223. configFns.mapCombo = function (rules) {
  4224. var Config = this.Config;
  4225. if (rules === false) {
  4226. return Config.mappedComboRules = [];
  4227. }
  4228. return Config.mappedComboRules = (Config.mappedComboRules || []).concat(rules || []);
  4229. };
  4230. configFns.packages = function (config) {
  4231. var name,
  4232. Config = this.Config,
  4233. ps = Config.packages = Config.packages || {};
  4234. if (config) {
  4235. S.each(config, function (cfg, key) {
  4236. // 兼容数组方式
  4237. name = cfg.name || key;
  4238. // 兼容 path
  4239. var baseUri = normalizeBase(cfg.base || cfg.path);
  4240. cfg.name = name;
  4241. cfg.base = baseUri.toString();
  4242. cfg.baseUri = baseUri;
  4243. cfg.runtime = S;
  4244. delete cfg.path;
  4245. if (ps[name]) {
  4246. ps[name].reset(cfg);
  4247. } else {
  4248. ps[name] = new Loader.Package(cfg);
  4249. }
  4250. });
  4251. return undefined;
  4252. } else if (config === false) {
  4253. Config.packages = {
  4254. };
  4255. return undefined;
  4256. } else {
  4257. return ps;
  4258. }
  4259. };
  4260. configFns.modules = function (modules) {
  4261. var self = this;
  4262. if (modules) {
  4263. S.each(modules, function (modCfg, modName) {
  4264. var mod = Utils.createModuleInfo(self, modName, modCfg);
  4265. // #485, invalid after add
  4266. if (mod.status == Loader.Status.INIT) {
  4267. S.mix(mod, modCfg);
  4268. }
  4269. });
  4270. }
  4271. };
  4272. configFns.base = function (base) {
  4273. var self = this,
  4274. Config = self.Config,
  4275. baseUri;
  4276. if (!base) {
  4277. return Config.base;
  4278. }
  4279. baseUri = normalizeBase(base);
  4280. Config.base = baseUri.toString();
  4281. Config.baseUri = baseUri;
  4282. return undefined;
  4283. };
  4284. function normalizeBase(base) {
  4285. var baseUri;
  4286. base = base.replace(/\\/g, '/');
  4287. if (base.charAt(base.length - 1) != '/') {
  4288. base += '/';
  4289. }
  4290. if (simulatedLocation) {
  4291. baseUri = simulatedLocation.resolve(base);
  4292. } else {
  4293. // add scheme for S.Uri
  4294. // currently only for nodejs
  4295. if (!S.startsWith(base, 'file:')) {
  4296. base = 'file:' + base;
  4297. }
  4298. baseUri = new S.Uri(base);
  4299. }
  4300. return baseUri;
  4301. }
  4302. })(KISSY);
  4303. /**
  4304. * combo loader for KISSY. using combo to load module files.
  4305. * @ignore
  4306. * @author yiminghe@gmail.com
  4307. */
  4308. (function (S, undefined) {
  4309. // ie11 is a new one!
  4310. var oldIE = S.UA.ie && S.UA.ie < 10;
  4311. function loadScripts(runtime, rss, callback, charset, timeout) {
  4312. var count = rss && rss.length,
  4313. errorList = [],
  4314. successList = [];
  4315. function complete() {
  4316. if (!(--count)) {
  4317. callback(successList, errorList);
  4318. }
  4319. }
  4320. S.each(rss, function (rs) {
  4321. var mod;
  4322. var config = {
  4323. timeout: timeout,
  4324. success: function () {
  4325. successList.push(rs);
  4326. if (mod && currentMod) {
  4327. // standard browser(except ie9) fire load after KISSY.add immediately
  4328. logger.debug('standard browser get mod name after load : ' + mod.name);
  4329. Utils.registerModule(runtime, mod.name, currentMod.fn, currentMod.config);
  4330. currentMod = undefined;
  4331. }
  4332. complete();
  4333. },
  4334. error: function () {
  4335. errorList.push(rs);
  4336. complete();
  4337. },
  4338. charset: charset
  4339. };
  4340. if (!rs.combine) {
  4341. mod = rs.mods[0];
  4342. if (mod.getType() == 'css') {
  4343. mod = undefined;
  4344. }
  4345. else if (oldIE) {
  4346. startLoadModName = mod.name;
  4347. startLoadModTime = S.now();
  4348. config.attrs = {
  4349. 'data-mod-name': mod.name
  4350. };
  4351. }
  4352. }
  4353. S.Config.loadModsFn(rs, config);
  4354. });
  4355. }
  4356. var Loader = S.Loader,
  4357. logger = S.getLogger('s/loader'),
  4358. Status = Loader.Status,
  4359. Utils = Loader.Utils,
  4360. getHash=Utils.getHash,
  4361. LOADING = Status.LOADING,
  4362. LOADED = Status.LOADED,
  4363. ERROR = Status.ERROR,
  4364. groupTag = S.now(),
  4365. ATTACHED = Status.ATTACHED;
  4366. ComboLoader.groupTag = groupTag;
  4367. /**
  4368. * @class KISSY.Loader.ComboLoader
  4369. * using combo to load module files
  4370. * @param runtime KISSY
  4371. * @param waitingModules
  4372. * @private
  4373. */
  4374. function ComboLoader(runtime, waitingModules) {
  4375. S.mix(this, {
  4376. runtime: runtime,
  4377. waitingModules: waitingModules
  4378. });
  4379. }
  4380. var currentMod;
  4381. var startLoadModName;
  4382. var startLoadModTime;
  4383. ComboLoader.add = function (name, fn, config, runtime) {
  4384. if (typeof name === 'function') {
  4385. config = fn;
  4386. fn = name;
  4387. if (oldIE) {
  4388. // http://groups.google.com/group/commonjs/browse_thread/thread/5a3358ece35e688e/43145ceccfb1dc02#43145ceccfb1dc02
  4389. name = findModuleNameByInteractive();
  4390. // S.log('oldIE get modName by interactive: ' + name);
  4391. Utils.registerModule(runtime, name, fn, config);
  4392. startLoadModName = null;
  4393. startLoadModTime = 0;
  4394. } else {
  4395. // 其他浏览器 onload 时,关联模块名与模块定义
  4396. currentMod = {
  4397. fn: fn,
  4398. config: config
  4399. };
  4400. }
  4401. } else {
  4402. if (oldIE) {
  4403. startLoadModName = null;
  4404. startLoadModTime = 0;
  4405. } else {
  4406. currentMod = undefined;
  4407. }
  4408. Utils.registerModule(runtime, name, fn, config);
  4409. }
  4410. };
  4411. // oldIE 特有,找到当前正在交互的脚本,根据脚本名确定模块名
  4412. // 如果找不到,返回发送前那个脚本
  4413. function findModuleNameByInteractive() {
  4414. var scripts = S.Env.host.document.getElementsByTagName('script'),
  4415. re,
  4416. i,
  4417. name,
  4418. script;
  4419. for (i = scripts.length - 1; i >= 0; i--) {
  4420. script = scripts[i];
  4421. if (script.readyState == 'interactive') {
  4422. re = script;
  4423. break;
  4424. }
  4425. }
  4426. if (re) {
  4427. name = re.getAttribute('data-mod-name');
  4428. } else {
  4429. // sometimes when read module file from cache,
  4430. // interactive status is not triggered
  4431. // module code is executed right after inserting into dom
  4432. // i has to preserve module name before insert module script into dom,
  4433. // then get it back here
  4434. logger.debug('can not find interactive script,time diff : ' + (S.now() - startLoadModTime));
  4435. logger.debug('old_ie get mod name from cache : ' + startLoadModName);
  4436. name = startLoadModName;
  4437. }
  4438. return name;
  4439. }
  4440. function debugRemoteModules(rss) {
  4441. S.each(rss, function (rs) {
  4442. var ms = [];
  4443. S.each(rs.mods, function (m) {
  4444. if (m.status == LOADED) {
  4445. ms.push(m.name);
  4446. }
  4447. });
  4448. if (ms.length) {
  4449. logger.info('load remote modules: "' + ms.join(', ') + '" from: "' + rs.fullpath + '"');
  4450. }
  4451. });
  4452. }
  4453. function getCommonPrefix(str1, str2) {
  4454. str1 = str1.split(/\//);
  4455. str2 = str2.split(/\//);
  4456. var l = Math.min(str1.length, str2.length);
  4457. for (var i = 0; i < l; i++) {
  4458. if (str1[i] !== str2[i]) {
  4459. break;
  4460. }
  4461. }
  4462. return str1.slice(0, i).join('/') + '/';
  4463. }
  4464. S.augment(ComboLoader, {
  4465. /**
  4466. * load modules asynchronously
  4467. */
  4468. use: function (normalizedModNames) {
  4469. var self = this,
  4470. allModNames,
  4471. comboUrls,
  4472. timeout = S.Config.timeout,
  4473. runtime = self.runtime;
  4474. allModNames = S.keys(self.calculate(normalizedModNames));
  4475. Utils.createModulesInfo(runtime, allModNames);
  4476. comboUrls = self.getComboUrls(allModNames);
  4477. // load css first to avoid page blink
  4478. S.each(comboUrls.css, function (cssOne) {
  4479. loadScripts(runtime, cssOne, function (success, error) {
  4480. if ('@DEBUG@') {
  4481. debugRemoteModules(success);
  4482. }
  4483. S.each(success, function (one) {
  4484. S.each(one.mods, function (mod) {
  4485. Utils.registerModule(runtime, mod.getName(), S.noop);
  4486. // notify all loader instance
  4487. mod.notifyAll();
  4488. });
  4489. });
  4490. S.each(error, function (one) {
  4491. S.each(one.mods, function (mod) {
  4492. var msg = mod.name +
  4493. ' is not loaded! can not find module in path : ' +
  4494. one.fullpath;
  4495. S.log(msg, 'error');
  4496. mod.status = ERROR;
  4497. // notify all loader instance
  4498. mod.notifyAll();
  4499. });
  4500. });
  4501. }, cssOne.charset, timeout);
  4502. });
  4503. // jss css download in parallel
  4504. S.each(comboUrls['js'], function (jsOne) {
  4505. loadScripts(runtime, jsOne, function (success) {
  4506. if ('@DEBUG@') {
  4507. debugRemoteModules(success);
  4508. }
  4509. S.each(jsOne, function (one) {
  4510. S.each(one.mods, function (mod) {
  4511. // fix #111
  4512. // https://github.com/kissyteam/kissy/issues/111
  4513. if (!mod.fn) {
  4514. var msg = mod.name +
  4515. ' is not loaded! can not find module in path : ' +
  4516. one.fullpath;
  4517. S.log(msg, 'error');
  4518. mod.status = ERROR;
  4519. }
  4520. // notify all loader instance
  4521. mod.notifyAll();
  4522. });
  4523. });
  4524. }, jsOne.charset, timeout);
  4525. });
  4526. },
  4527. /**
  4528. * calculate dependency
  4529. */
  4530. calculate: function (modNames, cache, ret) {
  4531. var i,
  4532. m,
  4533. mod,
  4534. modStatus,
  4535. self = this,
  4536. waitingModules = self.waitingModules,
  4537. runtime = self.runtime;
  4538. ret = ret || {};
  4539. // 提高性能,不用每个模块都再次全部依赖计算
  4540. // 做个缓存,每个模块对应的待动态加载模块
  4541. cache = cache || {};
  4542. for (i = 0; i < modNames.length; i++) {
  4543. m = modNames[i];
  4544. if (cache[m]) {
  4545. continue;
  4546. }
  4547. cache[m] = 1;
  4548. mod = Utils.createModuleInfo(runtime, m);
  4549. modStatus = mod.status;
  4550. if (modStatus === ERROR || modStatus === ATTACHED) {
  4551. continue;
  4552. }
  4553. if (modStatus != LOADED) {
  4554. if (!waitingModules.contains(m)) {
  4555. if (modStatus != LOADING) {
  4556. mod.status = LOADING;
  4557. ret[m] = 1;
  4558. }
  4559. mod.wait(function (mod) {
  4560. waitingModules.remove(mod.getName());
  4561. // notify current loader instance
  4562. waitingModules.notifyAll();
  4563. });
  4564. waitingModules.add(m);
  4565. }
  4566. }
  4567. self.calculate(mod.getNormalizedRequires(), cache, ret);
  4568. }
  4569. return ret;
  4570. },
  4571. /**
  4572. * get combo mods for modNames
  4573. */
  4574. getComboMods: function (modNames, comboPrefixes) {
  4575. var comboMods = {},
  4576. packageUri,
  4577. runtime = this.runtime,
  4578. i = 0,
  4579. l = modNames.length,
  4580. modName, mod, packageInfo, type, typedCombos, mods,
  4581. tag, charset, packagePath,
  4582. packageName, group, fullpath;
  4583. for (; i < l; ++i) {
  4584. modName = modNames[i];
  4585. mod = Utils.createModuleInfo(runtime, modName);
  4586. type = mod.getType();
  4587. fullpath = mod.getFullPath();
  4588. packageInfo = mod.getPackage();
  4589. packageName = packageInfo.getName();
  4590. charset = packageInfo.getCharset();
  4591. tag = packageInfo.getTag();
  4592. group = packageInfo.getGroup();
  4593. packagePath = packageInfo.getPrefixUriForCombo();
  4594. packageUri = packageInfo.getPackageUri();
  4595. var comboName = packageName;
  4596. // whether group packages can be combined (except default package and non-combo modules)
  4597. if ((mod.canBeCombined = packageInfo.isCombine() &&
  4598. S.startsWith(fullpath, packagePath)) && group) {
  4599. // combined package name
  4600. comboName = group + '_' + charset + '_' + groupTag;
  4601. var groupPrefixUri;
  4602. if (groupPrefixUri = comboPrefixes[comboName]) {
  4603. if (groupPrefixUri.isSameOriginAs(packageUri)) {
  4604. groupPrefixUri.setPath(getCommonPrefix(groupPrefixUri.getPath(),
  4605. packageUri.getPath()));
  4606. } else {
  4607. comboName = packageName;
  4608. comboPrefixes[packageName] = packageUri;
  4609. }
  4610. } else {
  4611. comboPrefixes[comboName] = packageUri.clone();
  4612. }
  4613. } else {
  4614. comboPrefixes[packageName] = packageUri;
  4615. }
  4616. typedCombos = comboMods[type] = comboMods[type] || {};
  4617. if (!(mods = typedCombos[comboName])) {
  4618. mods = typedCombos[comboName] = [];
  4619. mods.charset = charset;
  4620. mods.tags = [tag]; // [package tag]
  4621. } else {
  4622. if (mods.tags.length == 1 && mods.tags[0] == tag) {
  4623. } else {
  4624. mods.tags.push(tag);
  4625. }
  4626. }
  4627. mods.push(mod);
  4628. }
  4629. return comboMods;
  4630. },
  4631. /**
  4632. * Get combo urls
  4633. */
  4634. getComboUrls: function (modNames) {
  4635. var runtime = this.runtime,
  4636. Config = runtime.Config,
  4637. comboPrefix = Config.comboPrefix,
  4638. comboSep = Config.comboSep,
  4639. maxFileNum = Config.comboMaxFileNum,
  4640. maxUrlLength = Config.comboMaxUrlLength;
  4641. var comboPrefixes = {};
  4642. // {type, {comboName, [modInfo]}}}
  4643. var comboMods = this.getComboMods(modNames, comboPrefixes);
  4644. // {type, {comboName, [url]}}}
  4645. var comboRes = {};
  4646. // generate combo urls
  4647. for (var type in comboMods) {
  4648. comboRes[type] = {};
  4649. for (var comboName in comboMods[type]) {
  4650. var currentComboUrls = [];
  4651. var currentComboMods = [];
  4652. var mods = comboMods[type][comboName];
  4653. var tags = mods.tags;
  4654. var tag = tags.length > 1 ? getHash(tags.join('')) : tags[0];
  4655. var suffix = (tag ? '?t=' + encodeURIComponent(tag) + '.' + type : ''),
  4656. suffixLength = suffix.length,
  4657. basePrefix = comboPrefixes[comboName].toString(),
  4658. baseLen = basePrefix.length,
  4659. prefix = basePrefix + comboPrefix,
  4660. res = comboRes[type][comboName] = [];
  4661. var l = prefix.length;
  4662. res.charset = mods.charset;
  4663. res.mods = [];
  4664. function pushComboUrl() {
  4665. // map the whole combo path
  4666. //noinspection JSReferencingMutableVariableFromClosure
  4667. res.push({
  4668. combine: 1,
  4669. fullpath: Utils.getMappedPath(runtime, prefix +
  4670. currentComboUrls.join(comboSep) + suffix,
  4671. Config.mappedComboRules),
  4672. mods: currentComboMods
  4673. });
  4674. }
  4675. for (var i = 0; i < mods.length; i++) {
  4676. var currentMod = mods[i];
  4677. res.mods.push(currentMod);
  4678. // map individual module
  4679. var fullpath = currentMod.getFullPath();
  4680. if (!currentMod.canBeCombined) {
  4681. res.push({
  4682. combine: 0,
  4683. fullpath: fullpath,
  4684. mods: [currentMod]
  4685. });
  4686. continue;
  4687. }
  4688. // ignore query parameter
  4689. var path = fullpath.slice(baseLen).replace(/\?.*$/, '');
  4690. currentComboUrls.push(path);
  4691. currentComboMods.push(currentMod);
  4692. if (currentComboUrls.length > maxFileNum ||
  4693. (l + currentComboUrls.join(comboSep).length + suffixLength > maxUrlLength)) {
  4694. currentComboUrls.pop();
  4695. currentComboMods.pop();
  4696. pushComboUrl();
  4697. currentComboUrls = [];
  4698. currentComboMods = [];
  4699. i--;
  4700. }
  4701. }
  4702. if (currentComboUrls.length) {
  4703. pushComboUrl();
  4704. }
  4705. }
  4706. }
  4707. return comboRes;
  4708. }
  4709. });
  4710. Loader.ComboLoader = ComboLoader;
  4711. })(KISSY);
  4712. /*
  4713. 2013-09-11
  4714. - union simple loader and combo loader
  4715. 2013-07-25 阿古, yiminghe
  4716. - support group combo for packages
  4717. 2013-06-04 yiminghe@gmail.com
  4718. - refactor merge combo loader and simple loader
  4719. - support error callback
  4720. 2012-02-20 yiminghe note:
  4721. - three status
  4722. 0: initialized
  4723. LOADED: load into page
  4724. ATTACHED: fn executed
  4725. *//**
  4726. * @ignore
  4727. * mix loader into KISSY and infer KISSY baseUrl if not set
  4728. * @author yiminghe@gmail.com
  4729. */
  4730. (function (S, undefined) {
  4731. var Loader = S.Loader,
  4732. Env = S.Env,
  4733. logger = S.getLogger('s/loader'),
  4734. Utils = Loader.Utils,
  4735. processImmediate = S.setImmediate,
  4736. ComboLoader = Loader.ComboLoader;
  4737. function WaitingModules(fn) {
  4738. S.mix(this, {
  4739. fn: fn,
  4740. waitMods: {}
  4741. });
  4742. }
  4743. WaitingModules.prototype = {
  4744. constructor: WaitingModules,
  4745. notifyAll: function () {
  4746. var self = this,
  4747. fn = self.fn;
  4748. if (fn && S.isEmptyObject(self.waitMods)) {
  4749. self.fn = null;
  4750. fn();
  4751. }
  4752. },
  4753. add: function (modName) {
  4754. this.waitMods[modName] = 1;
  4755. },
  4756. remove: function (modName) {
  4757. delete this.waitMods[modName];
  4758. },
  4759. contains: function (modName) {
  4760. return this.waitMods[modName];
  4761. }
  4762. };
  4763. Loader.WaitingModules = WaitingModules;
  4764. S.mix(S, {
  4765. /**
  4766. * Registers a module with the KISSY global.
  4767. * @param {String} name module name.
  4768. * it must be set if combine is true in {@link KISSY#config}
  4769. * @param {Function} fn module definition function that is used to return
  4770. * this module value
  4771. * @param {KISSY} fn.S KISSY global instance
  4772. * @param {Object} [cfg] module optional config data
  4773. * @param {String[]} cfg.requires this module's required module name list
  4774. * @member KISSY
  4775. *
  4776. *
  4777. * // dom module's definition
  4778. * KISSY.add('dom', function(S, xx){
  4779. * return {css: function(el, name, val){}};
  4780. * },{
  4781. * requires:['xx']
  4782. * });
  4783. */
  4784. add: function (name, fn, cfg) {
  4785. ComboLoader.add(name, fn, cfg, S);
  4786. },
  4787. /**
  4788. * Attached one or more modules to global KISSY instance.
  4789. * @param {String|String[]} modNames moduleNames. 1-n modules to bind(use comma to separate)
  4790. * @param {Function} success callback function executed
  4791. * when KISSY has the required functionality.
  4792. * @param {KISSY} success.S KISSY instance
  4793. * @param success.x... used module values
  4794. * @member KISSY
  4795. *
  4796. *
  4797. * // loads and attached overlay,dd and its dependencies
  4798. * KISSY.use('overlay,dd', function(S, Overlay){});
  4799. */
  4800. use: function (modNames, success) {
  4801. var normalizedModNames,
  4802. loader,
  4803. error,
  4804. sync,
  4805. tryCount = 0,
  4806. waitingModules = new WaitingModules(loadReady);
  4807. if (S.isPlainObject(success)) {
  4808. //noinspection JSUnresolvedVariable
  4809. sync = success.sync;
  4810. //noinspection JSUnresolvedVariable
  4811. error = success.error;
  4812. //noinspection JSUnresolvedVariable
  4813. success = success.success;
  4814. }
  4815. modNames = Utils.getModNamesAsArray(modNames);
  4816. modNames = Utils.normalizeModNamesWithAlias(S, modNames);
  4817. normalizedModNames = Utils.unalias(S, modNames);
  4818. function loadReady() {
  4819. ++tryCount;
  4820. var errorList = [],
  4821. start = S.now(),
  4822. ret;
  4823. ret = Utils.attachModsRecursively(normalizedModNames, S, undefined, errorList);
  4824. logger.debug(tryCount + ' check duration ' + (S.now() - start));
  4825. if (ret) {
  4826. if (success) {
  4827. if (sync) {
  4828. success.apply(S, Utils.getModules(S, modNames));
  4829. } else {
  4830. // standalone error trace
  4831. processImmediate(function () {
  4832. success.apply(S, Utils.getModules(S, modNames));
  4833. });
  4834. }
  4835. }
  4836. } else if (errorList.length) {
  4837. if (error) {
  4838. if (sync) {
  4839. error.apply(S, errorList);
  4840. } else {
  4841. processImmediate(function () {
  4842. error.apply(S, errorList);
  4843. });
  4844. }
  4845. }
  4846. } else {
  4847. logger.debug(tryCount + ' reload ' + modNames);
  4848. waitingModules.fn = loadReady;
  4849. loader.use(normalizedModNames);
  4850. }
  4851. }
  4852. loader = new ComboLoader(S, waitingModules);
  4853. // in case modules is loaded statically
  4854. // synchronous check
  4855. // but always async for loader
  4856. if (sync) {
  4857. waitingModules.notifyAll();
  4858. } else {
  4859. processImmediate(function () {
  4860. waitingModules.notifyAll();
  4861. });
  4862. }
  4863. return S;
  4864. },
  4865. /**
  4866. * get module value from KISSY module cache
  4867. * @param {String} moduleName module name
  4868. * @member KISSY
  4869. * @return {*} value of module which name is moduleName
  4870. */
  4871. require: function (moduleName) {
  4872. var moduleNames = Utils.unalias(S, Utils.normalizeModNamesWithAlias(S, [moduleName]));
  4873. if (Utils.attachModsRecursively(moduleNames, S)) {
  4874. return Utils.getModules(S, moduleNames)[1];
  4875. }
  4876. return undefined;
  4877. }
  4878. });
  4879. Env.mods = {}; // all added mods
  4880. })(KISSY);
  4881. /*
  4882. 2013-06-04 yiminghe@gmail.com
  4883. - refactor merge combo loader and simple loader
  4884. - support error callback
  4885. *//**
  4886. * @ignore
  4887. * init loader, set config
  4888. */
  4889. (function (S) {
  4890. var doc = S.Env.host && S.Env.host.document;
  4891. // var logger = S.getLogger('s/loader');
  4892. var Utils = S.Loader.Utils;
  4893. var TIMESTAMP = '20131121234724';
  4894. function returnJson(s) {
  4895. return (new Function('return ' + s))();
  4896. }
  4897. var baseReg = /^(.*)(seed|kissy)(?:-min)?\.js[^/]*/i,
  4898. baseTestReg = /(seed|kissy)(?:-min)?\.js/i;
  4899. function getBaseInfoFromOneScript(script) {
  4900. // can not use KISSY.Uri
  4901. // /??x.js,dom.js for tbcdn
  4902. var src = script.src || '';
  4903. if (!src.match(baseTestReg)) {
  4904. return 0;
  4905. }
  4906. var baseInfo = script.getAttribute('data-config');
  4907. if (baseInfo) {
  4908. baseInfo = returnJson(baseInfo);
  4909. } else {
  4910. baseInfo = {};
  4911. }
  4912. var comboPrefix = baseInfo.comboPrefix = baseInfo.comboPrefix || '??';
  4913. var comboSep = baseInfo.comboSep = baseInfo.comboSep || ',';
  4914. var parts ,
  4915. base,
  4916. index = src.indexOf(comboPrefix);
  4917. // no combo
  4918. if (index == -1) {
  4919. base = src.replace(baseReg, '$1');
  4920. } else {
  4921. base = src.substring(0, index);
  4922. // a.tbcdn.cn??y.js, ie does not insert / after host
  4923. // a.tbcdn.cn/combo? comboPrefix=/combo?
  4924. if (base.charAt(base.length - 1) != '/') {
  4925. base += '/';
  4926. }
  4927. parts = src.substring(index + comboPrefix.length).split(comboSep);
  4928. S.each(parts, function (part) {
  4929. if (part.match(baseTestReg)) {
  4930. base += part.replace(baseReg, '$1');
  4931. return false;
  4932. }
  4933. return undefined;
  4934. });
  4935. }
  4936. return S.mix({
  4937. base: base,
  4938. // kissy 's tag will be determined by build time and user specified url
  4939. tag: Utils.getHash(TIMESTAMP + src)
  4940. }, baseInfo);
  4941. }
  4942. /**
  4943. * get base from seed.js
  4944. * @return {Object} base for kissy
  4945. * @ignore
  4946. *
  4947. * for example:
  4948. * @example
  4949. * http://a.tbcdn.cn/??s/kissy/x.y.z/seed-min.js,p/global/global.js
  4950. * note about custom combo rules, such as yui3:
  4951. * combo-prefix='combo?' combo-sep='&'
  4952. */
  4953. function getBaseInfo() {
  4954. // get base from current script file path
  4955. // notice: timestamp
  4956. var scripts = doc.getElementsByTagName('script'),
  4957. i,
  4958. info;
  4959. for (i = scripts.length - 1; i >= 0; i--) {
  4960. if (info = getBaseInfoFromOneScript(scripts[i])) {
  4961. return info;
  4962. }
  4963. }
  4964. S.log('must load kissy by file name in browser environment: seed.js or seed-min.js', 'error');
  4965. return null;
  4966. }
  4967. S.config({
  4968. charset: 'utf-8',
  4969. lang: 'zh-cn',
  4970. tag: TIMESTAMP
  4971. });
  4972. if (S.UA.nodejs) {
  4973. // nodejs: no tag
  4974. // noinspection JSUnresolvedVariable
  4975. S.config({
  4976. charset: 'utf-8',
  4977. base: __dirname.replace(/\\/g, '/').replace(/\/$/, '') + '/'
  4978. });
  4979. // ejecta
  4980. } else if (doc && doc.getElementsByTagName) {
  4981. // will transform base to absolute path
  4982. S.config(S.mix({
  4983. // 2k(2048) url length
  4984. comboMaxUrlLength: 2000,
  4985. // file limit number for a single combo url
  4986. comboMaxFileNum: 40
  4987. }, getBaseInfo()));
  4988. }
  4989. })(KISSY);/**
  4990. * @ignore
  4991. * i18n plugin for kissy loader
  4992. * @author yiminghe@gmail.com
  4993. */
  4994. KISSY.add('i18n', {
  4995. alias: function (S, name) {
  4996. return name + '/i18n/' + S.Config.lang;
  4997. }
  4998. });/**
  4999. * this code can only run at browser environment
  5000. * @ignore
  5001. * @author lifesinger@gmail.com, yiminghe@gmail.com
  5002. */
  5003. (function (S, undefined) {
  5004. var win = S.Env.host,
  5005. logger = S.getLogger('s/web'),
  5006. UA = S.UA,
  5007. doc = win['document'],
  5008. docElem = doc && doc.documentElement,
  5009. location = win.location,
  5010. EMPTY = '',
  5011. domReady = 0,
  5012. callbacks = [],
  5013. // The number of poll times.
  5014. POLL_RETIRES = 500,
  5015. // The poll interval in milliseconds.
  5016. POLL_INTERVAL = 40,
  5017. // #id or id
  5018. RE_ID_STR = /^#?([\w-]+)$/,
  5019. RE_NOT_WHITESPACE = /\S/,
  5020. standardEventModel = !!(doc && doc.addEventListener),
  5021. DOM_READY_EVENT = 'DOMContentLoaded',
  5022. READY_STATE_CHANGE_EVENT = 'readystatechange',
  5023. LOAD_EVENT = 'load',
  5024. COMPLETE = 'complete',
  5025. addEventListener = standardEventModel ? function (el, type, fn) {
  5026. el.addEventListener(type, fn, false);
  5027. } : function (el, type, fn) {
  5028. el.attachEvent('on' + type, fn);
  5029. },
  5030. removeEventListener = standardEventModel ? function (el, type, fn) {
  5031. el.removeEventListener(type, fn, false);
  5032. } : function (el, type, fn) {
  5033. el.detachEvent('on' + type, fn);
  5034. };
  5035. S.mix(S, {
  5036. /**
  5037. * A crude way of determining if an object is a window
  5038. * @member KISSY
  5039. */
  5040. isWindow: function (obj) {
  5041. return obj != null && obj == obj.window;
  5042. },
  5043. /**
  5044. * get xml representation of data
  5045. * @param {String} data
  5046. * @member KISSY
  5047. */
  5048. parseXML: function (data) {
  5049. // already a xml
  5050. if (data.documentElement) {
  5051. return data;
  5052. }
  5053. var xml;
  5054. try {
  5055. // Standard
  5056. if (win['DOMParser']) {
  5057. xml = new DOMParser().parseFromString(data, 'text/xml');
  5058. } else { // IE
  5059. xml = new ActiveXObject('Microsoft.XMLDOM');
  5060. xml.async = false;
  5061. xml.loadXML(data);
  5062. }
  5063. } catch (e) {
  5064. logger.error('parseXML error :');
  5065. logger.error(e);
  5066. xml = undefined;
  5067. }
  5068. if (!xml || !xml.documentElement || xml.getElementsByTagName('parsererror').length) {
  5069. S.error('Invalid XML: ' + data);
  5070. }
  5071. return xml;
  5072. },
  5073. /**
  5074. * Evaluates a script in a global context.
  5075. * @member KISSY
  5076. */
  5077. globalEval: function (data) {
  5078. if (data && RE_NOT_WHITESPACE.test(data)) {
  5079. // http://weblogs.java.net/blog/driscoll/archive/2009/09/08/eval-javascript-global-context
  5080. // http://msdn.microsoft.com/en-us/library/ie/ms536420(v=vs.85).aspx always return null
  5081. ( win.execScript || function (data) {
  5082. win[ 'eval' ].call(win, data);
  5083. } )(data);
  5084. }
  5085. },
  5086. /**
  5087. * Specify a function to execute when the Dom is fully loaded.
  5088. * @param fn {Function} A function to execute after the Dom is ready
  5089. * @chainable
  5090. * @member KISSY
  5091. */
  5092. ready: function (fn) {
  5093. if (domReady) {
  5094. try {
  5095. fn(S);
  5096. } catch (e) {
  5097. S.log(e.stack || e, 'error');
  5098. setTimeout(function () {
  5099. throw e;
  5100. }, 0);
  5101. }
  5102. } else {
  5103. callbacks.push(fn);
  5104. }
  5105. return this;
  5106. },
  5107. /**
  5108. * Executes the supplied callback when the item with the supplied id is found.
  5109. * @param id {String} The id of the element, or an array of ids to look for.
  5110. * @param fn {Function} What to execute when the element is found.
  5111. * @member KISSY
  5112. */
  5113. available: function (id, fn) {
  5114. id = (id + EMPTY).match(RE_ID_STR)[1];
  5115. var retryCount = 1;
  5116. var timer = S.later(function () {
  5117. if (++retryCount > POLL_RETIRES) {
  5118. timer.cancel();
  5119. return;
  5120. }
  5121. var node = doc.getElementById(id);
  5122. if (node) {
  5123. fn(node);
  5124. timer.cancel();
  5125. }
  5126. }, POLL_INTERVAL, true);
  5127. }
  5128. });
  5129. function fireReady() {
  5130. if (domReady) {
  5131. return;
  5132. }
  5133. // nodejs
  5134. if (doc && !UA.nodejs) {
  5135. removeEventListener(win, LOAD_EVENT, fireReady);
  5136. }
  5137. domReady = 1;
  5138. for (var i = 0; i < callbacks.length; i++) {
  5139. try {
  5140. callbacks[i](S);
  5141. } catch (e) {
  5142. S.log(e.stack || e, 'error');
  5143. setTimeout(function () {
  5144. throw e;
  5145. }, 0);
  5146. }
  5147. }
  5148. }
  5149. // Binds ready events.
  5150. function bindReady() {
  5151. // Catch cases where ready() is called after the
  5152. // browser event has already occurred.
  5153. if (!doc || doc.readyState === COMPLETE) {
  5154. fireReady();
  5155. return;
  5156. }
  5157. // A fallback to window.onload, that will always work
  5158. addEventListener(win, LOAD_EVENT, fireReady);
  5159. // w3c mode
  5160. if (standardEventModel) {
  5161. var domReady = function () {
  5162. removeEventListener(doc, DOM_READY_EVENT, domReady);
  5163. fireReady();
  5164. };
  5165. addEventListener(doc, DOM_READY_EVENT, domReady);
  5166. }
  5167. // IE event model is used
  5168. else {
  5169. var stateChange = function () {
  5170. if (doc.readyState === COMPLETE) {
  5171. removeEventListener(doc, READY_STATE_CHANGE_EVENT, stateChange);
  5172. fireReady();
  5173. }
  5174. };
  5175. // ensure firing before onload (but completed after all inner iframes is loaded)
  5176. // maybe late but safe also for iframes
  5177. addEventListener(doc, READY_STATE_CHANGE_EVENT, stateChange);
  5178. // If IE and not a frame
  5179. // continually check to see if the document is ready
  5180. var notframe,
  5181. doScroll = docElem && docElem.doScroll;
  5182. try {
  5183. notframe = (win['frameElement'] === null);
  5184. } catch (e) {
  5185. notframe = false;
  5186. }
  5187. // can not use in iframe,parent window is dom ready so doScroll is ready too
  5188. if (doScroll && notframe) {
  5189. var readyScroll = function () {
  5190. try {
  5191. // Ref: http://javascript.nwbox.com/IEContentLoaded/
  5192. doScroll('left');
  5193. fireReady();
  5194. } catch (ex) {
  5195. setTimeout(readyScroll, POLL_INTERVAL);
  5196. }
  5197. };
  5198. readyScroll();
  5199. }
  5200. }
  5201. }
  5202. // If url contains '?ks-debug', debug mode will turn on automatically.
  5203. if (location && (location.search || EMPTY).indexOf('ks-debug') !== -1) {
  5204. S.Config.debug = true;
  5205. }
  5206. // bind on start
  5207. // in case when you bind but the DOMContentLoaded has triggered
  5208. // then you has to wait onload
  5209. // worst case no callback at all
  5210. bindReady();
  5211. if (UA.ie) {
  5212. try {
  5213. doc.execCommand('BackgroundImageCache', false, true);
  5214. } catch (e) {
  5215. }
  5216. }
  5217. })(KISSY, undefined);
  5218. /**
  5219. * @ignore
  5220. * Default KISSY Gallery and core alias.
  5221. * @author yiminghe@gmail.com
  5222. */
  5223. (function (S) {
  5224. // compatibility
  5225. S.config({
  5226. modules: {
  5227. core: {
  5228. alias: ['dom', 'event', 'io', 'anim', 'base', 'node', 'json', 'ua', 'cookie']
  5229. },
  5230. ajax: {
  5231. alias: ['io']
  5232. },
  5233. 'rich-base': {
  5234. alias: ['base']
  5235. }
  5236. }
  5237. });
  5238. if (typeof location != 'undefined') {
  5239. var https = S.startsWith(location.href, 'https');
  5240. var prefix = https ? 'https://s.tbcdn.cn/s/kissy/' : 'http://a.tbcdn.cn/s/kissy/';
  5241. S.config({
  5242. packages: {
  5243. gallery: {
  5244. base: prefix
  5245. },
  5246. mobile: {
  5247. base: prefix
  5248. }
  5249. }
  5250. });
  5251. }
  5252. })(KISSY);
  5253. (function(config,Features,UA){
  5254. /*Generated By KISSY Module Compiler*/
  5255. config({
  5256. 'anim': {requires: ['dom','anim/base','anim/timer',KISSY.Features.isTransitionSupported() ? "anim/transition" : ""]}
  5257. });
  5258. /*Generated By KISSY Module Compiler*/
  5259. config({
  5260. 'anim/base': {requires: ['dom','promise']}
  5261. });
  5262. /*Generated By KISSY Module Compiler*/
  5263. config({
  5264. 'anim/timer': {requires: ['dom','anim/base']}
  5265. });
  5266. /*Generated By KISSY Module Compiler*/
  5267. config({
  5268. 'anim/transition': {requires: ['dom','event/dom','anim/base']}
  5269. });
  5270. /*Generated By KISSY Module Compiler*/
  5271. config({
  5272. 'base': {requires: ['event/custom']}
  5273. });
  5274. /*Generated By KISSY Module Compiler*/
  5275. config({
  5276. 'button': {requires: ['node','component/control']}
  5277. });
  5278. /*Generated By KISSY Module Compiler*/
  5279. config({
  5280. 'color': {requires: ['base']}
  5281. });
  5282. /*Generated By KISSY Module Compiler*/
  5283. config({
  5284. 'combobox': {requires: ['node','component/control','menu','base','io']}
  5285. });
  5286. /*Generated By KISSY Module Compiler*/
  5287. config({
  5288. 'component/container': {requires: ['component/control','component/manager']}
  5289. });
  5290. /*Generated By KISSY Module Compiler*/
  5291. config({
  5292. 'component/control': {requires: ['node','base','promise','component/manager','xtemplate/runtime']}
  5293. });
  5294. /*Generated By KISSY Module Compiler*/
  5295. config({
  5296. 'component/extension/align': {requires: ['node']}
  5297. });
  5298. /*Generated By KISSY Module Compiler*/
  5299. config({
  5300. 'component/extension/delegate-children': {requires: ['node','component/manager']}
  5301. });
  5302. /*Generated By KISSY Module Compiler*/
  5303. config({
  5304. 'component/plugin/drag': {requires: ['base','dd']}
  5305. });
  5306. /*Generated By KISSY Module Compiler*/
  5307. config({
  5308. 'component/plugin/resize': {requires: ['resizable']}
  5309. });
  5310. /*Generated By KISSY Module Compiler*/
  5311. config({
  5312. 'date/format': {requires: ['date/gregorian','i18n!date']}
  5313. });
  5314. /*Generated By KISSY Module Compiler*/
  5315. config({
  5316. 'date/gregorian': {requires: ['i18n!date']}
  5317. });
  5318. /*Generated By KISSY Module Compiler*/
  5319. config({
  5320. 'date/picker': {requires: ['node','date/gregorian','i18n!date/picker','component/control','date/format']}
  5321. });
  5322. /*Generated By KISSY Module Compiler*/
  5323. config({
  5324. 'date/popup-picker': {requires: ['date/picker','component/extension/align','component/extension/shim']}
  5325. });
  5326. /*Generated By KISSY Module Compiler*/
  5327. config({
  5328. 'dd': {requires: ['node','base']}
  5329. });
  5330. /*Generated By KISSY Module Compiler*/
  5331. config({
  5332. 'dd/plugin/constrain': {requires: ['base','node']}
  5333. });
  5334. /*Generated By KISSY Module Compiler*/
  5335. config({
  5336. 'dd/plugin/proxy': {requires: ['node','base','dd']}
  5337. });
  5338. /*Generated By KISSY Module Compiler*/
  5339. config({
  5340. 'dd/plugin/scroll': {requires: ['dd','base','node']}
  5341. });
  5342. config({
  5343. "dom/basic": {
  5344. "alias": [
  5345. 'dom/base',
  5346. Features.isIELessThan(9) ? 'dom/ie' : '',
  5347. Features.isClassListSupported() ? '' : 'dom/class-list'
  5348. ]
  5349. },
  5350. "dom": {
  5351. "alias": [
  5352. 'dom/basic',
  5353. !Features.isQuerySelectorSupported() ? 'dom/selector' : ''
  5354. ]
  5355. }
  5356. });/*Generated By KISSY Module Compiler*/
  5357. config({
  5358. 'dom/class-list': {requires: ['dom/base']}
  5359. });
  5360. /*Generated By KISSY Module Compiler*/
  5361. config({
  5362. 'dom/ie': {requires: ['dom/base']}
  5363. });
  5364. /*Generated By KISSY Module Compiler*/
  5365. config({
  5366. 'dom/selector': {requires: ['dom/basic']}
  5367. });
  5368. /*Generated By KISSY Module Compiler*/
  5369. config({
  5370. 'editor': {requires: ['node','html-parser','component/control','event']}
  5371. });
  5372. /*Generated By KISSY Module Compiler*/
  5373. config({
  5374. 'event': {requires: ['event/dom','event/custom']}
  5375. });
  5376. /*Generated By KISSY Module Compiler*/
  5377. config({
  5378. 'event/custom': {requires: ['event/base']}
  5379. });
  5380. config({
  5381. "event/dom": {
  5382. "alias": [
  5383. "event/dom/base",
  5384. Features.isTouchEventSupported() || Features.isMsPointerSupported() ?
  5385. 'event/dom/touch' : '',
  5386. Features.isDeviceMotionSupported() ?
  5387. 'event/dom/shake' : '',
  5388. Features.isHashChangeSupported() ?
  5389. '' : 'event/dom/hashchange',
  5390. Features.isIELessThan(9) ?
  5391. 'event/dom/ie' : '',
  5392. UA.ie ? '' : 'event/dom/focusin'
  5393. ]
  5394. }
  5395. });/*Generated By KISSY Module Compiler*/
  5396. config({
  5397. 'event/dom/base': {requires: ['event/base','dom']}
  5398. });
  5399. /*Generated By KISSY Module Compiler*/
  5400. config({
  5401. 'event/dom/focusin': {requires: ['event/dom/base']}
  5402. });
  5403. /*Generated By KISSY Module Compiler*/
  5404. config({
  5405. 'event/dom/hashchange': {requires: ['event/dom/base','dom']}
  5406. });
  5407. /*Generated By KISSY Module Compiler*/
  5408. config({
  5409. 'event/dom/ie': {requires: ['event/dom/base','dom']}
  5410. });
  5411. /*Generated By KISSY Module Compiler*/
  5412. config({
  5413. 'event/dom/shake': {requires: ['event/dom/base']}
  5414. });
  5415. /*Generated By KISSY Module Compiler*/
  5416. config({
  5417. 'event/dom/touch': {requires: ['event/dom/base','dom']}
  5418. });
  5419. /*Generated By KISSY Module Compiler*/
  5420. config({
  5421. 'filter-menu': {requires: ['menu','node','component/extension/content-render']}
  5422. });
  5423. /*Generated By KISSY Module Compiler*/
  5424. config({
  5425. 'io': {requires: ['dom','event/custom','promise','event']}
  5426. });
  5427. /*Generated By KISSY Module Compiler*/
  5428. config({
  5429. 'kison': {requires: ['base']}
  5430. });
  5431. /*Generated By KISSY Module Compiler*/
  5432. config({
  5433. 'menu': {requires: ['node','component/container','component/extension/delegate-children','component/control','component/extension/content-render','component/extension/align','component/extension/shim']}
  5434. });
  5435. /*Generated By KISSY Module Compiler*/
  5436. config({
  5437. 'menubutton': {requires: ['node','button','component/extension/content-render','menu']}
  5438. });
  5439. /*Generated By KISSY Module Compiler*/
  5440. config({
  5441. 'mvc': {requires: ['base','node','io','json']}
  5442. });
  5443. /*Generated By KISSY Module Compiler*/
  5444. config({
  5445. 'node': {requires: ['dom','event/dom','anim']}
  5446. });
  5447. /*Generated By KISSY Module Compiler*/
  5448. config({
  5449. 'overlay': {requires: ['component/container','component/extension/shim','component/extension/align','node','component/extension/content-render']}
  5450. });
  5451. /*Generated By KISSY Module Compiler*/
  5452. config({
  5453. 'resizable': {requires: ['node','base','dd']}
  5454. });
  5455. /*Generated By KISSY Module Compiler*/
  5456. config({
  5457. 'resizable/plugin/proxy': {requires: ['base','node']}
  5458. });
  5459. config({
  5460. "scroll-view": {
  5461. "alias": [
  5462. Features.isTouchEventSupported()||Features.isMsPointerSupported()
  5463. ? 'scroll-view/drag' : 'scroll-view/base'
  5464. ]
  5465. }
  5466. });/*Generated By KISSY Module Compiler*/
  5467. config({
  5468. 'scroll-view/base': {requires: ['node','anim','component/container','component/extension/content-render']}
  5469. });
  5470. /*Generated By KISSY Module Compiler*/
  5471. config({
  5472. 'scroll-view/drag': {requires: ['scroll-view/base','node','anim']}
  5473. });
  5474. /*Generated By KISSY Module Compiler*/
  5475. config({
  5476. 'scroll-view/plugin/pull-to-refresh': {requires: ['base']}
  5477. });
  5478. /*Generated By KISSY Module Compiler*/
  5479. config({
  5480. 'scroll-view/plugin/scrollbar': {requires: ['base','node','component/control']}
  5481. });
  5482. /*Generated By KISSY Module Compiler*/
  5483. config({
  5484. 'separator': {requires: ['component/control']}
  5485. });
  5486. /*Generated By KISSY Module Compiler*/
  5487. config({
  5488. 'split-button': {requires: ['component/container','button','menubutton']}
  5489. });
  5490. /*Generated By KISSY Module Compiler*/
  5491. config({
  5492. 'stylesheet': {requires: ['dom']}
  5493. });
  5494. /*Generated By KISSY Module Compiler*/
  5495. config({
  5496. 'swf': {requires: ['dom','json','base']}
  5497. });
  5498. /*Generated By KISSY Module Compiler*/
  5499. config({
  5500. 'tabs': {requires: ['component/container','toolbar','button']}
  5501. });
  5502. /*Generated By KISSY Module Compiler*/
  5503. config({
  5504. 'toolbar': {requires: ['component/container','component/extension/delegate-children','node']}
  5505. });
  5506. /*Generated By KISSY Module Compiler*/
  5507. config({
  5508. 'tree': {requires: ['node','component/container','component/extension/content-render','component/extension/delegate-children']}
  5509. });
  5510. /*Generated By KISSY Module Compiler*/
  5511. config({
  5512. 'xtemplate': {requires: ['xtemplate/runtime','xtemplate/compiler']}
  5513. });
  5514. /*Generated By KISSY Module Compiler*/
  5515. config({
  5516. 'xtemplate/compiler': {requires: ['xtemplate/runtime']}
  5517. });
  5518. /*Generated By KISSY Module Compiler*/
  5519. config({
  5520. 'xtemplate/nodejs': {requires: ['xtemplate']}
  5521. });
  5522. })(function(c){
  5523. KISSY.config('modules', c);
  5524. },KISSY.Features,KISSY.UA);
  5525. /**
  5526. * @ignore
  5527. * 1. export KISSY 's functionality to module system
  5528. * 2. export light-weighted json parse
  5529. */
  5530. (function (S) {
  5531. // empty mod for conditional loading
  5532. S.add('empty', S.noop);
  5533. S.add('ua', function () {
  5534. return S.UA;
  5535. });
  5536. S.add('uri', function () {
  5537. return S.Uri;
  5538. });
  5539. S.add('path', function () {
  5540. return S.Path
  5541. });
  5542. var UA = S.UA,
  5543. Env = S.Env,
  5544. win = Env.host,
  5545. doc = win.document || {},
  5546. documentMode = doc.documentMode,
  5547. nativeJson = ((UA.nodejs && typeof global === 'object') ? global : win).JSON;
  5548. // ie 8.0.7600.16315@win7 json bug!
  5549. if (documentMode && documentMode < 9) {
  5550. nativeJson = null;
  5551. }
  5552. if (nativeJson) {
  5553. S.add('json', function () {
  5554. return S.JSON = nativeJson;
  5555. });
  5556. // light weight json parse
  5557. S.parseJson = function (data) {
  5558. return nativeJson.parse(data);
  5559. };
  5560. } else {
  5561. // Json RegExp
  5562. var INVALID_CHARS_REG = /^[\],:{}\s]*$/,
  5563. INVALID_BRACES_REG = /(?:^|:|,)(?:\s*\[)+/g,
  5564. INVALID_ESCAPES_REG = /\\(?:["\\\/bfnrt]|u[\da-fA-F]{4})/g,
  5565. INVALID_TOKENS_REG = /"[^"\\\r\n]*"|true|false|null|-?(?:\d+\.|)\d+(?:[eE][+-]?\d+|)/g;
  5566. S.parseJson = function (data) {
  5567. if (data === null) {
  5568. return data;
  5569. }
  5570. if (typeof data === "string") {
  5571. // for ie
  5572. data = S.trim(data);
  5573. if (data) {
  5574. // from json2
  5575. if (INVALID_CHARS_REG.test(data.replace(INVALID_ESCAPES_REG, "@")
  5576. .replace(INVALID_TOKENS_REG, "]")
  5577. .replace(INVALID_BRACES_REG, ""))) {
  5578. return ( new Function("return " + data) )();
  5579. }
  5580. }
  5581. }
  5582. return S.error("Invalid Json: " + data);
  5583. };
  5584. }
  5585. // exports for nodejs
  5586. if (S.UA.nodejs) {
  5587. S.KISSY = S;
  5588. module.exports = S;
  5589. }
  5590. })(KISSY);