| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593 | /*! WebUploader 0.1.2 *//** * @fileOverview 让内部各个部件的代码可以用[amd](https://github.com/amdjs/amdjs-api/wiki/AMD)模块定义方式组织起来。 * * AMD API 内部的简单不完全实现,请忽略。只有当WebUploader被合并成一个文件的时候才会引入。 */(function( root, factory ) {    var modules = {},        // 内部require, 简单不完全实现。        // https://github.com/amdjs/amdjs-api/wiki/require        _require = function( deps, callback ) {            var args, len, i;            // 如果deps不是数组,则直接返回指定module            if ( typeof deps === 'string' ) {                return getModule( deps );            } else {                args = [];                for( len = deps.length, i = 0; i < len; i++ ) {                    args.push( getModule( deps[ i ] ) );                }                return callback.apply( null, args );            }        },        // 内部define,暂时不支持不指定id.        _define = function( id, deps, factory ) {            if ( arguments.length === 2 ) {                factory = deps;                deps = null;            }            _require( deps || [], function() {                setModule( id, factory, arguments );            });        },        // 设置module, 兼容CommonJs写法。        setModule = function( id, factory, args ) {            var module = {                    exports: factory                },                returned;            if ( typeof factory === 'function' ) {                args.length || (args = [ _require, module.exports, module ]);                returned = factory.apply( null, args );                returned !== undefined && (module.exports = returned);            }            modules[ id ] = module.exports;        },        // 根据id获取module        getModule = function( id ) {            var module = modules[ id ] || root[ id ];            if ( !module ) {                throw new Error( '`' + id + '` is undefined' );            }            return module;        },        // 将所有modules,将路径ids装换成对象。        exportsTo = function( obj ) {            var key, host, parts, part, last, ucFirst;            // make the first character upper case.            ucFirst = function( str ) {                return str && (str.charAt( 0 ).toUpperCase() + str.substr( 1 ));            };            for ( key in modules ) {                host = obj;                if ( !modules.hasOwnProperty( key ) ) {                    continue;                }                parts = key.split('/');                last = ucFirst( parts.pop() );                while( (part = ucFirst( parts.shift() )) ) {                    host[ part ] = host[ part ] || {};                    host = host[ part ];                }                host[ last ] = modules[ key ];            }        },        exports = factory( root, _define, _require ),        origin;    // exports every module.    exportsTo( exports );    if ( typeof module === 'object' && typeof module.exports === 'object' ) {        // For CommonJS and CommonJS-like environments where a proper window is present,        module.exports = exports;    } else if ( typeof define === 'function' && define.amd ) {        // Allow using this built library as an AMD module        // in another project. That other project will only        // see this AMD call, not the internal modules in        // the closure below.        define([], exports );    } else {        // Browser globals case. Just assign the        // result to a property on the global.        origin = root.WebUploader;        root.WebUploader = exports;        root.WebUploader.noConflict = function() {            root.WebUploader = origin;        };    }})( this, function( window, define, require ) {    /**     * @fileOverview jQuery or Zepto     */    define('dollar-third',[],function() {        return window.jQuery || window.Zepto;    });    /**     * @fileOverview Dom 操作相关     */    define('dollar',[        'dollar-third'    ], function( _ ) {        return _;    });    /**     * @fileOverview 使用jQuery的Promise     */    define('promise-third',[        'dollar'    ], function( $ ) {        return {            Deferred: $.Deferred,            when: $.when,                isPromise: function( anything ) {                return anything && typeof anything.then === 'function';            }        };    });    /**     * @fileOverview Promise/A+     */    define('promise',[        'promise-third'    ], function( _ ) {        return _;    });    /**     * @fileOverview 基础类方法。     */        /**     * Web Uploader内部类的详细说明,以下提及的功能类,都可以在`WebUploader`这个变量中访问到。     *     * As you know, Web Uploader的每个文件都是用过[AMD](https://github.com/amdjs/amdjs-api/wiki/AMD)规范中的`define`组织起来的, 每个Module都会有个module id.     * 默认module id该文件的路径,而此路径将会转化成名字空间存放在WebUploader中。如:     *     * * module `base`:WebUploader.Base     * * module `file`: WebUploader.File     * * module `lib/dnd`: WebUploader.Lib.Dnd     * * module `runtime/html5/dnd`: WebUploader.Runtime.Html5.Dnd     *     *     * 以下文档将可能省略`WebUploader`前缀。     * @module WebUploader     * @title WebUploader API文档     */    define('base',[        'dollar',        'promise'    ], function( $, promise ) {            var noop = function() {},            call = Function.call;            // http://jsperf.com/uncurrythis        // 反科里化        function uncurryThis( fn ) {            return function() {                return call.apply( fn, arguments );            };        }            function bindFn( fn, context ) {            return function() {                return fn.apply( context, arguments );            };        }            function createObject( proto ) {            var f;                if ( Object.create ) {                return Object.create( proto );            } else {                f = function() {};                f.prototype = proto;                return new f();            }        }                /**         * 基础类,提供一些简单常用的方法。         * @class Base         */        return {                /**             * @property {String} version 当前版本号。             */            version: '0.1.2',                /**             * @property {jQuery|Zepto} $ 引用依赖的jQuery或者Zepto对象。             */            $: $,                Deferred: promise.Deferred,                isPromise: promise.isPromise,                when: promise.when,                /**             * @description  简单的浏览器检查结果。             *             * * `webkit`  webkit版本号,如果浏览器为非webkit内核,此属性为`undefined`。             * * `chrome`  chrome浏览器版本号,如果浏览器为chrome,此属性为`undefined`。             * * `ie`  ie浏览器版本号,如果浏览器为非ie,此属性为`undefined`。**暂不支持ie10+**             * * `firefox`  firefox浏览器版本号,如果浏览器为非firefox,此属性为`undefined`。             * * `safari`  safari浏览器版本号,如果浏览器为非safari,此属性为`undefined`。             * * `opera`  opera浏览器版本号,如果浏览器为非opera,此属性为`undefined`。             *             * @property {Object} [browser]             */            browser: (function( ua ) {                var ret = {},                    webkit = ua.match( /WebKit\/([\d.]+)/ ),                    chrome = ua.match( /Chrome\/([\d.]+)/ ) ||                        ua.match( /CriOS\/([\d.]+)/ ),                        ie = ua.match( /MSIE\s([\d\.]+)/ ) ||                        ua.match(/(?:trident)(?:.*rv:([\w.]+))?/i),                    firefox = ua.match( /Firefox\/([\d.]+)/ ),                    safari = ua.match( /Safari\/([\d.]+)/ ),                    opera = ua.match( /OPR\/([\d.]+)/ );                    webkit && (ret.webkit = parseFloat( webkit[ 1 ] ));                chrome && (ret.chrome = parseFloat( chrome[ 1 ] ));                ie && (ret.ie = parseFloat( ie[ 1 ] ));                firefox && (ret.firefox = parseFloat( firefox[ 1 ] ));                safari && (ret.safari = parseFloat( safari[ 1 ] ));                opera && (ret.opera = parseFloat( opera[ 1 ] ));                    return ret;            })( navigator.userAgent ),                /**             * @description  操作系统检查结果。             *             * * `android`  如果在android浏览器环境下,此值为对应的android版本号,否则为`undefined`。             * * `ios` 如果在ios浏览器环境下,此值为对应的ios版本号,否则为`undefined`。             * @property {Object} [os]             */            os: (function( ua ) {                var ret = {},                        // osx = !!ua.match( /\(Macintosh\; Intel / ),                    android = ua.match( /(?:Android);?[\s\/]+([\d.]+)?/ ),                    ios = ua.match( /(?:iPad|iPod|iPhone).*OS\s([\d_]+)/ );                    // osx && (ret.osx = true);                android && (ret.android = parseFloat( android[ 1 ] ));                ios && (ret.ios = parseFloat( ios[ 1 ].replace( /_/g, '.' ) ));                    return ret;            })( navigator.userAgent ),                /**             * 实现类与类之间的继承。             * @method inherits             * @grammar Base.inherits( super ) => child             * @grammar Base.inherits( super, protos ) => child             * @grammar Base.inherits( super, protos, statics ) => child             * @param  {Class} super 父类             * @param  {Object | Function} [protos] 子类或者对象。如果对象中包含constructor,子类将是用此属性值。             * @param  {Function} [protos.constructor] 子类构造器,不指定的话将创建个临时的直接执行父类构造器的方法。             * @param  {Object} [statics] 静态属性或方法。             * @return {Class} 返回子类。             * @example             * function Person() {             *     console.log( 'Super' );             * }             * Person.prototype.hello = function() {             *     console.log( 'hello' );             * };             *             * var Manager = Base.inherits( Person, {             *     world: function() {             *         console.log( 'World' );             *     }             * });             *             * // 因为没有指定构造器,父类的构造器将会执行。             * var instance = new Manager();    // => Super             *             * // 继承子父类的方法             * instance.hello();    // => hello             * instance.world();    // => World             *             * // 子类的__super__属性指向父类             * console.log( Manager.__super__ === Person );    // => true             */            inherits: function( Super, protos, staticProtos ) {                var child;                    if ( typeof protos === 'function' ) {                    child = protos;                    protos = null;                } else if ( protos && protos.hasOwnProperty('constructor') ) {                    child = protos.constructor;                } else {                    child = function() {                        return Super.apply( this, arguments );                    };                }                    // 复制静态方法                $.extend( true, child, Super, staticProtos || {} );                    /* jshint camelcase: false */                    // 让子类的__super__属性指向父类。                child.__super__ = Super.prototype;                    // 构建原型,添加原型方法或属性。                // 暂时用Object.create实现。                child.prototype = createObject( Super.prototype );                protos && $.extend( true, child.prototype, protos );                    return child;            },                /**             * 一个不做任何事情的方法。可以用来赋值给默认的callback.             * @method noop             */            noop: noop,                /**             * 返回一个新的方法,此方法将已指定的`context`来执行。             * @grammar Base.bindFn( fn, context ) => Function             * @method bindFn             * @example             * var doSomething = function() {             *         console.log( this.name );             *     },             *     obj = {             *         name: 'Object Name'             *     },             *     aliasFn = Base.bind( doSomething, obj );             *             *  aliasFn();    // => Object Name             *             */            bindFn: bindFn,                /**             * 引用Console.log如果存在的话,否则引用一个[空函数loop](#WebUploader:Base.log)。             * @grammar Base.log( args... ) => undefined             * @method log             */            log: (function() {                if ( window.console ) {                    return bindFn( console.log, console );                }                return noop;            })(),                nextTick: (function() {                    return function( cb ) {                    setTimeout( cb, 1 );                };                    // @bug 当浏览器不在当前窗口时就停了。                // var next = window.requestAnimationFrame ||                //     window.webkitRequestAnimationFrame ||                //     window.mozRequestAnimationFrame ||                //     function( cb ) {                //         window.setTimeout( cb, 1000 / 60 );                //     };                    // // fix: Uncaught TypeError: Illegal invocation                // return bindFn( next, window );            })(),                /**             * 被[uncurrythis](http://www.2ality.com/2011/11/uncurrying-this.html)的数组slice方法。             * 将用来将非数组对象转化成数组对象。             * @grammar Base.slice( target, start[, end] ) => Array             * @method slice             * @example             * function doSomthing() {             *     var args = Base.slice( arguments, 1 );             *     console.log( args );             * }             *             * doSomthing( 'ignored', 'arg2', 'arg3' );    // => Array ["arg2", "arg3"]             */            slice: uncurryThis( [].slice ),                /**             * 生成唯一的ID             * @method guid             * @grammar Base.guid() => String             * @grammar Base.guid( prefx ) => String             */            guid: (function() {                var counter = 0;                    return function( prefix ) {                    var guid = (+new Date()).toString( 32 ),                        i = 0;                        for ( ; i < 5; i++ ) {                        guid += Math.floor( Math.random() * 65535 ).toString( 32 );                    }                        return (prefix || 'wu_') + guid + (counter++).toString( 32 );                };            })(),                /**             * 格式化文件大小, 输出成带单位的字符串             * @method formatSize             * @grammar Base.formatSize( size ) => String             * @grammar Base.formatSize( size, pointLength ) => String             * @grammar Base.formatSize( size, pointLength, units ) => String             * @param {Number} size 文件大小             * @param {Number} [pointLength=2] 精确到的小数点数。             * @param {Array} [units=[ 'B', 'K', 'M', 'G', 'TB' ]] 单位数组。从字节,到千字节,一直往上指定。如果单位数组里面只指定了到了K(千字节),同时文件大小大于M, 此方法的输出将还是显示成多少K.             * @example             * console.log( Base.formatSize( 100 ) );    // => 100B             * console.log( Base.formatSize( 1024 ) );    // => 1.00K             * console.log( Base.formatSize( 1024, 0 ) );    // => 1K             * console.log( Base.formatSize( 1024 * 1024 ) );    // => 1.00M             * console.log( Base.formatSize( 1024 * 1024 * 1024 ) );    // => 1.00G             * console.log( Base.formatSize( 1024 * 1024 * 1024, 0, ['B', 'KB', 'MB'] ) );    // => 1024MB             */            formatSize: function( size, pointLength, units ) {                var unit;                    units = units || [ 'B', 'K', 'M', 'G', 'TB' ];                    while ( (unit = units.shift()) && size > 1024 ) {                    size = size / 1024;                }                    return (unit === 'B' ? size : size.toFixed( pointLength || 2 )) +                        unit;            }        };    });    /**     * 事件处理类,可以独立使用,也可以扩展给对象使用。     * @fileOverview Mediator     */    define('mediator',[        'base'    ], function( Base ) {        var $ = Base.$,            slice = [].slice,            separator = /\s+/,            protos;            // 根据条件过滤出事件handlers.        function findHandlers( arr, name, callback, context ) {            return $.grep( arr, function( handler ) {                return handler &&                        (!name || handler.e === name) &&                        (!callback || handler.cb === callback ||                        handler.cb._cb === callback) &&                        (!context || handler.ctx === context);            });        }            function eachEvent( events, callback, iterator ) {            // 不支持对象,只支持多个event用空格隔开            $.each( (events || '').split( separator ), function( _, key ) {                iterator( key, callback );            });        }            function triggerHanders( events, args ) {            var stoped = false,                i = -1,                len = events.length,                handler;                while ( ++i < len ) {                handler = events[ i ];                    if ( handler.cb.apply( handler.ctx2, args ) === false ) {                    stoped = true;                    break;                }            }                return !stoped;        }            protos = {                /**             * 绑定事件。             *             * `callback`方法在执行时,arguments将会来源于trigger的时候携带的参数。如             * ```javascript             * var obj = {};             *             * // 使得obj有事件行为             * Mediator.installTo( obj );             *             * obj.on( 'testa', function( arg1, arg2 ) {             *     console.log( arg1, arg2 ); // => 'arg1', 'arg2'             * });             *             * obj.trigger( 'testa', 'arg1', 'arg2' );             * ```             *             * 如果`callback`中,某一个方法`return false`了,则后续的其他`callback`都不会被执行到。             * 切会影响到`trigger`方法的返回值,为`false`。             *             * `on`还可以用来添加一个特殊事件`all`, 这样所有的事件触发都会响应到。同时此类`callback`中的arguments有一个不同处,             * 就是第一个参数为`type`,记录当前是什么事件在触发。此类`callback`的优先级比脚低,会再正常`callback`执行完后触发。             * ```javascript             * obj.on( 'all', function( type, arg1, arg2 ) {             *     console.log( type, arg1, arg2 ); // => 'testa', 'arg1', 'arg2'             * });             * ```             *             * @method on             * @grammar on( name, callback[, context] ) => self             * @param  {String}   name     事件名,支持多个事件用空格隔开             * @param  {Function} callback 事件处理器             * @param  {Object}   [context]  事件处理器的上下文。             * @return {self} 返回自身,方便链式             * @chainable             * @class Mediator             */            on: function( name, callback, context ) {                var me = this,                    set;                    if ( !callback ) {                    return this;                }                    set = this._events || (this._events = []);                    eachEvent( name, callback, function( name, callback ) {                    var handler = { e: name };                        handler.cb = callback;                    handler.ctx = context;                    handler.ctx2 = context || me;                    handler.id = set.length;                        set.push( handler );                });                    return this;            },                /**             * 绑定事件,且当handler执行完后,自动解除绑定。             * @method once             * @grammar once( name, callback[, context] ) => self             * @param  {String}   name     事件名             * @param  {Function} callback 事件处理器             * @param  {Object}   [context]  事件处理器的上下文。             * @return {self} 返回自身,方便链式             * @chainable             */            once: function( name, callback, context ) {                var me = this;                    if ( !callback ) {                    return me;                }                    eachEvent( name, callback, function( name, callback ) {                    var once = function() {                            me.off( name, once );                            return callback.apply( context || me, arguments );                        };                        once._cb = callback;                    me.on( name, once, context );                });                    return me;            },                /**             * 解除事件绑定             * @method off             * @grammar off( [name[, callback[, context] ] ] ) => self             * @param  {String}   [name]     事件名             * @param  {Function} [callback] 事件处理器             * @param  {Object}   [context]  事件处理器的上下文。             * @return {self} 返回自身,方便链式             * @chainable             */            off: function( name, cb, ctx ) {                var events = this._events;                    if ( !events ) {                    return this;                }                    if ( !name && !cb && !ctx ) {                    this._events = [];                    return this;                }                    eachEvent( name, cb, function( name, cb ) {                    $.each( findHandlers( events, name, cb, ctx ), function() {                        delete events[ this.id ];                    });                });                    return this;            },                /**             * 触发事件             * @method trigger             * @grammar trigger( name[, args...] ) => self             * @param  {String}   type     事件名             * @param  {*} [...] 任意参数             * @return {Boolean} 如果handler中return false了,则返回false, 否则返回true             */            trigger: function( type ) {                var args, events, allEvents;                    if ( !this._events || !type ) {                    return this;                }                    args = slice.call( arguments, 1 );                events = findHandlers( this._events, type );                allEvents = findHandlers( this._events, 'all' );                    return triggerHanders( events, args ) &&                        triggerHanders( allEvents, arguments );            }        };            /**         * 中介者,它本身是个单例,但可以通过[installTo](#WebUploader:Mediator:installTo)方法,使任何对象具备事件行为。         * 主要目的是负责模块与模块之间的合作,降低耦合度。         *         * @class Mediator         */        return $.extend({                /**             * 可以通过这个接口,使任何对象具备事件功能。             * @method installTo             * @param  {Object} obj 需要具备事件行为的对象。             * @return {Object} 返回obj.             */            installTo: function( obj ) {                return $.extend( obj, protos );            }            }, protos );    });    /**     * @fileOverview Uploader上传类     */    define('uploader',[        'base',        'mediator'    ], function( Base, Mediator ) {            var $ = Base.$;            /**         * 上传入口类。         * @class Uploader         * @constructor         * @grammar new Uploader( opts ) => Uploader         * @example         * var uploader = WebUploader.Uploader({         *     swf: 'path_of_swf/Uploader.swf',         *         *     // 开起分片上传。         *     chunked: true         * });         */        function Uploader( opts ) {            this.options = $.extend( true, {}, Uploader.options, opts );            this._init( this.options );        }            // default Options        // widgets中有相应扩展        Uploader.options = {};        Mediator.installTo( Uploader.prototype );            // 批量添加纯命令式方法。        $.each({            upload: 'start-upload',            stop: 'stop-upload',            getFile: 'get-file',            getFiles: 'get-files',            addFile: 'add-file',            addFiles: 'add-file',            sort: 'sort-files',            removeFile: 'remove-file',            skipFile: 'skip-file',            retry: 'retry',            isInProgress: 'is-in-progress',            makeThumb: 'make-thumb',            getDimension: 'get-dimension',            addButton: 'add-btn',            getRuntimeType: 'get-runtime-type',            refresh: 'refresh',            disable: 'disable',            enable: 'enable',            reset: 'reset'        }, function( fn, command ) {            Uploader.prototype[ fn ] = function() {                return this.request( command, arguments );            };        });            $.extend( Uploader.prototype, {            state: 'pending',                _init: function( opts ) {                var me = this;                    me.request( 'init', opts, function() {                    me.state = 'ready';                    me.trigger('ready');                });            },                /**             * 获取或者设置Uploader配置项。             * @method option             * @grammar option( key ) => *             * @grammar option( key, val ) => self             * @example             *             * // 初始状态图片上传前不会压缩             * var uploader = new WebUploader.Uploader({             *     resize: null;             * });             *             * // 修改后图片上传前,尝试将图片压缩到1600 * 1600             * uploader.options( 'resize', {             *     width: 1600,             *     height: 1600             * });             */            option: function( key, val ) {                var opts = this.options;                    // setter                if ( arguments.length > 1 ) {                        if ( $.isPlainObject( val ) &&                            $.isPlainObject( opts[ key ] ) ) {                        $.extend( opts[ key ], val );                    } else {                        opts[ key ] = val;                    }                    } else {    // getter                    return key ? opts[ key ] : opts;                }            },                /**             * 获取文件统计信息。返回一个包含一下信息的对象。             * * `successNum` 上传成功的文件数             * * `uploadFailNum` 上传失败的文件数             * * `cancelNum` 被删除的文件数             * * `invalidNum` 无效的文件数             * * `queueNum` 还在队列中的文件数             * @method getStats             * @grammar getStats() => Object             */            getStats: function() {                // return this._mgr.getStats.apply( this._mgr, arguments );                var stats = this.request('get-stats');                    return {                    successNum: stats.numOfSuccess,                        // who care?                    // queueFailNum: 0,                    cancelNum: stats.numOfCancel,                    invalidNum: stats.numOfInvalid,                    uploadFailNum: stats.numOfUploadFailed,                    queueNum: stats.numOfQueue                };            },                // 需要重写此方法来来支持opts.onEvent和instance.onEvent的处理器            trigger: function( type/*, args...*/ ) {                var args = [].slice.call( arguments, 1 ),                    opts = this.options,                    name = 'on' + type.substring( 0, 1 ).toUpperCase() +                        type.substring( 1 );                    if (                        // 调用通过on方法注册的handler.                        Mediator.trigger.apply( this, arguments ) === false ||                            // 调用opts.onEvent                        $.isFunction( opts[ name ] ) &&                        opts[ name ].apply( this, args ) === false ||                            // 调用this.onEvent                        $.isFunction( this[ name ] ) &&                        this[ name ].apply( this, args ) === false ||                            // 广播所有uploader的事件。                        Mediator.trigger.apply( Mediator,                        [ this, type ].concat( args ) ) === false ) {                        return false;                }                    return true;            },                // widgets/widget.js将补充此方法的详细文档。            request: Base.noop        });            /**         * 创建Uploader实例,等同于new Uploader( opts );         * @method create         * @class Base         * @static         * @grammar Base.create( opts ) => Uploader         */        Base.create = Uploader.create = function( opts ) {            return new Uploader( opts );        };            // 暴露Uploader,可以通过它来扩展业务逻辑。        Base.Uploader = Uploader;            return Uploader;    });    /**     * @fileOverview Runtime管理器,负责Runtime的选择, 连接     */    define('runtime/runtime',[        'base',        'mediator'    ], function( Base, Mediator ) {            var $ = Base.$,            factories = {},                // 获取对象的第一个key            getFirstKey = function( obj ) {                for ( var key in obj ) {                    if ( obj.hasOwnProperty( key ) ) {                        return key;                    }                }                return null;            };            // 接口类。        function Runtime( options ) {            this.options = $.extend({                container: document.body            }, options );            this.uid = Base.guid('rt_');        }            $.extend( Runtime.prototype, {                getContainer: function() {                var opts = this.options,                    parent, container;                    if ( this._container ) {                    return this._container;                }                    parent = $( opts.container || document.body );                container = $( document.createElement('div') );                    container.attr( 'id', 'rt_' + this.uid );                container.css({                    position: 'absolute',                    top: '0px',                    left: '0px',                    width: '1px',                    height: '1px',                    overflow: 'hidden'                });                    parent.append( container );                parent.addClass('webuploader-container');                this._container = container;                return container;            },                init: Base.noop,            exec: Base.noop,                destroy: function() {                if ( this._container ) {                    this._container.parentNode.removeChild( this.__container );                }                    this.off();            }        });            Runtime.orders = 'html5,flash';                /**         * 添加Runtime实现。         * @param {String} type    类型         * @param {Runtime} factory 具体Runtime实现。         */        Runtime.addRuntime = function( type, factory ) {            factories[ type ] = factory;        };            Runtime.hasRuntime = function( type ) {            return !!(type ? factories[ type ] : getFirstKey( factories ));        };            Runtime.create = function( opts, orders ) {            var type, runtime;                orders = orders || Runtime.orders;            $.each( orders.split( /\s*,\s*/g ), function() {                if ( factories[ this ] ) {                    type = this;                    return false;                }            });                type = type || getFirstKey( factories );                if ( !type ) {                throw new Error('Runtime Error');            }                runtime = new factories[ type ]( opts );            return runtime;        };            Mediator.installTo( Runtime.prototype );        return Runtime;    });        /**     * @fileOverview Runtime管理器,负责Runtime的选择, 连接     */    define('runtime/client',[        'base',        'mediator',        'runtime/runtime'    ], function( Base, Mediator, Runtime ) {            var cache;            cache = (function() {            var obj = {};                return {                add: function( runtime ) {                    obj[ runtime.uid ] = runtime;                },                    get: function( ruid, standalone ) {                    var i;                        if ( ruid ) {                        return obj[ ruid ];                    }                        for ( i in obj ) {                        // 有些类型不能重用,比如filepicker.                        if ( standalone && obj[ i ].__standalone ) {                            continue;                        }                            return obj[ i ];                    }                        return null;                },                    remove: function( runtime ) {                    delete obj[ runtime.uid ];                }            };        })();            function RuntimeClient( component, standalone ) {            var deferred = Base.Deferred(),                runtime;                this.uid = Base.guid('client_');                // 允许runtime没有初始化之前,注册一些方法在初始化后执行。            this.runtimeReady = function( cb ) {                return deferred.done( cb );            };                this.connectRuntime = function( opts, cb ) {                    // already connected.                if ( runtime ) {                    throw new Error('already connected!');                }                    deferred.done( cb );                    if ( typeof opts === 'string' && cache.get( opts ) ) {                    runtime = cache.get( opts );                }                    // 像filePicker只能独立存在,不能公用。                runtime = runtime || cache.get( null, standalone );                    // 需要创建                if ( !runtime ) {                    runtime = Runtime.create( opts, opts.runtimeOrder );                    runtime.__promise = deferred.promise();                    runtime.once( 'ready', deferred.resolve );                    runtime.init();                    cache.add( runtime );                    runtime.__client = 1;                } else {                    // 来自cache                    Base.$.extend( runtime.options, opts );                    runtime.__promise.then( deferred.resolve );                    runtime.__client++;                }                    standalone && (runtime.__standalone = standalone);                return runtime;            };                this.getRuntime = function() {                return runtime;            };                this.disconnectRuntime = function() {                if ( !runtime ) {                    return;                }                    runtime.__client--;                    if ( runtime.__client <= 0 ) {                    cache.remove( runtime );                    delete runtime.__promise;                    runtime.destroy();                }                    runtime = null;            };                this.exec = function() {                if ( !runtime ) {                    return;                }                    var args = Base.slice( arguments );                component && args.unshift( component );                    return runtime.exec.apply( this, args );            };                this.getRuid = function() {                return runtime && runtime.uid;            };                this.destroy = (function( destroy ) {                return function() {                    destroy && destroy.apply( this, arguments );                    this.trigger('destroy');                    this.off();                    this.exec('destroy');                    this.disconnectRuntime();                };            })( this.destroy );        }            Mediator.installTo( RuntimeClient.prototype );        return RuntimeClient;    });    /**     * @fileOverview 错误信息     */    define('lib/dnd',[        'base',        'mediator',        'runtime/client'    ], function( Base, Mediator, RuntimeClent ) {            var $ = Base.$;            function DragAndDrop( opts ) {            opts = this.options = $.extend({}, DragAndDrop.options, opts );                opts.container = $( opts.container );                if ( !opts.container.length ) {                return;            }                RuntimeClent.call( this, 'DragAndDrop' );        }            DragAndDrop.options = {            accept: null,            disableGlobalDnd: false        };            Base.inherits( RuntimeClent, {            constructor: DragAndDrop,                init: function() {                var me = this;                    me.connectRuntime( me.options, function() {                    me.exec('init');                    me.trigger('ready');                });            },                destroy: function() {                this.disconnectRuntime();            }        });            Mediator.installTo( DragAndDrop.prototype );            return DragAndDrop;    });    /**     * @fileOverview 组件基类。     */    define('widgets/widget',[        'base',        'uploader'    ], function( Base, Uploader ) {            var $ = Base.$,            _init = Uploader.prototype._init,            IGNORE = {},            widgetClass = [];            function isArrayLike( obj ) {            if ( !obj ) {                return false;            }                var length = obj.length,                type = $.type( obj );                if ( obj.nodeType === 1 && length ) {                return true;            }                return type === 'array' || type !== 'function' && type !== 'string' &&                    (length === 0 || typeof length === 'number' && length > 0 &&                    (length - 1) in obj);        }            function Widget( uploader ) {            this.owner = uploader;            this.options = uploader.options;        }            $.extend( Widget.prototype, {                init: Base.noop,                // 类Backbone的事件监听声明,监听uploader实例上的事件            // widget直接无法监听事件,事件只能通过uploader来传递            invoke: function( apiName, args ) {                    /*                    {                        'make-thumb': 'makeThumb'                    }                 */                var map = this.responseMap;                    // 如果无API响应声明则忽略                if ( !map || !(apiName in map) || !(map[ apiName ] in this) ||                        !$.isFunction( this[ map[ apiName ] ] ) ) {                        return IGNORE;                }                    return this[ map[ apiName ] ].apply( this, args );                },                /**             * 发送命令。当传入`callback`或者`handler`中返回`promise`时。返回一个当所有`handler`中的promise都完成后完成的新`promise`。             * @method request             * @grammar request( command, args ) => * | Promise             * @grammar request( command, args, callback ) => Promise             * @for  Uploader             */            request: function() {                return this.owner.request.apply( this.owner, arguments );            }        });            // 扩展Uploader.        $.extend( Uploader.prototype, {                // 覆写_init用来初始化widgets            _init: function() {                var me = this,                    widgets = me._widgets = [];                    $.each( widgetClass, function( _, klass ) {                    widgets.push( new klass( me ) );                });                    return _init.apply( me, arguments );            },                request: function( apiName, args, callback ) {                var i = 0,                    widgets = this._widgets,                    len = widgets.length,                    rlts = [],                    dfds = [],                    widget, rlt, promise, key;                    args = isArrayLike( args ) ? args : [ args ];                    for ( ; i < len; i++ ) {                    widget = widgets[ i ];                    rlt = widget.invoke( apiName, args );                        if ( rlt !== IGNORE ) {                            // Deferred对象                        if ( Base.isPromise( rlt ) ) {                            dfds.push( rlt );                        } else {                            rlts.push( rlt );                        }                    }                }                    // 如果有callback,则用异步方式。                if ( callback || dfds.length ) {                    promise = Base.when.apply( Base, dfds );                    key = promise.pipe ? 'pipe' : 'then';                        // 很重要不能删除。删除了会死循环。                    // 保证执行顺序。让callback总是在下一个tick中执行。                    return promise[ key ](function() {                                var deferred = Base.Deferred(),                                    args = arguments;                                    setTimeout(function() {                                    deferred.resolve.apply( deferred, args );                                }, 1 );                                    return deferred.promise();                            })[ key ]( callback || Base.noop );                } else {                    return rlts[ 0 ];                }            }        });            /**         * 添加组件         * @param  {object} widgetProto 组件原型,构造函数通过constructor属性定义         * @param  {object} responseMap API名称与函数实现的映射         * @example         *     Uploader.register( {         *         init: function( options ) {},         *         makeThumb: function() {}         *     }, {         *         'make-thumb': 'makeThumb'         *     } );         */        Uploader.register = Widget.register = function( responseMap, widgetProto ) {            var map = { init: 'init' },                klass;                if ( arguments.length === 1 ) {                widgetProto = responseMap;                widgetProto.responseMap = map;            } else {                widgetProto.responseMap = $.extend( map, responseMap );            }                klass = Base.inherits( Widget, widgetProto );            widgetClass.push( klass );                return klass;        };            return Widget;    });    /**     * @fileOverview DragAndDrop Widget。     */    define('widgets/filednd',[        'base',        'uploader',        'lib/dnd',        'widgets/widget'    ], function( Base, Uploader, Dnd ) {        var $ = Base.$;            Uploader.options.dnd = '';            /**         * @property {Selector} [dnd=undefined]  指定Drag And Drop拖拽的容器,如果不指定,则不启动。         * @namespace options         * @for Uploader         */            /**         * @event dndAccept         * @param {DataTransferItemList} items DataTransferItem         * @description 阻止此事件可以拒绝某些类型的文件拖入进来。目前只有 chrome 提供这样的 API,且只能通过 mime-type 验证。         * @for  Uploader         */        return Uploader.register({            init: function( opts ) {                    if ( !opts.dnd ||                        this.request('predict-runtime-type') !== 'html5' ) {                    return;                }                    var me = this,                    deferred = Base.Deferred(),                    options = $.extend({}, {                        disableGlobalDnd: opts.disableGlobalDnd,                        container: opts.dnd,                        accept: opts.accept                    }),                    dnd;                    dnd = new Dnd( options );                    dnd.once( 'ready', deferred.resolve );                dnd.on( 'drop', function( files ) {                    me.request( 'add-file', [ files ]);                });                    // 检测文件是否全部允许添加。                dnd.on( 'accept', function( items ) {                    return me.owner.trigger( 'dndAccept', items );                });                    dnd.init();                    return deferred.promise();            }        });    });        /**     * @fileOverview 错误信息     */    define('lib/filepaste',[        'base',        'mediator',        'runtime/client'    ], function( Base, Mediator, RuntimeClent ) {            var $ = Base.$;            function FilePaste( opts ) {            opts = this.options = $.extend({}, opts );            opts.container = $( opts.container || document.body );            RuntimeClent.call( this, 'FilePaste' );        }            Base.inherits( RuntimeClent, {            constructor: FilePaste,                init: function() {                var me = this;                    me.connectRuntime( me.options, function() {                    me.exec('init');                    me.trigger('ready');                });            },                destroy: function() {                this.exec('destroy');                this.disconnectRuntime();                this.off();            }        });            Mediator.installTo( FilePaste.prototype );            return FilePaste;    });    /**     * @fileOverview 组件基类。     */    define('widgets/filepaste',[        'base',        'uploader',        'lib/filepaste',        'widgets/widget'    ], function( Base, Uploader, FilePaste ) {        var $ = Base.$;            /**         * @property {Selector} [paste=undefined]  指定监听paste事件的容器,如果不指定,不启用此功能。此功能为通过粘贴来添加截屏的图片。建议设置为`document.body`.         * @namespace options         * @for Uploader         */        return Uploader.register({            init: function( opts ) {                    if ( !opts.paste ||                        this.request('predict-runtime-type') !== 'html5' ) {                    return;                }                    var me = this,                    deferred = Base.Deferred(),                    options = $.extend({}, {                        container: opts.paste,                        accept: opts.accept                    }),                    paste;                    paste = new FilePaste( options );                    paste.once( 'ready', deferred.resolve );                paste.on( 'paste', function( files ) {                    me.owner.request( 'add-file', [ files ]);                });                paste.init();                    return deferred.promise();            }        });    });    /**     * @fileOverview Blob     */    define('lib/blob',[        'base',        'runtime/client'    ], function( Base, RuntimeClient ) {            function Blob( ruid, source ) {            var me = this;                me.source = source;            me.ruid = ruid;                RuntimeClient.call( me, 'Blob' );                this.uid = source.uid || this.uid;            this.type = source.type || '';            this.size = source.size || 0;                if ( ruid ) {                me.connectRuntime( ruid );            }        }            Base.inherits( RuntimeClient, {            constructor: Blob,                slice: function( start, end ) {                return this.exec( 'slice', start, end );            },                getSource: function() {                return this.source;            }        });            return Blob;    });    /**     * 为了统一化Flash的File和HTML5的File而存在。     * 以至于要调用Flash里面的File,也可以像调用HTML5版本的File一下。     * @fileOverview File     */    define('lib/file',[        'base',        'lib/blob'    ], function( Base, Blob ) {            var uid = 1,            rExt = /\.([^.]+)$/;            function File( ruid, file ) {            var ext;                Blob.apply( this, arguments );            this.name = file.name || ('untitled' + uid++);            ext = rExt.exec( file.name ) ? RegExp.$1.toLowerCase() : '';                // todo 支持其他类型文件的转换。                // 如果有mimetype, 但是文件名里面没有找出后缀规律            if ( !ext && this.type ) {                ext = /\/(jpg|jpeg|png|gif|bmp)$/i.exec( this.type ) ?                        RegExp.$1.toLowerCase() : '';                this.name += '.' + ext;            }                // 如果没有指定mimetype, 但是知道文件后缀。            if ( !this.type &&  ~'jpg,jpeg,png,gif,bmp'.indexOf( ext ) ) {                this.type = 'image/' + (ext === 'jpg' ? 'jpeg' : ext);            }                this.ext = ext;            this.lastModifiedDate = file.lastModifiedDate ||                    (new Date()).toLocaleString();        }            return Base.inherits( Blob, File );    });        /**     * @fileOverview 错误信息     */    define('lib/filepicker',[        'base',        'runtime/client',        'lib/file'    ], function( Base, RuntimeClent, File ) {            var $ = Base.$;            function FilePicker( opts ) {            opts = this.options = $.extend({}, FilePicker.options, opts );                opts.container = $( opts.id );                if ( !opts.container.length ) {                throw new Error('按钮指定错误');            }                opts.innerHTML = opts.innerHTML || opts.label ||                    opts.container.html() || '';                opts.button = $( opts.button || document.createElement('div') );            opts.button.html( opts.innerHTML );            opts.container.html( opts.button );                RuntimeClent.call( this, 'FilePicker', true );        }            FilePicker.options = {            button: null,            container: null,            label: null,            innerHTML: null,            multiple: true,            accept: null,            name: 'file'        };            Base.inherits( RuntimeClent, {            constructor: FilePicker,                init: function() {                var me = this,                    opts = me.options,                    button = opts.button;                    button.addClass('webuploader-pick');                    me.on( 'all', function( type ) {                    var files;                        switch ( type ) {                        case 'mouseenter':                            button.addClass('webuploader-pick-hover');                            break;                            case 'mouseleave':                            button.removeClass('webuploader-pick-hover');                            break;                            case 'change':                            files = me.exec('getFiles');                            me.trigger( 'select', $.map( files, function( file ) {                                file = new File( me.getRuid(), file );                                    // 记录来源。                                file._refer = opts.container;                                return file;                            }), opts.container );                            break;                    }                });                    me.connectRuntime( opts, function() {                    me.refresh();                    me.exec( 'init', opts );                    me.trigger('ready');                });                    $( window ).on( 'resize', function() {                    me.refresh();                });            },                refresh: function() {                var shimContainer = this.getRuntime().getContainer(),                    button = this.options.button,                    width = button.outerWidth ?                            button.outerWidth() : button.width(),                        height = button.outerHeight ?                            button.outerHeight() : button.height(),                        pos = button.offset();                    width && height && shimContainer.css({                    bottom: 'auto',                    right: 'auto',                    width: width + 'px',                    height: height + 'px'                }).offset( pos );            },                enable: function() {                var btn = this.options.button;                    btn.removeClass('webuploader-pick-disable');                this.refresh();            },                disable: function() {                var btn = this.options.button;                    this.getRuntime().getContainer().css({                    top: '-99999px'                });                    btn.addClass('webuploader-pick-disable');            },                destroy: function() {                if ( this.runtime ) {                    this.exec('destroy');                    this.disconnectRuntime();                }            }        });            return FilePicker;    });        /**     * @fileOverview 文件选择相关     */    define('widgets/filepicker',[        'base',        'uploader',        'lib/filepicker',        'widgets/widget'    ], function( Base, Uploader, FilePicker ) {        var $ = Base.$;            $.extend( Uploader.options, {                /**             * @property {Selector | Object} [pick=undefined]             * @namespace options             * @for Uploader             * @description 指定选择文件的按钮容器,不指定则不创建按钮。             *             * * `id` {Seletor} 指定选择文件的按钮容器,不指定则不创建按钮。             * * `label` {String} 请采用 `innerHTML` 代替             * * `innerHTML` {String} 指定按钮文字。不指定时优先从指定的容器中看是否自带文字。             * * `multiple` {Boolean} 是否开起同时选择多个文件能力。             */            pick: null,                /**             * @property {Arroy} [accept=null]             * @namespace options             * @for Uploader             * @description 指定接受哪些类型的文件。 由于目前还有ext转mimeType表,所以这里需要分开指定。             *             * * `title` {String} 文字描述             * * `extensions` {String} 允许的文件后缀,不带点,多个用逗号分割。             * * `mimeTypes` {String} 多个用逗号分割。             *             * 如:             *             * ```             * {             *     title: 'Images',             *     extensions: 'gif,jpg,jpeg,bmp,png',             *     mimeTypes: 'image/*'             * }             * ```             */            accept: null/*{                title: 'Images',                extensions: 'gif,jpg,jpeg,bmp,png',                mimeTypes: 'image/*'            }*/        });            return Uploader.register({            'add-btn': 'addButton',            refresh: 'refresh',            disable: 'disable',            enable: 'enable'        }, {                init: function( opts ) {                this.pickers = [];                return opts.pick && this.addButton( opts.pick );            },                refresh: function() {                $.each( this.pickers, function() {                    this.refresh();                });            },                /**             * @method addButton             * @for Uploader             * @grammar addButton( pick ) => Promise             * @description             * 添加文件选择按钮,如果一个按钮不够,需要调用此方法来添加。参数跟[options.pick](#WebUploader:Uploader:options)一致。             * @example             * uploader.addButton({             *     id: '#btnContainer',             *     innerHTML: '选择文件'             * });             */            addButton: function( pick ) {                var me = this,                    opts = me.options,                    accept = opts.accept,                    options, picker, deferred;                    if ( !pick ) {                    return;                }                    deferred = Base.Deferred();                $.isPlainObject( pick ) || (pick = {                    id: pick                });                    options = $.extend({}, pick, {                    accept: $.isPlainObject( accept ) ? [ accept ] : accept,                    swf: opts.swf,                    runtimeOrder: opts.runtimeOrder                });                    picker = new FilePicker( options );                    picker.once( 'ready', deferred.resolve );                picker.on( 'select', function( files ) {                    me.owner.request( 'add-file', [ files ]);                });                picker.init();                    this.pickers.push( picker );                    return deferred.promise();            },                disable: function() {                $.each( this.pickers, function() {                    this.disable();                });            },                enable: function() {                $.each( this.pickers, function() {                    this.enable();                });            }        });    });    /**     * @fileOverview 文件属性封装     */    define('file',[        'base',        'mediator'    ], function( Base, Mediator ) {            var $ = Base.$,            idPrefix = 'WU_FILE_',            idSuffix = 0,            rExt = /\.([^.]+)$/,            statusMap = {};            function gid() {            return idPrefix + idSuffix++;        }            /**         * 文件类         * @class File         * @constructor 构造函数         * @grammar new File( source ) => File         * @param {Lib.File} source [lib.File](#Lib.File)实例, 此source对象是带有Runtime信息的。         */        function WUFile( source ) {                /**             * 文件名,包括扩展名(后缀)             * @property name             * @type {string}             */            this.name = source.name || 'Untitled';                /**             * 文件体积(字节)             * @property size             * @type {uint}             * @default 0             */            this.size = source.size || 0;                /**             * 文件MIMETYPE类型,与文件类型的对应关系请参考[http://t.cn/z8ZnFny](http://t.cn/z8ZnFny)             * @property type             * @type {string}             * @default 'application'             */            this.type = source.type || 'application';                /**             * 文件最后修改日期             * @property lastModifiedDate             * @type {int}             * @default 当前时间戳             */            this.lastModifiedDate = source.lastModifiedDate || (new Date() * 1);                /**             * 文件ID,每个对象具有唯一ID,与文件名无关             * @property id             * @type {string}             */            this.id = gid();                /**             * 文件扩展名,通过文件名获取,例如test.png的扩展名为png             * @property ext             * @type {string}             */            this.ext = rExt.exec( this.name ) ? RegExp.$1 : '';                    /**             * 状态文字说明。在不同的status语境下有不同的用途。             * @property statusText             * @type {string}             */            this.statusText = '';                // 存储文件状态,防止通过属性直接修改            statusMap[ this.id ] = WUFile.Status.INITED;                this.source = source;            this.loaded = 0;                this.on( 'error', function( msg ) {                this.setStatus( WUFile.Status.ERROR, msg );            });        }            $.extend( WUFile.prototype, {                /**             * 设置状态,状态变化时会触发`change`事件。             * @method setStatus             * @grammar setStatus( status[, statusText] );             * @param {File.Status|String} status [文件状态值](#WebUploader:File:File.Status)             * @param {String} [statusText=''] 状态说明,常在error时使用,用http, abort,server等来标记是由于什么原因导致文件错误。             */            setStatus: function( status, text ) {                    var prevStatus = statusMap[ this.id ];                    typeof text !== 'undefined' && (this.statusText = text);                    if ( status !== prevStatus ) {                    statusMap[ this.id ] = status;                    /**                     * 文件状态变化                     * @event statuschange                     */                    this.trigger( 'statuschange', status, prevStatus );                }                },                /**             * 获取文件状态             * @return {File.Status}             * @example                     文件状态具体包括以下几种类型:                     {                         // 初始化                        INITED:     0,                        // 已入队列                        QUEUED:     1,                        // 正在上传                        PROGRESS:     2,                        // 上传出错                        ERROR:         3,                        // 上传成功                        COMPLETE:     4,                        // 上传取消                        CANCELLED:     5                    }             */            getStatus: function() {                return statusMap[ this.id ];            },                /**             * 获取文件原始信息。             * @return {*}             */            getSource: function() {                return this.source;            },                destory: function() {                delete statusMap[ this.id ];            }        });            Mediator.installTo( WUFile.prototype );            /**         * 文件状态值,具体包括以下几种类型:         * * `inited` 初始状态         * * `queued` 已经进入队列, 等待上传         * * `progress` 上传中         * * `complete` 上传完成。         * * `error` 上传出错,可重试         * * `interrupt` 上传中断,可续传。         * * `invalid` 文件不合格,不能重试上传。会自动从队列中移除。         * * `cancelled` 文件被移除。         * @property {Object} Status         * @namespace File         * @class File         * @static         */        WUFile.Status = {            INITED:     'inited',    // 初始状态            QUEUED:     'queued',    // 已经进入队列, 等待上传            PROGRESS:   'progress',    // 上传中            ERROR:      'error',    // 上传出错,可重试            COMPLETE:   'complete',    // 上传完成。            CANCELLED:  'cancelled',    // 上传取消。            INTERRUPT:  'interrupt',    // 上传中断,可续传。            INVALID:    'invalid'    // 文件不合格,不能重试上传。        };            return WUFile;    });        /**     * @fileOverview 文件队列     */    define('queue',[        'base',        'mediator',        'file'    ], function( Base, Mediator, WUFile ) {            var $ = Base.$,            STATUS = WUFile.Status;            /**         * 文件队列, 用来存储各个状态中的文件。         * @class Queue         * @extends Mediator         */        function Queue() {                /**             * 统计文件数。             * * `numOfQueue` 队列中的文件数。             * * `numOfSuccess` 上传成功的文件数             * * `numOfCancel` 被移除的文件数             * * `numOfProgress` 正在上传中的文件数             * * `numOfUploadFailed` 上传错误的文件数。             * * `numOfInvalid` 无效的文件数。             * @property {Object} stats             */            this.stats = {                numOfQueue: 0,                numOfSuccess: 0,                numOfCancel: 0,                numOfProgress: 0,                numOfUploadFailed: 0,                numOfInvalid: 0            };                // 上传队列,仅包括等待上传的文件            this._queue = [];                // 存储所有文件            this._map = {};        }            $.extend( Queue.prototype, {                /**             * 将新文件加入对队列尾部             *             * @method append             * @param  {File} file   文件对象             */            append: function( file ) {                this._queue.push( file );                this._fileAdded( file );                return this;            },                /**             * 将新文件加入对队列头部             *             * @method prepend             * @param  {File} file   文件对象             */            prepend: function( file ) {                this._queue.unshift( file );                this._fileAdded( file );                return this;            },                /**             * 获取文件对象             *             * @method getFile             * @param  {String} fileId   文件ID             * @return {File}             */            getFile: function( fileId ) {                if ( typeof fileId !== 'string' ) {                    return fileId;                }                return this._map[ fileId ];            },                /**             * 从队列中取出一个指定状态的文件。             * @grammar fetch( status ) => File             * @method fetch             * @param {String} status [文件状态值](#WebUploader:File:File.Status)             * @return {File} [File](#WebUploader:File)             */            fetch: function( status ) {                var len = this._queue.length,                    i, file;                    status = status || STATUS.QUEUED;                    for ( i = 0; i < len; i++ ) {                    file = this._queue[ i ];                        if ( status === file.getStatus() ) {                        return file;                    }                }                    return null;            },                /**             * 对队列进行排序,能够控制文件上传顺序。             * @grammar sort( fn ) => undefined             * @method sort             * @param {Function} fn 排序方法             */            sort: function( fn ) {                if ( typeof fn === 'function' ) {                    this._queue.sort( fn );                }            },                /**             * 获取指定类型的文件列表, 列表中每一个成员为[File](#WebUploader:File)对象。             * @grammar getFiles( [status1[, status2 ...]] ) => Array             * @method getFiles             * @param {String} [status] [文件状态值](#WebUploader:File:File.Status)             */            getFiles: function() {                var sts = [].slice.call( arguments, 0 ),                    ret = [],                    i = 0,                    len = this._queue.length,                    file;                    for ( ; i < len; i++ ) {                    file = this._queue[ i ];                        if ( sts.length && !~$.inArray( file.getStatus(), sts ) ) {                        continue;                    }                        ret.push( file );                }                    return ret;            },                _fileAdded: function( file ) {                var me = this,                    existing = this._map[ file.id ];                    if ( !existing ) {                    this._map[ file.id ] = file;                        file.on( 'statuschange', function( cur, pre ) {                        me._onFileStatusChange( cur, pre );                    });                }                    file.setStatus( STATUS.QUEUED );            },                _onFileStatusChange: function( curStatus, preStatus ) {                var stats = this.stats;                    switch ( preStatus ) {                    case STATUS.PROGRESS:                        stats.numOfProgress--;                        break;                        case STATUS.QUEUED:                        stats.numOfQueue --;                        break;                        case STATUS.ERROR:                        stats.numOfUploadFailed--;                        break;                        case STATUS.INVALID:                        stats.numOfInvalid--;                        break;                }                    switch ( curStatus ) {                    case STATUS.QUEUED:                        stats.numOfQueue++;                        break;                        case STATUS.PROGRESS:                        stats.numOfProgress++;                        break;                        case STATUS.ERROR:                        stats.numOfUploadFailed++;                        break;                        case STATUS.COMPLETE:                        stats.numOfSuccess++;                        break;                        case STATUS.CANCELLED:                        stats.numOfCancel++;                        break;                        case STATUS.INVALID:                        stats.numOfInvalid++;                        break;                }            }            });            Mediator.installTo( Queue.prototype );            return Queue;    });    /**     * @fileOverview 队列     */    define('widgets/queue',[        'base',        'uploader',        'queue',        'file',        'lib/file',        'runtime/client',        'widgets/widget'    ], function( Base, Uploader, Queue, WUFile, File, RuntimeClient ) {            var $ = Base.$,            rExt = /\.\w+$/,            Status = WUFile.Status;            return Uploader.register({            'sort-files': 'sortFiles',            'add-file': 'addFiles',            'get-file': 'getFile',            'fetch-file': 'fetchFile',            'get-stats': 'getStats',            'get-files': 'getFiles',            'remove-file': 'removeFile',            'retry': 'retry',            'reset': 'reset',            'accept-file': 'acceptFile'        }, {                init: function( opts ) {                var me = this,                    deferred, len, i, item, arr, accept, runtime;                    if ( $.isPlainObject( opts.accept ) ) {                    opts.accept = [ opts.accept ];                }                    // accept中的中生成匹配正则。                if ( opts.accept ) {                    arr = [];                        for ( i = 0, len = opts.accept.length; i < len; i++ ) {                        item = opts.accept[ i ].extensions;                        item && arr.push( item );                    }                        if ( arr.length ) {                        accept = '\\.' + arr.join(',')                                .replace( /,/g, '$|\\.' )                                .replace( /\*/g, '.*' ) + '$';                    }                        me.accept = new RegExp( accept, 'i' );                }                    me.queue = new Queue();                me.stats = me.queue.stats;                    // 如果当前不是html5运行时,那就算了。                // 不执行后续操作                if ( this.request('predict-runtime-type') !== 'html5' ) {                    return;                }                    // 创建一个 html5 运行时的 placeholder                // 以至于外部添加原生 File 对象的时候能正确包裹一下供 webuploader 使用。                deferred = Base.Deferred();                runtime = new RuntimeClient('Placeholder');                runtime.connectRuntime({                    runtimeOrder: 'html5'                }, function() {                    me._ruid = runtime.getRuid();                    deferred.resolve();                });                return deferred.promise();            },                    // 为了支持外部直接添加一个原生File对象。            _wrapFile: function( file ) {                if ( !(file instanceof WUFile) ) {                        if ( !(file instanceof File) ) {                        if ( !this._ruid ) {                            throw new Error('Can\'t add external files.');                        }                        file = new File( this._ruid, file );                    }                        file = new WUFile( file );                }                    return file;            },                // 判断文件是否可以被加入队列            acceptFile: function( file ) {                var invalid = !file || file.size < 6 || this.accept &&                            // 如果名字中有后缀,才做后缀白名单处理。                        rExt.exec( file.name ) && !this.accept.test( file.name );                    return !invalid;            },                    /**             * @event beforeFileQueued             * @param {File} file File对象             * @description 当文件被加入队列之前触发,此事件的handler返回值为`false`,则此文件不会被添加进入队列。             * @for  Uploader             */                /**             * @event fileQueued             * @param {File} file File对象             * @description 当文件被加入队列以后触发。             * @for  Uploader             */                _addFile: function( file ) {                var me = this;                    file = me._wrapFile( file );                    // 不过类型判断允许不允许,先派送 `beforeFileQueued`                if ( !me.owner.trigger( 'beforeFileQueued', file ) ) {                    return;                }                    // 类型不匹配,则派送错误事件,并返回。                if ( !me.acceptFile( file ) ) {                    me.owner.trigger( 'error', 'Q_TYPE_DENIED', file );                    return;                }                    me.queue.append( file );                me.owner.trigger( 'fileQueued', file );                return file;            },                getFile: function( fileId ) {                return this.queue.getFile( fileId );            },                /**             * @event filesQueued             * @param {File} files 数组,内容为原始File(lib/File)对象。             * @description 当一批文件添加进队列以后触发。             * @for  Uploader             */                /**             * @method addFiles             * @grammar addFiles( file ) => undefined             * @grammar addFiles( [file1, file2 ...] ) => undefined             * @param {Array of File or File} [files] Files 对象 数组             * @description 添加文件到队列             * @for  Uploader             */            addFiles: function( files ) {                var me = this;                    if ( !files.length ) {                    files = [ files ];                }                    files = $.map( files, function( file ) {                    return me._addFile( file );                });                    me.owner.trigger( 'filesQueued', files );                    if ( me.options.auto ) {                    me.request('start-upload');                }            },                getStats: function() {                return this.stats;            },                /**             * @event fileDequeued             * @param {File} file File对象             * @description 当文件被移除队列后触发。             * @for  Uploader             */                /**             * @method removeFile             * @grammar removeFile( file ) => undefined             * @grammar removeFile( id ) => undefined             * @param {File|id} file File对象或这File对象的id             * @description 移除某一文件。             * @for  Uploader             * @example             *             * $li.on('click', '.remove-this', function() {             *     uploader.removeFile( file );             * })             */            removeFile: function( file ) {                var me = this;                    file = file.id ? file : me.queue.getFile( file );                    file.setStatus( Status.CANCELLED );                me.owner.trigger( 'fileDequeued', file );            },                /**             * @method getFiles             * @grammar getFiles() => Array             * @grammar getFiles( status1, status2, status... ) => Array             * @description 返回指定状态的文件集合,不传参数将返回所有状态的文件。             * @for  Uploader             * @example             * console.log( uploader.getFiles() );    // => all files             * console.log( uploader.getFiles('error') )    // => all error files.             */            getFiles: function() {                return this.queue.getFiles.apply( this.queue, arguments );            },                fetchFile: function() {                return this.queue.fetch.apply( this.queue, arguments );            },                /**             * @method retry             * @grammar retry() => undefined             * @grammar retry( file ) => undefined             * @description 重试上传,重试指定文件,或者从出错的文件开始重新上传。             * @for  Uploader             * @example             * function retry() {             *     uploader.retry();             * }             */            retry: function( file, noForceStart ) {                var me = this,                    files, i, len;                    if ( file ) {                    file = file.id ? file : me.queue.getFile( file );                    file.setStatus( Status.QUEUED );                    noForceStart || me.request('start-upload');                    return;                }                    files = me.queue.getFiles( Status.ERROR );                i = 0;                len = files.length;                    for ( ; i < len; i++ ) {                    file = files[ i ];                    file.setStatus( Status.QUEUED );                }                    me.request('start-upload');            },                /**             * @method sort             * @grammar sort( fn ) => undefined             * @description 排序队列中的文件,在上传之前调整可以控制上传顺序。             * @for  Uploader             */            sortFiles: function() {                return this.queue.sort.apply( this.queue, arguments );            },                /**             * @method reset             * @grammar reset() => undefined             * @description 重置uploader。目前只重置了队列。             * @for  Uploader             * @example             * uploader.reset();             */            reset: function() {                this.queue = new Queue();                this.stats = this.queue.stats;            }        });        });    /**     * @fileOverview 添加获取Runtime相关信息的方法。     */    define('widgets/runtime',[        'uploader',        'runtime/runtime',        'widgets/widget'    ], function( Uploader, Runtime ) {            Uploader.support = function() {            return Runtime.hasRuntime.apply( Runtime, arguments );        };            return Uploader.register({            'predict-runtime-type': 'predictRuntmeType'        }, {                init: function() {                if ( !this.predictRuntmeType() ) {                    throw Error('Runtime Error');                }            },                /**             * 预测Uploader将采用哪个`Runtime`             * @grammar predictRuntmeType() => String             * @method predictRuntmeType             * @for  Uploader             */            predictRuntmeType: function() {                var orders = this.options.runtimeOrder || Runtime.orders,                    type = this.type,                    i, len;                    if ( !type ) {                    orders = orders.split( /\s*,\s*/g );                        for ( i = 0, len = orders.length; i < len; i++ ) {                        if ( Runtime.hasRuntime( orders[ i ] ) ) {                            this.type = type = orders[ i ];                            break;                        }                    }                }                    return type;            }        });    });    /**     * @fileOverview Transport     */    define('lib/transport',[        'base',        'runtime/client',        'mediator'    ], function( Base, RuntimeClient, Mediator ) {            var $ = Base.$;            function Transport( opts ) {            var me = this;                opts = me.options = $.extend( true, {}, Transport.options, opts || {} );            RuntimeClient.call( this, 'Transport' );                this._blob = null;            this._formData = opts.formData || {};            this._headers = opts.headers || {};                this.on( 'progress', this._timeout );            this.on( 'load error', function() {                me.trigger( 'progress', 1 );                clearTimeout( me._timer );            });        }            Transport.options = {            server: '',            method: 'POST',                // 跨域时,是否允许携带cookie, 只有html5 runtime才有效            withCredentials: false,            fileVal: 'file',            timeout: 2 * 60 * 1000,    // 2分钟            formData: {},            headers: {},            sendAsBinary: false        };            $.extend( Transport.prototype, {                // 添加Blob, 只能添加一次,最后一次有效。            appendBlob: function( key, blob, filename ) {                var me = this,                    opts = me.options;                    if ( me.getRuid() ) {                    me.disconnectRuntime();                }                    // 连接到blob归属的同一个runtime.                me.connectRuntime( blob.ruid, function() {                    me.exec('init');                });                    me._blob = blob;                opts.fileVal = key || opts.fileVal;                opts.filename = filename || opts.filename;            },                // 添加其他字段            append: function( key, value ) {                if ( typeof key === 'object' ) {                    $.extend( this._formData, key );                } else {                    this._formData[ key ] = value;                }            },                setRequestHeader: function( key, value ) {                if ( typeof key === 'object' ) {                    $.extend( this._headers, key );                } else {                    this._headers[ key ] = value;                }            },                send: function( method ) {                this.exec( 'send', method );                this._timeout();            },                abort: function() {                clearTimeout( this._timer );                return this.exec('abort');            },                destroy: function() {                this.trigger('destroy');                this.off();                this.exec('destroy');                this.disconnectRuntime();            },                getResponse: function() {                return this.exec('getResponse');            },                getResponseAsJson: function() {                return this.exec('getResponseAsJson');            },                getStatus: function() {                return this.exec('getStatus');            },                _timeout: function() {                var me = this,                    duration = me.options.timeout;                    if ( !duration ) {                    return;                }                    clearTimeout( me._timer );                me._timer = setTimeout(function() {                    me.abort();                    me.trigger( 'error', 'timeout' );                }, duration );            }            });            // 让Transport具备事件功能。        Mediator.installTo( Transport.prototype );            return Transport;    });    /**     * @fileOverview 负责文件上传相关。     */    define('widgets/upload',[        'base',        'uploader',        'file',        'lib/transport',        'widgets/widget'    ], function( Base, Uploader, WUFile, Transport ) {            var $ = Base.$,            isPromise = Base.isPromise,            Status = WUFile.Status;            // 添加默认配置项        $.extend( Uploader.options, {                    /**             * @property {Boolean} [prepareNextFile=false]             * @namespace options             * @for Uploader             * @description 是否允许在文件传输时提前把下一个文件准备好。             * 对于一个文件的准备工作比较耗时,比如图片压缩,md5序列化。             * 如果能提前在当前文件传输期处理,可以节省总体耗时。             */            prepareNextFile: false,                /**             * @property {Boolean} [chunked=false]             * @namespace options             * @for Uploader             * @description 是否要分片处理大文件上传。             */            chunked: false,                /**             * @property {Boolean} [chunkSize=5242880]             * @namespace options             * @for Uploader             * @description 如果要分片,分多大一片? 默认大小为5M.             */            chunkSize: 5 * 1024 * 1024,                /**             * @property {Boolean} [chunkRetry=2]             * @namespace options             * @for Uploader             * @description 如果某个分片由于网络问题出错,允许自动重传多少次?             */            chunkRetry: 2,                /**             * @property {Boolean} [threads=3]             * @namespace options             * @for Uploader             * @description 上传并发数。允许同时最大上传进程数。             */            threads: 3,                    /**             * @property {Object} [formData]             * @namespace options             * @for Uploader             * @description 文件上传请求的参数表,每次发送都会发送此对象中的参数。             */            formData: null                /**             * @property {Object} [fileVal='file']             * @namespace options             * @for Uploader             * @description 设置文件上传域的name。             */                /**             * @property {Object} [method='POST']             * @namespace options             * @for Uploader             * @description 文件上传方式,`POST`或者`GET`。             */                /**             * @property {Object} [sendAsBinary=false]             * @namespace options             * @for Uploader             * @description 是否已二进制的流的方式发送文件,这样整个上传内容`php://input`都为文件内容,             * 其他参数在$_GET数组中。             */        });            // 负责将文件切片。        function CuteFile( file, chunkSize ) {            var pending = [],                blob = file.source,                total = blob.size,                chunks = chunkSize ? Math.ceil( total / chunkSize ) : 1,                start = 0,                index = 0,                len;                while ( index < chunks ) {                len = Math.min( chunkSize, total - start );                    pending.push({                    file: file,                    start: start,                    end: chunkSize ? (start + len) : total,                    total: total,                    chunks: chunks,                    chunk: index++                });                start += len;            }                file.blocks = pending.concat();            file.remaning = pending.length;                return {                file: file,                    has: function() {                    return !!pending.length;                },                    fetch: function() {                    return pending.shift();                }            };        }            Uploader.register({            'start-upload': 'start',            'stop-upload': 'stop',            'skip-file': 'skipFile',            'is-in-progress': 'isInProgress'        }, {                init: function() {                var owner = this.owner;                    this.runing = false;                    // 记录当前正在传的数据,跟threads相关                this.pool = [];                    // 缓存即将上传的文件。                this.pending = [];                    // 跟踪还有多少分片没有完成上传。                this.remaning = 0;                this.__tick = Base.bindFn( this._tick, this );                    owner.on( 'uploadComplete', function( file ) {                    // 把其他块取消了。                    file.blocks && $.each( file.blocks, function( _, v ) {                        v.transport && (v.transport.abort(), v.transport.destroy());                        delete v.transport;                    });                        delete file.blocks;                    delete file.remaning;                });            },                /**             * @event startUpload             * @description 当开始上传流程时触发。             * @for  Uploader             */                /**             * 开始上传。此方法可以从初始状态调用开始上传流程,也可以从暂停状态调用,继续上传流程。             * @grammar upload() => undefined             * @method upload             * @for  Uploader             */            start: function() {                var me = this;                    // 移出invalid的文件                $.each( me.request( 'get-files', Status.INVALID ), function() {                    me.request( 'remove-file', this );                });                    if ( me.runing ) {                    return;                }                    me.runing = true;                    // 如果有暂停的,则续传                $.each( me.pool, function( _, v ) {                    var file = v.file;                        if ( file.getStatus() === Status.INTERRUPT ) {                        file.setStatus( Status.PROGRESS );                        me._trigged = false;                        v.transport && v.transport.send();                    }                });                    me._trigged = false;                me.owner.trigger('startUpload');                Base.nextTick( me.__tick );            },                /**             * @event stopUpload             * @description 当开始上传流程暂停时触发。             * @for  Uploader             */                /**             * 暂停上传。第一个参数为是否中断上传当前正在上传的文件。             * @grammar stop() => undefined             * @grammar stop( true ) => undefined             * @method stop             * @for  Uploader             */            stop: function( interrupt ) {                var me = this;                    if ( me.runing === false ) {                    return;                }                    me.runing = false;                    interrupt && $.each( me.pool, function( _, v ) {                    v.transport && v.transport.abort();                    v.file.setStatus( Status.INTERRUPT );                });                    me.owner.trigger('stopUpload');            },                /**             * 判断`Uplaode`r是否正在上传中。             * @grammar isInProgress() => Boolean             * @method isInProgress             * @for  Uploader             */            isInProgress: function() {                return !!this.runing;            },                getStats: function() {                return this.request('get-stats');            },                /**             * 掉过一个文件上传,直接标记指定文件为已上传状态。             * @grammar skipFile( file ) => undefined             * @method skipFile             * @for  Uploader             */            skipFile: function( file, status ) {                file = this.request( 'get-file', file );                    file.setStatus( status || Status.COMPLETE );                file.skipped = true;                    // 如果正在上传。                file.blocks && $.each( file.blocks, function( _, v ) {                    var _tr = v.transport;                        if ( _tr ) {                        _tr.abort();                        _tr.destroy();                        delete v.transport;                    }                });                    this.owner.trigger( 'uploadSkip', file );            },                /**             * @event uploadFinished             * @description 当所有文件上传结束时触发。             * @for  Uploader             */            _tick: function() {                var me = this,                    opts = me.options,                    fn, val;                    // 上一个promise还没有结束,则等待完成后再执行。                if ( me._promise ) {                    return me._promise.always( me.__tick );                }                    // 还有位置,且还有文件要处理的话。                if ( me.pool.length < opts.threads && (val = me._nextBlock()) ) {                    me._trigged = false;                        fn = function( val ) {                        me._promise = null;                            // 有可能是reject过来的,所以要检测val的类型。                        val && val.file && me._startSend( val );                        Base.nextTick( me.__tick );                    };                        me._promise = isPromise( val ) ? val.always( fn ) : fn( val );                    // 没有要上传的了,且没有正在传输的了。                } else if ( !me.remaning && !me.getStats().numOfQueue ) {                    me.runing = false;                        me._trigged || Base.nextTick(function() {                        me.owner.trigger('uploadFinished');                    });                    me._trigged = true;                }            },                _nextBlock: function() {                var me = this,                    act = me._act,                    opts = me.options,                    next, done;                    // 如果当前文件还有没有需要传输的,则直接返回剩下的。                if ( act && act.has() &&                        act.file.getStatus() === Status.PROGRESS ) {                        // 是否提前准备下一个文件                    if ( opts.prepareNextFile && !me.pending.length ) {                        me._prepareNextFile();                    }                        return act.fetch();                    // 否则,如果正在运行,则准备下一个文件,并等待完成后返回下个分片。                } else if ( me.runing ) {                        // 如果缓存中有,则直接在缓存中取,没有则去queue中取。                    if ( !me.pending.length && me.getStats().numOfQueue ) {                        me._prepareNextFile();                    }                        next = me.pending.shift();                    done = function( file ) {                        if ( !file ) {                            return null;                        }                            act = CuteFile( file, opts.chunked ? opts.chunkSize : 0 );                        me._act = act;                        return act.fetch();                    };                        // 文件可能还在prepare中,也有可能已经完全准备好了。                    return isPromise( next ) ?                            next[ next.pipe ? 'pipe' : 'then']( done ) :                            done( next );                }            },                    /**             * @event uploadStart             * @param {File} file File对象             * @description 某个文件开始上传前触发,一个文件只会触发一次。             * @for  Uploader             */            _prepareNextFile: function() {                var me = this,                    file = me.request('fetch-file'),                    pending = me.pending,                    promise;                    if ( file ) {                    promise = me.request( 'before-send-file', file, function() {                            // 有可能文件被skip掉了。文件被skip掉后,状态坑定不是Queued.                        if ( file.getStatus() === Status.QUEUED ) {                            me.owner.trigger( 'uploadStart', file );                            file.setStatus( Status.PROGRESS );                            return file;                        }                            return me._finishFile( file );                    });                        // 如果还在pending中,则替换成文件本身。                    promise.done(function() {                        var idx = $.inArray( promise, pending );                            ~idx && pending.splice( idx, 1, file );                    });                        // befeore-send-file的钩子就有错误发生。                    promise.fail(function( reason ) {                        file.setStatus( Status.ERROR, reason );                        me.owner.trigger( 'uploadError', file, reason );                        me.owner.trigger( 'uploadComplete', file );                    });                        pending.push( promise );                }            },                // 让出位置了,可以让其他分片开始上传            _popBlock: function( block ) {                var idx = $.inArray( block, this.pool );                    this.pool.splice( idx, 1 );                block.file.remaning--;                this.remaning--;            },                // 开始上传,可以被掉过。如果promise被reject了,则表示跳过此分片。            _startSend: function( block ) {                var me = this,                    file = block.file,                    promise;                    me.pool.push( block );                me.remaning++;                    // 如果没有分片,则直接使用原始的。                // 不会丢失content-type信息。                block.blob = block.chunks === 1 ? file.source :                        file.source.slice( block.start, block.end );                    // hook, 每个分片发送之前可能要做些异步的事情。                promise = me.request( 'before-send', block, function() {                        // 有可能文件已经上传出错了,所以不需要再传输了。                    if ( file.getStatus() === Status.PROGRESS ) {                        me._doSend( block );                    } else {                        me._popBlock( block );                        Base.nextTick( me.__tick );                    }                });                    // 如果为fail了,则跳过此分片。                promise.fail(function() {                    if ( file.remaning === 1 ) {                        me._finishFile( file ).always(function() {                            block.percentage = 1;                            me._popBlock( block );                            me.owner.trigger( 'uploadComplete', file );                            Base.nextTick( me.__tick );                        });                    } else {                        block.percentage = 1;                        me._popBlock( block );                        Base.nextTick( me.__tick );                    }                });            },                    /**             * @event uploadBeforeSend             * @param {Object} object             * @param {Object} data 默认的上传参数,可以扩展此对象来控制上传参数。             * @description 当某个文件的分块在发送前触发,主要用来询问是否要添加附带参数,大文件在开起分片上传的前提下此事件可能会触发多次。             * @for  Uploader             */                /**             * @event uploadAccept             * @param {Object} object             * @param {Object} ret 服务端的返回数据,json格式,如果服务端不是json格式,从ret._raw中取数据,自行解析。             * @description 当某个文件上传到服务端响应后,会派送此事件来询问服务端响应是否有效。如果此事件handler返回值为`false`, 则此文件将派送`server`类型的`uploadError`事件。             * @for  Uploader             */                /**             * @event uploadProgress             * @param {File} file File对象             * @param {Number} percentage 上传进度             * @description 上传过程中触发,携带上传进度。             * @for  Uploader             */                    /**             * @event uploadError             * @param {File} file File对象             * @param {String} reason 出错的code             * @description 当文件上传出错时触发。             * @for  Uploader             */                /**             * @event uploadSuccess             * @param {File} file File对象             * @param {Object} response 服务端返回的数据             * @description 当文件上传成功时触发。             * @for  Uploader             */                /**             * @event uploadComplete             * @param {File} [file] File对象             * @description 不管成功或者失败,文件上传完成时触发。             * @for  Uploader             */                // 做上传操作。            _doSend: function( block ) {                var me = this,                    owner = me.owner,                    opts = me.options,                    file = block.file,                    tr = new Transport( opts ),                    data = $.extend({}, opts.formData ),                    headers = $.extend({}, opts.headers ),                    requestAccept, ret;                    block.transport = tr;                    tr.on( 'destroy', function() {                    delete block.transport;                    me._popBlock( block );                    Base.nextTick( me.__tick );                });                    // 广播上传进度。以文件为单位。                tr.on( 'progress', function( percentage ) {                    var totalPercent = 0,                        uploaded = 0;                        // 可能没有abort掉,progress还是执行进来了。                    // if ( !file.blocks ) {                    //     return;                    // }                        totalPercent = block.percentage = percentage;                        if ( block.chunks > 1 ) {    // 计算文件的整体速度。                        $.each( file.blocks, function( _, v ) {                            uploaded += (v.percentage || 0) * (v.end - v.start);                        });                            totalPercent = uploaded / file.size;                    }                        owner.trigger( 'uploadProgress', file, totalPercent || 0 );                });                    // 用来询问,是否返回的结果是有错误的。                requestAccept = function( reject ) {                    var fn;                        ret = tr.getResponseAsJson() || {};                    ret._raw = tr.getResponse();                    fn = function( value ) {                        reject = value;                    };                        // 服务端响应了,不代表成功了,询问是否响应正确。                    if ( !owner.trigger( 'uploadAccept', block, ret, fn ) ) {                        reject = reject || 'server';                    }                        return reject;                };                    // 尝试重试,然后广播文件上传出错。                tr.on( 'error', function( type, flag ) {                    block.retried = block.retried || 0;                        // 自动重试                    if ( block.chunks > 1 && ~'http,abort'.indexOf( type ) &&                            block.retried < opts.chunkRetry ) {                            block.retried++;                        tr.send();                        } else {                            // http status 500 ~ 600                        if ( !flag && type === 'server' ) {                            type = requestAccept( type );                        }                            file.setStatus( Status.ERROR, type );                        owner.trigger( 'uploadError', file, type );                        owner.trigger( 'uploadComplete', file );                    }                });                    // 上传成功                tr.on( 'load', function() {                    var reason;                        // 如果非预期,转向上传出错。                    if ( (reason = requestAccept()) ) {                        tr.trigger( 'error', reason, true );                        return;                    }                        // 全部上传完成。                    if ( file.remaning === 1 ) {                        me._finishFile( file, ret );                    } else {                        tr.destroy();                    }                });                    // 配置默认的上传字段。                data = $.extend( data, {                    id: file.id,                    name: file.name,                    type: file.type,                    lastModifiedDate: file.lastModifiedDate,                    size: file.size                });                    block.chunks > 1 && $.extend( data, {                    chunks: block.chunks,                    chunk: block.chunk                });                    // 在发送之间可以添加字段什么的。。。                // 如果默认的字段不够使用,可以通过监听此事件来扩展                owner.trigger( 'uploadBeforeSend', block, data, headers );                    // 开始发送。                tr.appendBlob( opts.fileVal, block.blob, file.name );                tr.append( data );                tr.setRequestHeader( headers );                tr.send();            },                // 完成上传。            _finishFile: function( file, ret, hds ) {                var owner = this.owner;                    return owner                        .request( 'after-send-file', arguments, function() {                            file.setStatus( Status.COMPLETE );                            owner.trigger( 'uploadSuccess', file, ret, hds );                        })                        .fail(function( reason ) {                                // 如果外部已经标记为invalid什么的,不再改状态。                            if ( file.getStatus() === Status.PROGRESS ) {                                file.setStatus( Status.ERROR, reason );                            }                                owner.trigger( 'uploadError', file, reason );                        })                        .always(function() {                            owner.trigger( 'uploadComplete', file );                        });            }            });    });    /**     * @fileOverview 各种验证,包括文件总大小是否超出、单文件是否超出和文件是否重复。     */        define('widgets/validator',[        'base',        'uploader',        'file',        'widgets/widget'    ], function( Base, Uploader, WUFile ) {            var $ = Base.$,            validators = {},            api;            /**         * @event error         * @param {String} type 错误类型。         * @description 当validate不通过时,会以派送错误事件的形式通知调用者。通过`upload.on('error', handler)`可以捕获到此类错误,目前有以下错误会在特定的情况下派送错来。         *         * * `Q_EXCEED_NUM_LIMIT` 在设置了`fileNumLimit`且尝试给`uploader`添加的文件数量超出这个值时派送。         * * `Q_EXCEED_SIZE_LIMIT` 在设置了`Q_EXCEED_SIZE_LIMIT`且尝试给`uploader`添加的文件总大小超出这个值时派送。         * @for  Uploader         */            // 暴露给外面的api        api = {                // 添加验证器            addValidator: function( type, cb ) {                validators[ type ] = cb;            },                // 移除验证器            removeValidator: function( type ) {                delete validators[ type ];            }        };            // 在Uploader初始化的时候启动Validators的初始化        Uploader.register({            init: function() {                var me = this;                $.each( validators, function() {                    this.call( me.owner );                });            }        });            /**         * @property {int} [fileNumLimit=undefined]         * @namespace options         * @for Uploader         * @description 验证文件总数量, 超出则不允许加入队列。         */        api.addValidator( 'fileNumLimit', function() {            var uploader = this,                opts = uploader.options,                count = 0,                max = opts.fileNumLimit >> 0,                flag = true;                if ( !max ) {                return;            }                uploader.on( 'beforeFileQueued', function( file ) {                    if ( count >= max && flag ) {                    flag = false;                    this.trigger( 'error', 'Q_EXCEED_NUM_LIMIT', max, file );                    setTimeout(function() {                        flag = true;                    }, 1 );                }                    return count >= max ? false : true;            });                uploader.on( 'fileQueued', function() {                count++;            });                uploader.on( 'fileDequeued', function() {                count--;            });                uploader.on( 'uploadFinished', function() {                count = 0;            });        });                /**         * @property {int} [fileSizeLimit=undefined]         * @namespace options         * @for Uploader         * @description 验证文件总大小是否超出限制, 超出则不允许加入队列。         */        api.addValidator( 'fileSizeLimit', function() {            var uploader = this,                opts = uploader.options,                count = 0,                max = opts.fileSizeLimit >> 0,                flag = true;                if ( !max ) {                return;            }                uploader.on( 'beforeFileQueued', function( file ) {                var invalid = count + file.size > max;                    if ( invalid && flag ) {                    flag = false;                    this.trigger( 'error', 'Q_EXCEED_SIZE_LIMIT', max, file );                    setTimeout(function() {                        flag = true;                    }, 1 );                }                    return invalid ? false : true;            });                uploader.on( 'fileQueued', function( file ) {                count += file.size;            });                uploader.on( 'fileDequeued', function( file ) {                count -= file.size;            });                uploader.on( 'uploadFinished', function() {                count = 0;            });        });            /**         * @property {int} [fileSingleSizeLimit=undefined]         * @namespace options         * @for Uploader         * @description 验证单个文件大小是否超出限制, 超出则不允许加入队列。         */        api.addValidator( 'fileSingleSizeLimit', function() {            var uploader = this,                opts = uploader.options,                max = opts.fileSingleSizeLimit;                if ( !max ) {                return;            }                uploader.on( 'beforeFileQueued', function( file ) {                    if ( file.size > max ) {                    file.setStatus( WUFile.Status.INVALID, 'exceed_size' );                    this.trigger( 'error', 'F_EXCEED_SIZE', file );                    return false;                }                });            });            /**         * @property {int} [duplicate=undefined]         * @namespace options         * @for Uploader         * @description 去重, 根据文件名字、文件大小和最后修改时间来生成hash Key.         */        api.addValidator( 'duplicate', function() {            var uploader = this,                opts = uploader.options,                mapping = {};                if ( opts.duplicate ) {                return;            }                function hashString( str ) {                var hash = 0,                    i = 0,                    len = str.length,                    _char;                    for ( ; i < len; i++ ) {                    _char = str.charCodeAt( i );                    hash = _char + (hash << 6) + (hash << 16) - hash;                }                    return hash;            }                uploader.on( 'beforeFileQueued', function( file ) {                var hash = file.__hash || (file.__hash = hashString( file.name +                        file.size + file.lastModifiedDate ));                    // 已经重复了                if ( mapping[ hash ] ) {                    this.trigger( 'error', 'F_DUPLICATE', file );                    return false;                }            });                uploader.on( 'fileQueued', function( file ) {                var hash = file.__hash;                    hash && (mapping[ hash ] = true);            });                uploader.on( 'fileDequeued', function( file ) {                var hash = file.__hash;                    hash && (delete mapping[ hash ]);            });        });            return api;    });        /**     * @fileOverview Runtime管理器,负责Runtime的选择, 连接     */    define('runtime/compbase',[],function() {            function CompBase( owner, runtime ) {                this.owner = owner;            this.options = owner.options;                this.getRuntime = function() {                return runtime;            };                this.getRuid = function() {                return runtime.uid;            };                this.trigger = function() {                return owner.trigger.apply( owner, arguments );            };        }            return CompBase;    });    /**     * @fileOverview Html5Runtime     */    define('runtime/html5/runtime',[        'base',        'runtime/runtime',        'runtime/compbase'    ], function( Base, Runtime, CompBase ) {            var type = 'html5',            components = {};            function Html5Runtime() {            var pool = {},                me = this,                destory = this.destory;                Runtime.apply( me, arguments );            me.type = type;                    // 这个方法的调用者,实际上是RuntimeClient            me.exec = function( comp, fn/*, args...*/) {                var client = this,                    uid = client.uid,                    args = Base.slice( arguments, 2 ),                    instance;                    if ( components[ comp ] ) {                    instance = pool[ uid ] = pool[ uid ] ||                            new components[ comp ]( client, me );                        if ( instance[ fn ] ) {                        return instance[ fn ].apply( instance, args );                    }                }            };                me.destory = function() {                // @todo 删除池子中的所有实例                return destory && destory.apply( this, arguments );            };        }            Base.inherits( Runtime, {            constructor: Html5Runtime,                // 不需要连接其他程序,直接执行callback            init: function() {                var me = this;                setTimeout(function() {                    me.trigger('ready');                }, 1 );            }            });            // 注册Components        Html5Runtime.register = function( name, component ) {            var klass = components[ name ] = Base.inherits( CompBase, component );            return klass;        };            // 注册html5运行时。        // 只有在支持的前提下注册。        if ( window.Blob && window.FileReader && window.DataView ) {            Runtime.addRuntime( type, Html5Runtime );        }            return Html5Runtime;    });    /**     * @fileOverview Blob Html实现     */    define('runtime/html5/blob',[        'runtime/html5/runtime',        'lib/blob'    ], function( Html5Runtime, Blob ) {            return Html5Runtime.register( 'Blob', {            slice: function( start, end ) {                var blob = this.owner.source,                    slice = blob.slice || blob.webkitSlice || blob.mozSlice;                    blob = slice.call( blob, start, end );                    return new Blob( this.getRuid(), blob );            }        });    });    /**     * @fileOverview FilePaste     */    define('runtime/html5/dnd',[        'base',        'runtime/html5/runtime',        'lib/file'    ], function( Base, Html5Runtime, File ) {            var $ = Base.$,            prefix = 'webuploader-dnd-';            return Html5Runtime.register( 'DragAndDrop', {            init: function() {                var elem = this.elem = this.options.container;                    this.dragEnterHandler = Base.bindFn( this._dragEnterHandler, this );                this.dragOverHandler = Base.bindFn( this._dragOverHandler, this );                this.dragLeaveHandler = Base.bindFn( this._dragLeaveHandler, this );                this.dropHandler = Base.bindFn( this._dropHandler, this );                this.dndOver = false;                    elem.on( 'dragenter', this.dragEnterHandler );                elem.on( 'dragover', this.dragOverHandler );                elem.on( 'dragleave', this.dragLeaveHandler );                elem.on( 'drop', this.dropHandler );                    if ( this.options.disableGlobalDnd ) {                    $( document ).on( 'dragover', this.dragOverHandler );                    $( document ).on( 'drop', this.dropHandler );                }            },                _dragEnterHandler: function( e ) {                var me = this,                    denied = me._denied || false,                    items;                    e = e.originalEvent || e;                    if ( !me.dndOver ) {                    me.dndOver = true;                        // 注意只有 chrome 支持。                    items = e.dataTransfer.items;                        if ( items && items.length ) {                        me._denied = denied = !me.trigger( 'accept', items );                    }                        me.elem.addClass( prefix + 'over' );                    me.elem[ denied ? 'addClass' :                            'removeClass' ]( prefix + 'denied' );                }                        e.dataTransfer.dropEffect = denied ? 'none' : 'copy';                    return false;            },                _dragOverHandler: function( e ) {                // 只处理框内的。                var parentElem = this.elem.parent().get( 0 );                if ( parentElem && !$.contains( parentElem, e.currentTarget ) ) {                    return false;                }                    clearTimeout( this._leaveTimer );                this._dragEnterHandler.call( this, e );                    return false;            },                _dragLeaveHandler: function() {                var me = this,                    handler;                    handler = function() {                    me.dndOver = false;                    me.elem.removeClass( prefix + 'over ' + prefix + 'denied' );                };                    clearTimeout( me._leaveTimer );                me._leaveTimer = setTimeout( handler, 100 );                return false;            },                _dropHandler: function( e ) {                var me = this,                    ruid = me.getRuid(),                    parentElem = me.elem.parent().get( 0 );                    // 只处理框内的。                if ( parentElem && !$.contains( parentElem, e.currentTarget ) ) {                    return false;                }                    me._getTansferFiles( e, function( results ) {                    me.trigger( 'drop', $.map( results, function( file ) {                        return new File( ruid, file );                    }) );                });                    me.dndOver = false;                me.elem.removeClass( prefix + 'over' );                return false;            },                // 如果传入 callback 则去查看文件夹,否则只管当前文件夹。            _getTansferFiles: function( e, callback ) {                var results  = [],                    promises = [],                    items, files, dataTransfer, file, item, i, len, canAccessFolder;                    e = e.originalEvent || e;                    dataTransfer = e.dataTransfer;                items = dataTransfer.items;                files = dataTransfer.files;                    canAccessFolder = !!(items && items[ 0 ].webkitGetAsEntry);                    for ( i = 0, len = files.length; i < len; i++ ) {                    file = files[ i ];                    item = items && items[ i ];                        if ( canAccessFolder && item.webkitGetAsEntry().isDirectory ) {                            promises.push( this._traverseDirectoryTree(                                item.webkitGetAsEntry(), results ) );                    } else {                        results.push( file );                    }                }                    Base.when.apply( Base, promises ).done(function() {                        if ( !results.length ) {                        return;                    }                        callback( results );                });            },                _traverseDirectoryTree: function( entry, results ) {                var deferred = Base.Deferred(),                    me = this;                    if ( entry.isFile ) {                    entry.file(function( file ) {                        results.push( file );                        deferred.resolve();                    });                } else if ( entry.isDirectory ) {                    entry.createReader().readEntries(function( entries ) {                        var len = entries.length,                            promises = [],                            arr = [],    // 为了保证顺序。                            i;                            for ( i = 0; i < len; i++ ) {                            promises.push( me._traverseDirectoryTree(                                    entries[ i ], arr ) );                        }                            Base.when.apply( Base, promises ).then(function() {                            results.push.apply( results, arr );                            deferred.resolve();                        }, deferred.reject );                    });                }                    return deferred.promise();            },                destroy: function() {                var elem = this.elem;                    elem.off( 'dragenter', this.dragEnterHandler );                elem.off( 'dragover', this.dragEnterHandler );                elem.off( 'dragleave', this.dragLeaveHandler );                elem.off( 'drop', this.dropHandler );                    if ( this.options.disableGlobalDnd ) {                    $( document ).off( 'dragover', this.dragOverHandler );                    $( document ).off( 'drop', this.dropHandler );                }            }        });    });        /**     * @fileOverview FilePaste     */    define('runtime/html5/filepaste',[        'base',        'runtime/html5/runtime',        'lib/file'    ], function( Base, Html5Runtime, File ) {            return Html5Runtime.register( 'FilePaste', {            init: function() {                var opts = this.options,                    elem = this.elem = opts.container,                    accept = '.*',                    arr, i, len, item;                    // accetp的mimeTypes中生成匹配正则。                if ( opts.accept ) {                    arr = [];                        for ( i = 0, len = opts.accept.length; i < len; i++ ) {                        item = opts.accept[ i ].mimeTypes;                        item && arr.push( item );                    }                        if ( arr.length ) {                        accept = arr.join(',');                        accept = accept.replace( /,/g, '|' ).replace( /\*/g, '.*' );                    }                }                this.accept = accept = new RegExp( accept, 'i' );                this.hander = Base.bindFn( this._pasteHander, this );                elem.on( 'paste', this.hander );            },                _pasteHander: function( e ) {                var allowed = [],                    ruid = this.getRuid(),                    items, item, blob, i, len;                    e = e.originalEvent || e;                items = e.clipboardData.items;                    for ( i = 0, len = items.length; i < len; i++ ) {                    item = items[ i ];                        if ( item.kind !== 'file' || !(blob = item.getAsFile()) ) {                        continue;                    }                        allowed.push( new File( ruid, blob ) );                }                    if ( allowed.length ) {                    // 不阻止非文件粘贴(文字粘贴)的事件冒泡                    e.preventDefault();                    e.stopPropagation();                    this.trigger( 'paste', allowed );                }            },                destroy: function() {                this.elem.off( 'paste', this.hander );            }        });    });        /**     * @fileOverview FilePicker     */    define('runtime/html5/filepicker',[        'base',        'runtime/html5/runtime'    ], function( Base, Html5Runtime ) {            var $ = Base.$;            return Html5Runtime.register( 'FilePicker', {            init: function() {                var container = this.getRuntime().getContainer(),                    me = this,                    owner = me.owner,                    opts = me.options,                    lable = $( document.createElement('label') ),                    input = $( document.createElement('input') ),                    arr, i, len, mouseHandler;                    input.attr( 'type', 'file' );                input.attr( 'name', opts.name );                input.addClass('webuploader-element-invisible');                    lable.on( 'click', function() {                    input.trigger('click');                });                    lable.css({                    opacity: 0,                    width: '100%',                    height: '100%',                    display: 'block',                    cursor: 'pointer',                    background: '#ffffff'                });                    if ( opts.multiple ) {                    input.attr( 'multiple', 'multiple' );                }                    // @todo Firefox不支持单独指定后缀                if ( opts.accept && opts.accept.length > 0 ) {                    arr = [];                        for ( i = 0, len = opts.accept.length; i < len; i++ ) {                        arr.push( opts.accept[ i ].mimeTypes );                    }                        input.attr( 'accept', arr.join(',') );                }                    container.append( input );                container.append( lable );                    mouseHandler = function( e ) {                    owner.trigger( e.type );                };                    input.on( 'change', function( e ) {                    var fn = arguments.callee,                        clone;                        me.files = e.target.files;                        // reset input                    clone = this.cloneNode( true );                    this.parentNode.replaceChild( clone, this );                        input.off();                    input = $( clone ).on( 'change', fn )                            .on( 'mouseenter mouseleave', mouseHandler );                        owner.trigger('change');                });                    lable.on( 'mouseenter mouseleave', mouseHandler );                },                    getFiles: function() {                return this.files;            },                destroy: function() {                // todo            }        });    });    /**     * @fileOverview Transport     * @todo 支持chunked传输,优势:     * 可以将大文件分成小块,挨个传输,可以提高大文件成功率,当失败的时候,也只需要重传那小部分,     * 而不需要重头再传一次。另外断点续传也需要用chunked方式。     */    define('runtime/html5/transport',[        'base',        'runtime/html5/runtime'    ], function( Base, Html5Runtime ) {            var noop = Base.noop,            $ = Base.$;            return Html5Runtime.register( 'Transport', {            init: function() {                this._status = 0;                this._response = null;            },                send: function() {                var owner = this.owner,                    opts = this.options,                    xhr = this._initAjax(),                    blob = owner._blob,                    server = opts.server,                    formData, binary, fr;                    if ( opts.sendAsBinary ) {                    server += (/\?/.test( server ) ? '&' : '?') +                            $.param( owner._formData );                        binary = blob.getSource();                } else {                    formData = new FormData();                    $.each( owner._formData, function( k, v ) {                        formData.append( k, v );                    });                        formData.append( opts.fileVal, blob.getSource(),                            opts.filename || owner._formData.name || '' );                }                    if ( opts.withCredentials && 'withCredentials' in xhr ) {                    xhr.open( opts.method, server, true );                    xhr.withCredentials = true;                } else {                    xhr.open( opts.method, server );                }                    this._setRequestHeader( xhr, opts.headers );                    if ( binary ) {                    xhr.overrideMimeType('application/octet-stream');                        // android直接发送blob会导致服务端接收到的是空文件。                    // bug详情。                    // https://code.google.com/p/android/issues/detail?id=39882                    // 所以先用fileReader读取出来再通过arraybuffer的方式发送。                    if ( Base.os.android ) {                        fr = new FileReader();                            fr.onload = function() {                            xhr.send( this.result );                            fr = fr.onload = null;                        };                            fr.readAsArrayBuffer( binary );                    } else {                        xhr.send( binary );                    }                } else {                    xhr.send( formData );                }            },                getResponse: function() {                return this._response;            },                getResponseAsJson: function() {                return this._parseJson( this._response );            },                getStatus: function() {                return this._status;            },                abort: function() {                var xhr = this._xhr;                    if ( xhr ) {                    xhr.upload.onprogress = noop;                    xhr.onreadystatechange = noop;                    xhr.abort();                        this._xhr = xhr = null;                }            },                destroy: function() {                this.abort();            },                _initAjax: function() {                var me = this,                    xhr = new XMLHttpRequest(),                    opts = this.options;                    if ( opts.withCredentials && !('withCredentials' in xhr) &&                        typeof XDomainRequest !== 'undefined' ) {                    xhr = new XDomainRequest();                }                    xhr.upload.onprogress = function( e ) {                    var percentage = 0;                        if ( e.lengthComputable ) {                        percentage = e.loaded / e.total;                    }                        return me.trigger( 'progress', percentage );                };                    xhr.onreadystatechange = function() {                        if ( xhr.readyState !== 4 ) {                        return;                    }                        xhr.upload.onprogress = noop;                    xhr.onreadystatechange = noop;                    me._xhr = null;                    me._status = xhr.status;                        if ( xhr.status >= 200 && xhr.status < 300 ) {                        me._response = xhr.responseText;                        return me.trigger('load');                    } else if ( xhr.status >= 500 && xhr.status < 600 ) {                        me._response = xhr.responseText;                        return me.trigger( 'error', 'server' );                    }                            return me.trigger( 'error', me._status ? 'http' : 'abort' );                };                    me._xhr = xhr;                return xhr;            },                _setRequestHeader: function( xhr, headers ) {                $.each( headers, function( key, val ) {                    xhr.setRequestHeader( key, val );                });            },                _parseJson: function( str ) {                var json;                    try {                    json = JSON.parse( str );                } catch ( ex ) {                    json = {};                }                    return json;            }        });    });    /**     * @fileOverview FlashRuntime     */    define('runtime/flash/runtime',[        'base',        'runtime/runtime',        'runtime/compbase'    ], function( Base, Runtime, CompBase ) {            var $ = Base.$,            type = 'flash',            components = {};                function getFlashVersion() {            var version;                try {                version = navigator.plugins[ 'Shockwave Flash' ];                version = version.description;            } catch ( ex ) {                try {                    version = new ActiveXObject('ShockwaveFlash.ShockwaveFlash')                            .GetVariable('$version');                } catch ( ex2 ) {                    version = '0.0';                }            }            version = version.match( /\d+/g );            return parseFloat( version[ 0 ] + '.' + version[ 1 ], 10 );        }            function FlashRuntime() {            var pool = {},                clients = {},                destory = this.destory,                me = this,                jsreciver = Base.guid('webuploader_');                Runtime.apply( me, arguments );            me.type = type;                    // 这个方法的调用者,实际上是RuntimeClient            me.exec = function( comp, fn/*, args...*/ ) {                var client = this,                    uid = client.uid,                    args = Base.slice( arguments, 2 ),                    instance;                    clients[ uid ] = client;                    if ( components[ comp ] ) {                    if ( !pool[ uid ] ) {                        pool[ uid ] = new components[ comp ]( client, me );                    }                        instance = pool[ uid ];                        if ( instance[ fn ] ) {                        return instance[ fn ].apply( instance, args );                    }                }                    return me.flashExec.apply( client, arguments );            };                function handler( evt, obj ) {                var type = evt.type || evt,                    parts, uid;                    parts = type.split('::');                uid = parts[ 0 ];                type = parts[ 1 ];                    // console.log.apply( console, arguments );                    if ( type === 'Ready' && uid === me.uid ) {                    me.trigger('ready');                } else if ( clients[ uid ] ) {                    clients[ uid ].trigger( type.toLowerCase(), evt, obj );                }                    // Base.log( evt, obj );            }                // flash的接受器。            window[ jsreciver ] = function() {                var args = arguments;                    // 为了能捕获得到。                setTimeout(function() {                    handler.apply( null, args );                }, 1 );            };                this.jsreciver = jsreciver;                this.destory = function() {                // @todo 删除池子中的所有实例                return destory && destory.apply( this, arguments );            };                this.flashExec = function( comp, fn ) {                var flash = me.getFlash(),                    args = Base.slice( arguments, 2 );                    return flash.exec( this.uid, comp, fn, args );            };                // @todo        }            Base.inherits( Runtime, {            constructor: FlashRuntime,                init: function() {                var container = this.getContainer(),                    opts = this.options,                    html;                    // if not the minimal height, shims are not initialized                // in older browsers (e.g FF3.6, IE6,7,8, Safari 4.0,5.0, etc)                container.css({                    position: 'absolute',                    top: '-8px',                    left: '-8px',                    width: '9px',                    height: '9px',                    overflow: 'hidden'                });                    // insert flash object                html = '<object id="' + this.uid + '" type="application/' +                        'x-shockwave-flash" data="' +  opts.swf + '" ';                    if ( Base.browser.ie ) {                    html += 'classid="clsid:d27cdb6e-ae6d-11cf-96b8-444553540000" ';                }                    html += 'width="100%" height="100%" style="outline:0">'  +                    '<param name="movie" value="' + opts.swf + '" />' +                    '<param name="flashvars" value="uid=' + this.uid +                    '&jsreciver=' + this.jsreciver + '" />' +                    '<param name="wmode" value="transparent" />' +                    '<param name="allowscriptaccess" value="always" />' +                '</object>';                    container.html( html );            },                getFlash: function() {                if ( this._flash ) {                    return this._flash;                }                    this._flash = $( '#' + this.uid ).get( 0 );                return this._flash;            }            });            FlashRuntime.register = function( name, component ) {            component = components[ name ] = Base.inherits( CompBase, $.extend({                    // @todo fix this later                flashExec: function() {                    var owner = this.owner,                        runtime = this.getRuntime();                        return runtime.flashExec.apply( owner, arguments );                }            }, component ) );                return component;        };            if ( getFlashVersion() >= 11.4 ) {            Runtime.addRuntime( type, FlashRuntime );        }            return FlashRuntime;    });    /**     * @fileOverview FilePicker     */    define('runtime/flash/filepicker',[        'base',        'runtime/flash/runtime'    ], function( Base, FlashRuntime ) {        var $ = Base.$;            return FlashRuntime.register( 'FilePicker', {            init: function( opts ) {                var copy = $.extend({}, opts ),                    len, i;                    // 修复Flash再没有设置title的情况下无法弹出flash文件选择框的bug.                len = copy.accept && copy.accept.length;                for (  i = 0; i < len; i++ ) {                    if ( !copy.accept[ i ].title ) {                        copy.accept[ i ].title = 'Files';                    }                }                    delete copy.button;                delete copy.container;                    this.flashExec( 'FilePicker', 'init', copy );            },                destroy: function() {                // todo            }        });    });    /**     * @fileOverview  Transport flash实现     */    define('runtime/flash/transport',[        'base',        'runtime/flash/runtime',        'runtime/client'    ], function( Base, FlashRuntime, RuntimeClient ) {        var $ = Base.$;            return FlashRuntime.register( 'Transport', {            init: function() {                this._status = 0;                this._response = null;                this._responseJson = null;            },                send: function() {                var owner = this.owner,                    opts = this.options,                    xhr = this._initAjax(),                    blob = owner._blob,                    server = opts.server,                    binary;                    xhr.connectRuntime( blob.ruid );                    if ( opts.sendAsBinary ) {                    server += (/\?/.test( server ) ? '&' : '?') +                            $.param( owner._formData );                        binary = blob.uid;                } else {                    $.each( owner._formData, function( k, v ) {                        xhr.exec( 'append', k, v );                    });                        xhr.exec( 'appendBlob', opts.fileVal, blob.uid,                            opts.filename || owner._formData.name || '' );                }                    this._setRequestHeader( xhr, opts.headers );                xhr.exec( 'send', {                    method: opts.method,                    url: server                }, binary );            },                getStatus: function() {                return this._status;            },                getResponse: function() {                return this._response;            },                getResponseAsJson: function() {                return this._responseJson;            },                abort: function() {                var xhr = this._xhr;                    if ( xhr ) {                    xhr.exec('abort');                    xhr.destroy();                    this._xhr = xhr = null;                }            },                destroy: function() {                this.abort();            },                _initAjax: function() {                var me = this,                    xhr = new RuntimeClient('XMLHttpRequest');                    xhr.on( 'uploadprogress progress', function( e ) {                    return me.trigger( 'progress', e.loaded / e.total );                });                    xhr.on( 'load', function() {                    var status = xhr.exec('getStatus'),                        err = '';                        xhr.off();                    me._xhr = null;                        if ( status >= 200 && status < 300 ) {                        me._response = xhr.exec('getResponse');                        me._responseJson = xhr.exec('getResponseAsJson');                    } else if ( status >= 500 && status < 600 ) {                        me._response = xhr.exec('getResponse');                        me._responseJson = xhr.exec('getResponseAsJson');                        err = 'server';                    } else {                        err = 'http';                    }                        xhr.destroy();                    xhr = null;                        return err ? me.trigger( 'error', err ) : me.trigger('load');                });                    xhr.on( 'error', function() {                    xhr.off();                    me._xhr = null;                    me.trigger( 'error', 'http' );                });                    me._xhr = xhr;                return xhr;            },                _setRequestHeader: function( xhr, headers ) {                $.each( headers, function( key, val ) {                    xhr.exec( 'setRequestHeader', key, val );                });            }        });    });    /**     * @fileOverview 没有图像处理的版本。     */    define('preset/withoutimage',[        'base',            // widgets        'widgets/filednd',        'widgets/filepaste',        'widgets/filepicker',        'widgets/queue',        'widgets/runtime',        'widgets/upload',        'widgets/validator',            // runtimes        // html5        'runtime/html5/blob',        'runtime/html5/dnd',        'runtime/html5/filepaste',        'runtime/html5/filepicker',        'runtime/html5/transport',            // flash        'runtime/flash/filepicker',        'runtime/flash/transport'    ], function( Base ) {        return Base;    });    define('webuploader',[        'preset/withoutimage'    ], function( preset ) {        return preset;    });    return require('webuploader');});
 |