1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690769176927693769476957696769776987699770077017702770377047705770677077708770977107711771277137714771577167717771877197720772177227723772477257726772777287729773077317732773377347735773677377738773977407741774277437744774577467747774877497750775177527753775477557756775777587759776077617762776377647765776677677768776977707771777277737774777577767777777877797780778177827783778477857786778777887789779077917792779377947795779677977798779978007801780278037804780578067807780878097810781178127813781478157816781778187819782078217822782378247825782678277828782978307831783278337834783578367837783878397840784178427843784478457846784778487849785078517852785378547855785678577858785978607861786278637864786578667867786878697870787178727873787478757876787778787879788078817882788378847885788678877888788978907891789278937894789578967897789878997900790179027903790479057906790779087909791079117912791379147915791679177918791979207921792279237924792579267927792879297930793179327933793479357936793779387939794079417942794379447945794679477948794979507951795279537954795579567957795879597960796179627963796479657966796779687969797079717972797379747975797679777978797979807981798279837984798579867987798879897990799179927993799479957996799779987999800080018002800380048005800680078008800980108011801280138014801580168017801880198020802180228023802480258026802780288029803080318032803380348035803680378038803980408041804280438044804580468047804880498050805180528053805480558056805780588059806080618062806380648065806680678068806980708071807280738074807580768077807880798080808180828083808480858086808780888089809080918092809380948095809680978098809981008101810281038104810581068107810881098110811181128113811481158116811781188119812081218122812381248125812681278128812981308131813281338134813581368137813881398140814181428143814481458146814781488149815081518152815381548155815681578158815981608161816281638164816581668167816881698170817181728173817481758176817781788179818081818182818381848185818681878188818981908191819281938194819581968197819881998200820182028203820482058206820782088209821082118212821382148215821682178218821982208221822282238224822582268227822882298230823182328233823482358236823782388239824082418242824382448245824682478248824982508251825282538254825582568257825882598260826182628263826482658266826782688269827082718272827382748275827682778278827982808281828282838284828582868287828882898290829182928293829482958296829782988299830083018302830383048305830683078308830983108311831283138314831583168317831883198320832183228323832483258326832783288329833083318332833383348335833683378338833983408341834283438344834583468347834883498350835183528353835483558356835783588359836083618362836383648365836683678368836983708371837283738374837583768377837883798380838183828383838483858386838783888389839083918392839383948395839683978398839984008401840284038404840584068407840884098410841184128413841484158416841784188419842084218422842384248425842684278428842984308431843284338434843584368437843884398440844184428443844484458446844784488449845084518452845384548455845684578458845984608461846284638464846584668467846884698470847184728473847484758476847784788479848084818482848384848485848684878488848984908491849284938494849584968497849884998500850185028503850485058506850785088509851085118512851385148515851685178518851985208521852285238524852585268527852885298530853185328533853485358536853785388539854085418542854385448545854685478548854985508551855285538554855585568557855885598560856185628563856485658566856785688569857085718572857385748575857685778578857985808581858285838584858585868587858885898590859185928593859485958596859785988599860086018602860386048605860686078608860986108611861286138614861586168617861886198620862186228623862486258626862786288629863086318632863386348635863686378638863986408641864286438644864586468647864886498650865186528653865486558656865786588659866086618662866386648665866686678668866986708671867286738674867586768677867886798680868186828683868486858686868786888689869086918692869386948695869686978698869987008701870287038704870587068707870887098710871187128713871487158716871787188719872087218722872387248725872687278728872987308731873287338734873587368737873887398740874187428743874487458746874787488749875087518752875387548755875687578758875987608761876287638764876587668767876887698770877187728773877487758776877787788779878087818782878387848785878687878788878987908791879287938794879587968797879887998800880188028803880488058806880788088809881088118812881388148815881688178818881988208821882288238824882588268827882888298830883188328833883488358836883788388839884088418842884388448845884688478848884988508851885288538854885588568857885888598860886188628863886488658866886788688869887088718872887388748875887688778878887988808881888288838884888588868887888888898890889188928893889488958896889788988899890089018902890389048905890689078908890989108911891289138914891589168917891889198920892189228923892489258926892789288929893089318932893389348935893689378938893989408941894289438944894589468947894889498950895189528953895489558956895789588959896089618962896389648965896689678968896989708971897289738974897589768977897889798980898189828983898489858986898789888989899089918992899389948995899689978998899990009001900290039004900590069007900890099010901190129013901490159016901790189019902090219022902390249025902690279028902990309031903290339034903590369037903890399040904190429043904490459046904790489049905090519052905390549055905690579058905990609061906290639064906590669067906890699070907190729073907490759076907790789079908090819082908390849085908690879088908990909091909290939094909590969097909890999100910191029103910491059106910791089109911091119112911391149115911691179118911991209121912291239124912591269127912891299130913191329133913491359136913791389139914091419142914391449145914691479148914991509151915291539154915591569157915891599160916191629163916491659166916791689169917091719172917391749175917691779178917991809181918291839184918591869187918891899190919191929193919491959196919791989199920092019202920392049205920692079208920992109211921292139214921592169217921892199220922192229223922492259226922792289229923092319232923392349235923692379238923992409241924292439244924592469247924892499250925192529253925492559256925792589259926092619262926392649265926692679268926992709271927292739274927592769277927892799280928192829283928492859286928792889289929092919292929392949295929692979298929993009301930293039304930593069307930893099310931193129313931493159316931793189319932093219322932393249325932693279328932993309331933293339334933593369337933893399340934193429343934493459346934793489349935093519352935393549355935693579358935993609361936293639364936593669367936893699370937193729373937493759376937793789379938093819382938393849385938693879388938993909391939293939394939593969397939893999400940194029403940494059406940794089409941094119412941394149415941694179418941994209421942294239424942594269427942894299430943194329433943494359436943794389439944094419442944394449445944694479448944994509451945294539454945594569457945894599460946194629463946494659466946794689469947094719472947394749475947694779478947994809481948294839484948594869487948894899490949194929493949494959496949794989499950095019502950395049505950695079508950995109511951295139514951595169517951895199520952195229523952495259526952795289529953095319532953395349535953695379538953995409541954295439544954595469547954895499550955195529553955495559556955795589559956095619562956395649565956695679568956995709571957295739574957595769577957895799580958195829583958495859586958795889589959095919592959395949595959695979598959996009601960296039604960596069607960896099610961196129613961496159616961796189619962096219622962396249625962696279628962996309631963296339634963596369637963896399640964196429643964496459646964796489649965096519652965396549655965696579658965996609661966296639664966596669667966896699670967196729673967496759676967796789679968096819682968396849685968696879688968996909691969296939694969596969697969896999700970197029703970497059706970797089709971097119712971397149715971697179718971997209721972297239724972597269727972897299730973197329733973497359736973797389739974097419742974397449745974697479748974997509751975297539754975597569757975897599760976197629763976497659766976797689769977097719772977397749775977697779778977997809781978297839784978597869787978897899790979197929793979497959796979797989799980098019802980398049805980698079808980998109811981298139814981598169817981898199820982198229823982498259826982798289829983098319832983398349835983698379838983998409841984298439844984598469847984898499850985198529853985498559856985798589859986098619862986398649865986698679868986998709871987298739874987598769877987898799880988198829883988498859886988798889889989098919892989398949895989698979898989999009901990299039904990599069907990899099910991199129913991499159916991799189919992099219922992399249925992699279928992999309931993299339934993599369937993899399940994199429943994499459946994799489949995099519952995399549955995699579958995999609961996299639964996599669967996899699970997199729973997499759976997799789979998099819982998399849985998699879988998999909991999299939994999599969997999899991000010001100021000310004100051000610007100081000910010100111001210013100141001510016100171001810019100201002110022100231002410025100261002710028100291003010031100321003310034100351003610037100381003910040100411004210043100441004510046100471004810049100501005110052100531005410055100561005710058100591006010061100621006310064100651006610067100681006910070100711007210073100741007510076100771007810079100801008110082100831008410085100861008710088100891009010091100921009310094100951009610097100981009910100101011010210103101041010510106101071010810109101101011110112101131011410115101161011710118101191012010121101221012310124101251012610127101281012910130101311013210133101341013510136101371013810139101401014110142101431014410145101461014710148101491015010151101521015310154101551015610157101581015910160101611016210163101641016510166101671016810169101701017110172101731017410175101761017710178101791018010181101821018310184101851018610187101881018910190101911019210193101941019510196101971019810199102001020110202102031020410205102061020710208102091021010211102121021310214102151021610217102181021910220102211022210223102241022510226102271022810229102301023110232102331023410235102361023710238102391024010241102421024310244102451024610247102481024910250102511025210253102541025510256102571025810259102601026110262102631026410265102661026710268102691027010271102721027310274102751027610277102781027910280102811028210283102841028510286102871028810289102901029110292102931029410295102961029710298102991030010301103021030310304103051030610307103081030910310103111031210313103141031510316103171031810319103201032110322103231032410325103261032710328103291033010331103321033310334103351033610337103381033910340103411034210343103441034510346103471034810349103501035110352103531035410355103561035710358103591036010361103621036310364103651036610367103681036910370103711037210373103741037510376103771037810379103801038110382103831038410385103861038710388103891039010391103921039310394103951039610397103981039910400104011040210403104041040510406104071040810409104101041110412104131041410415104161041710418104191042010421104221042310424104251042610427104281042910430104311043210433104341043510436104371043810439104401044110442104431044410445104461044710448104491045010451104521045310454104551045610457104581045910460104611046210463104641046510466104671046810469104701047110472104731047410475104761047710478104791048010481104821048310484104851048610487104881048910490104911049210493104941049510496104971049810499105001050110502105031050410505105061050710508105091051010511105121051310514105151051610517105181051910520105211052210523105241052510526105271052810529105301053110532105331053410535105361053710538105391054010541105421054310544105451054610547105481054910550105511055210553105541055510556105571055810559105601056110562105631056410565105661056710568105691057010571105721057310574105751057610577105781057910580105811058210583105841058510586105871058810589105901059110592105931059410595105961059710598105991060010601106021060310604106051060610607106081060910610106111061210613106141061510616106171061810619106201062110622106231062410625106261062710628106291063010631106321063310634106351063610637106381063910640106411064210643106441064510646106471064810649106501065110652106531065410655106561065710658106591066010661106621066310664106651066610667106681066910670106711067210673106741067510676106771067810679106801068110682106831068410685106861068710688106891069010691106921069310694106951069610697106981069910700107011070210703107041070510706107071070810709107101071110712107131071410715107161071710718107191072010721107221072310724107251072610727107281072910730107311073210733107341073510736107371073810739107401074110742107431074410745107461074710748107491075010751107521075310754107551075610757107581075910760107611076210763107641076510766107671076810769107701077110772107731077410775107761077710778107791078010781107821078310784107851078610787107881078910790107911079210793107941079510796107971079810799108001080110802108031080410805108061080710808108091081010811108121081310814108151081610817108181081910820108211082210823108241082510826108271082810829108301083110832108331083410835108361083710838108391084010841108421084310844108451084610847108481084910850108511085210853108541085510856108571085810859108601086110862108631086410865108661086710868108691087010871108721087310874108751087610877108781087910880108811088210883108841088510886108871088810889108901089110892108931089410895108961089710898108991090010901109021090310904109051090610907109081090910910109111091210913109141091510916109171091810919109201092110922109231092410925109261092710928109291093010931109321093310934109351093610937109381093910940109411094210943109441094510946109471094810949109501095110952109531095410955109561095710958109591096010961109621096310964109651096610967109681096910970109711097210973109741097510976109771097810979109801098110982109831098410985109861098710988109891099010991109921099310994109951099610997109981099911000110011100211003110041100511006110071100811009110101101111012110131101411015110161101711018110191102011021110221102311024110251102611027110281102911030110311103211033110341103511036110371103811039110401104111042110431104411045110461104711048110491105011051110521105311054110551105611057110581105911060110611106211063110641106511066110671106811069110701107111072110731107411075110761107711078110791108011081110821108311084110851108611087110881108911090110911109211093110941109511096110971109811099111001110111102111031110411105111061110711108111091111011111111121111311114111151111611117111181111911120111211112211123111241112511126111271112811129111301113111132111331113411135111361113711138111391114011141111421114311144111451114611147111481114911150111511115211153111541115511156111571115811159111601116111162111631116411165111661116711168111691117011171111721117311174111751117611177111781117911180111811118211183111841118511186111871118811189111901119111192111931119411195111961119711198111991120011201112021120311204112051120611207112081120911210112111121211213112141121511216112171121811219112201122111222112231122411225112261122711228112291123011231112321123311234112351123611237112381123911240112411124211243112441124511246112471124811249112501125111252112531125411255112561125711258112591126011261112621126311264112651126611267112681126911270112711127211273112741127511276112771127811279112801128111282112831128411285112861128711288112891129011291112921129311294112951129611297112981129911300113011130211303113041130511306113071130811309113101131111312113131131411315113161131711318113191132011321113221132311324113251132611327113281132911330113311133211333113341133511336113371133811339113401134111342113431134411345113461134711348113491135011351113521135311354113551135611357113581135911360113611136211363113641136511366113671136811369113701137111372113731137411375113761137711378113791138011381113821138311384113851138611387113881138911390113911139211393113941139511396113971139811399114001140111402114031140411405114061140711408114091141011411114121141311414114151141611417114181141911420114211142211423114241142511426114271142811429114301143111432114331143411435114361143711438114391144011441114421144311444114451144611447114481144911450114511145211453114541145511456114571145811459114601146111462114631146411465114661146711468114691147011471114721147311474114751147611477114781147911480114811148211483114841148511486114871148811489114901149111492114931149411495114961149711498114991150011501115021150311504115051150611507115081150911510115111151211513115141151511516115171151811519115201152111522115231152411525115261152711528115291153011531115321153311534115351153611537115381153911540115411154211543115441154511546115471154811549115501155111552115531155411555115561155711558115591156011561115621156311564115651156611567115681156911570115711157211573115741157511576115771157811579115801158111582115831158411585115861158711588115891159011591115921159311594115951159611597115981159911600116011160211603116041160511606116071160811609116101161111612116131161411615116161161711618116191162011621116221162311624116251162611627116281162911630116311163211633116341163511636116371163811639116401164111642116431164411645116461164711648116491165011651116521165311654116551165611657116581165911660116611166211663116641166511666116671166811669116701167111672116731167411675116761167711678116791168011681116821168311684116851168611687116881168911690116911169211693116941169511696116971169811699117001170111702117031170411705117061170711708117091171011711117121171311714117151171611717117181171911720117211172211723117241172511726117271172811729117301173111732117331173411735117361173711738117391174011741117421174311744117451174611747117481174911750117511175211753117541175511756117571175811759117601176111762117631176411765117661176711768117691177011771117721177311774117751177611777117781177911780117811178211783117841178511786117871178811789117901179111792117931179411795117961179711798117991180011801118021180311804118051180611807118081180911810118111181211813118141181511816118171181811819118201182111822118231182411825118261182711828118291183011831118321183311834118351183611837118381183911840118411184211843118441184511846118471184811849118501185111852118531185411855118561185711858118591186011861118621186311864118651186611867118681186911870118711187211873118741187511876118771187811879118801188111882118831188411885118861188711888118891189011891118921189311894118951189611897118981189911900119011190211903 |
- (function(f){
- if(typeof exports==="object"&&typeof module!=="undefined"){
- module.exports=f()
- }else if(typeof define==="function"&&define.amd){
- define([],f)
- }else{
- var g;
- if(typeof window!=="undefined"){
- g=window
- }else if(typeof global!=="undefined"){
- g=global
- }else if(typeof self!=="undefined"){
- g=self
- }else{
- g=this
- }
- g.flvjs = f()
- }
- })(function(){
- var define,module,exports;
- return (function e(t,n,r){
- function s(o,u){
- if(!n[o]){
- if(!t[o]){
- var a=typeof require=="function"&&require;
- if(!u&&a) return a(o,!0);
- if(i) return i(o,!0);
- var f=new Error("Cannot find module '"+o+"'");
- throw f.code="MODULE_NOT_FOUND",f
- }var l=n[o]={exports:{}};
- t[o][0].call(l.exports,function(e){
- var n=t[o][1][e];return s(n?n:e)
- },l,l.exports,e,t,n,r)}return n[o].exports}
- var i=typeof require=="function"&&require;
- for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(_dereq_,module,exports){
- (function (process,global){
- /*!
- * @overview es6-promise - a tiny implementation of Promises/A+.
- * @copyright Copyright (c) 2014 Yehuda Katz, Tom Dale, Stefan Penner and contributors (Conversion to ES6 API by Jake Archibald)
- * @license Licensed under MIT license
- * See https://raw.githubusercontent.com/stefanpenner/es6-promise/master/LICENSE
- * @version 4.1.0
- */
- (function (global, factory) {
- typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
- typeof define === 'function' && define.amd ? define(factory) :
- (global.ES6Promise = factory());
- }(this, (function () { 'use strict';
- function objectOrFunction(x) {
- return typeof x === 'function' || typeof x === 'object' && x !== null;
- }
- function isFunction(x) {
- return typeof x === 'function';
- }
- var _isArray = undefined;
- if (!Array.isArray) {
- _isArray = function (x) {
- return Object.prototype.toString.call(x) === '[object Array]';
- };
- } else {
- _isArray = Array.isArray;
- }
- var isArray = _isArray;
- var len = 0;
- var vertxNext = undefined;
- var customSchedulerFn = undefined;
- var asap = function asap(callback, arg) {
- queue[len] = callback;
- queue[len + 1] = arg;
- len += 2;
- if (len === 2) {
- // If len is 2, that means that we need to schedule an async flush.
- // If additional callbacks are queued before the queue is flushed, they
- // will be processed by this flush that we are scheduling.
- if (customSchedulerFn) {
- customSchedulerFn(flush);
- } else {
- scheduleFlush();
- }
- }
- };
- function setScheduler(scheduleFn) {
- customSchedulerFn = scheduleFn;
- }
- function setAsap(asapFn) {
- asap = asapFn;
- }
- var browserWindow = typeof window !== 'undefined' ? window : undefined;
- var browserGlobal = browserWindow || {};
- var BrowserMutationObserver = browserGlobal.MutationObserver || browserGlobal.WebKitMutationObserver;
- var isNode = typeof self === 'undefined' && typeof process !== 'undefined' && ({}).toString.call(process) === '[object process]';
- // test for web worker but not in IE10
- var isWorker = typeof Uint8ClampedArray !== 'undefined' && typeof importScripts !== 'undefined' && typeof MessageChannel !== 'undefined';
- // node
- function useNextTick() {
- // node version 0.10.x displays a deprecation warning when nextTick is used recursively
- // see https://github.com/cujojs/when/issues/410 for details
- return function () {
- return process.nextTick(flush);
- };
- }
- // vertx
- function useVertxTimer() {
- if (typeof vertxNext !== 'undefined') {
- return function () {
- vertxNext(flush);
- };
- }
- return useSetTimeout();
- }
- function useMutationObserver() {
- var iterations = 0;
- var observer = new BrowserMutationObserver(flush);
- var node = document.createTextNode('');
- observer.observe(node, { characterData: true });
- return function () {
- node.data = iterations = ++iterations % 2;
- };
- }
- // web worker
- function useMessageChannel() {
- var channel = new MessageChannel();
- channel.port1.onmessage = flush;
- return function () {
- return channel.port2.postMessage(0);
- };
- }
- function useSetTimeout() {
- // Store setTimeout reference so es6-promise will be unaffected by
- // other code modifying setTimeout (like sinon.useFakeTimers())
- var globalSetTimeout = setTimeout;
- return function () {
- return globalSetTimeout(flush, 1);
- };
- }
- var queue = new Array(1000);
- function flush() {
- for (var i = 0; i < len; i += 2) {
- var callback = queue[i];
- var arg = queue[i + 1];
- callback(arg);
- queue[i] = undefined;
- queue[i + 1] = undefined;
- }
- len = 0;
- }
- function attemptVertx() {
- try {
- var r = _dereq_;
- var vertx = r('vertx');
- vertxNext = vertx.runOnLoop || vertx.runOnContext;
- return useVertxTimer();
- } catch (e) {
- return useSetTimeout();
- }
- }
- var scheduleFlush = undefined;
- // Decide what async method to use to triggering processing of queued callbacks:
- if (isNode) {
- scheduleFlush = useNextTick();
- } else if (BrowserMutationObserver) {
- scheduleFlush = useMutationObserver();
- } else if (isWorker) {
- scheduleFlush = useMessageChannel();
- } else if (browserWindow === undefined && typeof _dereq_ === 'function') {
- scheduleFlush = attemptVertx();
- } else {
- scheduleFlush = useSetTimeout();
- }
- function then(onFulfillment, onRejection) {
- var _arguments = arguments;
- var parent = this;
- var child = new this.constructor(noop);
- if (child[PROMISE_ID] === undefined) {
- makePromise(child);
- }
- var _state = parent._state;
- if (_state) {
- (function () {
- var callback = _arguments[_state - 1];
- asap(function () {
- return invokeCallback(_state, child, callback, parent._result);
- });
- })();
- } else {
- subscribe(parent, child, onFulfillment, onRejection);
- }
- return child;
- }
- /**
- `Promise.resolve` returns a promise that will become resolved with the
- passed `value`. It is shorthand for the following:
- ```javascript
- let promise = new Promise(function(resolve, reject){
- resolve(1);
- });
- promise.then(function(value){
- // value === 1
- });
- ```
- Instead of writing the above, your code now simply becomes the following:
- ```javascript
- let promise = Promise.resolve(1);
- promise.then(function(value){
- // value === 1
- });
- ```
- @method resolve
- @static
- @param {Any} value value that the returned promise will be resolved with
- Useful for tooling.
- @return {Promise} a promise that will become fulfilled with the given
- `value`
- */
- function resolve(object) {
- /*jshint validthis:true */
- var Constructor = this;
- if (object && typeof object === 'object' && object.constructor === Constructor) {
- return object;
- }
- var promise = new Constructor(noop);
- _resolve(promise, object);
- return promise;
- }
- var PROMISE_ID = Math.random().toString(36).substring(16);
- function noop() {}
- var PENDING = void 0;
- var FULFILLED = 1;
- var REJECTED = 2;
- var GET_THEN_ERROR = new ErrorObject();
- function selfFulfillment() {
- return new TypeError("You cannot resolve a promise with itself");
- }
- function cannotReturnOwn() {
- return new TypeError('A promises callback cannot return that same promise.');
- }
- function getThen(promise) {
- try {
- return promise.then;
- } catch (error) {
- GET_THEN_ERROR.error = error;
- return GET_THEN_ERROR;
- }
- }
- function tryThen(then, value, fulfillmentHandler, rejectionHandler) {
- try {
- then.call(value, fulfillmentHandler, rejectionHandler);
- } catch (e) {
- return e;
- }
- }
- function handleForeignThenable(promise, thenable, then) {
- asap(function (promise) {
- var sealed = false;
- var error = tryThen(then, thenable, function (value) {
- if (sealed) {
- return;
- }
- sealed = true;
- if (thenable !== value) {
- _resolve(promise, value);
- } else {
- fulfill(promise, value);
- }
- }, function (reason) {
- if (sealed) {
- return;
- }
- sealed = true;
- _reject(promise, reason);
- }, 'Settle: ' + (promise._label || ' unknown promise'));
- if (!sealed && error) {
- sealed = true;
- _reject(promise, error);
- }
- }, promise);
- }
- function handleOwnThenable(promise, thenable) {
- if (thenable._state === FULFILLED) {
- fulfill(promise, thenable._result);
- } else if (thenable._state === REJECTED) {
- _reject(promise, thenable._result);
- } else {
- subscribe(thenable, undefined, function (value) {
- return _resolve(promise, value);
- }, function (reason) {
- return _reject(promise, reason);
- });
- }
- }
- function handleMaybeThenable(promise, maybeThenable, then$$) {
- if (maybeThenable.constructor === promise.constructor && then$$ === then && maybeThenable.constructor.resolve === resolve) {
- handleOwnThenable(promise, maybeThenable);
- } else {
- if (then$$ === GET_THEN_ERROR) {
- _reject(promise, GET_THEN_ERROR.error);
- GET_THEN_ERROR.error = null;
- } else if (then$$ === undefined) {
- fulfill(promise, maybeThenable);
- } else if (isFunction(then$$)) {
- handleForeignThenable(promise, maybeThenable, then$$);
- } else {
- fulfill(promise, maybeThenable);
- }
- }
- }
- function _resolve(promise, value) {
- if (promise === value) {
- _reject(promise, selfFulfillment());
- } else if (objectOrFunction(value)) {
- handleMaybeThenable(promise, value, getThen(value));
- } else {
- fulfill(promise, value);
- }
- }
- function publishRejection(promise) {
- if (promise._onerror) {
- promise._onerror(promise._result);
- }
- publish(promise);
- }
- function fulfill(promise, value) {
- if (promise._state !== PENDING) {
- return;
- }
- promise._result = value;
- promise._state = FULFILLED;
- if (promise._subscribers.length !== 0) {
- asap(publish, promise);
- }
- }
- function _reject(promise, reason) {
- if (promise._state !== PENDING) {
- return;
- }
- promise._state = REJECTED;
- promise._result = reason;
- asap(publishRejection, promise);
- }
- function subscribe(parent, child, onFulfillment, onRejection) {
- var _subscribers = parent._subscribers;
- var length = _subscribers.length;
- parent._onerror = null;
- _subscribers[length] = child;
- _subscribers[length + FULFILLED] = onFulfillment;
- _subscribers[length + REJECTED] = onRejection;
- if (length === 0 && parent._state) {
- asap(publish, parent);
- }
- }
- function publish(promise) {
- var subscribers = promise._subscribers;
- var settled = promise._state;
- if (subscribers.length === 0) {
- return;
- }
- var child = undefined,
- callback = undefined,
- detail = promise._result;
- for (var i = 0; i < subscribers.length; i += 3) {
- child = subscribers[i];
- callback = subscribers[i + settled];
- if (child) {
- invokeCallback(settled, child, callback, detail);
- } else {
- callback(detail);
- }
- }
- promise._subscribers.length = 0;
- }
- function ErrorObject() {
- this.error = null;
- }
- var TRY_CATCH_ERROR = new ErrorObject();
- function tryCatch(callback, detail) {
- try {
- return callback(detail);
- } catch (e) {
- TRY_CATCH_ERROR.error = e;
- return TRY_CATCH_ERROR;
- }
- }
- function invokeCallback(settled, promise, callback, detail) {
- var hasCallback = isFunction(callback),
- value = undefined,
- error = undefined,
- succeeded = undefined,
- failed = undefined;
- if (hasCallback) {
- value = tryCatch(callback, detail);
- if (value === TRY_CATCH_ERROR) {
- failed = true;
- error = value.error;
- value.error = null;
- } else {
- succeeded = true;
- }
- if (promise === value) {
- _reject(promise, cannotReturnOwn());
- return;
- }
- } else {
- value = detail;
- succeeded = true;
- }
- if (promise._state !== PENDING) {
- // noop
- } else if (hasCallback && succeeded) {
- _resolve(promise, value);
- } else if (failed) {
- _reject(promise, error);
- } else if (settled === FULFILLED) {
- fulfill(promise, value);
- } else if (settled === REJECTED) {
- _reject(promise, value);
- }
- }
- function initializePromise(promise, resolver) {
- try {
- resolver(function resolvePromise(value) {
- _resolve(promise, value);
- }, function rejectPromise(reason) {
- _reject(promise, reason);
- });
- } catch (e) {
- _reject(promise, e);
- }
- }
- var id = 0;
- function nextId() {
- return id++;
- }
- function makePromise(promise) {
- promise[PROMISE_ID] = id++;
- promise._state = undefined;
- promise._result = undefined;
- promise._subscribers = [];
- }
- function Enumerator(Constructor, input) {
- this._instanceConstructor = Constructor;
- this.promise = new Constructor(noop);
- if (!this.promise[PROMISE_ID]) {
- makePromise(this.promise);
- }
- if (isArray(input)) {
- this._input = input;
- this.length = input.length;
- this._remaining = input.length;
- this._result = new Array(this.length);
- if (this.length === 0) {
- fulfill(this.promise, this._result);
- } else {
- this.length = this.length || 0;
- this._enumerate();
- if (this._remaining === 0) {
- fulfill(this.promise, this._result);
- }
- }
- } else {
- _reject(this.promise, validationError());
- }
- }
- function validationError() {
- return new Error('Array Methods must be provided an Array');
- };
- Enumerator.prototype._enumerate = function () {
- var length = this.length;
- var _input = this._input;
- for (var i = 0; this._state === PENDING && i < length; i++) {
- this._eachEntry(_input[i], i);
- }
- };
- Enumerator.prototype._eachEntry = function (entry, i) {
- var c = this._instanceConstructor;
- var resolve$$ = c.resolve;
- if (resolve$$ === resolve) {
- var _then = getThen(entry);
- if (_then === then && entry._state !== PENDING) {
- this._settledAt(entry._state, i, entry._result);
- } else if (typeof _then !== 'function') {
- this._remaining--;
- this._result[i] = entry;
- } else if (c === Promise) {
- var promise = new c(noop);
- handleMaybeThenable(promise, entry, _then);
- this._willSettleAt(promise, i);
- } else {
- this._willSettleAt(new c(function (resolve$$) {
- return resolve$$(entry);
- }), i);
- }
- } else {
- this._willSettleAt(resolve$$(entry), i);
- }
- };
- Enumerator.prototype._settledAt = function (state, i, value) {
- var promise = this.promise;
- if (promise._state === PENDING) {
- this._remaining--;
- if (state === REJECTED) {
- _reject(promise, value);
- } else {
- this._result[i] = value;
- }
- }
- if (this._remaining === 0) {
- fulfill(promise, this._result);
- }
- };
- Enumerator.prototype._willSettleAt = function (promise, i) {
- var enumerator = this;
- subscribe(promise, undefined, function (value) {
- return enumerator._settledAt(FULFILLED, i, value);
- }, function (reason) {
- return enumerator._settledAt(REJECTED, i, reason);
- });
- };
- /**
- `Promise.all` accepts an array of promises, and returns a new promise which
- is fulfilled with an array of fulfillment values for the passed promises, or
- rejected with the reason of the first passed promise to be rejected. It casts all
- elements of the passed iterable to promises as it runs this algorithm.
- Example:
- ```javascript
- let promise1 = resolve(1);
- let promise2 = resolve(2);
- let promise3 = resolve(3);
- let promises = [ promise1, promise2, promise3 ];
- Promise.all(promises).then(function(array){
- // The array here would be [ 1, 2, 3 ];
- });
- ```
- If any of the `promises` given to `all` are rejected, the first promise
- that is rejected will be given as an argument to the returned promises's
- rejection handler. For example:
- Example:
- ```javascript
- let promise1 = resolve(1);
- let promise2 = reject(new Error("2"));
- let promise3 = reject(new Error("3"));
- let promises = [ promise1, promise2, promise3 ];
- Promise.all(promises).then(function(array){
- // Code here never runs because there are rejected promises!
- }, function(error) {
- // error.message === "2"
- });
- ```
- @method all
- @static
- @param {Array} entries array of promises
- @param {String} label optional string for labeling the promise.
- Useful for tooling.
- @return {Promise} promise that is fulfilled when all `promises` have been
- fulfilled, or rejected if any of them become rejected.
- @static
- */
- function all(entries) {
- return new Enumerator(this, entries).promise;
- }
- /**
- `Promise.race` returns a new promise which is settled in the same way as the
- first passed promise to settle.
- Example:
- ```javascript
- let promise1 = new Promise(function(resolve, reject){
- setTimeout(function(){
- resolve('promise 1');
- }, 200);
- });
- let promise2 = new Promise(function(resolve, reject){
- setTimeout(function(){
- resolve('promise 2');
- }, 100);
- });
- Promise.race([promise1, promise2]).then(function(result){
- // result === 'promise 2' because it was resolved before promise1
- // was resolved.
- });
- ```
- `Promise.race` is deterministic in that only the state of the first
- settled promise matters. For example, even if other promises given to the
- `promises` array argument are resolved, but the first settled promise has
- become rejected before the other promises became fulfilled, the returned
- promise will become rejected:
- ```javascript
- let promise1 = new Promise(function(resolve, reject){
- setTimeout(function(){
- resolve('promise 1');
- }, 200);
- });
- let promise2 = new Promise(function(resolve, reject){
- setTimeout(function(){
- reject(new Error('promise 2'));
- }, 100);
- });
- Promise.race([promise1, promise2]).then(function(result){
- // Code here never runs
- }, function(reason){
- // reason.message === 'promise 2' because promise 2 became rejected before
- // promise 1 became fulfilled
- });
- ```
- An example real-world use case is implementing timeouts:
- ```javascript
- Promise.race([ajax('foo.json'), timeout(5000)])
- ```
- @method race
- @static
- @param {Array} promises array of promises to observe
- Useful for tooling.
- @return {Promise} a promise which settles in the same way as the first passed
- promise to settle.
- */
- function race(entries) {
- /*jshint validthis:true */
- var Constructor = this;
- if (!isArray(entries)) {
- return new Constructor(function (_, reject) {
- return reject(new TypeError('You must pass an array to race.'));
- });
- } else {
- return new Constructor(function (resolve, reject) {
- var length = entries.length;
- for (var i = 0; i < length; i++) {
- Constructor.resolve(entries[i]).then(resolve, reject);
- }
- });
- }
- }
- /**
- `Promise.reject` returns a promise rejected with the passed `reason`.
- It is shorthand for the following:
- ```javascript
- let promise = new Promise(function(resolve, reject){
- reject(new Error('WHOOPS'));
- });
- promise.then(function(value){
- // Code here doesn't run because the promise is rejected!
- }, function(reason){
- // reason.message === 'WHOOPS'
- });
- ```
- Instead of writing the above, your code now simply becomes the following:
- ```javascript
- let promise = Promise.reject(new Error('WHOOPS'));
- promise.then(function(value){
- // Code here doesn't run because the promise is rejected!
- }, function(reason){
- // reason.message === 'WHOOPS'
- });
- ```
- @method reject
- @static
- @param {Any} reason value that the returned promise will be rejected with.
- Useful for tooling.
- @return {Promise} a promise rejected with the given `reason`.
- */
- function reject(reason) {
- /*jshint validthis:true */
- var Constructor = this;
- var promise = new Constructor(noop);
- _reject(promise, reason);
- return promise;
- }
- function needsResolver() {
- throw new TypeError('You must pass a resolver function as the first argument to the promise constructor');
- }
- function needsNew() {
- throw new TypeError("Failed to construct 'Promise': Please use the 'new' operator, this object constructor cannot be called as a function.");
- }
- /**
- Promise objects represent the eventual result of an asynchronous operation. The
- primary way of interacting with a promise is through its `then` method, which
- registers callbacks to receive either a promise's eventual value or the reason
- why the promise cannot be fulfilled.
- Terminology
- -----------
- - `promise` is an object or function with a `then` method whose behavior conforms to this specification.
- - `thenable` is an object or function that defines a `then` method.
- - `value` is any legal JavaScript value (including undefined, a thenable, or a promise).
- - `exception` is a value that is thrown using the throw statement.
- - `reason` is a value that indicates why a promise was rejected.
- - `settled` the final resting state of a promise, fulfilled or rejected.
- A promise can be in one of three states: pending, fulfilled, or rejected.
- Promises that are fulfilled have a fulfillment value and are in the fulfilled
- state. Promises that are rejected have a rejection reason and are in the
- rejected state. A fulfillment value is never a thenable.
- Promises can also be said to *resolve* a value. If this value is also a
- promise, then the original promise's settled state will match the value's
- settled state. So a promise that *resolves* a promise that rejects will
- itself reject, and a promise that *resolves* a promise that fulfills will
- itself fulfill.
- Basic Usage:
- ------------
- ```js
- let promise = new Promise(function(resolve, reject) {
- // on success
- resolve(value);
- // on failure
- reject(reason);
- });
- promise.then(function(value) {
- // on fulfillment
- }, function(reason) {
- // on rejection
- });
- ```
- Advanced Usage:
- ---------------
- Promises shine when abstracting away asynchronous interactions such as
- `XMLHttpRequest`s.
- ```js
- function getJSON(url) {
- return new Promise(function(resolve, reject){
- let xhr = new XMLHttpRequest();
- xhr.open('GET', url);
- xhr.onreadystatechange = handler;
- xhr.responseType = 'json';
- xhr.setRequestHeader('Accept', 'application/json');
- xhr.send();
- function handler() {
- if (this.readyState === this.DONE) {
- if (this.status === 200) {
- resolve(this.response);
- } else {
- reject(new Error('getJSON: `' + url + '` failed with status: [' + this.status + ']'));
- }
- }
- };
- });
- }
- getJSON('/posts.json').then(function(json) {
- // on fulfillment
- }, function(reason) {
- // on rejection
- });
- ```
- Unlike callbacks, promises are great composable primitives.
- ```js
- Promise.all([
- getJSON('/posts'),
- getJSON('/comments')
- ]).then(function(values){
- values[0] // => postsJSON
- values[1] // => commentsJSON
- return values;
- });
- ```
- @class Promise
- @param {function} resolver
- Useful for tooling.
- @constructor
- */
- function Promise(resolver) {
- this[PROMISE_ID] = nextId();
- this._result = this._state = undefined;
- this._subscribers = [];
- if (noop !== resolver) {
- typeof resolver !== 'function' && needsResolver();
- this instanceof Promise ? initializePromise(this, resolver) : needsNew();
- }
- }
- Promise.all = all;
- Promise.race = race;
- Promise.resolve = resolve;
- Promise.reject = reject;
- Promise._setScheduler = setScheduler;
- Promise._setAsap = setAsap;
- Promise._asap = asap;
- Promise.prototype = {
- constructor: Promise,
- /**
- The primary way of interacting with a promise is through its `then` method,
- which registers callbacks to receive either a promise's eventual value or the
- reason why the promise cannot be fulfilled.
-
- ```js
- findUser().then(function(user){
- // user is available
- }, function(reason){
- // user is unavailable, and you are given the reason why
- });
- ```
-
- Chaining
- --------
-
- The return value of `then` is itself a promise. This second, 'downstream'
- promise is resolved with the return value of the first promise's fulfillment
- or rejection handler, or rejected if the handler throws an exception.
-
- ```js
- findUser().then(function (user) {
- return user.name;
- }, function (reason) {
- return 'default name';
- }).then(function (userName) {
- // If `findUser` fulfilled, `userName` will be the user's name, otherwise it
- // will be `'default name'`
- });
-
- findUser().then(function (user) {
- throw new Error('Found user, but still unhappy');
- }, function (reason) {
- throw new Error('`findUser` rejected and we're unhappy');
- }).then(function (value) {
- // never reached
- }, function (reason) {
- // if `findUser` fulfilled, `reason` will be 'Found user, but still unhappy'.
- // If `findUser` rejected, `reason` will be '`findUser` rejected and we're unhappy'.
- });
- ```
- If the downstream promise does not specify a rejection handler, rejection reasons will be propagated further downstream.
-
- ```js
- findUser().then(function (user) {
- throw new PedagogicalException('Upstream error');
- }).then(function (value) {
- // never reached
- }).then(function (value) {
- // never reached
- }, function (reason) {
- // The `PedgagocialException` is propagated all the way down to here
- });
- ```
-
- Assimilation
- ------------
-
- Sometimes the value you want to propagate to a downstream promise can only be
- retrieved asynchronously. This can be achieved by returning a promise in the
- fulfillment or rejection handler. The downstream promise will then be pending
- until the returned promise is settled. This is called *assimilation*.
-
- ```js
- findUser().then(function (user) {
- return findCommentsByAuthor(user);
- }).then(function (comments) {
- // The user's comments are now available
- });
- ```
-
- If the assimliated promise rejects, then the downstream promise will also reject.
-
- ```js
- findUser().then(function (user) {
- return findCommentsByAuthor(user);
- }).then(function (comments) {
- // If `findCommentsByAuthor` fulfills, we'll have the value here
- }, function (reason) {
- // If `findCommentsByAuthor` rejects, we'll have the reason here
- });
- ```
-
- Simple Example
- --------------
-
- Synchronous Example
-
- ```javascript
- let result;
-
- try {
- result = findResult();
- // success
- } catch(reason) {
- // failure
- }
- ```
-
- Errback Example
-
- ```js
- findResult(function(result, err){
- if (err) {
- // failure
- } else {
- // success
- }
- });
- ```
-
- Promise Example;
-
- ```javascript
- findResult().then(function(result){
- // success
- }, function(reason){
- // failure
- });
- ```
-
- Advanced Example
- --------------
-
- Synchronous Example
-
- ```javascript
- let author, books;
-
- try {
- author = findAuthor();
- books = findBooksByAuthor(author);
- // success
- } catch(reason) {
- // failure
- }
- ```
-
- Errback Example
-
- ```js
-
- function foundBooks(books) {
-
- }
-
- function failure(reason) {
-
- }
-
- findAuthor(function(author, err){
- if (err) {
- failure(err);
- // failure
- } else {
- try {
- findBoooksByAuthor(author, function(books, err) {
- if (err) {
- failure(err);
- } else {
- try {
- foundBooks(books);
- } catch(reason) {
- failure(reason);
- }
- }
- });
- } catch(error) {
- failure(err);
- }
- // success
- }
- });
- ```
-
- Promise Example;
-
- ```javascript
- findAuthor().
- then(findBooksByAuthor).
- then(function(books){
- // found books
- }).catch(function(reason){
- // something went wrong
- });
- ```
-
- @method then
- @param {Function} onFulfilled
- @param {Function} onRejected
- Useful for tooling.
- @return {Promise}
- */
- then: then,
- /**
- `catch` is simply sugar for `then(undefined, onRejection)` which makes it the same
- as the catch block of a try/catch statement.
-
- ```js
- function findAuthor(){
- throw new Error('couldn't find that author');
- }
-
- // synchronous
- try {
- findAuthor();
- } catch(reason) {
- // something went wrong
- }
-
- // async with promises
- findAuthor().catch(function(reason){
- // something went wrong
- });
- ```
-
- @method catch
- @param {Function} onRejection
- Useful for tooling.
- @return {Promise}
- */
- 'catch': function _catch(onRejection) {
- return this.then(null, onRejection);
- }
- };
- function polyfill() {
- var local = undefined;
- if (typeof global !== 'undefined') {
- local = global;
- } else if (typeof self !== 'undefined') {
- local = self;
- } else {
- try {
- local = Function('return this')();
- } catch (e) {
- throw new Error('polyfill failed because global object is unavailable in this environment');
- }
- }
- var P = local.Promise;
- if (P) {
- var promiseToString = null;
- try {
- promiseToString = Object.prototype.toString.call(P.resolve());
- } catch (e) {
- // silently ignored
- }
- if (promiseToString === '[object Promise]' && !P.cast) {
- return;
- }
- }
- local.Promise = Promise;
- }
- // Strange compat..
- Promise.polyfill = polyfill;
- Promise.Promise = Promise;
- return Promise;
- })));
- }).call(this,_dereq_('_process'),typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
- },{"_process":3}],2:[function(_dereq_,module,exports){
- // Copyright Joyent, Inc. and other Node contributors.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a
- // copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to permit
- // persons to whom the Software is furnished to do so, subject to the
- // following conditions:
- //
- // The above copyright notice and this permission notice shall be included
- // in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
- // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
- // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
- // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
- // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
- // USE OR OTHER DEALINGS IN THE SOFTWARE.
- function EventEmitter() {
- this._events = this._events || {};
- this._maxListeners = this._maxListeners || undefined;
- }
- module.exports = EventEmitter;
- // Backwards-compat with node 0.10.x
- EventEmitter.EventEmitter = EventEmitter;
- EventEmitter.prototype._events = undefined;
- EventEmitter.prototype._maxListeners = undefined;
- // By default EventEmitters will print a warning if more than 10 listeners are
- // added to it. This is a useful default which helps finding memory leaks.
- EventEmitter.defaultMaxListeners = 10;
- // Obviously not all Emitters should be limited to 10. This function allows
- // that to be increased. Set to zero for unlimited.
- EventEmitter.prototype.setMaxListeners = function(n) {
- if (!isNumber(n) || n < 0 || isNaN(n))
- throw TypeError('n must be a positive number');
- this._maxListeners = n;
- return this;
- };
- EventEmitter.prototype.emit = function(type) {
- var er, handler, len, args, i, listeners;
- if (!this._events)
- this._events = {};
- // If there is no 'error' event listener then throw.
- if (type === 'error') {
- if (!this._events.error ||
- (isObject(this._events.error) && !this._events.error.length)) {
- er = arguments[1];
- if (er instanceof Error) {
- throw er; // Unhandled 'error' event
- } else {
- // At least give some kind of context to the user
- var err = new Error('Uncaught, unspecified "error" event. (' + er + ')');
- err.context = er;
- throw err;
- }
- }
- }
- handler = this._events[type];
- if (isUndefined(handler))
- return false;
- if (isFunction(handler)) {
- switch (arguments.length) {
- // fast cases
- case 1:
- handler.call(this);
- break;
- case 2:
- handler.call(this, arguments[1]);
- break;
- case 3:
- handler.call(this, arguments[1], arguments[2]);
- break;
- // slower
- default:
- args = Array.prototype.slice.call(arguments, 1);
- handler.apply(this, args);
- }
- } else if (isObject(handler)) {
- args = Array.prototype.slice.call(arguments, 1);
- listeners = handler.slice();
- len = listeners.length;
- for (i = 0; i < len; i++)
- listeners[i].apply(this, args);
- }
- return true;
- };
- EventEmitter.prototype.addListener = function(type, listener) {
- var m;
- if (!isFunction(listener))
- throw TypeError('listener must be a function');
- if (!this._events)
- this._events = {};
- // To avoid recursion in the case that type === "newListener"! Before
- // adding it to the listeners, first emit "newListener".
- if (this._events.newListener)
- this.emit('newListener', type,
- isFunction(listener.listener) ?
- listener.listener : listener);
- if (!this._events[type])
- // Optimize the case of one listener. Don't need the extra array object.
- this._events[type] = listener;
- else if (isObject(this._events[type]))
- // If we've already got an array, just append.
- this._events[type].push(listener);
- else
- // Adding the second element, need to change to array.
- this._events[type] = [this._events[type], listener];
- // Check for listener leak
- if (isObject(this._events[type]) && !this._events[type].warned) {
- if (!isUndefined(this._maxListeners)) {
- m = this._maxListeners;
- } else {
- m = EventEmitter.defaultMaxListeners;
- }
- if (m && m > 0 && this._events[type].length > m) {
- this._events[type].warned = true;
- console.error('(node) warning: possible EventEmitter memory ' +
- 'leak detected. %d listeners added. ' +
- 'Use emitter.setMaxListeners() to increase limit.',
- this._events[type].length);
- if (typeof console.trace === 'function') {
- // not supported in IE 10
- console.trace();
- }
- }
- }
- return this;
- };
- EventEmitter.prototype.on = EventEmitter.prototype.addListener;
- EventEmitter.prototype.once = function(type, listener) {
- if (!isFunction(listener))
- throw TypeError('listener must be a function');
- var fired = false;
- function g() {
- this.removeListener(type, g);
- if (!fired) {
- fired = true;
- listener.apply(this, arguments);
- }
- }
- g.listener = listener;
- this.on(type, g);
- return this;
- };
- // emits a 'removeListener' event iff the listener was removed
- EventEmitter.prototype.removeListener = function(type, listener) {
- var list, position, length, i;
- if (!isFunction(listener))
- throw TypeError('listener must be a function');
- if (!this._events || !this._events[type])
- return this;
- list = this._events[type];
- length = list.length;
- position = -1;
- if (list === listener ||
- (isFunction(list.listener) && list.listener === listener)) {
- delete this._events[type];
- if (this._events.removeListener)
- this.emit('removeListener', type, listener);
- } else if (isObject(list)) {
- for (i = length; i-- > 0;) {
- if (list[i] === listener ||
- (list[i].listener && list[i].listener === listener)) {
- position = i;
- break;
- }
- }
- if (position < 0)
- return this;
- if (list.length === 1) {
- list.length = 0;
- delete this._events[type];
- } else {
- list.splice(position, 1);
- }
- if (this._events.removeListener)
- this.emit('removeListener', type, listener);
- }
- return this;
- };
- EventEmitter.prototype.removeAllListeners = function(type) {
- var key, listeners;
- if (!this._events)
- return this;
- // not listening for removeListener, no need to emit
- if (!this._events.removeListener) {
- if (arguments.length === 0)
- this._events = {};
- else if (this._events[type])
- delete this._events[type];
- return this;
- }
- // emit removeListener for all listeners on all events
- if (arguments.length === 0) {
- for (key in this._events) {
- if (key === 'removeListener') continue;
- this.removeAllListeners(key);
- }
- this.removeAllListeners('removeListener');
- this._events = {};
- return this;
- }
- listeners = this._events[type];
- if (isFunction(listeners)) {
- this.removeListener(type, listeners);
- } else if (listeners) {
- // LIFO order
- while (listeners.length)
- this.removeListener(type, listeners[listeners.length - 1]);
- }
- delete this._events[type];
- return this;
- };
- EventEmitter.prototype.listeners = function(type) {
- var ret;
- if (!this._events || !this._events[type])
- ret = [];
- else if (isFunction(this._events[type]))
- ret = [this._events[type]];
- else
- ret = this._events[type].slice();
- return ret;
- };
- EventEmitter.prototype.listenerCount = function(type) {
- if (this._events) {
- var evlistener = this._events[type];
- if (isFunction(evlistener))
- return 1;
- else if (evlistener)
- return evlistener.length;
- }
- return 0;
- };
- EventEmitter.listenerCount = function(emitter, type) {
- return emitter.listenerCount(type);
- };
- function isFunction(arg) {
- return typeof arg === 'function';
- }
- function isNumber(arg) {
- return typeof arg === 'number';
- }
- function isObject(arg) {
- return typeof arg === 'object' && arg !== null;
- }
- function isUndefined(arg) {
- return arg === void 0;
- }
- },{}],3:[function(_dereq_,module,exports){
- // shim for using process in browser
- var process = module.exports = {};
- // cached from whatever global is present so that test runners that stub it
- // don't break things. But we need to wrap it in a try catch in case it is
- // wrapped in strict mode code which doesn't define any globals. It's inside a
- // function because try/catches deoptimize in certain engines.
- var cachedSetTimeout;
- var cachedClearTimeout;
- function defaultSetTimout() {
- throw new Error('setTimeout has not been defined');
- }
- function defaultClearTimeout () {
- throw new Error('clearTimeout has not been defined');
- }
- (function () {
- try {
- if (typeof setTimeout === 'function') {
- cachedSetTimeout = setTimeout;
- } else {
- cachedSetTimeout = defaultSetTimout;
- }
- } catch (e) {
- cachedSetTimeout = defaultSetTimout;
- }
- try {
- if (typeof clearTimeout === 'function') {
- cachedClearTimeout = clearTimeout;
- } else {
- cachedClearTimeout = defaultClearTimeout;
- }
- } catch (e) {
- cachedClearTimeout = defaultClearTimeout;
- }
- } ())
- function runTimeout(fun) {
- if (cachedSetTimeout === setTimeout) {
- //normal enviroments in sane situations
- return setTimeout(fun, 0);
- }
- // if setTimeout wasn't available but was latter defined
- if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {
- cachedSetTimeout = setTimeout;
- return setTimeout(fun, 0);
- }
- try {
- // when when somebody has screwed with setTimeout but no I.E. maddness
- return cachedSetTimeout(fun, 0);
- } catch(e){
- try {
- // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
- return cachedSetTimeout.call(null, fun, 0);
- } catch(e){
- // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error
- return cachedSetTimeout.call(this, fun, 0);
- }
- }
- }
- function runClearTimeout(marker) {
- if (cachedClearTimeout === clearTimeout) {
- //normal enviroments in sane situations
- return clearTimeout(marker);
- }
- // if clearTimeout wasn't available but was latter defined
- if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {
- cachedClearTimeout = clearTimeout;
- return clearTimeout(marker);
- }
- try {
- // when when somebody has screwed with setTimeout but no I.E. maddness
- return cachedClearTimeout(marker);
- } catch (e){
- try {
- // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
- return cachedClearTimeout.call(null, marker);
- } catch (e){
- // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.
- // Some versions of I.E. have different rules for clearTimeout vs setTimeout
- return cachedClearTimeout.call(this, marker);
- }
- }
- }
- var queue = [];
- var draining = false;
- var currentQueue;
- var queueIndex = -1;
- function cleanUpNextTick() {
- if (!draining || !currentQueue) {
- return;
- }
- draining = false;
- if (currentQueue.length) {
- queue = currentQueue.concat(queue);
- } else {
- queueIndex = -1;
- }
- if (queue.length) {
- drainQueue();
- }
- }
- function drainQueue() {
- if (draining) {
- return;
- }
- var timeout = runTimeout(cleanUpNextTick);
- draining = true;
- var len = queue.length;
- while(len) {
- currentQueue = queue;
- queue = [];
- while (++queueIndex < len) {
- if (currentQueue) {
- currentQueue[queueIndex].run();
- }
- }
- queueIndex = -1;
- len = queue.length;
- }
- currentQueue = null;
- draining = false;
- runClearTimeout(timeout);
- }
- process.nextTick = function (fun) {
- var args = new Array(arguments.length - 1);
- if (arguments.length > 1) {
- for (var i = 1; i < arguments.length; i++) {
- args[i - 1] = arguments[i];
- }
- }
- queue.push(new Item(fun, args));
- if (queue.length === 1 && !draining) {
- runTimeout(drainQueue);
- }
- };
- // v8 likes predictible objects
- function Item(fun, array) {
- this.fun = fun;
- this.array = array;
- }
- Item.prototype.run = function () {
- this.fun.apply(null, this.array);
- };
- process.title = 'browser';
- process.browser = true;
- process.env = {};
- process.argv = [];
- process.version = ''; // empty string to avoid regexp issues
- process.versions = {};
- function noop() {}
- process.on = noop;
- process.addListener = noop;
- process.once = noop;
- process.off = noop;
- process.removeListener = noop;
- process.removeAllListeners = noop;
- process.emit = noop;
- process.prependListener = noop;
- process.prependOnceListener = noop;
- process.listeners = function (name) { return [] }
- process.binding = function (name) {
- throw new Error('process.binding is not supported');
- };
- process.cwd = function () { return '/' };
- process.chdir = function (dir) {
- throw new Error('process.chdir is not supported');
- };
- process.umask = function() { return 0; };
- },{}],4:[function(_dereq_,module,exports){
- var bundleFn = arguments[3];
- var sources = arguments[4];
- var cache = arguments[5];
- var stringify = JSON.stringify;
- module.exports = function (fn, options) {
- var wkey;
- var cacheKeys = Object.keys(cache);
- for (var i = 0, l = cacheKeys.length; i < l; i++) {
- var key = cacheKeys[i];
- var exp = cache[key].exports;
- // Using babel as a transpiler to use esmodule, the export will always
- // be an object with the default export as a property of it. To ensure
- // the existing api and babel esmodule exports are both supported we
- // check for both
- if (exp === fn || exp && exp.default === fn) {
- wkey = key;
- break;
- }
- }
- if (!wkey) {
- wkey = Math.floor(Math.pow(16, 8) * Math.random()).toString(16);
- var wcache = {};
- for (var i = 0, l = cacheKeys.length; i < l; i++) {
- var key = cacheKeys[i];
- wcache[key] = key;
- }
- sources[wkey] = [
- Function(['require','module','exports'], '(' + fn + ')(self)'),
- wcache
- ];
- }
- var skey = Math.floor(Math.pow(16, 8) * Math.random()).toString(16);
- var scache = {}; scache[wkey] = wkey;
- sources[skey] = [
- Function(['require'], (
- // try to call default if defined to also support babel esmodule
- // exports
- 'var f = require(' + stringify(wkey) + ');' +
- '(f.default ? f.default : f)(self);'
- )),
- scache
- ];
- var workerSources = {};
- resolveSources(skey);
- function resolveSources(key) {
- workerSources[key] = true;
- for (var depPath in sources[key][1]) {
- var depKey = sources[key][1][depPath];
- if (!workerSources[depKey]) {
- resolveSources(depKey);
- }
- }
- }
- var src = '(' + bundleFn + ')({'
- + Object.keys(workerSources).map(function (key) {
- return stringify(key) + ':['
- + sources[key][0]
- + ',' + stringify(sources[key][1]) + ']'
- ;
- }).join(',')
- + '},{},[' + stringify(skey) + '])'
- ;
- var URL = window.URL || window.webkitURL || window.mozURL || window.msURL;
- var blob = new Blob([src], { type: 'text/javascript' });
- if (options && options.bare) { return blob; }
- var workerUrl = URL.createObjectURL(blob);
- var worker = new Worker(workerUrl);
- worker.objectURL = workerUrl;
- return worker;
- };
- },{}],5:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.createDefaultConfig = createDefaultConfig;
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var defaultConfig = exports.defaultConfig = {
- enableWorker: false,
- enableStashBuffer: true,
- stashInitialSize: undefined,
- isLive: false,
- lazyLoad: true,
- lazyLoadMaxDuration: 3 * 60,
- lazyLoadRecoverDuration: 30,
- deferLoadAfterSourceOpen: true,
- // autoCleanupSourceBuffer: default as false, leave unspecified
- autoCleanupMaxBackwardDuration: 3 * 60,
- autoCleanupMinBackwardDuration: 2 * 60,
- statisticsInfoReportInterval: 600,
- fixAudioTimestampGap: true,
- accurateSeek: false,
- seekType: 'range', // [range, param, custom]
- seekParamStart: 'bstart',
- seekParamEnd: 'bend',
- rangeLoadZeroStart: false,
- customSeekHandler: undefined,
- reuseRedirectedURL: false
- // referrerPolicy: leave as unspecified
- };
- function createDefaultConfig() {
- return Object.assign({}, defaultConfig);
- }
- },{}],6:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _ioController = _dereq_('../io/io-controller.js');
- var _ioController2 = _interopRequireDefault(_ioController);
- var _config = _dereq_('../config.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var Features = function () {
- function Features() {
- _classCallCheck(this, Features);
- }
- _createClass(Features, null, [{
- key: 'supportMSEH264Playback',
- value: function supportMSEH264Playback() {
- return window.MediaSource && window.MediaSource.isTypeSupported('video/mp4; codecs="avc1.42E01E,mp4a.40.2"');
- }
- }, {
- key: 'supportNetworkStreamIO',
- value: function supportNetworkStreamIO() {
- var ioctl = new _ioController2.default({}, (0, _config.createDefaultConfig)());
- var loaderType = ioctl.loaderType;
- ioctl.destroy();
- return loaderType == 'fetch-stream-loader' || loaderType == 'xhr-moz-chunked-loader';
- }
- }, {
- key: 'getNetworkLoaderTypeName',
- value: function getNetworkLoaderTypeName() {
- var ioctl = new _ioController2.default({}, (0, _config.createDefaultConfig)());
- var loaderType = ioctl.loaderType;
- ioctl.destroy();
- return loaderType;
- }
- }, {
- key: 'supportNativeMediaPlayback',
- value: function supportNativeMediaPlayback(mimeType) {
- if (Features.videoElement == undefined) {
- Features.videoElement = window.document.createElement('video');
- }
- var canPlay = Features.videoElement.canPlayType(mimeType);
- return canPlay === 'probably' || canPlay == 'maybe';
- }
- }, {
- key: 'getFeatureList',
- value: function getFeatureList() {
- var features = {
- mseFlvPlayback: false,
- mseLiveFlvPlayback: false,
- networkStreamIO: false,
- networkLoaderName: '',
- nativeMP4H264Playback: false,
- nativeWebmVP8Playback: false,
- nativeWebmVP9Playback: false
- };
- features.mseFlvPlayback = Features.supportMSEH264Playback();
- features.networkStreamIO = Features.supportNetworkStreamIO();
- features.networkLoaderName = Features.getNetworkLoaderTypeName();
- features.mseLiveFlvPlayback = features.mseFlvPlayback && features.networkStreamIO;
- features.nativeMP4H264Playback = Features.supportNativeMediaPlayback('video/mp4; codecs="avc1.42001E, mp4a.40.2"');
- features.nativeWebmVP8Playback = Features.supportNativeMediaPlayback('video/webm; codecs="vp8.0, vorbis"');
- features.nativeWebmVP9Playback = Features.supportNativeMediaPlayback('video/webm; codecs="vp9"');
- return features;
- }
- }]);
- return Features;
- }();
- exports.default = Features;
- },{"../config.js":5,"../io/io-controller.js":23}],7:[function(_dereq_,module,exports){
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var MediaInfo = function () {
- function MediaInfo() {
- _classCallCheck(this, MediaInfo);
- this.mimeType = null;
- this.duration = null;
- this.hasAudio = null;
- this.hasVideo = null;
- this.audioCodec = null;
- this.videoCodec = null;
- this.audioDataRate = null;
- this.videoDataRate = null;
- this.audioSampleRate = null;
- this.audioChannelCount = null;
- this.width = null;
- this.height = null;
- this.fps = null;
- this.profile = null;
- this.level = null;
- this.refFrames = null;
- this.chromaFormat = null;
- this.sarNum = null;
- this.sarDen = null;
- this.metadata = null;
- this.segments = null; // MediaInfo[]
- this.segmentCount = null;
- this.hasKeyframesIndex = null;
- this.keyframesIndex = null;
- }
- _createClass(MediaInfo, [{
- key: "isComplete",
- value: function isComplete() {
- var audioInfoComplete = this.hasAudio === false || this.hasAudio === true && this.audioCodec != null && this.audioSampleRate != null && this.audioChannelCount != null;
- var videoInfoComplete = this.hasVideo === false || this.hasVideo === true && this.videoCodec != null && this.width != null && this.height != null && this.fps != null && this.profile != null && this.level != null && this.refFrames != null && this.chromaFormat != null && this.sarNum != null && this.sarDen != null;
- // keyframesIndex may not be present
- return this.mimeType != null && this.duration != null && this.metadata != null && this.hasKeyframesIndex != null && audioInfoComplete && videoInfoComplete;
- }
- }, {
- key: "isSeekable",
- value: function isSeekable() {
- return this.hasKeyframesIndex === true;
- }
- }, {
- key: "getNearestKeyframe",
- value: function getNearestKeyframe(milliseconds) {
- if (this.keyframesIndex == null) {
- return null;
- }
- var table = this.keyframesIndex;
- var keyframeIdx = this._search(table.times, milliseconds);
- return {
- index: keyframeIdx,
- milliseconds: table.times[keyframeIdx],
- fileposition: table.filepositions[keyframeIdx]
- };
- }
- }, {
- key: "_search",
- value: function _search(list, value) {
- var idx = 0;
- var last = list.length - 1;
- var mid = 0;
- var lbound = 0;
- var ubound = last;
- if (value < list[0]) {
- idx = 0;
- lbound = ubound + 1; // skip search
- }
- while (lbound <= ubound) {
- mid = lbound + Math.floor((ubound - lbound) / 2);
- if (mid === last || value >= list[mid] && value < list[mid + 1]) {
- idx = mid;
- break;
- } else if (list[mid] < value) {
- lbound = mid + 1;
- } else {
- ubound = mid - 1;
- }
- }
- return idx;
- }
- }]);
- return MediaInfo;
- }();
- exports.default = MediaInfo;
- },{}],8:[function(_dereq_,module,exports){
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- // Represents an media sample (audio / video)
- var SampleInfo = exports.SampleInfo = function SampleInfo(dts, pts, duration, originalDts, isSync) {
- _classCallCheck(this, SampleInfo);
- this.dts = dts;
- this.pts = pts;
- this.duration = duration;
- this.originalDts = originalDts;
- this.isSyncPoint = isSync;
- this.fileposition = null;
- };
- // Media Segment concept is defined in Media Source Extensions spec.
- // Particularly in ISO BMFF format, an Media Segment contains a moof box followed by a mdat box.
- var MediaSegmentInfo = exports.MediaSegmentInfo = function () {
- function MediaSegmentInfo() {
- _classCallCheck(this, MediaSegmentInfo);
- this.beginDts = 0;
- this.endDts = 0;
- this.beginPts = 0;
- this.endPts = 0;
- this.originalBeginDts = 0;
- this.originalEndDts = 0;
- this.syncPoints = []; // SampleInfo[n], for video IDR frames only
- this.firstSample = null; // SampleInfo
- this.lastSample = null; // SampleInfo
- }
- _createClass(MediaSegmentInfo, [{
- key: "appendSyncPoint",
- value: function appendSyncPoint(sampleInfo) {
- // also called Random Access Point
- sampleInfo.isSyncPoint = true;
- this.syncPoints.push(sampleInfo);
- }
- }]);
- return MediaSegmentInfo;
- }();
- // Ordered list for recording video IDR frames, sorted by originalDts
- var IDRSampleList = exports.IDRSampleList = function () {
- function IDRSampleList() {
- _classCallCheck(this, IDRSampleList);
- this._list = [];
- }
- _createClass(IDRSampleList, [{
- key: "clear",
- value: function clear() {
- this._list = [];
- }
- }, {
- key: "appendArray",
- value: function appendArray(syncPoints) {
- var list = this._list;
- if (syncPoints.length === 0) {
- return;
- }
- if (list.length > 0 && syncPoints[0].originalDts < list[list.length - 1].originalDts) {
- this.clear();
- }
- Array.prototype.push.apply(list, syncPoints);
- }
- }, {
- key: "getLastSyncPointBeforeDts",
- value: function getLastSyncPointBeforeDts(dts) {
- if (this._list.length == 0) {
- return null;
- }
- var list = this._list;
- var idx = 0;
- var last = list.length - 1;
- var mid = 0;
- var lbound = 0;
- var ubound = last;
- if (dts < list[0].dts) {
- idx = 0;
- lbound = ubound + 1;
- }
- while (lbound <= ubound) {
- mid = lbound + Math.floor((ubound - lbound) / 2);
- if (mid === last || dts >= list[mid].dts && dts < list[mid + 1].dts) {
- idx = mid;
- break;
- } else if (list[mid].dts < dts) {
- lbound = mid + 1;
- } else {
- ubound = mid - 1;
- }
- }
- return this._list[idx];
- }
- }]);
- return IDRSampleList;
- }();
- // Data structure for recording information of media segments in single track.
- var MediaSegmentInfoList = exports.MediaSegmentInfoList = function () {
- function MediaSegmentInfoList(type) {
- _classCallCheck(this, MediaSegmentInfoList);
- this._type = type;
- this._list = [];
- this._lastAppendLocation = -1; // cached last insert location
- }
- _createClass(MediaSegmentInfoList, [{
- key: "isEmpty",
- value: function isEmpty() {
- return this._list.length === 0;
- }
- }, {
- key: "clear",
- value: function clear() {
- this._list = [];
- this._lastAppendLocation = -1;
- }
- }, {
- key: "_searchNearestSegmentBefore",
- value: function _searchNearestSegmentBefore(originalBeginDts) {
- var list = this._list;
- if (list.length === 0) {
- return -2;
- }
- var last = list.length - 1;
- var mid = 0;
- var lbound = 0;
- var ubound = last;
- var idx = 0;
- if (originalBeginDts < list[0].originalBeginDts) {
- idx = -1;
- return idx;
- }
- while (lbound <= ubound) {
- mid = lbound + Math.floor((ubound - lbound) / 2);
- if (mid === last || originalBeginDts > list[mid].lastSample.originalDts && originalBeginDts < list[mid + 1].originalBeginDts) {
- idx = mid;
- break;
- } else if (list[mid].originalBeginDts < originalBeginDts) {
- lbound = mid + 1;
- } else {
- ubound = mid - 1;
- }
- }
- return idx;
- }
- }, {
- key: "_searchNearestSegmentAfter",
- value: function _searchNearestSegmentAfter(originalBeginDts) {
- return this._searchNearestSegmentBefore(originalBeginDts) + 1;
- }
- }, {
- key: "append",
- value: function append(mediaSegmentInfo) {
- var list = this._list;
- var msi = mediaSegmentInfo;
- var lastAppendIdx = this._lastAppendLocation;
- var insertIdx = 0;
- if (lastAppendIdx !== -1 && lastAppendIdx < list.length && msi.originalBeginDts >= list[lastAppendIdx].lastSample.originalDts && (lastAppendIdx === list.length - 1 || lastAppendIdx < list.length - 1 && msi.originalBeginDts < list[lastAppendIdx + 1].originalBeginDts)) {
- insertIdx = lastAppendIdx + 1; // use cached location idx
- } else {
- if (list.length > 0) {
- insertIdx = this._searchNearestSegmentBefore(msi.originalBeginDts) + 1;
- }
- }
- this._lastAppendLocation = insertIdx;
- this._list.splice(insertIdx, 0, msi);
- }
- }, {
- key: "getLastSegmentBefore",
- value: function getLastSegmentBefore(originalBeginDts) {
- var idx = this._searchNearestSegmentBefore(originalBeginDts);
- if (idx >= 0) {
- return this._list[idx];
- } else {
- // -1
- return null;
- }
- }
- }, {
- key: "getLastSampleBefore",
- value: function getLastSampleBefore(originalBeginDts) {
- var segment = this.getLastSegmentBefore(originalBeginDts);
- if (segment != null) {
- return segment.lastSample;
- } else {
- return null;
- }
- }
- }, {
- key: "getLastSyncPointBefore",
- value: function getLastSyncPointBefore(originalBeginDts) {
- var segmentIdx = this._searchNearestSegmentBefore(originalBeginDts);
- var syncPoints = this._list[segmentIdx].syncPoints;
- while (syncPoints.length === 0 && segmentIdx > 0) {
- segmentIdx--;
- syncPoints = this._list[segmentIdx].syncPoints;
- }
- if (syncPoints.length > 0) {
- return syncPoints[syncPoints.length - 1];
- } else {
- return null;
- }
- }
- }, {
- key: "type",
- get: function get() {
- return this._type;
- }
- }, {
- key: "length",
- get: function get() {
- return this._list.length;
- }
- }]);
- return MediaSegmentInfoList;
- }();
- },{}],9:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _browser = _dereq_('../utils/browser.js');
- var _browser2 = _interopRequireDefault(_browser);
- var _mseEvents = _dereq_('./mse-events.js');
- var _mseEvents2 = _interopRequireDefault(_mseEvents);
- var _mediaSegmentInfo = _dereq_('./media-segment-info.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- // Media Source Extensions controller
- var MSEController = function () {
- function MSEController(config) {
- _classCallCheck(this, MSEController);
- this.TAG = 'MSEController';
- this._config = config;
- this._emitter = new _events2.default();
- if (this._config.isLive && this._config.autoCleanupSourceBuffer == undefined) {
- // For live stream, do auto cleanup by default
- this._config.autoCleanupSourceBuffer = true;
- }
- this.e = {
- onSourceOpen: this._onSourceOpen.bind(this),
- onSourceEnded: this._onSourceEnded.bind(this),
- onSourceClose: this._onSourceClose.bind(this),
- onSourceBufferError: this._onSourceBufferError.bind(this),
- onSourceBufferUpdateEnd: this._onSourceBufferUpdateEnd.bind(this)
- };
- this._mediaSource = null;
- this._mediaSourceObjectURL = null;
- this._mediaElement = null;
- this._isBufferFull = false;
- this._hasPendingEos = false;
- this._requireSetMediaDuration = false;
- this._pendingMediaDuration = 0;
- this._pendingSourceBufferInit = [];
- this._mimeTypes = {
- video: null,
- audio: null
- };
- this._sourceBuffers = {
- video: null,
- audio: null
- };
- this._lastInitSegments = {
- video: null,
- audio: null
- };
- this._pendingSegments = {
- video: [],
- audio: []
- };
- this._pendingRemoveRanges = {
- video: [],
- audio: []
- };
- this._idrList = new _mediaSegmentInfo.IDRSampleList();
- }
- _createClass(MSEController, [{
- key: 'destroy',
- value: function destroy() {
- if (this._mediaElement || this._mediaSource) {
- this.detachMediaElement();
- }
- this.e = null;
- this._emitter.removeAllListeners();
- this._emitter = null;
- }
- }, {
- key: 'on',
- value: function on(event, listener) {
- this._emitter.addListener(event, listener);
- }
- }, {
- key: 'off',
- value: function off(event, listener) {
- this._emitter.removeListener(event, listener);
- }
- }, {
- key: 'attachMediaElement',
- value: function attachMediaElement(mediaElement) {
- if (this._mediaSource) {
- throw new _exception.IllegalStateException('MediaSource has been attached to an HTMLMediaElement!');
- }
- var ms = this._mediaSource = new window.MediaSource();
- ms.addEventListener('sourceopen', this.e.onSourceOpen);
- ms.addEventListener('sourceended', this.e.onSourceEnded);
- ms.addEventListener('sourceclose', this.e.onSourceClose);
- this._mediaElement = mediaElement;
- this._mediaSourceObjectURL = window.URL.createObjectURL(this._mediaSource);
- mediaElement.src = this._mediaSourceObjectURL;
- }
- }, {
- key: 'detachMediaElement',
- value: function detachMediaElement() {
- if (this._mediaSource) {
- var ms = this._mediaSource;
- for (var type in this._sourceBuffers) {
- // pending segments should be discard
- var ps = this._pendingSegments[type];
- ps.splice(0, ps.length);
- this._pendingSegments[type] = null;
- this._pendingRemoveRanges[type] = null;
- this._lastInitSegments[type] = null;
- // remove all sourcebuffers
- var sb = this._sourceBuffers[type];
- if (sb) {
- if (ms.readyState !== 'closed') {
- ms.removeSourceBuffer(sb);
- sb.removeEventListener('error', this.e.onSourceBufferError);
- sb.removeEventListener('updateend', this.e.onSourceBufferUpdateEnd);
- }
- this._mimeTypes[type] = null;
- this._sourceBuffers[type] = null;
- }
- }
- if (ms.readyState === 'open') {
- try {
- ms.endOfStream();
- } catch (error) {
- _logger2.default.e(this.TAG, error.message);
- }
- }
- ms.removeEventListener('sourceopen', this.e.onSourceOpen);
- ms.removeEventListener('sourceended', this.e.onSourceEnded);
- ms.removeEventListener('sourceclose', this.e.onSourceClose);
- this._pendingSourceBufferInit = [];
- this._isBufferFull = false;
- this._idrList.clear();
- this._mediaSource = null;
- }
- if (this._mediaElement) {
- this._mediaElement.src = '';
- this._mediaElement.removeAttribute('src');
- this._mediaElement = null;
- }
- if (this._mediaSourceObjectURL) {
- window.URL.revokeObjectURL(this._mediaSourceObjectURL);
- this._mediaSourceObjectURL = null;
- }
- }
- }, {
- key: 'appendInitSegment',
- value: function appendInitSegment(initSegment, deferred) {
- if (!this._mediaSource || this._mediaSource.readyState !== 'open') {
- // sourcebuffer creation requires mediaSource.readyState === 'open'
- // so we defer the sourcebuffer creation, until sourceopen event triggered
- this._pendingSourceBufferInit.push(initSegment);
- // make sure that this InitSegment is in the front of pending segments queue
- this._pendingSegments[initSegment.type].push(initSegment);
- return;
- }
- var is = initSegment;
- var mimeType = '' + is.container;
- if (is.codec && is.codec.length > 0) {
- mimeType += ';codecs=' + is.codec;
- }
- var firstInitSegment = false;
- _logger2.default.v(this.TAG, 'Received Initialization Segment, mimeType: ' + mimeType);
- this._lastInitSegments[is.type] = is;
- if (mimeType !== this._mimeTypes[is.type]) {
- if (!this._mimeTypes[is.type]) {
- // empty, first chance create sourcebuffer
- firstInitSegment = true;
- try {
- var sb = this._sourceBuffers[is.type] = this._mediaSource.addSourceBuffer(mimeType);
- sb.addEventListener('error', this.e.onSourceBufferError);
- sb.addEventListener('updateend', this.e.onSourceBufferUpdateEnd);
- } catch (error) {
- _logger2.default.e(this.TAG, error.message);
- this._emitter.emit(_mseEvents2.default.ERROR, { code: error.code, msg: error.message });
- return;
- }
- } else {
- _logger2.default.v(this.TAG, 'Notice: ' + is.type + ' mimeType changed, origin: ' + this._mimeTypes[is.type] + ', target: ' + mimeType);
- }
- this._mimeTypes[is.type] = mimeType;
- }
- if (!deferred) {
- // deferred means this InitSegment has been pushed to pendingSegments queue
- this._pendingSegments[is.type].push(is);
- }
- if (!firstInitSegment) {
- // append immediately only if init segment in subsequence
- if (this._sourceBuffers[is.type] && !this._sourceBuffers[is.type].updating) {
- this._doAppendSegments();
- }
- }
- if (_browser2.default.safari && is.container === 'audio/mpeg' && is.mediaDuration > 0) {
- // 'audio/mpeg' track under Safari may cause MediaElement's duration to be NaN
- // Manually correct MediaSource.duration to make progress bar seekable, and report right duration
- this._requireSetMediaDuration = true;
- this._pendingMediaDuration = is.mediaDuration / 1000; // in seconds
- this._updateMediaSourceDuration();
- }
- }
- }, {
- key: 'appendMediaSegment',
- value: function appendMediaSegment(mediaSegment) {
- var ms = mediaSegment;
- this._pendingSegments[ms.type].push(ms);
- if (this._config.autoCleanupSourceBuffer && this._needCleanupSourceBuffer()) {
- this._doCleanupSourceBuffer();
- }
- var sb = this._sourceBuffers[ms.type];
- if (sb && !sb.updating && !this._hasPendingRemoveRanges()) {
- this._doAppendSegments();
- }
- }
- }, {
- key: 'seek',
- value: function seek(seconds) {
- // remove all appended buffers
- for (var type in this._sourceBuffers) {
- if (!this._sourceBuffers[type]) {
- continue;
- }
- // abort current buffer append algorithm
- var sb = this._sourceBuffers[type];
- if (this._mediaSource.readyState === 'open') {
- try {
- // If range removal algorithm is running, InvalidStateError will be throwed
- // Ignore it.
- sb.abort();
- } catch (error) {
- _logger2.default.e(this.TAG, error.message);
- }
- }
- // IDRList should be clear
- this._idrList.clear();
- // pending segments should be discard
- var ps = this._pendingSegments[type];
- ps.splice(0, ps.length);
- if (this._mediaSource.readyState === 'closed') {
- // Parent MediaSource object has been detached from HTMLMediaElement
- continue;
- }
- // record ranges to be remove from SourceBuffer
- for (var i = 0; i < sb.buffered.length; i++) {
- var start = sb.buffered.start(i);
- var end = sb.buffered.end(i);
- this._pendingRemoveRanges[type].push({ start: start, end: end });
- }
- // if sb is not updating, let's remove ranges now!
- if (!sb.updating) {
- this._doRemoveRanges();
- }
- // Safari 10 may get InvalidStateError in the later appendBuffer() after SourceBuffer.remove() call
- // Internal parser's state may be invalid at this time. Re-append last InitSegment to workaround.
- // Related issue: https://bugs.webkit.org/show_bug.cgi?id=159230
- if (_browser2.default.safari) {
- var lastInitSegment = this._lastInitSegments[type];
- if (lastInitSegment) {
- this._pendingSegments[type].push(lastInitSegment);
- if (!sb.updating) {
- this._doAppendSegments();
- }
- }
- }
- }
- }
- }, {
- key: 'endOfStream',
- value: function endOfStream() {
- var ms = this._mediaSource;
- var sb = this._sourceBuffers;
- if (!ms || ms.readyState !== 'open') {
- if (ms && ms.readyState === 'closed' && this._hasPendingSegments()) {
- // If MediaSource hasn't turned into open state, and there're pending segments
- // Mark pending endOfStream, defer call until all pending segments appended complete
- this._hasPendingEos = true;
- }
- return;
- }
- if (sb.video && sb.video.updating || sb.audio && sb.audio.updating) {
- // If any sourcebuffer is updating, defer endOfStream operation
- // See _onSourceBufferUpdateEnd()
- this._hasPendingEos = true;
- } else {
- this._hasPendingEos = false;
- // Notify media data loading complete
- // This is helpful for correcting total duration to match last media segment
- // Otherwise MediaElement's ended event may not be triggered
- ms.endOfStream();
- }
- }
- }, {
- key: 'getNearestKeyframe',
- value: function getNearestKeyframe(dts) {
- return this._idrList.getLastSyncPointBeforeDts(dts);
- }
- }, {
- key: '_needCleanupSourceBuffer',
- value: function _needCleanupSourceBuffer() {
- if (!this._config.autoCleanupSourceBuffer) {
- return false;
- }
- var currentTime = this._mediaElement.currentTime;
- for (var type in this._sourceBuffers) {
- var sb = this._sourceBuffers[type];
- if (sb) {
- var buffered = sb.buffered;
- if (buffered.length >= 1) {
- if (currentTime - buffered.start(0) >= this._config.autoCleanupMaxBackwardDuration) {
- return true;
- }
- }
- }
- }
- return false;
- }
- }, {
- key: '_doCleanupSourceBuffer',
- value: function _doCleanupSourceBuffer() {
- var currentTime = this._mediaElement.currentTime;
- for (var type in this._sourceBuffers) {
- var sb = this._sourceBuffers[type];
- if (sb) {
- var buffered = sb.buffered;
- var doRemove = false;
- for (var i = 0; i < buffered.length; i++) {
- var start = buffered.start(i);
- var end = buffered.end(i);
- if (start <= currentTime && currentTime < end + 3) {
- // padding 3 seconds
- if (currentTime - start >= this._config.autoCleanupMaxBackwardDuration) {
- doRemove = true;
- var removeEnd = currentTime - this._config.autoCleanupMinBackwardDuration;
- this._pendingRemoveRanges[type].push({ start: start, end: removeEnd });
- }
- } else if (end < currentTime) {
- doRemove = true;
- this._pendingRemoveRanges[type].push({ start: start, end: end });
- }
- }
- if (doRemove && !sb.updating) {
- this._doRemoveRanges();
- }
- }
- }
- }
- }, {
- key: '_updateMediaSourceDuration',
- value: function _updateMediaSourceDuration() {
- var sb = this._sourceBuffers;
- if (this._mediaElement.readyState === 0 || this._mediaSource.readyState !== 'open') {
- return;
- }
- if (sb.video && sb.video.updating || sb.audio && sb.audio.updating) {
- return;
- }
- var current = this._mediaSource.duration;
- var target = this._pendingMediaDuration;
- if (target > 0 && (isNaN(current) || target > current)) {
- _logger2.default.v(this.TAG, 'Update MediaSource duration from ' + current + ' to ' + target);
- this._mediaSource.duration = target;
- }
- this._requireSetMediaDuration = false;
- this._pendingMediaDuration = 0;
- }
- }, {
- key: '_doRemoveRanges',
- value: function _doRemoveRanges() {
- for (var type in this._pendingRemoveRanges) {
- if (!this._sourceBuffers[type] || this._sourceBuffers[type].updating) {
- continue;
- }
- var sb = this._sourceBuffers[type];
- var ranges = this._pendingRemoveRanges[type];
- while (ranges.length && !sb.updating) {
- var range = ranges.shift();
- sb.remove(range.start, range.end);
- }
- }
- }
- }, {
- key: '_doAppendSegments',
- value: function _doAppendSegments() {
- var pendingSegments = this._pendingSegments;
- for (var type in pendingSegments) {
- if (!this._sourceBuffers[type] || this._sourceBuffers[type].updating) {
- continue;
- }
- if (pendingSegments[type].length > 0) {
- var segment = pendingSegments[type].shift();
- if (segment.timestampOffset) {
- // For MPEG audio stream in MSE, if unbuffered-seeking occurred
- // We need explicitly set timestampOffset to the desired point in timeline for mpeg SourceBuffer.
- var currentOffset = this._sourceBuffers[type].timestampOffset;
- var targetOffset = segment.timestampOffset / 1000; // in seconds
- var delta = Math.abs(currentOffset - targetOffset);
- if (delta > 0.1) {
- // If time delta > 100ms
- _logger2.default.v(this.TAG, 'Update MPEG audio timestampOffset from ' + currentOffset + ' to ' + targetOffset);
- this._sourceBuffers[type].timestampOffset = targetOffset;
- }
- delete segment.timestampOffset;
- }
- if (!segment.data || segment.data.byteLength === 0) {
- // Ignore empty buffer
- continue;
- }
- try {
- this._sourceBuffers[type].appendBuffer(segment.data);
- this._isBufferFull = false;
- if (type === 'video' && segment.hasOwnProperty('info')) {
- this._idrList.appendArray(segment.info.syncPoints);
- }
- } catch (error) {
- this._pendingSegments[type].unshift(segment);
- if (error.code === 22) {
- // QuotaExceededError
- /* Notice that FireFox may not throw QuotaExceededError if SourceBuffer is full
- * Currently we can only do lazy-load to avoid SourceBuffer become scattered.
- * SourceBuffer eviction policy may be changed in future version of FireFox.
- *
- * Related issues:
- * https://bugzilla.mozilla.org/show_bug.cgi?id=1279885
- * https://bugzilla.mozilla.org/show_bug.cgi?id=1280023
- */
- // report buffer full, abort network IO
- if (!this._isBufferFull) {
- this._emitter.emit(_mseEvents2.default.BUFFER_FULL);
- }
- this._isBufferFull = true;
- } else {
- _logger2.default.e(this.TAG, error.message);
- this._emitter.emit(_mseEvents2.default.ERROR, { code: error.code, msg: error.message });
- }
- }
- }
- }
- }
- }, {
- key: '_onSourceOpen',
- value: function _onSourceOpen() {
- _logger2.default.v(this.TAG, 'MediaSource onSourceOpen');
- this._mediaSource.removeEventListener('sourceopen', this.e.onSourceOpen);
- // deferred sourcebuffer creation / initialization
- if (this._pendingSourceBufferInit.length > 0) {
- var pendings = this._pendingSourceBufferInit;
- while (pendings.length) {
- var segment = pendings.shift();
- this.appendInitSegment(segment, true);
- }
- }
- // there may be some pending media segments, append them
- if (this._hasPendingSegments()) {
- this._doAppendSegments();
- }
- this._emitter.emit(_mseEvents2.default.SOURCE_OPEN);
- }
- }, {
- key: '_onSourceEnded',
- value: function _onSourceEnded() {
- // fired on endOfStream
- _logger2.default.v(this.TAG, 'MediaSource onSourceEnded');
- }
- }, {
- key: '_onSourceClose',
- value: function _onSourceClose() {
- // fired on detaching from media element
- _logger2.default.v(this.TAG, 'MediaSource onSourceClose');
- if (this._mediaSource && this.e != null) {
- this._mediaSource.removeEventListener('sourceopen', this.e.onSourceOpen);
- this._mediaSource.removeEventListener('sourceended', this.e.onSourceEnded);
- this._mediaSource.removeEventListener('sourceclose', this.e.onSourceClose);
- }
- }
- }, {
- key: '_hasPendingSegments',
- value: function _hasPendingSegments() {
- var ps = this._pendingSegments;
- return ps.video.length > 0 || ps.audio.length > 0;
- }
- }, {
- key: '_hasPendingRemoveRanges',
- value: function _hasPendingRemoveRanges() {
- var prr = this._pendingRemoveRanges;
- return prr.video.length > 0 || prr.audio.length > 0;
- }
- }, {
- key: '_onSourceBufferUpdateEnd',
- value: function _onSourceBufferUpdateEnd() {
- if (this._requireSetMediaDuration) {
- this._updateMediaSourceDuration();
- } else if (this._hasPendingRemoveRanges()) {
- this._doRemoveRanges();
- } else if (this._hasPendingSegments()) {
- this._doAppendSegments();
- } else if (this._hasPendingEos) {
- this.endOfStream();
- }
- this._emitter.emit(_mseEvents2.default.UPDATE_END);
- }
- }, {
- key: '_onSourceBufferError',
- value: function _onSourceBufferError(e) {
- _logger2.default.e(this.TAG, 'SourceBuffer Error: ' + e);
- // this error might not always be fatal, just ignore it
- }
- }]);
- return MSEController;
- }();
- exports.default = MSEController;
- },{"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./media-segment-info.js":8,"./mse-events.js":10,"events":2}],10:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var MSEEvents = {
- ERROR: 'error',
- SOURCE_OPEN: 'source_open',
- UPDATE_END: 'update_end',
- BUFFER_FULL: 'buffer_full'
- };
- exports.default = MSEEvents;
- },{}],11:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _loggingControl = _dereq_('../utils/logging-control.js');
- var _loggingControl2 = _interopRequireDefault(_loggingControl);
- var _transmuxingController = _dereq_('./transmuxing-controller.js');
- var _transmuxingController2 = _interopRequireDefault(_transmuxingController);
- var _transmuxingEvents = _dereq_('./transmuxing-events.js');
- var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents);
- var _transmuxingWorker = _dereq_('./transmuxing-worker.js');
- var _transmuxingWorker2 = _interopRequireDefault(_transmuxingWorker);
- var _mediaInfo = _dereq_('./media-info.js');
- var _mediaInfo2 = _interopRequireDefault(_mediaInfo);
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var Transmuxer = function () {
- function Transmuxer(mediaDataSource, config) {
- _classCallCheck(this, Transmuxer);
- this.TAG = 'Transmuxer';
- this._emitter = new _events2.default();
- if (config.enableWorker && typeof Worker !== 'undefined') {
- try {
- var work = _dereq_('webworkify');
- this._worker = work(_transmuxingWorker2.default);
- this._workerDestroying = false;
- this._worker.addEventListener('message', this._onWorkerMessage.bind(this));
- this._worker.postMessage({ cmd: 'init', param: [mediaDataSource, config] });
- this.e = {
- onLoggingConfigChanged: this._onLoggingConfigChanged.bind(this)
- };
- _loggingControl2.default.registerListener(this.e.onLoggingConfigChanged);
- this._worker.postMessage({ cmd: 'logging_config', param: _loggingControl2.default.getConfig() });
- } catch (error) {
- _logger2.default.e(this.TAG, 'Error while initialize transmuxing worker, fallback to inline transmuxing');
- this._worker = null;
- this._controller = new _transmuxingController2.default(mediaDataSource, config);
- }
- } else {
- this._controller = new _transmuxingController2.default(mediaDataSource, config);
- }
- if (this._controller) {
- var ctl = this._controller;
- ctl.on(_transmuxingEvents2.default.IO_ERROR, this._onIOError.bind(this));
- ctl.on(_transmuxingEvents2.default.DEMUX_ERROR, this._onDemuxError.bind(this));
- ctl.on(_transmuxingEvents2.default.INIT_SEGMENT, this._onInitSegment.bind(this));
- ctl.on(_transmuxingEvents2.default.MEDIA_SEGMENT, this._onMediaSegment.bind(this));
- ctl.on(_transmuxingEvents2.default.LOADING_COMPLETE, this._onLoadingComplete.bind(this));
- ctl.on(_transmuxingEvents2.default.RECOVERED_EARLY_EOF, this._onRecoveredEarlyEof.bind(this));
- ctl.on(_transmuxingEvents2.default.MEDIA_INFO, this._onMediaInfo.bind(this));
- ctl.on(_transmuxingEvents2.default.STATISTICS_INFO, this._onStatisticsInfo.bind(this));
- ctl.on(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, this._onRecommendSeekpoint.bind(this));
- }
- }
- _createClass(Transmuxer, [{
- key: 'destroy',
- value: function destroy() {
- if (this._worker) {
- if (!this._workerDestroying) {
- this._workerDestroying = true;
- this._worker.postMessage({ cmd: 'destroy' });
- _loggingControl2.default.removeListener(this.e.onLoggingConfigChanged);
- this.e = null;
- }
- } else {
- this._controller.destroy();
- this._controller = null;
- }
- this._emitter.removeAllListeners();
- this._emitter = null;
- }
- }, {
- key: 'on',
- value: function on(event, listener) {
- this._emitter.addListener(event, listener);
- }
- }, {
- key: 'off',
- value: function off(event, listener) {
- this._emitter.removeListener(event, listener);
- }
- }, {
- key: 'hasWorker',
- value: function hasWorker() {
- return this._worker != null;
- }
- }, {
- key: 'open',
- value: function open() {
- if (this._worker) {
- this._worker.postMessage({ cmd: 'start' });
- } else {
- this._controller.start();
- }
- }
- }, {
- key: 'close',
- value: function close() {
- if (this._worker) {
- this._worker.postMessage({ cmd: 'stop' });
- } else {
- this._controller.stop();
- }
- }
- }, {
- key: 'seek',
- value: function seek(milliseconds) {
- if (this._worker) {
- this._worker.postMessage({ cmd: 'seek', param: milliseconds });
- } else {
- this._controller.seek(milliseconds);
- }
- }
- }, {
- key: 'pause',
- value: function pause() {
- if (this._worker) {
- this._worker.postMessage({ cmd: 'pause' });
- } else {
- this._controller.pause();
- }
- }
- }, {
- key: 'resume',
- value: function resume() {
- if (this._worker) {
- this._worker.postMessage({ cmd: 'resume' });
- } else {
- this._controller.resume();
- }
- }
- }, {
- key: '_onInitSegment',
- value: function _onInitSegment(type, initSegment) {
- var _this = this;
- // do async invoke
- Promise.resolve().then(function () {
- _this._emitter.emit(_transmuxingEvents2.default.INIT_SEGMENT, type, initSegment);
- });
- }
- }, {
- key: '_onMediaSegment',
- value: function _onMediaSegment(type, mediaSegment) {
- var _this2 = this;
- Promise.resolve().then(function () {
- _this2._emitter.emit(_transmuxingEvents2.default.MEDIA_SEGMENT, type, mediaSegment);
- });
- }
- }, {
- key: '_onLoadingComplete',
- value: function _onLoadingComplete() {
- var _this3 = this;
- Promise.resolve().then(function () {
- _this3._emitter.emit(_transmuxingEvents2.default.LOADING_COMPLETE);
- });
- }
- }, {
- key: '_onRecoveredEarlyEof',
- value: function _onRecoveredEarlyEof() {
- var _this4 = this;
- Promise.resolve().then(function () {
- _this4._emitter.emit(_transmuxingEvents2.default.RECOVERED_EARLY_EOF);
- });
- }
- }, {
- key: '_onMediaInfo',
- value: function _onMediaInfo(mediaInfo) {
- var _this5 = this;
- Promise.resolve().then(function () {
- _this5._emitter.emit(_transmuxingEvents2.default.MEDIA_INFO, mediaInfo);
- });
- }
- }, {
- key: '_onStatisticsInfo',
- value: function _onStatisticsInfo(statisticsInfo) {
- var _this6 = this;
- Promise.resolve().then(function () {
- _this6._emitter.emit(_transmuxingEvents2.default.STATISTICS_INFO, statisticsInfo);
- });
- }
- }, {
- key: '_onIOError',
- value: function _onIOError(type, info) {
- var _this7 = this;
- Promise.resolve().then(function () {
- _this7._emitter.emit(_transmuxingEvents2.default.IO_ERROR, type, info);
- });
- }
- }, {
- key: '_onDemuxError',
- value: function _onDemuxError(type, info) {
- var _this8 = this;
- Promise.resolve().then(function () {
- _this8._emitter.emit(_transmuxingEvents2.default.DEMUX_ERROR, type, info);
- });
- }
- }, {
- key: '_onRecommendSeekpoint',
- value: function _onRecommendSeekpoint(milliseconds) {
- var _this9 = this;
- Promise.resolve().then(function () {
- _this9._emitter.emit(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, milliseconds);
- });
- }
- }, {
- key: '_onLoggingConfigChanged',
- value: function _onLoggingConfigChanged(config) {
- if (this._worker) {
- this._worker.postMessage({ cmd: 'logging_config', param: config });
- }
- }
- }, {
- key: '_onWorkerMessage',
- value: function _onWorkerMessage(e) {
- var message = e.data;
- var data = message.data;
- if (message.msg === 'destroyed' || this._workerDestroying) {
- this._workerDestroying = false;
- this._worker.terminate();
- this._worker = null;
- return;
- }
- switch (message.msg) {
- case _transmuxingEvents2.default.INIT_SEGMENT:
- case _transmuxingEvents2.default.MEDIA_SEGMENT:
- this._emitter.emit(message.msg, data.type, data.data);
- break;
- case _transmuxingEvents2.default.LOADING_COMPLETE:
- case _transmuxingEvents2.default.RECOVERED_EARLY_EOF:
- this._emitter.emit(message.msg);
- break;
- case _transmuxingEvents2.default.MEDIA_INFO:
- Object.setPrototypeOf(data, _mediaInfo2.default.prototype);
- this._emitter.emit(message.msg, data);
- break;
- case _transmuxingEvents2.default.STATISTICS_INFO:
- this._emitter.emit(message.msg, data);
- break;
- case _transmuxingEvents2.default.IO_ERROR:
- case _transmuxingEvents2.default.DEMUX_ERROR:
- this._emitter.emit(message.msg, data.type, data.info);
- break;
- case _transmuxingEvents2.default.RECOMMEND_SEEKPOINT:
- this._emitter.emit(message.msg, data);
- break;
- case 'logcat_callback':
- _logger2.default.emitter.emit('log', data.type, data.logcat);
- break;
- default:
- break;
- }
- }
- }]);
- return Transmuxer;
- }();
- exports.default = Transmuxer;
- },{"../utils/logger.js":41,"../utils/logging-control.js":42,"./media-info.js":7,"./transmuxing-controller.js":12,"./transmuxing-events.js":13,"./transmuxing-worker.js":14,"events":2,"webworkify":4}],12:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _browser = _dereq_('../utils/browser.js');
- var _browser2 = _interopRequireDefault(_browser);
- var _mediaInfo = _dereq_('./media-info.js');
- var _mediaInfo2 = _interopRequireDefault(_mediaInfo);
- var _flvDemuxer = _dereq_('../demux/flv-demuxer.js');
- var _flvDemuxer2 = _interopRequireDefault(_flvDemuxer);
- var _mp4Remuxer = _dereq_('../remux/mp4-remuxer.js');
- var _mp4Remuxer2 = _interopRequireDefault(_mp4Remuxer);
- var _demuxErrors = _dereq_('../demux/demux-errors.js');
- var _demuxErrors2 = _interopRequireDefault(_demuxErrors);
- var _ioController = _dereq_('../io/io-controller.js');
- var _ioController2 = _interopRequireDefault(_ioController);
- var _transmuxingEvents = _dereq_('./transmuxing-events.js');
- var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents);
- var _loader = _dereq_('../io/loader.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- // Transmuxing (IO, Demuxing, Remuxing) controller, with multipart support
- var TransmuxingController = function () {
- function TransmuxingController(mediaDataSource, config) {
- _classCallCheck(this, TransmuxingController);
- this.TAG = 'TransmuxingController';
- this._emitter = new _events2.default();
- this._config = config;
- // treat single part media as multipart media, which has only one segment
- if (!mediaDataSource.segments) {
- mediaDataSource.segments = [{
- duration: mediaDataSource.duration,
- filesize: mediaDataSource.filesize,
- url: mediaDataSource.url
- }];
- }
- // fill in default IO params if not exists
- if (typeof mediaDataSource.cors !== 'boolean') {
- mediaDataSource.cors = true;
- }
- if (typeof mediaDataSource.withCredentials !== 'boolean') {
- mediaDataSource.withCredentials = false;
- }
- this._mediaDataSource = mediaDataSource;
- this._currentSegmentIndex = 0;
- var totalDuration = 0;
- this._mediaDataSource.segments.forEach(function (segment) {
- // timestampBase for each segment, and calculate total duration
- segment.timestampBase = totalDuration;
- totalDuration += segment.duration;
- // params needed by IOController
- segment.cors = mediaDataSource.cors;
- segment.withCredentials = mediaDataSource.withCredentials;
- // referrer policy control, if exist
- if (config.referrerPolicy) {
- segment.referrerPolicy = config.referrerPolicy;
- }
- });
- if (!isNaN(totalDuration) && this._mediaDataSource.duration !== totalDuration) {
- this._mediaDataSource.duration = totalDuration;
- }
- this._mediaInfo = null;
- this._demuxer = null;
- this._remuxer = null;
- this._ioctl = null;
- this._pendingSeekTime = null;
- this._pendingResolveSeekPoint = null;
- this._statisticsReporter = null;
- }
- _createClass(TransmuxingController, [{
- key: 'destroy',
- value: function destroy() {
- this._mediaInfo = null;
- this._mediaDataSource = null;
- if (this._statisticsReporter) {
- this._disableStatisticsReporter();
- }
- if (this._ioctl) {
- this._ioctl.destroy();
- this._ioctl = null;
- }
- if (this._demuxer) {
- this._demuxer.destroy();
- this._demuxer = null;
- }
- if (this._remuxer) {
- this._remuxer.destroy();
- this._remuxer = null;
- }
- this._emitter.removeAllListeners();
- this._emitter = null;
- }
- }, {
- key: 'on',
- value: function on(event, listener) {
- this._emitter.addListener(event, listener);
- }
- }, {
- key: 'off',
- value: function off(event, listener) {
- this._emitter.removeListener(event, listener);
- }
- }, {
- key: 'start',
- value: function start() {
- this._loadSegment(0);
- this._enableStatisticsReporter();
- }
- }, {
- key: '_loadSegment',
- value: function _loadSegment(segmentIndex, optionalFrom) {
- this._currentSegmentIndex = segmentIndex;
- var dataSource = this._mediaDataSource.segments[segmentIndex];
- var ioctl = this._ioctl = new _ioController2.default(dataSource, this._config, segmentIndex);
- ioctl.onError = this._onIOException.bind(this);
- ioctl.onSeeked = this._onIOSeeked.bind(this);
- ioctl.onComplete = this._onIOComplete.bind(this);
- ioctl.onRedirect = this._onIORedirect.bind(this);
- ioctl.onRecoveredEarlyEof = this._onIORecoveredEarlyEof.bind(this);
- if (optionalFrom) {
- this._demuxer.bindDataSource(this._ioctl);
- } else {
- ioctl.onDataArrival = this._onInitChunkArrival.bind(this);
- }
- ioctl.open(optionalFrom);
- }
- }, {
- key: 'stop',
- value: function stop() {
- this._internalAbort();
- this._disableStatisticsReporter();
- }
- }, {
- key: '_internalAbort',
- value: function _internalAbort() {
- if (this._ioctl) {
- this._ioctl.destroy();
- this._ioctl = null;
- }
- }
- }, {
- key: 'pause',
- value: function pause() {
- // take a rest
- if (this._ioctl && this._ioctl.isWorking()) {
- this._ioctl.pause();
- this._disableStatisticsReporter();
- }
- }
- }, {
- key: 'resume',
- value: function resume() {
- if (this._ioctl && this._ioctl.isPaused()) {
- this._ioctl.resume();
- this._enableStatisticsReporter();
- }
- }
- }, {
- key: 'seek',
- value: function seek(milliseconds) {
- if (this._mediaInfo == null || !this._mediaInfo.isSeekable()) {
- return;
- }
- var targetSegmentIndex = this._searchSegmentIndexContains(milliseconds);
- if (targetSegmentIndex === this._currentSegmentIndex) {
- // intra-segment seeking
- var segmentInfo = this._mediaInfo.segments[targetSegmentIndex];
- if (segmentInfo == undefined) {
- // current segment loading started, but mediainfo hasn't received yet
- // wait for the metadata loaded, then seek to expected position
- this._pendingSeekTime = milliseconds;
- } else {
- var keyframe = segmentInfo.getNearestKeyframe(milliseconds);
- this._remuxer.seek(keyframe.milliseconds);
- this._ioctl.seek(keyframe.fileposition);
- // Will be resolved in _onRemuxerMediaSegmentArrival()
- this._pendingResolveSeekPoint = keyframe.milliseconds;
- }
- } else {
- // cross-segment seeking
- var targetSegmentInfo = this._mediaInfo.segments[targetSegmentIndex];
- if (targetSegmentInfo == undefined) {
- // target segment hasn't been loaded. We need metadata then seek to expected time
- this._pendingSeekTime = milliseconds;
- this._internalAbort();
- this._remuxer.seek();
- this._remuxer.insertDiscontinuity();
- this._loadSegment(targetSegmentIndex);
- // Here we wait for the metadata loaded, then seek to expected position
- } else {
- // We have target segment's metadata, direct seek to target position
- var _keyframe = targetSegmentInfo.getNearestKeyframe(milliseconds);
- this._internalAbort();
- this._remuxer.seek(milliseconds);
- this._remuxer.insertDiscontinuity();
- this._demuxer.resetMediaInfo();
- this._demuxer.timestampBase = this._mediaDataSource.segments[targetSegmentIndex].timestampBase;
- this._loadSegment(targetSegmentIndex, _keyframe.fileposition);
- this._pendingResolveSeekPoint = _keyframe.milliseconds;
- this._reportSegmentMediaInfo(targetSegmentIndex);
- }
- }
- this._enableStatisticsReporter();
- }
- }, {
- key: '_searchSegmentIndexContains',
- value: function _searchSegmentIndexContains(milliseconds) {
- var segments = this._mediaDataSource.segments;
- var idx = segments.length - 1;
- for (var i = 0; i < segments.length; i++) {
- if (milliseconds < segments[i].timestampBase) {
- idx = i - 1;
- break;
- }
- }
- return idx;
- }
- }, {
- key: '_onInitChunkArrival',
- value: function _onInitChunkArrival(data, byteStart) {
- var _this = this;
- var probeData = null;
- var consumed = 0;
- if (byteStart > 0) {
- // IOController seeked immediately after opened, byteStart > 0 callback may received
- this._demuxer.bindDataSource(this._ioctl);
- this._demuxer.timestampBase = this._mediaDataSource.segments[this._currentSegmentIndex].timestampBase;
- consumed = this._demuxer.parseChunks(data, byteStart);
- } else if ((probeData = _flvDemuxer2.default.probe(data)).match) {
- // Always create new FLVDemuxer
- this._demuxer = new _flvDemuxer2.default(probeData, this._config);
- if (!this._remuxer) {
- this._remuxer = new _mp4Remuxer2.default(this._config);
- }
- var mds = this._mediaDataSource;
- if (mds.duration != undefined && !isNaN(mds.duration)) {
- this._demuxer.overridedDuration = mds.duration;
- }
- if (typeof mds.hasAudio === 'boolean') {
- this._demuxer.overridedHasAudio = mds.hasAudio;
- }
- if (typeof mds.hasVideo === 'boolean') {
- this._demuxer.overridedHasVideo = mds.hasVideo;
- }
- this._demuxer.timestampBase = mds.segments[this._currentSegmentIndex].timestampBase;
- this._demuxer.onError = this._onDemuxException.bind(this);
- this._demuxer.onMediaInfo = this._onMediaInfo.bind(this);
- this._remuxer.bindDataSource(this._demuxer.bindDataSource(this._ioctl));
- this._remuxer.onInitSegment = this._onRemuxerInitSegmentArrival.bind(this);
- this._remuxer.onMediaSegment = this._onRemuxerMediaSegmentArrival.bind(this);
- consumed = this._demuxer.parseChunks(data, byteStart);
- } else {
- probeData = null;
- _logger2.default.e(this.TAG, 'Non-FLV, Unsupported media type!');
- Promise.resolve().then(function () {
- _this._internalAbort();
- });
- this._emitter.emit(_transmuxingEvents2.default.DEMUX_ERROR, _demuxErrors2.default.FORMAT_UNSUPPORTED, 'Non-FLV, Unsupported media type');
- consumed = 0;
- }
- return consumed;
- }
- }, {
- key: '_onMediaInfo',
- value: function _onMediaInfo(mediaInfo) {
- var _this2 = this;
- if (this._mediaInfo == null) {
- // Store first segment's mediainfo as global mediaInfo
- this._mediaInfo = Object.assign({}, mediaInfo);
- this._mediaInfo.keyframesIndex = null;
- this._mediaInfo.segments = [];
- this._mediaInfo.segmentCount = this._mediaDataSource.segments.length;
- Object.setPrototypeOf(this._mediaInfo, _mediaInfo2.default.prototype);
- }
- var segmentInfo = Object.assign({}, mediaInfo);
- Object.setPrototypeOf(segmentInfo, _mediaInfo2.default.prototype);
- this._mediaInfo.segments[this._currentSegmentIndex] = segmentInfo;
- // notify mediaInfo update
- this._reportSegmentMediaInfo(this._currentSegmentIndex);
- if (this._pendingSeekTime != null) {
- Promise.resolve().then(function () {
- var target = _this2._pendingSeekTime;
- _this2._pendingSeekTime = null;
- _this2.seek(target);
- });
- }
- }
- }, {
- key: '_onIOSeeked',
- value: function _onIOSeeked() {
- this._remuxer.insertDiscontinuity();
- }
- }, {
- key: '_onIOComplete',
- value: function _onIOComplete(extraData) {
- var segmentIndex = extraData;
- var nextSegmentIndex = segmentIndex + 1;
- if (nextSegmentIndex < this._mediaDataSource.segments.length) {
- this._internalAbort();
- this._loadSegment(nextSegmentIndex);
- } else {
- this._remuxer.flushStashedSamples();
- this._emitter.emit(_transmuxingEvents2.default.LOADING_COMPLETE);
- this._disableStatisticsReporter();
- }
- }
- }, {
- key: '_onIORedirect',
- value: function _onIORedirect(redirectedURL) {
- var segmentIndex = this._ioctl.extraData;
- this._mediaDataSource.segments[segmentIndex].redirectedURL = redirectedURL;
- }
- }, {
- key: '_onIORecoveredEarlyEof',
- value: function _onIORecoveredEarlyEof() {
- this._emitter.emit(_transmuxingEvents2.default.RECOVERED_EARLY_EOF);
- }
- }, {
- key: '_onIOException',
- value: function _onIOException(type, info) {
- _logger2.default.e(this.TAG, 'IOException: type = ' + type + ', code = ' + info.code + ', msg = ' + info.msg);
- this._emitter.emit(_transmuxingEvents2.default.IO_ERROR, type, info);
- this._disableStatisticsReporter();
- }
- }, {
- key: '_onDemuxException',
- value: function _onDemuxException(type, info) {
- _logger2.default.e(this.TAG, 'DemuxException: type = ' + type + ', info = ' + info);
- this._emitter.emit(_transmuxingEvents2.default.DEMUX_ERROR, type, info);
- }
- }, {
- key: '_onRemuxerInitSegmentArrival',
- value: function _onRemuxerInitSegmentArrival(type, initSegment) {
- this._emitter.emit(_transmuxingEvents2.default.INIT_SEGMENT, type, initSegment);
- }
- }, {
- key: '_onRemuxerMediaSegmentArrival',
- value: function _onRemuxerMediaSegmentArrival(type, mediaSegment) {
- if (this._pendingSeekTime != null) {
- // Media segments after new-segment cross-seeking should be dropped.
- return;
- }
- this._emitter.emit(_transmuxingEvents2.default.MEDIA_SEGMENT, type, mediaSegment);
- // Resolve pending seekPoint
- if (this._pendingResolveSeekPoint != null && type === 'video') {
- var syncPoints = mediaSegment.info.syncPoints;
- var seekpoint = this._pendingResolveSeekPoint;
- this._pendingResolveSeekPoint = null;
- // Safari: Pass PTS for recommend_seekpoint
- if (_browser2.default.safari && syncPoints.length > 0 && syncPoints[0].originalDts === seekpoint) {
- seekpoint = syncPoints[0].pts;
- }
- // else: use original DTS (keyframe.milliseconds)
- this._emitter.emit(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, seekpoint);
- }
- }
- }, {
- key: '_enableStatisticsReporter',
- value: function _enableStatisticsReporter() {
- if (this._statisticsReporter == null) {
- this._statisticsReporter = self.setInterval(this._reportStatisticsInfo.bind(this), this._config.statisticsInfoReportInterval);
- }
- }
- }, {
- key: '_disableStatisticsReporter',
- value: function _disableStatisticsReporter() {
- if (this._statisticsReporter) {
- self.clearInterval(this._statisticsReporter);
- this._statisticsReporter = null;
- }
- }
- }, {
- key: '_reportSegmentMediaInfo',
- value: function _reportSegmentMediaInfo(segmentIndex) {
- var segmentInfo = this._mediaInfo.segments[segmentIndex];
- var exportInfo = Object.assign({}, segmentInfo);
- exportInfo.duration = this._mediaInfo.duration;
- exportInfo.segmentCount = this._mediaInfo.segmentCount;
- delete exportInfo.segments;
- delete exportInfo.keyframesIndex;
- this._emitter.emit(_transmuxingEvents2.default.MEDIA_INFO, exportInfo);
- }
- }, {
- key: '_reportStatisticsInfo',
- value: function _reportStatisticsInfo() {
- var info = {};
- info.url = this._ioctl.currentURL;
- info.hasRedirect = this._ioctl.hasRedirect;
- if (info.hasRedirect) {
- info.redirectedURL = this._ioctl.currentRedirectedURL;
- }
- info.speed = this._ioctl.currentSpeed;
- info.loaderType = this._ioctl.loaderType;
- info.currentSegmentIndex = this._currentSegmentIndex;
- info.totalSegmentCount = this._mediaDataSource.segments.length;
- this._emitter.emit(_transmuxingEvents2.default.STATISTICS_INFO, info);
- }
- }]);
- return TransmuxingController;
- }();
- exports.default = TransmuxingController;
- },{"../demux/demux-errors.js":16,"../demux/flv-demuxer.js":18,"../io/io-controller.js":23,"../io/loader.js":24,"../remux/mp4-remuxer.js":38,"../utils/browser.js":39,"../utils/logger.js":41,"./media-info.js":7,"./transmuxing-events.js":13,"events":2}],13:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var TransmuxingEvents = {
- IO_ERROR: 'io_error',
- DEMUX_ERROR: 'demux_error',
- INIT_SEGMENT: 'init_segment',
- MEDIA_SEGMENT: 'media_segment',
- LOADING_COMPLETE: 'loading_complete',
- RECOVERED_EARLY_EOF: 'recovered_early_eof',
- MEDIA_INFO: 'media_info',
- STATISTICS_INFO: 'statistics_info',
- RECOMMEND_SEEKPOINT: 'recommend_seekpoint'
- };
- exports.default = TransmuxingEvents;
- },{}],14:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _loggingControl = _dereq_('../utils/logging-control.js');
- var _loggingControl2 = _interopRequireDefault(_loggingControl);
- var _polyfill = _dereq_('../utils/polyfill.js');
- var _polyfill2 = _interopRequireDefault(_polyfill);
- var _transmuxingController = _dereq_('./transmuxing-controller.js');
- var _transmuxingController2 = _interopRequireDefault(_transmuxingController);
- var _transmuxingEvents = _dereq_('./transmuxing-events.js');
- var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents);
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- /* post message to worker:
- data: {
- cmd: string
- param: any
- }
- receive message from worker:
- data: {
- msg: string,
- data: any
- }
- */
- var TransmuxingWorker = function TransmuxingWorker(self) {
- var TAG = 'TransmuxingWorker';
- var controller = null;
- var logcatListener = onLogcatCallback.bind(this);
- _polyfill2.default.install();
- self.addEventListener('message', function (e) {
- switch (e.data.cmd) {
- case 'init':
- controller = new _transmuxingController2.default(e.data.param[0], e.data.param[1]);
- controller.on(_transmuxingEvents2.default.IO_ERROR, onIOError.bind(this));
- controller.on(_transmuxingEvents2.default.DEMUX_ERROR, onDemuxError.bind(this));
- controller.on(_transmuxingEvents2.default.INIT_SEGMENT, onInitSegment.bind(this));
- controller.on(_transmuxingEvents2.default.MEDIA_SEGMENT, onMediaSegment.bind(this));
- controller.on(_transmuxingEvents2.default.LOADING_COMPLETE, onLoadingComplete.bind(this));
- controller.on(_transmuxingEvents2.default.RECOVERED_EARLY_EOF, onRecoveredEarlyEof.bind(this));
- controller.on(_transmuxingEvents2.default.MEDIA_INFO, onMediaInfo.bind(this));
- controller.on(_transmuxingEvents2.default.STATISTICS_INFO, onStatisticsInfo.bind(this));
- controller.on(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, onRecommendSeekpoint.bind(this));
- break;
- case 'destroy':
- if (controller) {
- controller.destroy();
- controller = null;
- }
- self.postMessage({ msg: 'destroyed' });
- break;
- case 'start':
- controller.start();
- break;
- case 'stop':
- controller.stop();
- break;
- case 'seek':
- controller.seek(e.data.param);
- break;
- case 'pause':
- controller.pause();
- break;
- case 'resume':
- controller.resume();
- break;
- case 'logging_config':
- {
- var config = e.data.param;
- _loggingControl2.default.applyConfig(config);
- if (config.enableCallback === true) {
- _loggingControl2.default.addLogListener(logcatListener);
- } else {
- _loggingControl2.default.removeLogListener(logcatListener);
- }
- break;
- }
- }
- });
- function onInitSegment(type, initSegment) {
- var obj = {
- msg: _transmuxingEvents2.default.INIT_SEGMENT,
- data: {
- type: type,
- data: initSegment
- }
- };
- self.postMessage(obj, [initSegment.data]); // data: ArrayBuffer
- }
- function onMediaSegment(type, mediaSegment) {
- var obj = {
- msg: _transmuxingEvents2.default.MEDIA_SEGMENT,
- data: {
- type: type,
- data: mediaSegment
- }
- };
- self.postMessage(obj, [mediaSegment.data]); // data: ArrayBuffer
- }
- function onLoadingComplete() {
- var obj = {
- msg: _transmuxingEvents2.default.LOADING_COMPLETE
- };
- self.postMessage(obj);
- }
- function onRecoveredEarlyEof() {
- var obj = {
- msg: _transmuxingEvents2.default.RECOVERED_EARLY_EOF
- };
- self.postMessage(obj);
- }
- function onMediaInfo(mediaInfo) {
- var obj = {
- msg: _transmuxingEvents2.default.MEDIA_INFO,
- data: mediaInfo
- };
- self.postMessage(obj);
- }
- function onStatisticsInfo(statInfo) {
- var obj = {
- msg: _transmuxingEvents2.default.STATISTICS_INFO,
- data: statInfo
- };
- self.postMessage(obj);
- }
- function onIOError(type, info) {
- self.postMessage({
- msg: _transmuxingEvents2.default.IO_ERROR,
- data: {
- type: type,
- info: info
- }
- });
- }
- function onDemuxError(type, info) {
- self.postMessage({
- msg: _transmuxingEvents2.default.DEMUX_ERROR,
- data: {
- type: type,
- info: info
- }
- });
- }
- function onRecommendSeekpoint(milliseconds) {
- self.postMessage({
- msg: _transmuxingEvents2.default.RECOMMEND_SEEKPOINT,
- data: milliseconds
- });
- }
- function onLogcatCallback(type, str) {
- self.postMessage({
- msg: 'logcat_callback',
- data: {
- type: type,
- logcat: str
- }
- });
- }
- }; /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- exports.default = TransmuxingWorker;
- },{"../utils/logger.js":41,"../utils/logging-control.js":42,"../utils/polyfill.js":43,"./transmuxing-controller.js":12,"./transmuxing-events.js":13}],15:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _utf8Conv = _dereq_('../utils/utf8-conv.js');
- var _utf8Conv2 = _interopRequireDefault(_utf8Conv);
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var le = function () {
- var buf = new ArrayBuffer(2);
- new DataView(buf).setInt16(0, 256, true); // little-endian write
- return new Int16Array(buf)[0] === 256; // platform-spec read, if equal then LE
- }();
- var AMF = function () {
- function AMF() {
- _classCallCheck(this, AMF);
- }
- _createClass(AMF, null, [{
- key: 'parseScriptData',
- value: function parseScriptData(arrayBuffer, dataOffset, dataSize) {
- var data = {};
- try {
- var name = AMF.parseValue(arrayBuffer, dataOffset, dataSize);
- var value = AMF.parseValue(arrayBuffer, dataOffset + name.size, dataSize - name.size);
- data[name.data] = value.data;
- } catch (e) {
- _logger2.default.e('AMF', e.toString());
- }
- return data;
- }
- }, {
- key: 'parseObject',
- value: function parseObject(arrayBuffer, dataOffset, dataSize) {
- if (dataSize < 3) {
- throw new _exception.IllegalStateException('Data not enough when parse ScriptDataObject');
- }
- var name = AMF.parseString(arrayBuffer, dataOffset, dataSize);
- var value = AMF.parseValue(arrayBuffer, dataOffset + name.size, dataSize - name.size);
- var isObjectEnd = value.objectEnd;
- return {
- data: {
- name: name.data,
- value: value.data
- },
- size: name.size + value.size,
- objectEnd: isObjectEnd
- };
- }
- }, {
- key: 'parseVariable',
- value: function parseVariable(arrayBuffer, dataOffset, dataSize) {
- return AMF.parseObject(arrayBuffer, dataOffset, dataSize);
- }
- }, {
- key: 'parseString',
- value: function parseString(arrayBuffer, dataOffset, dataSize) {
- if (dataSize < 2) {
- throw new _exception.IllegalStateException('Data not enough when parse String');
- }
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var length = v.getUint16(0, !le);
- var str = void 0;
- if (length > 0) {
- str = (0, _utf8Conv2.default)(new Uint8Array(arrayBuffer, dataOffset + 2, length));
- } else {
- str = '';
- }
- return {
- data: str,
- size: 2 + length
- };
- }
- }, {
- key: 'parseLongString',
- value: function parseLongString(arrayBuffer, dataOffset, dataSize) {
- if (dataSize < 4) {
- throw new _exception.IllegalStateException('Data not enough when parse LongString');
- }
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var length = v.getUint32(0, !le);
- var str = void 0;
- if (length > 0) {
- str = (0, _utf8Conv2.default)(new Uint8Array(arrayBuffer, dataOffset + 4, length));
- } else {
- str = '';
- }
- return {
- data: str,
- size: 4 + length
- };
- }
- }, {
- key: 'parseDate',
- value: function parseDate(arrayBuffer, dataOffset, dataSize) {
- if (dataSize < 10) {
- throw new _exception.IllegalStateException('Data size invalid when parse Date');
- }
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var timestamp = v.getFloat64(0, !le);
- var localTimeOffset = v.getInt16(8, !le);
- timestamp += localTimeOffset * 60 * 1000; // get UTC time
- return {
- data: new Date(timestamp),
- size: 8 + 2
- };
- }
- }, {
- key: 'parseValue',
- value: function parseValue(arrayBuffer, dataOffset, dataSize) {
- if (dataSize < 1) {
- throw new _exception.IllegalStateException('Data not enough when parse Value');
- }
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var offset = 1;
- var type = v.getUint8(0);
- var value = void 0;
- var objectEnd = false;
- try {
- switch (type) {
- case 0:
- // Number(Double) type
- value = v.getFloat64(1, !le);
- offset += 8;
- break;
- case 1:
- {
- // Boolean type
- var b = v.getUint8(1);
- value = b ? true : false;
- offset += 1;
- break;
- }
- case 2:
- {
- // String type
- var amfstr = AMF.parseString(arrayBuffer, dataOffset + 1, dataSize - 1);
- value = amfstr.data;
- offset += amfstr.size;
- break;
- }
- case 3:
- {
- // Object(s) type
- value = {};
- var terminal = 0; // workaround for malformed Objects which has missing ScriptDataObjectEnd
- if ((v.getUint32(dataSize - 4, !le) & 0x00FFFFFF) === 9) {
- terminal = 3;
- }
- while (offset < dataSize - 4) {
- // 4 === type(UI8) + ScriptDataObjectEnd(UI24)
- var amfobj = AMF.parseObject(arrayBuffer, dataOffset + offset, dataSize - offset - terminal);
- if (amfobj.objectEnd) break;
- value[amfobj.data.name] = amfobj.data.value;
- offset += amfobj.size;
- }
- if (offset <= dataSize - 3) {
- var marker = v.getUint32(offset - 1, !le) & 0x00FFFFFF;
- if (marker === 9) {
- offset += 3;
- }
- }
- break;
- }
- case 8:
- {
- // ECMA array type (Mixed array)
- value = {};
- offset += 4; // ECMAArrayLength(UI32)
- var _terminal = 0; // workaround for malformed MixedArrays which has missing ScriptDataObjectEnd
- if ((v.getUint32(dataSize - 4, !le) & 0x00FFFFFF) === 9) {
- _terminal = 3;
- }
- while (offset < dataSize - 8) {
- // 8 === type(UI8) + ECMAArrayLength(UI32) + ScriptDataVariableEnd(UI24)
- var amfvar = AMF.parseVariable(arrayBuffer, dataOffset + offset, dataSize - offset - _terminal);
- if (amfvar.objectEnd) break;
- value[amfvar.data.name] = amfvar.data.value;
- offset += amfvar.size;
- }
- if (offset <= dataSize - 3) {
- var _marker = v.getUint32(offset - 1, !le) & 0x00FFFFFF;
- if (_marker === 9) {
- offset += 3;
- }
- }
- break;
- }
- case 9:
- // ScriptDataObjectEnd
- value = undefined;
- offset = 1;
- objectEnd = true;
- break;
- case 10:
- {
- // Strict array type
- // ScriptDataValue[n]. NOTE: according to video_file_format_spec_v10_1.pdf
- value = [];
- var strictArrayLength = v.getUint32(1, !le);
- offset += 4;
- for (var i = 0; i < strictArrayLength; i++) {
- var val = AMF.parseValue(arrayBuffer, dataOffset + offset, dataSize - offset);
- value.push(val.data);
- offset += val.size;
- }
- break;
- }
- case 11:
- {
- // Date type
- var date = AMF.parseDate(arrayBuffer, dataOffset + 1, dataSize - 1);
- value = date.data;
- offset += date.size;
- break;
- }
- case 12:
- {
- // Long string type
- var amfLongStr = AMF.parseString(arrayBuffer, dataOffset + 1, dataSize - 1);
- value = amfLongStr.data;
- offset += amfLongStr.size;
- break;
- }
- default:
- // ignore and skip
- offset = dataSize;
- _logger2.default.w('AMF', 'Unsupported AMF value type ' + type);
- }
- } catch (e) {
- _logger2.default.e('AMF', e.toString());
- }
- return {
- data: value,
- size: offset,
- objectEnd: objectEnd
- };
- }
- }]);
- return AMF;
- }();
- exports.default = AMF;
- },{"../utils/exception.js":40,"../utils/logger.js":41,"../utils/utf8-conv.js":44}],16:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var DemuxErrors = {
- OK: 'OK',
- FORMAT_ERROR: 'FormatError',
- FORMAT_UNSUPPORTED: 'FormatUnsupported',
- CODEC_UNSUPPORTED: 'CodecUnsupported'
- };
- exports.default = DemuxErrors;
- },{}],17:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _exception = _dereq_('../utils/exception.js');
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- // Exponential-Golomb buffer decoder
- var ExpGolomb = function () {
- function ExpGolomb(uint8array) {
- _classCallCheck(this, ExpGolomb);
- this.TAG = 'ExpGolomb';
- this._buffer = uint8array;
- this._buffer_index = 0;
- this._total_bytes = uint8array.byteLength;
- this._total_bits = uint8array.byteLength * 8;
- this._current_word = 0;
- this._current_word_bits_left = 0;
- }
- _createClass(ExpGolomb, [{
- key: 'destroy',
- value: function destroy() {
- this._buffer = null;
- }
- }, {
- key: '_fillCurrentWord',
- value: function _fillCurrentWord() {
- var buffer_bytes_left = this._total_bytes - this._buffer_index;
- if (buffer_bytes_left <= 0) throw new _exception.IllegalStateException('ExpGolomb: _fillCurrentWord() but no bytes available');
- var bytes_read = Math.min(4, buffer_bytes_left);
- var word = new Uint8Array(4);
- word.set(this._buffer.subarray(this._buffer_index, this._buffer_index + bytes_read));
- this._current_word = new DataView(word.buffer).getUint32(0, false);
- this._buffer_index += bytes_read;
- this._current_word_bits_left = bytes_read * 8;
- }
- }, {
- key: 'readBits',
- value: function readBits(bits) {
- if (bits > 32) throw new _exception.InvalidArgumentException('ExpGolomb: readBits() bits exceeded max 32bits!');
- if (bits <= this._current_word_bits_left) {
- var _result = this._current_word >>> 32 - bits;
- this._current_word <<= bits;
- this._current_word_bits_left -= bits;
- return _result;
- }
- var result = this._current_word_bits_left ? this._current_word : 0;
- result = result >>> 32 - this._current_word_bits_left;
- var bits_need_left = bits - this._current_word_bits_left;
- this._fillCurrentWord();
- var bits_read_next = Math.min(bits_need_left, this._current_word_bits_left);
- var result2 = this._current_word >>> 32 - bits_read_next;
- this._current_word <<= bits_read_next;
- this._current_word_bits_left -= bits_read_next;
- result = result << bits_read_next | result2;
- return result;
- }
- }, {
- key: 'readBool',
- value: function readBool() {
- return this.readBits(1) === 1;
- }
- }, {
- key: 'readByte',
- value: function readByte() {
- return this.readBits(8);
- }
- }, {
- key: '_skipLeadingZero',
- value: function _skipLeadingZero() {
- var zero_count = void 0;
- for (zero_count = 0; zero_count < this._current_word_bits_left; zero_count++) {
- if (0 !== (this._current_word & 0x80000000 >>> zero_count)) {
- this._current_word <<= zero_count;
- this._current_word_bits_left -= zero_count;
- return zero_count;
- }
- }
- this._fillCurrentWord();
- return zero_count + this._skipLeadingZero();
- }
- }, {
- key: 'readUEG',
- value: function readUEG() {
- // unsigned exponential golomb
- var leading_zeros = this._skipLeadingZero();
- return this.readBits(leading_zeros + 1) - 1;
- }
- }, {
- key: 'readSEG',
- value: function readSEG() {
- // signed exponential golomb
- var value = this.readUEG();
- if (value & 0x01) {
- return value + 1 >>> 1;
- } else {
- return -1 * (value >>> 1);
- }
- }
- }]);
- return ExpGolomb;
- }();
- exports.default = ExpGolomb;
- },{"../utils/exception.js":40}],18:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _amfParser = _dereq_('./amf-parser.js');
- var _amfParser2 = _interopRequireDefault(_amfParser);
- var _spsParser = _dereq_('./sps-parser.js');
- var _spsParser2 = _interopRequireDefault(_spsParser);
- var _demuxErrors = _dereq_('./demux-errors.js');
- var _demuxErrors2 = _interopRequireDefault(_demuxErrors);
- var _mediaInfo = _dereq_('../core/media-info.js');
- var _mediaInfo2 = _interopRequireDefault(_mediaInfo);
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- function Swap16(src) {
- return src >>> 8 & 0xFF | (src & 0xFF) << 8;
- }
- function Swap32(src) {
- return (src & 0xFF000000) >>> 24 | (src & 0x00FF0000) >>> 8 | (src & 0x0000FF00) << 8 | (src & 0x000000FF) << 24;
- }
- function ReadBig32(array, index) {
- return array[index] << 24 | array[index + 1] << 16 | array[index + 2] << 8 | array[index + 3];
- }
- var FLVDemuxer = function () {
- function FLVDemuxer(probeData, config) {
- _classCallCheck(this, FLVDemuxer);
- this.TAG = 'FLVDemuxer';
- this._config = config;
- this._onError = null;
- this._onMediaInfo = null;
- this._onTrackMetadata = null;
- this._onDataAvailable = null;
- this._dataOffset = probeData.dataOffset;
- this._firstParse = true;
- this._dispatch = false;
- this._hasAudio = probeData.hasAudioTrack;
- this._hasVideo = probeData.hasVideoTrack;
- this._hasAudioFlagOverrided = false;
- this._hasVideoFlagOverrided = false;
- this._audioInitialMetadataDispatched = false;
- this._videoInitialMetadataDispatched = false;
- this._mediaInfo = new _mediaInfo2.default();
- this._mediaInfo.hasAudio = this._hasAudio;
- this._mediaInfo.hasVideo = this._hasVideo;
- this._metadata = null;
- this._audioMetadata = null;
- this._videoMetadata = null;
- this._naluLengthSize = 4;
- this._timestampBase = 0; // int32, in milliseconds
- this._timescale = 1000;
- this._duration = 0; // int32, in milliseconds
- this._durationOverrided = false;
- this._referenceFrameRate = {
- fixed: true,
- fps: 23.976,
- fps_num: 23976,
- fps_den: 1000
- };
- this._flvSoundRateTable = [5500, 11025, 22050, 44100, 48000];
- this._mpegSamplingRates = [96000, 88200, 64000, 48000, 44100, 32000, 24000, 22050, 16000, 12000, 11025, 8000, 7350];
- this._mpegAudioV10SampleRateTable = [44100, 48000, 32000, 0];
- this._mpegAudioV20SampleRateTable = [22050, 24000, 16000, 0];
- this._mpegAudioV25SampleRateTable = [11025, 12000, 8000, 0];
- this._mpegAudioL1BitRateTable = [0, 32, 64, 96, 128, 160, 192, 224, 256, 288, 320, 352, 384, 416, 448, -1];
- this._mpegAudioL2BitRateTable = [0, 32, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 384, -1];
- this._mpegAudioL3BitRateTable = [0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, -1];
- this._videoTrack = { type: 'video', id: 1, sequenceNumber: 0, samples: [], length: 0 };
- this._audioTrack = { type: 'audio', id: 2, sequenceNumber: 0, samples: [], length: 0 };
- this._littleEndian = function () {
- var buf = new ArrayBuffer(2);
- new DataView(buf).setInt16(0, 256, true); // little-endian write
- return new Int16Array(buf)[0] === 256; // platform-spec read, if equal then LE
- }();
- }
- _createClass(FLVDemuxer, [{
- key: 'destroy',
- value: function destroy() {
- this._mediaInfo = null;
- this._metadata = null;
- this._audioMetadata = null;
- this._videoMetadata = null;
- this._videoTrack = null;
- this._audioTrack = null;
- this._onError = null;
- this._onMediaInfo = null;
- this._onTrackMetadata = null;
- this._onDataAvailable = null;
- }
- }, {
- key: 'bindDataSource',
- value: function bindDataSource(loader) {
- loader.onDataArrival = this.parseChunks.bind(this);
- return this;
- }
- // prototype: function(type: string, metadata: any): void
- }, {
- key: 'resetMediaInfo',
- value: function resetMediaInfo() {
- this._mediaInfo = new _mediaInfo2.default();
- }
- }, {
- key: '_isInitialMetadataDispatched',
- value: function _isInitialMetadataDispatched() {
- if (this._hasAudio && this._hasVideo) {
- // both audio & video
- return this._audioInitialMetadataDispatched && this._videoInitialMetadataDispatched;
- }
- if (this._hasAudio && !this._hasVideo) {
- // audio only
- return this._audioInitialMetadataDispatched;
- }
- if (!this._hasAudio && this._hasVideo) {
- // video only
- return this._videoInitialMetadataDispatched;
- }
- return false;
- }
- // function parseChunks(chunk: ArrayBuffer, byteStart: number): number;
- }, {
- key: 'parseChunks',
- value: function parseChunks(chunk, byteStart) {
- if (!this._onError || !this._onMediaInfo || !this._onTrackMetadata || !this._onDataAvailable) {
- throw new _exception.IllegalStateException('Flv: onError & onMediaInfo & onTrackMetadata & onDataAvailable callback must be specified');
- }
- var offset = 0;
- var le = this._littleEndian;
- if (byteStart === 0) {
- // buffer with FLV header
- if (chunk.byteLength > 13) {
- var probeData = FLVDemuxer.probe(chunk);
- offset = probeData.dataOffset;
- } else {
- return 0;
- }
- }
- if (this._firstParse) {
- // handle PreviousTagSize0 before Tag1
- this._firstParse = false;
- if (byteStart + offset !== this._dataOffset) {
- _logger2.default.w(this.TAG, 'First time parsing but chunk byteStart invalid!');
- }
- var v = new DataView(chunk, offset);
- var prevTagSize0 = v.getUint32(0, !le);
- if (prevTagSize0 !== 0) {
- _logger2.default.w(this.TAG, 'PrevTagSize0 !== 0 !!!');
- }
- offset += 4;
- }
- while (offset < chunk.byteLength) {
- this._dispatch = true;
- var _v = new DataView(chunk, offset);
- if (offset + 11 + 4 > chunk.byteLength) {
- // data not enough for parsing an flv tag
- break;
- }
- var tagType = _v.getUint8(0);
- var dataSize = _v.getUint32(0, !le) & 0x00FFFFFF;
- if (offset + 11 + dataSize + 4 > chunk.byteLength) {
- // data not enough for parsing actual data body
- break;
- }
- if (tagType !== 8 && tagType !== 9 && tagType !== 18) {
- _logger2.default.w(this.TAG, 'Unsupported tag type ' + tagType + ', skipped');
- // consume the whole tag (skip it)
- offset += 11 + dataSize + 4;
- continue;
- }
- var ts2 = _v.getUint8(4);
- var ts1 = _v.getUint8(5);
- var ts0 = _v.getUint8(6);
- var ts3 = _v.getUint8(7);
- var timestamp = ts0 | ts1 << 8 | ts2 << 16 | ts3 << 24;
- var streamId = _v.getUint32(7, !le) & 0x00FFFFFF;
- if (streamId !== 0) {
- _logger2.default.w(this.TAG, 'Meet tag which has StreamID != 0!');
- }
- var dataOffset = offset + 11;
- switch (tagType) {
- case 8:
- // Audio
- this._parseAudioData(chunk, dataOffset, dataSize, timestamp);
- break;
- case 9:
- // Video
- this._parseVideoData(chunk, dataOffset, dataSize, timestamp, byteStart + offset);
- break;
- case 18:
- // ScriptDataObject
- this._parseScriptData(chunk, dataOffset, dataSize);
- break;
- }
- var prevTagSize = _v.getUint32(11 + dataSize, !le);
- if (prevTagSize !== 11 + dataSize) {
- _logger2.default.w(this.TAG, 'Invalid PrevTagSize ' + prevTagSize);
- }
- offset += 11 + dataSize + 4; // tagBody + dataSize + prevTagSize
- }
- // dispatch parsed frames to consumer (typically, the remuxer)
- if (this._isInitialMetadataDispatched()) {
- if (this._dispatch && (this._audioTrack.length || this._videoTrack.length)) {
- this._onDataAvailable(this._audioTrack, this._videoTrack);
- }
- }
- return offset; // consumed bytes, just equals latest offset index
- }
- }, {
- key: '_parseScriptData',
- value: function _parseScriptData(arrayBuffer, dataOffset, dataSize) {
- var scriptData = _amfParser2.default.parseScriptData(arrayBuffer, dataOffset, dataSize);
- if (scriptData.hasOwnProperty('onMetaData')) {
- if (scriptData.onMetaData == null || _typeof(scriptData.onMetaData) !== 'object') {
- _logger2.default.w(this.TAG, 'Invalid onMetaData structure!');
- return;
- }
- if (this._metadata) {
- _logger2.default.w(this.TAG, 'Found another onMetaData tag!');
- }
- this._metadata = scriptData;
- var onMetaData = this._metadata.onMetaData;
- if (typeof onMetaData.hasAudio === 'boolean') {
- // hasAudio
- if (this._hasAudioFlagOverrided === false) {
- this._hasAudio = onMetaData.hasAudio;
- this._mediaInfo.hasAudio = this._hasAudio;
- }
- }
- if (typeof onMetaData.hasVideo === 'boolean') {
- // hasVideo
- if (this._hasVideoFlagOverrided === false) {
- this._hasVideo = onMetaData.hasVideo;
- this._mediaInfo.hasVideo = this._hasVideo;
- }
- }
- if (typeof onMetaData.audiodatarate === 'number') {
- // audiodatarate
- this._mediaInfo.audioDataRate = onMetaData.audiodatarate;
- }
- if (typeof onMetaData.videodatarate === 'number') {
- // videodatarate
- this._mediaInfo.videoDataRate = onMetaData.videodatarate;
- }
- if (typeof onMetaData.width === 'number') {
- // width
- this._mediaInfo.width = onMetaData.width;
- }
- if (typeof onMetaData.height === 'number') {
- // height
- this._mediaInfo.height = onMetaData.height;
- }
- if (typeof onMetaData.duration === 'number') {
- // duration
- if (!this._durationOverrided) {
- var duration = Math.floor(onMetaData.duration * this._timescale);
- this._duration = duration;
- this._mediaInfo.duration = duration;
- }
- } else {
- this._mediaInfo.duration = 0;
- }
- if (typeof onMetaData.framerate === 'number') {
- // framerate
- var fps_num = Math.floor(onMetaData.framerate * 1000);
- if (fps_num > 0) {
- var fps = fps_num / 1000;
- this._referenceFrameRate.fixed = true;
- this._referenceFrameRate.fps = fps;
- this._referenceFrameRate.fps_num = fps_num;
- this._referenceFrameRate.fps_den = 1000;
- this._mediaInfo.fps = fps;
- }
- }
- if (_typeof(onMetaData.keyframes) === 'object') {
- // keyframes
- this._mediaInfo.hasKeyframesIndex = true;
- var keyframes = onMetaData.keyframes;
- this._mediaInfo.keyframesIndex = this._parseKeyframesIndex(keyframes);
- onMetaData.keyframes = null; // keyframes has been extracted, remove it
- } else {
- this._mediaInfo.hasKeyframesIndex = false;
- }
- this._dispatch = false;
- this._mediaInfo.metadata = onMetaData;
- _logger2.default.v(this.TAG, 'Parsed onMetaData');
- if (this._mediaInfo.isComplete()) {
- this._onMediaInfo(this._mediaInfo);
- }
- }
- }
- }, {
- key: '_parseKeyframesIndex',
- value: function _parseKeyframesIndex(keyframes) {
- var times = [];
- var filepositions = [];
- // ignore first keyframe which is actually AVC Sequence Header (AVCDecoderConfigurationRecord)
- for (var i = 1; i < keyframes.times.length; i++) {
- var time = this._timestampBase + Math.floor(keyframes.times[i] * 1000);
- times.push(time);
- filepositions.push(keyframes.filepositions[i]);
- }
- return {
- times: times,
- filepositions: filepositions
- };
- }
- }, {
- key: '_parseAudioData',
- value: function _parseAudioData(arrayBuffer, dataOffset, dataSize, tagTimestamp) {
- if (dataSize <= 1) {
- _logger2.default.w(this.TAG, 'Flv: Invalid audio packet, missing SoundData payload!');
- return;
- }
- if (this._hasAudioFlagOverrided === true && this._hasAudio === false) {
- // If hasAudio: false indicated explicitly in MediaDataSource,
- // Ignore all the audio packets
- return;
- }
- var le = this._littleEndian;
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var soundSpec = v.getUint8(0);
- var soundFormat = soundSpec >>> 4;
- if (soundFormat !== 2 && soundFormat !== 10) {
- // MP3 or AAC
- this._onError(_demuxErrors2.default.CODEC_UNSUPPORTED, 'Flv: Unsupported audio codec idx: ' + soundFormat);
- return;
- }
- var soundRate = 0;
- var soundRateIndex = (soundSpec & 12) >>> 2;
- if (soundRateIndex >= 0 && soundRateIndex <= 4) {
- soundRate = this._flvSoundRateTable[soundRateIndex];
- } else {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid audio sample rate idx: ' + soundRateIndex);
- return;
- }
- var soundSize = (soundSpec & 2) >>> 1; // unused
- var soundType = soundSpec & 1;
- var meta = this._audioMetadata;
- var track = this._audioTrack;
- if (!meta) {
- if (this._hasAudio === false && this._hasAudioFlagOverrided === false) {
- this._hasAudio = true;
- this._mediaInfo.hasAudio = true;
- }
- // initial metadata
- meta = this._audioMetadata = {};
- meta.type = 'audio';
- meta.id = track.id;
- meta.timescale = this._timescale;
- meta.duration = this._duration;
- meta.audioSampleRate = soundRate;
- meta.channelCount = soundType === 0 ? 1 : 2;
- }
- if (soundFormat === 10) {
- // AAC
- var aacData = this._parseAACAudioData(arrayBuffer, dataOffset + 1, dataSize - 1);
- if (aacData == undefined) {
- return;
- }
- if (aacData.packetType === 0) {
- // AAC sequence header (AudioSpecificConfig)
- if (meta.config) {
- _logger2.default.w(this.TAG, 'Found another AudioSpecificConfig!');
- }
- var misc = aacData.data;
- meta.audioSampleRate = misc.samplingRate;
- meta.channelCount = misc.channelCount;
- meta.codec = misc.codec;
- meta.originalCodec = misc.originalCodec;
- meta.config = misc.config;
- // The decode result of an aac sample is 1024 PCM samples
- meta.refSampleDuration = 1024 / meta.audioSampleRate * meta.timescale;
- _logger2.default.v(this.TAG, 'Parsed AudioSpecificConfig');
- if (this._isInitialMetadataDispatched()) {
- // Non-initial metadata, force dispatch (or flush) parsed frames to remuxer
- if (this._dispatch && (this._audioTrack.length || this._videoTrack.length)) {
- this._onDataAvailable(this._audioTrack, this._videoTrack);
- }
- } else {
- this._audioInitialMetadataDispatched = true;
- }
- // then notify new metadata
- this._dispatch = false;
- this._onTrackMetadata('audio', meta);
- var mi = this._mediaInfo;
- mi.audioCodec = meta.originalCodec;
- mi.audioSampleRate = meta.audioSampleRate;
- mi.audioChannelCount = meta.channelCount;
- if (mi.hasVideo) {
- if (mi.videoCodec != null) {
- mi.mimeType = 'video/x-flv; codecs="' + mi.videoCodec + ',' + mi.audioCodec + '"';
- }
- } else {
- mi.mimeType = 'video/x-flv; codecs="' + mi.audioCodec + '"';
- }
- if (mi.isComplete()) {
- this._onMediaInfo(mi);
- }
- } else if (aacData.packetType === 1) {
- // AAC raw frame data
- var dts = this._timestampBase + tagTimestamp;
- var aacSample = { unit: aacData.data, length: aacData.data.byteLength, dts: dts, pts: dts };
- track.samples.push(aacSample);
- track.length += aacData.data.length;
- } else {
- _logger2.default.e(this.TAG, 'Flv: Unsupported AAC data type ' + aacData.packetType);
- }
- } else if (soundFormat === 2) {
- // MP3
- if (!meta.codec) {
- // We need metadata for mp3 audio track, extract info from frame header
- var _misc = this._parseMP3AudioData(arrayBuffer, dataOffset + 1, dataSize - 1, true);
- if (_misc == undefined) {
- return;
- }
- meta.audioSampleRate = _misc.samplingRate;
- meta.channelCount = _misc.channelCount;
- meta.codec = _misc.codec;
- meta.originalCodec = _misc.originalCodec;
- // The decode result of an mp3 sample is 1152 PCM samples
- meta.refSampleDuration = 1152 / meta.audioSampleRate * meta.timescale;
- _logger2.default.v(this.TAG, 'Parsed MPEG Audio Frame Header');
- this._audioInitialMetadataDispatched = true;
- this._onTrackMetadata('audio', meta);
- var _mi = this._mediaInfo;
- _mi.audioCodec = meta.codec;
- _mi.audioSampleRate = meta.audioSampleRate;
- _mi.audioChannelCount = meta.channelCount;
- _mi.audioDataRate = _misc.bitRate;
- if (_mi.hasVideo) {
- if (_mi.videoCodec != null) {
- _mi.mimeType = 'video/x-flv; codecs="' + _mi.videoCodec + ',' + _mi.audioCodec + '"';
- }
- } else {
- _mi.mimeType = 'video/x-flv; codecs="' + _mi.audioCodec + '"';
- }
- if (_mi.isComplete()) {
- this._onMediaInfo(_mi);
- }
- }
- // This packet is always a valid audio packet, extract it
- var data = this._parseMP3AudioData(arrayBuffer, dataOffset + 1, dataSize - 1, false);
- if (data == undefined) {
- return;
- }
- var _dts = this._timestampBase + tagTimestamp;
- var mp3Sample = { unit: data, length: data.byteLength, dts: _dts, pts: _dts };
- track.samples.push(mp3Sample);
- track.length += data.length;
- }
- }
- }, {
- key: '_parseAACAudioData',
- value: function _parseAACAudioData(arrayBuffer, dataOffset, dataSize) {
- if (dataSize <= 1) {
- _logger2.default.w(this.TAG, 'Flv: Invalid AAC packet, missing AACPacketType or/and Data!');
- return;
- }
- var result = {};
- var array = new Uint8Array(arrayBuffer, dataOffset, dataSize);
- result.packetType = array[0];
- if (array[0] === 0) {
- result.data = this._parseAACAudioSpecificConfig(arrayBuffer, dataOffset + 1, dataSize - 1);
- } else {
- result.data = array.subarray(1);
- }
- return result;
- }
- }, {
- key: '_parseAACAudioSpecificConfig',
- value: function _parseAACAudioSpecificConfig(arrayBuffer, dataOffset, dataSize) {
- var array = new Uint8Array(arrayBuffer, dataOffset, dataSize);
- var config = null;
- /* Audio Object Type:
- 0: Null
- 1: AAC Main
- 2: AAC LC
- 3: AAC SSR (Scalable Sample Rate)
- 4: AAC LTP (Long Term Prediction)
- 5: HE-AAC / SBR (Spectral Band Replication)
- 6: AAC Scalable
- */
- var audioObjectType = 0;
- var originalAudioObjectType = 0;
- var audioExtensionObjectType = null;
- var samplingIndex = 0;
- var extensionSamplingIndex = null;
- // 5 bits
- audioObjectType = originalAudioObjectType = array[0] >>> 3;
- // 4 bits
- samplingIndex = (array[0] & 0x07) << 1 | array[1] >>> 7;
- if (samplingIndex < 0 || samplingIndex >= this._mpegSamplingRates.length) {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: AAC invalid sampling frequency index!');
- return;
- }
- var samplingFrequence = this._mpegSamplingRates[samplingIndex];
- // 4 bits
- var channelConfig = (array[1] & 0x78) >>> 3;
- if (channelConfig < 0 || channelConfig >= 8) {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: AAC invalid channel configuration');
- return;
- }
- if (audioObjectType === 5) {
- // HE-AAC?
- // 4 bits
- extensionSamplingIndex = (array[1] & 0x07) << 1 | array[2] >>> 7;
- // 5 bits
- audioExtensionObjectType = (array[2] & 0x7C) >>> 2;
- }
- // workarounds for various browsers
- var userAgent = self.navigator.userAgent.toLowerCase();
- if (userAgent.indexOf('firefox') !== -1) {
- // firefox: use SBR (HE-AAC) if freq less than 24kHz
- if (samplingIndex >= 6) {
- audioObjectType = 5;
- config = new Array(4);
- extensionSamplingIndex = samplingIndex - 3;
- } else {
- // use LC-AAC
- audioObjectType = 2;
- config = new Array(2);
- extensionSamplingIndex = samplingIndex;
- }
- } else if (userAgent.indexOf('android') !== -1) {
- // android: always use LC-AAC
- audioObjectType = 2;
- config = new Array(2);
- extensionSamplingIndex = samplingIndex;
- } else {
- // for other browsers, e.g. chrome...
- // Always use HE-AAC to make it easier to switch aac codec profile
- audioObjectType = 5;
- extensionSamplingIndex = samplingIndex;
- config = new Array(4);
- if (samplingIndex >= 6) {
- extensionSamplingIndex = samplingIndex - 3;
- } else if (channelConfig === 1) {
- // Mono channel
- audioObjectType = 2;
- config = new Array(2);
- extensionSamplingIndex = samplingIndex;
- }
- }
- config[0] = audioObjectType << 3;
- config[0] |= (samplingIndex & 0x0F) >>> 1;
- config[1] = (samplingIndex & 0x0F) << 7;
- config[1] |= (channelConfig & 0x0F) << 3;
- if (audioObjectType === 5) {
- config[1] |= (extensionSamplingIndex & 0x0F) >>> 1;
- config[2] = (extensionSamplingIndex & 0x01) << 7;
- // extended audio object type: force to 2 (LC-AAC)
- config[2] |= 2 << 2;
- config[3] = 0;
- }
- return {
- config: config,
- samplingRate: samplingFrequence,
- channelCount: channelConfig,
- codec: 'mp4a.40.' + audioObjectType,
- originalCodec: 'mp4a.40.' + originalAudioObjectType
- };
- }
- }, {
- key: '_parseMP3AudioData',
- value: function _parseMP3AudioData(arrayBuffer, dataOffset, dataSize, requestHeader) {
- if (dataSize < 4) {
- _logger2.default.w(this.TAG, 'Flv: Invalid MP3 packet, header missing!');
- return;
- }
- var le = this._littleEndian;
- var array = new Uint8Array(arrayBuffer, dataOffset, dataSize);
- var result = null;
- if (requestHeader) {
- if (array[0] !== 0xFF) {
- return;
- }
- var ver = array[1] >>> 3 & 0x03;
- var layer = (array[1] & 0x06) >> 1;
- var bitrate_index = (array[2] & 0xF0) >>> 4;
- var sampling_freq_index = (array[2] & 0x0C) >>> 2;
- var channel_mode = array[3] >>> 6 & 0x03;
- var channel_count = channel_mode !== 3 ? 2 : 1;
- var sample_rate = 0;
- var bit_rate = 0;
- var object_type = 34; // Layer-3, listed in MPEG-4 Audio Object Types
- var codec = 'mp3';
- switch (ver) {
- case 0:
- // MPEG 2.5
- sample_rate = this._mpegAudioV25SampleRateTable[sampling_freq_index];
- break;
- case 2:
- // MPEG 2
- sample_rate = this._mpegAudioV20SampleRateTable[sampling_freq_index];
- break;
- case 3:
- // MPEG 1
- sample_rate = this._mpegAudioV10SampleRateTable[sampling_freq_index];
- break;
- }
- switch (layer) {
- case 1:
- // Layer 3
- object_type = 34;
- if (bitrate_index < this._mpegAudioL3BitRateTable.length) {
- bit_rate = this._mpegAudioL3BitRateTable[bitrate_index];
- }
- break;
- case 2:
- // Layer 2
- object_type = 33;
- if (bitrate_index < this._mpegAudioL2BitRateTable.length) {
- bit_rate = this._mpegAudioL2BitRateTable[bitrate_index];
- }
- break;
- case 3:
- // Layer 1
- object_type = 32;
- if (bitrate_index < this._mpegAudioL1BitRateTable.length) {
- bit_rate = this._mpegAudioL1BitRateTable[bitrate_index];
- }
- break;
- }
- result = {
- bitRate: bit_rate,
- samplingRate: sample_rate,
- channelCount: channel_count,
- codec: codec,
- originalCodec: codec
- };
- } else {
- result = array;
- }
- return result;
- }
- }, {
- key: '_parseVideoData',
- value: function _parseVideoData(arrayBuffer, dataOffset, dataSize, tagTimestamp, tagPosition) {
- if (dataSize <= 1) {
- _logger2.default.w(this.TAG, 'Flv: Invalid video packet, missing VideoData payload!');
- return;
- }
- if (this._hasVideoFlagOverrided === true && this._hasVideo === false) {
- // If hasVideo: false indicated explicitly in MediaDataSource,
- // Ignore all the video packets
- return;
- }
- var spec = new Uint8Array(arrayBuffer, dataOffset, dataSize)[0];
- var frameType = (spec & 240) >>> 4;
- var codecId = spec & 15;
- if (codecId !== 7) {
- this._onError(_demuxErrors2.default.CODEC_UNSUPPORTED, 'Flv: Unsupported codec in video frame: ' + codecId);
- return;
- }
- this._parseAVCVideoPacket(arrayBuffer, dataOffset + 1, dataSize - 1, tagTimestamp, tagPosition, frameType);
- }
- }, {
- key: '_parseAVCVideoPacket',
- value: function _parseAVCVideoPacket(arrayBuffer, dataOffset, dataSize, tagTimestamp, tagPosition, frameType) {
- if (dataSize < 4) {
- _logger2.default.w(this.TAG, 'Flv: Invalid AVC packet, missing AVCPacketType or/and CompositionTime');
- return;
- }
- var le = this._littleEndian;
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var packetType = v.getUint8(0);
- var cts_unsigned = v.getUint32(0, !le) & 0x00FFFFFF;
- var cts = cts_unsigned << 8 >> 8; // convert to 24-bit signed int
- if (packetType === 0) {
- // AVCDecoderConfigurationRecord
- this._parseAVCDecoderConfigurationRecord(arrayBuffer, dataOffset + 4, dataSize - 4);
- } else if (packetType === 1) {
- // One or more Nalus
- this._parseAVCVideoData(arrayBuffer, dataOffset + 4, dataSize - 4, tagTimestamp, tagPosition, frameType, cts);
- } else if (packetType === 2) {
- // empty, AVC end of sequence
- } else {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid video packet type ' + packetType);
- return;
- }
- }
- }, {
- key: '_parseAVCDecoderConfigurationRecord',
- value: function _parseAVCDecoderConfigurationRecord(arrayBuffer, dataOffset, dataSize) {
- if (dataSize < 7) {
- _logger2.default.w(this.TAG, 'Flv: Invalid AVCDecoderConfigurationRecord, lack of data!');
- return;
- }
- var meta = this._videoMetadata;
- var track = this._videoTrack;
- var le = this._littleEndian;
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- if (!meta) {
- if (this._hasVideo === false && this._hasVideoFlagOverrided === false) {
- this._hasVideo = true;
- this._mediaInfo.hasVideo = true;
- }
- meta = this._videoMetadata = {};
- meta.type = 'video';
- meta.id = track.id;
- meta.timescale = this._timescale;
- meta.duration = this._duration;
- } else {
- if (typeof meta.avcc !== 'undefined') {
- _logger2.default.w(this.TAG, 'Found another AVCDecoderConfigurationRecord!');
- }
- }
- var version = v.getUint8(0); // configurationVersion
- var avcProfile = v.getUint8(1); // avcProfileIndication
- var profileCompatibility = v.getUint8(2); // profile_compatibility
- var avcLevel = v.getUint8(3); // AVCLevelIndication
- if (version !== 1 || avcProfile === 0) {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid AVCDecoderConfigurationRecord');
- return;
- }
- this._naluLengthSize = (v.getUint8(4) & 3) + 1; // lengthSizeMinusOne
- if (this._naluLengthSize !== 3 && this._naluLengthSize !== 4) {
- // holy shit!!!
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Strange NaluLengthSizeMinusOne: ' + (this._naluLengthSize - 1));
- return;
- }
- var spsCount = v.getUint8(5) & 31; // numOfSequenceParameterSets
- if (spsCount === 0) {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid AVCDecoderConfigurationRecord: No SPS');
- return;
- } else if (spsCount > 1) {
- _logger2.default.w(this.TAG, 'Flv: Strange AVCDecoderConfigurationRecord: SPS Count = ' + spsCount);
- }
- var offset = 6;
- for (var i = 0; i < spsCount; i++) {
- var len = v.getUint16(offset, !le); // sequenceParameterSetLength
- offset += 2;
- if (len === 0) {
- continue;
- }
- // Notice: Nalu without startcode header (00 00 00 01)
- var sps = new Uint8Array(arrayBuffer, dataOffset + offset, len);
- offset += len;
- var config = _spsParser2.default.parseSPS(sps);
- if (i !== 0) {
- // ignore other sps's config
- continue;
- }
- meta.codecWidth = config.codec_size.width;
- meta.codecHeight = config.codec_size.height;
- meta.presentWidth = config.present_size.width;
- meta.presentHeight = config.present_size.height;
- meta.profile = config.profile_string;
- meta.level = config.level_string;
- meta.bitDepth = config.bit_depth;
- meta.chromaFormat = config.chroma_format;
- meta.sarRatio = config.sar_ratio;
- meta.frameRate = config.frame_rate;
- if (config.frame_rate.fixed === false || config.frame_rate.fps_num === 0 || config.frame_rate.fps_den === 0) {
- meta.frameRate = this._referenceFrameRate;
- }
- var fps_den = meta.frameRate.fps_den;
- var fps_num = meta.frameRate.fps_num;
- meta.refSampleDuration = meta.timescale * (fps_den / fps_num);
- var codecArray = sps.subarray(1, 4);
- var codecString = 'avc1.';
- for (var j = 0; j < 3; j++) {
- var h = codecArray[j].toString(16);
- if (h.length < 2) {
- h = '0' + h;
- }
- codecString += h;
- }
- meta.codec = codecString;
- var mi = this._mediaInfo;
- mi.width = meta.codecWidth;
- mi.height = meta.codecHeight;
- mi.fps = meta.frameRate.fps;
- mi.profile = meta.profile;
- mi.level = meta.level;
- mi.refFrames = config.ref_frames;
- mi.chromaFormat = config.chroma_format_string;
- mi.sarNum = meta.sarRatio.width;
- mi.sarDen = meta.sarRatio.height;
- mi.videoCodec = codecString;
- if (mi.hasAudio) {
- if (mi.audioCodec != null) {
- mi.mimeType = 'video/x-flv; codecs="' + mi.videoCodec + ',' + mi.audioCodec + '"';
- }
- } else {
- mi.mimeType = 'video/x-flv; codecs="' + mi.videoCodec + '"';
- }
- if (mi.isComplete()) {
- this._onMediaInfo(mi);
- }
- }
- var ppsCount = v.getUint8(offset); // numOfPictureParameterSets
- if (ppsCount === 0) {
- this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid AVCDecoderConfigurationRecord: No PPS');
- return;
- } else if (ppsCount > 1) {
- _logger2.default.w(this.TAG, 'Flv: Strange AVCDecoderConfigurationRecord: PPS Count = ' + ppsCount);
- }
- offset++;
- for (var _i = 0; _i < ppsCount; _i++) {
- var _len = v.getUint16(offset, !le); // pictureParameterSetLength
- offset += 2;
- if (_len === 0) {
- continue;
- }
- // pps is useless for extracting video information
- offset += _len;
- }
- meta.avcc = new Uint8Array(dataSize);
- meta.avcc.set(new Uint8Array(arrayBuffer, dataOffset, dataSize), 0);
- _logger2.default.v(this.TAG, 'Parsed AVCDecoderConfigurationRecord');
- if (this._isInitialMetadataDispatched()) {
- // flush parsed frames
- if (this._dispatch && (this._audioTrack.length || this._videoTrack.length)) {
- this._onDataAvailable(this._audioTrack, this._videoTrack);
- }
- } else {
- this._videoInitialMetadataDispatched = true;
- }
- // notify new metadata
- this._dispatch = false;
- this._onTrackMetadata('video', meta);
- }
- }, {
- key: '_parseAVCVideoData',
- value: function _parseAVCVideoData(arrayBuffer, dataOffset, dataSize, tagTimestamp, tagPosition, frameType, cts) {
- var le = this._littleEndian;
- var v = new DataView(arrayBuffer, dataOffset, dataSize);
- var units = [],
- length = 0;
- var offset = 0;
- var lengthSize = this._naluLengthSize;
- var dts = this._timestampBase + tagTimestamp;
- var keyframe = frameType === 1; // from FLV Frame Type constants
- while (offset < dataSize) {
- if (offset + 4 >= dataSize) {
- _logger2.default.w(this.TAG, 'Malformed Nalu near timestamp ' + dts + ', offset = ' + offset + ', dataSize = ' + dataSize);
- break; // data not enough for next Nalu
- }
- // Nalu with length-header (AVC1)
- var naluSize = v.getUint32(offset, !le); // Big-Endian read
- if (lengthSize === 3) {
- naluSize >>>= 8;
- }
- if (naluSize > dataSize - lengthSize) {
- _logger2.default.w(this.TAG, 'Malformed Nalus near timestamp ' + dts + ', NaluSize > DataSize!');
- return;
- }
- var unitType = v.getUint8(offset + lengthSize) & 0x1F;
- if (unitType === 5) {
- // IDR
- keyframe = true;
- }
- var data = new Uint8Array(arrayBuffer, dataOffset + offset, lengthSize + naluSize);
- var unit = { type: unitType, data: data };
- units.push(unit);
- length += data.byteLength;
- offset += lengthSize + naluSize;
- }
- if (units.length) {
- var track = this._videoTrack;
- var avcSample = {
- units: units,
- length: length,
- isKeyframe: keyframe,
- dts: dts,
- cts: cts,
- pts: dts + cts
- };
- if (keyframe) {
- avcSample.fileposition = tagPosition;
- }
- track.samples.push(avcSample);
- track.length += length;
- }
- }
- }, {
- key: 'onTrackMetadata',
- get: function get() {
- return this._onTrackMetadata;
- },
- set: function set(callback) {
- this._onTrackMetadata = callback;
- }
- // prototype: function(mediaInfo: MediaInfo): void
- }, {
- key: 'onMediaInfo',
- get: function get() {
- return this._onMediaInfo;
- },
- set: function set(callback) {
- this._onMediaInfo = callback;
- }
- // prototype: function(type: number, info: string): void
- }, {
- key: 'onError',
- get: function get() {
- return this._onError;
- },
- set: function set(callback) {
- this._onError = callback;
- }
- // prototype: function(videoTrack: any, audioTrack: any): void
- }, {
- key: 'onDataAvailable',
- get: function get() {
- return this._onDataAvailable;
- },
- set: function set(callback) {
- this._onDataAvailable = callback;
- }
- // timestamp base for output samples, must be in milliseconds
- }, {
- key: 'timestampBase',
- get: function get() {
- return this._timestampBase;
- },
- set: function set(base) {
- this._timestampBase = base;
- }
- }, {
- key: 'overridedDuration',
- get: function get() {
- return this._duration;
- }
- // Force-override media duration. Must be in milliseconds, int32
- ,
- set: function set(duration) {
- this._durationOverrided = true;
- this._duration = duration;
- this._mediaInfo.duration = duration;
- }
- // Force-override audio track present flag, boolean
- }, {
- key: 'overridedHasAudio',
- set: function set(hasAudio) {
- this._hasAudioFlagOverrided = true;
- this._hasAudio = hasAudio;
- this._mediaInfo.hasAudio = hasAudio;
- }
- // Force-override video track present flag, boolean
- }, {
- key: 'overridedHasVideo',
- set: function set(hasVideo) {
- this._hasVideoFlagOverrided = true;
- this._hasVideo = hasVideo;
- this._mediaInfo.hasVideo = hasVideo;
- }
- }], [{
- key: 'probe',
- value: function probe(buffer) {
- var data = new Uint8Array(buffer);
- var mismatch = { match: false };
- if (data[0] !== 0x46 || data[1] !== 0x4C || data[2] !== 0x56 || data[3] !== 0x01) {
- return mismatch;
- }
- var hasAudio = (data[4] & 4) >>> 2 !== 0;
- var hasVideo = (data[4] & 1) !== 0;
- var offset = ReadBig32(data, 5);
- if (offset < 9) {
- return mismatch;
- }
- return {
- match: true,
- consumed: offset,
- dataOffset: offset,
- hasAudioTrack: hasAudio,
- hasVideoTrack: hasVideo
- };
- }
- }]);
- return FLVDemuxer;
- }();
- exports.default = FLVDemuxer;
- },{"../core/media-info.js":7,"../utils/exception.js":40,"../utils/logger.js":41,"./amf-parser.js":15,"./demux-errors.js":16,"./sps-parser.js":19}],19:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _expGolomb = _dereq_('./exp-golomb.js');
- var _expGolomb2 = _interopRequireDefault(_expGolomb);
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var SPSParser = function () {
- function SPSParser() {
- _classCallCheck(this, SPSParser);
- }
- _createClass(SPSParser, null, [{
- key: '_ebsp2rbsp',
- value: function _ebsp2rbsp(uint8array) {
- var src = uint8array;
- var src_length = src.byteLength;
- var dst = new Uint8Array(src_length);
- var dst_idx = 0;
- for (var i = 0; i < src_length; i++) {
- if (i >= 2) {
- // Unescape: Skip 0x03 after 00 00
- if (src[i] === 0x03 && src[i - 1] === 0x00 && src[i - 2] === 0x00) {
- continue;
- }
- }
- dst[dst_idx] = src[i];
- dst_idx++;
- }
- return new Uint8Array(dst.buffer, 0, dst_idx);
- }
- }, {
- key: 'parseSPS',
- value: function parseSPS(uint8array) {
- var rbsp = SPSParser._ebsp2rbsp(uint8array);
- var gb = new _expGolomb2.default(rbsp);
- gb.readByte();
- var profile_idc = gb.readByte(); // profile_idc
- gb.readByte(); // constraint_set_flags[5] + reserved_zero[3]
- var level_idc = gb.readByte(); // level_idc
- gb.readUEG(); // seq_parameter_set_id
- var profile_string = SPSParser.getProfileString(profile_idc);
- var level_string = SPSParser.getLevelString(level_idc);
- var chroma_format_idc = 1;
- var chroma_format = 420;
- var chroma_format_table = [0, 420, 422, 444];
- var bit_depth = 8;
- if (profile_idc === 100 || profile_idc === 110 || profile_idc === 122 || profile_idc === 244 || profile_idc === 44 || profile_idc === 83 || profile_idc === 86 || profile_idc === 118 || profile_idc === 128 || profile_idc === 138 || profile_idc === 144) {
- chroma_format_idc = gb.readUEG();
- if (chroma_format_idc === 3) {
- gb.readBits(1); // separate_colour_plane_flag
- }
- if (chroma_format_idc <= 3) {
- chroma_format = chroma_format_table[chroma_format_idc];
- }
- bit_depth = gb.readUEG() + 8; // bit_depth_luma_minus8
- gb.readUEG(); // bit_depth_chroma_minus8
- gb.readBits(1); // qpprime_y_zero_transform_bypass_flag
- if (gb.readBool()) {
- // seq_scaling_matrix_present_flag
- var scaling_list_count = chroma_format_idc !== 3 ? 8 : 12;
- for (var i = 0; i < scaling_list_count; i++) {
- if (gb.readBool()) {
- // seq_scaling_list_present_flag
- if (i < 6) {
- SPSParser._skipScalingList(gb, 16);
- } else {
- SPSParser._skipScalingList(gb, 64);
- }
- }
- }
- }
- }
- gb.readUEG(); // log2_max_frame_num_minus4
- var pic_order_cnt_type = gb.readUEG();
- if (pic_order_cnt_type === 0) {
- gb.readUEG(); // log2_max_pic_order_cnt_lsb_minus_4
- } else if (pic_order_cnt_type === 1) {
- gb.readBits(1); // delta_pic_order_always_zero_flag
- gb.readSEG(); // offset_for_non_ref_pic
- gb.readSEG(); // offset_for_top_to_bottom_field
- var num_ref_frames_in_pic_order_cnt_cycle = gb.readUEG();
- for (var _i = 0; _i < num_ref_frames_in_pic_order_cnt_cycle; _i++) {
- gb.readSEG(); // offset_for_ref_frame
- }
- }
- var ref_frames = gb.readUEG(); // max_num_ref_frames
- gb.readBits(1); // gaps_in_frame_num_value_allowed_flag
- var pic_width_in_mbs_minus1 = gb.readUEG();
- var pic_height_in_map_units_minus1 = gb.readUEG();
- var frame_mbs_only_flag = gb.readBits(1);
- if (frame_mbs_only_flag === 0) {
- gb.readBits(1); // mb_adaptive_frame_field_flag
- }
- gb.readBits(1); // direct_8x8_inference_flag
- var frame_crop_left_offset = 0;
- var frame_crop_right_offset = 0;
- var frame_crop_top_offset = 0;
- var frame_crop_bottom_offset = 0;
- var frame_cropping_flag = gb.readBool();
- if (frame_cropping_flag) {
- frame_crop_left_offset = gb.readUEG();
- frame_crop_right_offset = gb.readUEG();
- frame_crop_top_offset = gb.readUEG();
- frame_crop_bottom_offset = gb.readUEG();
- }
- var sar_width = 1,
- sar_height = 1;
- var fps = 0,
- fps_fixed = true,
- fps_num = 0,
- fps_den = 0;
- var vui_parameters_present_flag = gb.readBool();
- if (vui_parameters_present_flag) {
- if (gb.readBool()) {
- // aspect_ratio_info_present_flag
- var aspect_ratio_idc = gb.readByte();
- var sar_w_table = [1, 12, 10, 16, 40, 24, 20, 32, 80, 18, 15, 64, 160, 4, 3, 2];
- var sar_h_table = [1, 11, 11, 11, 33, 11, 11, 11, 33, 11, 11, 33, 99, 3, 2, 1];
- if (aspect_ratio_idc > 0 && aspect_ratio_idc < 16) {
- sar_width = sar_w_table[aspect_ratio_idc - 1];
- sar_height = sar_h_table[aspect_ratio_idc - 1];
- } else if (aspect_ratio_idc === 255) {
- sar_width = gb.readByte() << 8 | gb.readByte();
- sar_height = gb.readByte() << 8 | gb.readByte();
- }
- }
- if (gb.readBool()) {
- // overscan_info_present_flag
- gb.readBool(); // overscan_appropriate_flag
- }
- if (gb.readBool()) {
- // video_signal_type_present_flag
- gb.readBits(4); // video_format & video_full_range_flag
- if (gb.readBool()) {
- // colour_description_present_flag
- gb.readBits(24); // colour_primaries & transfer_characteristics & matrix_coefficients
- }
- }
- if (gb.readBool()) {
- // chroma_loc_info_present_flag
- gb.readUEG(); // chroma_sample_loc_type_top_field
- gb.readUEG(); // chroma_sample_loc_type_bottom_field
- }
- if (gb.readBool()) {
- // timing_info_present_flag
- var num_units_in_tick = gb.readBits(32);
- var time_scale = gb.readBits(32);
- fps_fixed = gb.readBool(); // fixed_frame_rate_flag
- fps_num = time_scale;
- fps_den = num_units_in_tick * 2;
- fps = fps_num / fps_den;
- }
- }
- var sarScale = 1;
- if (sar_width !== 1 || sar_height !== 1) {
- sarScale = sar_width / sar_height;
- }
- var crop_unit_x = 0,
- crop_unit_y = 0;
- if (chroma_format_idc === 0) {
- crop_unit_x = 1;
- crop_unit_y = 2 - frame_mbs_only_flag;
- } else {
- var sub_wc = chroma_format_idc === 3 ? 1 : 2;
- var sub_hc = chroma_format_idc === 1 ? 2 : 1;
- crop_unit_x = sub_wc;
- crop_unit_y = sub_hc * (2 - frame_mbs_only_flag);
- }
- var codec_width = (pic_width_in_mbs_minus1 + 1) * 16;
- var codec_height = (2 - frame_mbs_only_flag) * ((pic_height_in_map_units_minus1 + 1) * 16);
- codec_width -= (frame_crop_left_offset + frame_crop_right_offset) * crop_unit_x;
- codec_height -= (frame_crop_top_offset + frame_crop_bottom_offset) * crop_unit_y;
- var present_width = Math.ceil(codec_width * sarScale);
- gb.destroy();
- gb = null;
- return {
- profile_string: profile_string, // baseline, high, high10, ...
- level_string: level_string, // 3, 3.1, 4, 4.1, 5, 5.1, ...
- bit_depth: bit_depth, // 8bit, 10bit, ...
- ref_frames: ref_frames,
- chroma_format: chroma_format, // 4:2:0, 4:2:2, ...
- chroma_format_string: SPSParser.getChromaFormatString(chroma_format),
- frame_rate: {
- fixed: fps_fixed,
- fps: fps,
- fps_den: fps_den,
- fps_num: fps_num
- },
- sar_ratio: {
- width: sar_width,
- height: sar_height
- },
- codec_size: {
- width: codec_width,
- height: codec_height
- },
- present_size: {
- width: present_width,
- height: codec_height
- }
- };
- }
- }, {
- key: '_skipScalingList',
- value: function _skipScalingList(gb, count) {
- var last_scale = 8,
- next_scale = 8;
- var delta_scale = 0;
- for (var i = 0; i < count; i++) {
- if (next_scale !== 0) {
- delta_scale = gb.readSEG();
- next_scale = (last_scale + delta_scale + 256) % 256;
- }
- last_scale = next_scale === 0 ? last_scale : next_scale;
- }
- }
- }, {
- key: 'getProfileString',
- value: function getProfileString(profile_idc) {
- switch (profile_idc) {
- case 66:
- return 'Baseline';
- case 77:
- return 'Main';
- case 88:
- return 'Extended';
- case 100:
- return 'High';
- case 110:
- return 'High10';
- case 122:
- return 'High422';
- case 244:
- return 'High444';
- default:
- return 'Unknown';
- }
- }
- }, {
- key: 'getLevelString',
- value: function getLevelString(level_idc) {
- return (level_idc / 10).toFixed(1);
- }
- }, {
- key: 'getChromaFormatString',
- value: function getChromaFormatString(chroma) {
- switch (chroma) {
- case 420:
- return '4:2:0';
- case 422:
- return '4:2:2';
- case 444:
- return '4:4:4';
- default:
- return 'Unknown';
- }
- }
- }]);
- return SPSParser;
- }();
- exports.default = SPSParser;
- },{"./exp-golomb.js":17}],20:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _polyfill = _dereq_('./utils/polyfill.js');
- var _polyfill2 = _interopRequireDefault(_polyfill);
- var _features = _dereq_('./core/features.js');
- var _features2 = _interopRequireDefault(_features);
- var _flvPlayer = _dereq_('./player/flv-player.js');
- var _flvPlayer2 = _interopRequireDefault(_flvPlayer);
- var _nativePlayer = _dereq_('./player/native-player.js');
- var _nativePlayer2 = _interopRequireDefault(_nativePlayer);
- var _playerEvents = _dereq_('./player/player-events.js');
- var _playerEvents2 = _interopRequireDefault(_playerEvents);
- var _playerErrors = _dereq_('./player/player-errors.js');
- var _loggingControl = _dereq_('./utils/logging-control.js');
- var _loggingControl2 = _interopRequireDefault(_loggingControl);
- var _exception = _dereq_('./utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- // here are all the interfaces
- // install polyfills
- _polyfill2.default.install();
- // factory method
- function createPlayer(mediaDataSource, optionalConfig) {
- var mds = mediaDataSource;
- if (mds == null || (typeof mds === 'undefined' ? 'undefined' : _typeof(mds)) !== 'object') {
- throw new _exception.InvalidArgumentException('MediaDataSource must be an javascript object!');
- }
- if (!mds.hasOwnProperty('type')) {
- throw new _exception.InvalidArgumentException('MediaDataSource must has type field to indicate video file type!');
- }
- switch (mds.type) {
- case 'flv':
- return new _flvPlayer2.default(mds, optionalConfig);
- default:
- return new _nativePlayer2.default(mds, optionalConfig);
- }
- }
- // feature detection
- function isSupported() {
- return _features2.default.supportMSEH264Playback();
- }
- function getFeatureList() {
- return _features2.default.getFeatureList();
- }
- // interfaces
- var flvjs = {};
- flvjs.createPlayer = createPlayer;
- flvjs.isSupported = isSupported;
- flvjs.getFeatureList = getFeatureList;
- flvjs.Events = _playerEvents2.default;
- flvjs.ErrorTypes = _playerErrors.ErrorTypes;
- flvjs.ErrorDetails = _playerErrors.ErrorDetails;
- flvjs.FlvPlayer = _flvPlayer2.default;
- flvjs.NativePlayer = _nativePlayer2.default;
- flvjs.LoggingControl = _loggingControl2.default;
- Object.defineProperty(flvjs, 'version', {
- enumerable: true,
- get: function get() {
- // replaced by browserify-versionify transform
- return '1.4.0';
- }
- });
- exports.default = flvjs;
- },{"./core/features.js":6,"./player/flv-player.js":32,"./player/native-player.js":33,"./player/player-errors.js":34,"./player/player-events.js":35,"./utils/exception.js":40,"./utils/logging-control.js":42,"./utils/polyfill.js":43}],21:[function(_dereq_,module,exports){
- 'use strict';
- // entry/index file
- // make it compatible with browserify's umd wrapper
- module.exports = _dereq_('./flv.js').default;
- },{"./flv.js":20}],22:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _browser = _dereq_('../utils/browser.js');
- var _browser2 = _interopRequireDefault(_browser);
- var _loader = _dereq_('./loader.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
- function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- /* fetch + stream IO loader. Currently working on chrome 43+.
- * fetch provides a better alternative http API to XMLHttpRequest
- *
- * fetch spec https://fetch.spec.whatwg.org/
- * stream spec https://streams.spec.whatwg.org/
- */
- var FetchStreamLoader = function (_BaseLoader) {
- _inherits(FetchStreamLoader, _BaseLoader);
- _createClass(FetchStreamLoader, null, [{
- key: 'isSupported',
- value: function isSupported() {
- try {
- // fetch + stream is broken on Microsoft Edge. Disable before build 15048.
- // see https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/8196907/
- // Fixed in Jan 10, 2017. Build 15048+ removed from blacklist.
- var isWorkWellEdge = _browser2.default.msedge && _browser2.default.version.minor >= 15048;
- var browserNotBlacklisted = _browser2.default.msedge ? isWorkWellEdge : true;
- return self.fetch && self.ReadableStream && browserNotBlacklisted;
- } catch (e) {
- return false;
- }
- }
- }]);
- function FetchStreamLoader(seekHandler, config) {
- _classCallCheck(this, FetchStreamLoader);
- var _this = _possibleConstructorReturn(this, (FetchStreamLoader.__proto__ || Object.getPrototypeOf(FetchStreamLoader)).call(this, 'fetch-stream-loader'));
- _this.TAG = 'FetchStreamLoader';
- _this._seekHandler = seekHandler;
- _this._config = config;
- _this._needStash = true;
- _this._requestAbort = false;
- _this._contentLength = null;
- _this._receivedLength = 0;
- return _this;
- }
- _createClass(FetchStreamLoader, [{
- key: 'destroy',
- value: function destroy() {
- if (this.isWorking()) {
- this.abort();
- }
- _get(FetchStreamLoader.prototype.__proto__ || Object.getPrototypeOf(FetchStreamLoader.prototype), 'destroy', this).call(this);
- }
- }, {
- key: 'open',
- value: function open(dataSource, range) {
- var _this2 = this;
- this._dataSource = dataSource;
- this._range = range;
- var sourceURL = dataSource.url;
- if (this._config.reuseRedirectedURL && dataSource.redirectedURL != undefined) {
- sourceURL = dataSource.redirectedURL;
- }
- var seekConfig = this._seekHandler.getConfig(sourceURL, range);
- var headers = new self.Headers();
- if (_typeof(seekConfig.headers) === 'object') {
- var configHeaders = seekConfig.headers;
- for (var key in configHeaders) {
- if (configHeaders.hasOwnProperty(key)) {
- headers.append(key, configHeaders[key]);
- }
- }
- }
- var params = {
- method: 'GET',
- headers: headers,
- mode: 'cors',
- cache: 'default',
- // The default policy of Fetch API in the whatwg standard
- // Safari incorrectly indicates 'no-referrer' as default policy, fuck it
- referrerPolicy: 'no-referrer-when-downgrade'
- };
- // cors is enabled by default
- if (dataSource.cors === false) {
- // no-cors means 'disregard cors policy', which can only be used in ServiceWorker
- params.mode = 'same-origin';
- }
- // withCredentials is disabled by default
- if (dataSource.withCredentials) {
- params.credentials = 'include';
- }
- // referrerPolicy from config
- if (dataSource.referrerPolicy) {
- params.referrerPolicy = dataSource.referrerPolicy;
- }
- this._status = _loader.LoaderStatus.kConnecting;
- self.fetch(seekConfig.url, params).then(function (res) {
- if (_this2._requestAbort) {
- _this2._requestAbort = false;
- _this2._status = _loader.LoaderStatus.kIdle;
- return;
- }
- if (res.ok && res.status >= 200 && res.status <= 299) {
- if (res.url !== seekConfig.url) {
- if (_this2._onURLRedirect) {
- var redirectedURL = _this2._seekHandler.removeURLParameters(res.url);
- _this2._onURLRedirect(redirectedURL);
- }
- }
- var lengthHeader = res.headers.get('Content-Length');
- if (lengthHeader != null) {
- _this2._contentLength = parseInt(lengthHeader);
- if (_this2._contentLength !== 0) {
- if (_this2._onContentLengthKnown) {
- _this2._onContentLengthKnown(_this2._contentLength);
- }
- }
- }
- return _this2._pump.call(_this2, res.body.getReader());
- } else {
- _this2._status = _loader.LoaderStatus.kError;
- if (_this2._onError) {
- _this2._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: res.status, msg: res.statusText });
- } else {
- throw new _exception.RuntimeException('FetchStreamLoader: Http code invalid, ' + res.status + ' ' + res.statusText);
- }
- }
- }).catch(function (e) {
- _this2._status = _loader.LoaderStatus.kError;
- if (_this2._onError) {
- _this2._onError(_loader.LoaderErrors.EXCEPTION, { code: -1, msg: e.message });
- } else {
- throw e;
- }
- });
- }
- }, {
- key: 'abort',
- value: function abort() {
- this._requestAbort = true;
- }
- }, {
- key: '_pump',
- value: function _pump(reader) {
- var _this3 = this;
- // ReadableStreamReader
- return reader.read().then(function (result) {
- if (result.done) {
- _this3._status = _loader.LoaderStatus.kComplete;
- if (_this3._onComplete) {
- _this3._onComplete(_this3._range.from, _this3._range.from + _this3._receivedLength - 1);
- }
- } else {
- if (_this3._requestAbort === true) {
- _this3._requestAbort = false;
- _this3._status = _loader.LoaderStatus.kComplete;
- return reader.cancel();
- }
- _this3._status = _loader.LoaderStatus.kBuffering;
- var chunk = result.value.buffer;
- var byteStart = _this3._range.from + _this3._receivedLength;
- _this3._receivedLength += chunk.byteLength;
- if (_this3._onDataArrival) {
- _this3._onDataArrival(chunk, byteStart, _this3._receivedLength);
- }
- _this3._pump(reader);
- }
- }).catch(function (e) {
- if (e.code === 11 && _browser2.default.msedge) {
- // InvalidStateError on Microsoft Edge
- // Workaround: Edge may throw InvalidStateError after ReadableStreamReader.cancel() call
- // Ignore the unknown exception.
- // Related issue: https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/11265202/
- return;
- }
- _this3._status = _loader.LoaderStatus.kError;
- var type = 0;
- var info = null;
- if ((e.code === 19 || e.message === 'network error') && ( // NETWORK_ERR
- _this3._contentLength === null || _this3._contentLength !== null && _this3._receivedLength < _this3._contentLength)) {
- type = _loader.LoaderErrors.EARLY_EOF;
- info = { code: e.code, msg: 'Fetch stream meet Early-EOF' };
- } else {
- type = _loader.LoaderErrors.EXCEPTION;
- info = { code: e.code, msg: e.message };
- }
- if (_this3._onError) {
- _this3._onError(type, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- });
- }
- }]);
- return FetchStreamLoader;
- }(_loader.BaseLoader);
- exports.default = FetchStreamLoader;
- },{"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],23:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _speedSampler = _dereq_('./speed-sampler.js');
- var _speedSampler2 = _interopRequireDefault(_speedSampler);
- var _loader = _dereq_('./loader.js');
- var _fetchStreamLoader = _dereq_('./fetch-stream-loader.js');
- var _fetchStreamLoader2 = _interopRequireDefault(_fetchStreamLoader);
- var _xhrMozChunkedLoader = _dereq_('./xhr-moz-chunked-loader.js');
- var _xhrMozChunkedLoader2 = _interopRequireDefault(_xhrMozChunkedLoader);
- var _xhrMsstreamLoader = _dereq_('./xhr-msstream-loader.js');
- var _xhrMsstreamLoader2 = _interopRequireDefault(_xhrMsstreamLoader);
- var _xhrRangeLoader = _dereq_('./xhr-range-loader.js');
- var _xhrRangeLoader2 = _interopRequireDefault(_xhrRangeLoader);
- var _websocketLoader = _dereq_('./websocket-loader.js');
- var _websocketLoader2 = _interopRequireDefault(_websocketLoader);
- var _rangeSeekHandler = _dereq_('./range-seek-handler.js');
- var _rangeSeekHandler2 = _interopRequireDefault(_rangeSeekHandler);
- var _paramSeekHandler = _dereq_('./param-seek-handler.js');
- var _paramSeekHandler2 = _interopRequireDefault(_paramSeekHandler);
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /**
- * DataSource: {
- * url: string,
- * filesize: number,
- * cors: boolean,
- * withCredentials: boolean
- * }
- *
- */
- // Manage IO Loaders
- var IOController = function () {
- function IOController(dataSource, config, extraData) {
- _classCallCheck(this, IOController);
- this.TAG = 'IOController';
- this._config = config;
- this._extraData = extraData;
- this._stashInitialSize = 1024 * 384; // default initial size: 384KB
- if (config.stashInitialSize != undefined && config.stashInitialSize > 0) {
- // apply from config
- this._stashInitialSize = config.stashInitialSize;
- }
- this._stashUsed = 0;
- this._stashSize = this._stashInitialSize;
- this._bufferSize = 1024 * 1024 * 3; // initial size: 3MB
- this._stashBuffer = new ArrayBuffer(this._bufferSize);
- this._stashByteStart = 0;
- this._enableStash = true;
- if (config.enableStashBuffer === false) {
- this._enableStash = false;
- }
- this._loader = null;
- this._loaderClass = null;
- this._seekHandler = null;
- this._dataSource = dataSource;
- this._isWebSocketURL = /wss?:\/\/(.+?)/.test(dataSource.url);
- this._refTotalLength = dataSource.filesize ? dataSource.filesize : null;
- this._totalLength = this._refTotalLength;
- this._fullRequestFlag = false;
- this._currentRange = null;
- this._redirectedURL = null;
- this._speedNormalized = 0;
- this._speedSampler = new _speedSampler2.default();
- this._speedNormalizeList = [64, 128, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096];
- this._isEarlyEofReconnecting = false;
- this._paused = false;
- this._resumeFrom = 0;
- this._onDataArrival = null;
- this._onSeeked = null;
- this._onError = null;
- this._onComplete = null;
- this._onRedirect = null;
- this._onRecoveredEarlyEof = null;
- this._selectSeekHandler();
- this._selectLoader();
- this._createLoader();
- }
- _createClass(IOController, [{
- key: 'destroy',
- value: function destroy() {
- if (this._loader.isWorking()) {
- this._loader.abort();
- }
- this._loader.destroy();
- this._loader = null;
- this._loaderClass = null;
- this._dataSource = null;
- this._stashBuffer = null;
- this._stashUsed = this._stashSize = this._bufferSize = this._stashByteStart = 0;
- this._currentRange = null;
- this._speedSampler = null;
- this._isEarlyEofReconnecting = false;
- this._onDataArrival = null;
- this._onSeeked = null;
- this._onError = null;
- this._onComplete = null;
- this._onRedirect = null;
- this._onRecoveredEarlyEof = null;
- this._extraData = null;
- }
- }, {
- key: 'isWorking',
- value: function isWorking() {
- return this._loader && this._loader.isWorking() && !this._paused;
- }
- }, {
- key: 'isPaused',
- value: function isPaused() {
- return this._paused;
- }
- }, {
- key: '_selectSeekHandler',
- value: function _selectSeekHandler() {
- var config = this._config;
- if (config.seekType === 'range') {
- this._seekHandler = new _rangeSeekHandler2.default(this._config.rangeLoadZeroStart);
- } else if (config.seekType === 'param') {
- var paramStart = config.seekParamStart || 'bstart';
- var paramEnd = config.seekParamEnd || 'bend';
- this._seekHandler = new _paramSeekHandler2.default(paramStart, paramEnd);
- } else if (config.seekType === 'custom') {
- if (typeof config.customSeekHandler !== 'function') {
- throw new _exception.InvalidArgumentException('Custom seekType specified in config but invalid customSeekHandler!');
- }
- this._seekHandler = new config.customSeekHandler();
- } else {
- throw new _exception.InvalidArgumentException('Invalid seekType in config: ' + config.seekType);
- }
- }
- }, {
- key: '_selectLoader',
- value: function _selectLoader() {
- if (this._isWebSocketURL) {
- this._loaderClass = _websocketLoader2.default;
- } else if (_fetchStreamLoader2.default.isSupported()) {
- this._loaderClass = _fetchStreamLoader2.default;
- } else if (_xhrMozChunkedLoader2.default.isSupported()) {
- this._loaderClass = _xhrMozChunkedLoader2.default;
- } else if (_xhrRangeLoader2.default.isSupported()) {
- this._loaderClass = _xhrRangeLoader2.default;
- } else {
- throw new _exception.RuntimeException('Your browser doesn\'t support xhr with arraybuffer responseType!');
- }
- }
- }, {
- key: '_createLoader',
- value: function _createLoader() {
- this._loader = new this._loaderClass(this._seekHandler, this._config);
- if (this._loader.needStashBuffer === false) {
- this._enableStash = false;
- }
- this._loader.onContentLengthKnown = this._onContentLengthKnown.bind(this);
- this._loader.onURLRedirect = this._onURLRedirect.bind(this);
- this._loader.onDataArrival = this._onLoaderChunkArrival.bind(this);
- this._loader.onComplete = this._onLoaderComplete.bind(this);
- this._loader.onError = this._onLoaderError.bind(this);
- }
- }, {
- key: 'open',
- value: function open(optionalFrom) {
- this._currentRange = { from: 0, to: -1 };
- if (optionalFrom) {
- this._currentRange.from = optionalFrom;
- }
- this._speedSampler.reset();
- if (!optionalFrom) {
- this._fullRequestFlag = true;
- }
- this._loader.open(this._dataSource, Object.assign({}, this._currentRange));
- }
- }, {
- key: 'abort',
- value: function abort() {
- this._loader.abort();
- if (this._paused) {
- this._paused = false;
- this._resumeFrom = 0;
- }
- }
- }, {
- key: 'pause',
- value: function pause() {
- if (this.isWorking()) {
- this._loader.abort();
- if (this._stashUsed !== 0) {
- this._resumeFrom = this._stashByteStart;
- this._currentRange.to = this._stashByteStart - 1;
- } else {
- this._resumeFrom = this._currentRange.to + 1;
- }
- this._stashUsed = 0;
- this._stashByteStart = 0;
- this._paused = true;
- }
- }
- }, {
- key: 'resume',
- value: function resume() {
- if (this._paused) {
- this._paused = false;
- var bytes = this._resumeFrom;
- this._resumeFrom = 0;
- this._internalSeek(bytes, true);
- }
- }
- }, {
- key: 'seek',
- value: function seek(bytes) {
- this._paused = false;
- this._stashUsed = 0;
- this._stashByteStart = 0;
- this._internalSeek(bytes, true);
- }
- /**
- * When seeking request is from media seeking, unconsumed stash data should be dropped
- * However, stash data shouldn't be dropped if seeking requested from http reconnection
- *
- * @dropUnconsumed: Ignore and discard all unconsumed data in stash buffer
- */
- }, {
- key: '_internalSeek',
- value: function _internalSeek(bytes, dropUnconsumed) {
- if (this._loader.isWorking()) {
- this._loader.abort();
- }
- // dispatch & flush stash buffer before seek
- this._flushStashBuffer(dropUnconsumed);
- this._loader.destroy();
- this._loader = null;
- var requestRange = { from: bytes, to: -1 };
- this._currentRange = { from: requestRange.from, to: -1 };
- this._speedSampler.reset();
- this._stashSize = this._stashInitialSize;
- this._createLoader();
- this._loader.open(this._dataSource, requestRange);
- if (this._onSeeked) {
- this._onSeeked();
- }
- }
- }, {
- key: 'updateUrl',
- value: function updateUrl(url) {
- if (!url || typeof url !== 'string' || url.length === 0) {
- throw new _exception.InvalidArgumentException('Url must be a non-empty string!');
- }
- this._dataSource.url = url;
- // TODO: replace with new url
- }
- }, {
- key: '_expandBuffer',
- value: function _expandBuffer(expectedBytes) {
- var bufferNewSize = this._stashSize;
- while (bufferNewSize + 1024 * 1024 * 1 < expectedBytes) {
- bufferNewSize *= 2;
- }
- bufferNewSize += 1024 * 1024 * 1; // bufferSize = stashSize + 1MB
- if (bufferNewSize === this._bufferSize) {
- return;
- }
- var newBuffer = new ArrayBuffer(bufferNewSize);
- if (this._stashUsed > 0) {
- // copy existing data into new buffer
- var stashOldArray = new Uint8Array(this._stashBuffer, 0, this._stashUsed);
- var stashNewArray = new Uint8Array(newBuffer, 0, bufferNewSize);
- stashNewArray.set(stashOldArray, 0);
- }
- this._stashBuffer = newBuffer;
- this._bufferSize = bufferNewSize;
- }
- }, {
- key: '_normalizeSpeed',
- value: function _normalizeSpeed(input) {
- var list = this._speedNormalizeList;
- var last = list.length - 1;
- var mid = 0;
- var lbound = 0;
- var ubound = last;
- if (input < list[0]) {
- return list[0];
- }
- // binary search
- while (lbound <= ubound) {
- mid = lbound + Math.floor((ubound - lbound) / 2);
- if (mid === last || input >= list[mid] && input < list[mid + 1]) {
- return list[mid];
- } else if (list[mid] < input) {
- lbound = mid + 1;
- } else {
- ubound = mid - 1;
- }
- }
- }
- }, {
- key: '_adjustStashSize',
- value: function _adjustStashSize(normalized) {
- var stashSizeKB = 0;
- if (this._config.isLive) {
- // live stream: always use single normalized speed for size of stashSizeKB
- stashSizeKB = normalized;
- } else {
- if (normalized < 512) {
- stashSizeKB = normalized;
- } else if (normalized >= 512 && normalized <= 1024) {
- stashSizeKB = Math.floor(normalized * 1.5);
- } else {
- stashSizeKB = normalized * 2;
- }
- }
- if (stashSizeKB > 8192) {
- stashSizeKB = 8192;
- }
- var bufferSize = stashSizeKB * 1024 + 1024 * 1024 * 1; // stashSize + 1MB
- if (this._bufferSize < bufferSize) {
- this._expandBuffer(bufferSize);
- }
- this._stashSize = stashSizeKB * 1024;
- }
- }, {
- key: '_dispatchChunks',
- value: function _dispatchChunks(chunks, byteStart) {
- this._currentRange.to = byteStart + chunks.byteLength - 1;
- return this._onDataArrival(chunks, byteStart);
- }
- }, {
- key: '_onURLRedirect',
- value: function _onURLRedirect(redirectedURL) {
- this._redirectedURL = redirectedURL;
- if (this._onRedirect) {
- this._onRedirect(redirectedURL);
- }
- }
- }, {
- key: '_onContentLengthKnown',
- value: function _onContentLengthKnown(contentLength) {
- if (contentLength && this._fullRequestFlag) {
- this._totalLength = contentLength;
- this._fullRequestFlag = false;
- }
- }
- }, {
- key: '_onLoaderChunkArrival',
- value: function _onLoaderChunkArrival(chunk, byteStart, receivedLength) {
- if (!this._onDataArrival) {
- throw new _exception.IllegalStateException('IOController: No existing consumer (onDataArrival) callback!');
- }
- if (this._paused) {
- return;
- }
- if (this._isEarlyEofReconnecting) {
- // Auto-reconnect for EarlyEof succeed, notify to upper-layer by callback
- this._isEarlyEofReconnecting = false;
- if (this._onRecoveredEarlyEof) {
- this._onRecoveredEarlyEof();
- }
- }
- this._speedSampler.addBytes(chunk.byteLength);
- // adjust stash buffer size according to network speed dynamically
- var KBps = this._speedSampler.lastSecondKBps;
- if (KBps !== 0) {
- var normalized = this._normalizeSpeed(KBps);
- if (this._speedNormalized !== normalized) {
- this._speedNormalized = normalized;
- this._adjustStashSize(normalized);
- }
- }
- if (!this._enableStash) {
- // disable stash
- if (this._stashUsed === 0) {
- // dispatch chunk directly to consumer;
- // check ret value (consumed bytes) and stash unconsumed to stashBuffer
- var consumed = this._dispatchChunks(chunk, byteStart);
- if (consumed < chunk.byteLength) {
- // unconsumed data remain.
- var remain = chunk.byteLength - consumed;
- if (remain > this._bufferSize) {
- this._expandBuffer(remain);
- }
- var stashArray = new Uint8Array(this._stashBuffer, 0, this._bufferSize);
- stashArray.set(new Uint8Array(chunk, consumed), 0);
- this._stashUsed += remain;
- this._stashByteStart = byteStart + consumed;
- }
- } else {
- // else: Merge chunk into stashBuffer, and dispatch stashBuffer to consumer.
- if (this._stashUsed + chunk.byteLength > this._bufferSize) {
- this._expandBuffer(this._stashUsed + chunk.byteLength);
- }
- var _stashArray = new Uint8Array(this._stashBuffer, 0, this._bufferSize);
- _stashArray.set(new Uint8Array(chunk), this._stashUsed);
- this._stashUsed += chunk.byteLength;
- var _consumed = this._dispatchChunks(this._stashBuffer.slice(0, this._stashUsed), this._stashByteStart);
- if (_consumed < this._stashUsed && _consumed > 0) {
- // unconsumed data remain
- var remainArray = new Uint8Array(this._stashBuffer, _consumed);
- _stashArray.set(remainArray, 0);
- }
- this._stashUsed -= _consumed;
- this._stashByteStart += _consumed;
- }
- } else {
- // enable stash
- if (this._stashUsed === 0 && this._stashByteStart === 0) {
- // seeked? or init chunk?
- // This is the first chunk after seek action
- this._stashByteStart = byteStart;
- }
- if (this._stashUsed + chunk.byteLength <= this._stashSize) {
- // just stash
- var _stashArray2 = new Uint8Array(this._stashBuffer, 0, this._stashSize);
- _stashArray2.set(new Uint8Array(chunk), this._stashUsed);
- this._stashUsed += chunk.byteLength;
- } else {
- // stashUsed + chunkSize > stashSize, size limit exceeded
- var _stashArray3 = new Uint8Array(this._stashBuffer, 0, this._bufferSize);
- if (this._stashUsed > 0) {
- // There're stash datas in buffer
- // dispatch the whole stashBuffer, and stash remain data
- // then append chunk to stashBuffer (stash)
- var buffer = this._stashBuffer.slice(0, this._stashUsed);
- var _consumed2 = this._dispatchChunks(buffer, this._stashByteStart);
- if (_consumed2 < buffer.byteLength) {
- if (_consumed2 > 0) {
- var _remainArray = new Uint8Array(buffer, _consumed2);
- _stashArray3.set(_remainArray, 0);
- this._stashUsed = _remainArray.byteLength;
- this._stashByteStart += _consumed2;
- }
- } else {
- this._stashUsed = 0;
- this._stashByteStart += _consumed2;
- }
- if (this._stashUsed + chunk.byteLength > this._bufferSize) {
- this._expandBuffer(this._stashUsed + chunk.byteLength);
- _stashArray3 = new Uint8Array(this._stashBuffer, 0, this._bufferSize);
- }
- _stashArray3.set(new Uint8Array(chunk), this._stashUsed);
- this._stashUsed += chunk.byteLength;
- } else {
- // stash buffer empty, but chunkSize > stashSize (oh, holy shit)
- // dispatch chunk directly and stash remain data
- var _consumed3 = this._dispatchChunks(chunk, byteStart);
- if (_consumed3 < chunk.byteLength) {
- var _remain = chunk.byteLength - _consumed3;
- if (_remain > this._bufferSize) {
- this._expandBuffer(_remain);
- _stashArray3 = new Uint8Array(this._stashBuffer, 0, this._bufferSize);
- }
- _stashArray3.set(new Uint8Array(chunk, _consumed3), 0);
- this._stashUsed += _remain;
- this._stashByteStart = byteStart + _consumed3;
- }
- }
- }
- }
- }
- }, {
- key: '_flushStashBuffer',
- value: function _flushStashBuffer(dropUnconsumed) {
- if (this._stashUsed > 0) {
- var buffer = this._stashBuffer.slice(0, this._stashUsed);
- var consumed = this._dispatchChunks(buffer, this._stashByteStart);
- var remain = buffer.byteLength - consumed;
- if (consumed < buffer.byteLength) {
- if (dropUnconsumed) {
- _logger2.default.w(this.TAG, remain + ' bytes unconsumed data remain when flush buffer, dropped');
- } else {
- if (consumed > 0) {
- var stashArray = new Uint8Array(this._stashBuffer, 0, this._bufferSize);
- var remainArray = new Uint8Array(buffer, consumed);
- stashArray.set(remainArray, 0);
- this._stashUsed = remainArray.byteLength;
- this._stashByteStart += consumed;
- }
- return 0;
- }
- }
- this._stashUsed = 0;
- this._stashByteStart = 0;
- return remain;
- }
- return 0;
- }
- }, {
- key: '_onLoaderComplete',
- value: function _onLoaderComplete(from, to) {
- // Force-flush stash buffer, and drop unconsumed data
- this._flushStashBuffer(true);
- if (this._onComplete) {
- this._onComplete(this._extraData);
- }
- }
- }, {
- key: '_onLoaderError',
- value: function _onLoaderError(type, data) {
- _logger2.default.e(this.TAG, 'Loader error, code = ' + data.code + ', msg = ' + data.msg);
- this._flushStashBuffer(false);
- if (this._isEarlyEofReconnecting) {
- // Auto-reconnect for EarlyEof failed, throw UnrecoverableEarlyEof error to upper-layer
- this._isEarlyEofReconnecting = false;
- type = _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF;
- }
- switch (type) {
- case _loader.LoaderErrors.EARLY_EOF:
- {
- if (!this._config.isLive) {
- // Do internal http reconnect if not live stream
- if (this._totalLength) {
- var nextFrom = this._currentRange.to + 1;
- if (nextFrom < this._totalLength) {
- _logger2.default.w(this.TAG, 'Connection lost, trying reconnect...');
- this._isEarlyEofReconnecting = true;
- this._internalSeek(nextFrom, false);
- }
- return;
- }
- // else: We don't know totalLength, throw UnrecoverableEarlyEof
- }
- // live stream: throw UnrecoverableEarlyEof error to upper-layer
- type = _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF;
- break;
- }
- case _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF:
- case _loader.LoaderErrors.CONNECTING_TIMEOUT:
- case _loader.LoaderErrors.HTTP_STATUS_CODE_INVALID:
- case _loader.LoaderErrors.EXCEPTION:
- break;
- }
- if (this._onError) {
- this._onError(type, data);
- } else {
- throw new _exception.RuntimeException('IOException: ' + data.msg);
- }
- }
- }, {
- key: 'status',
- get: function get() {
- return this._loader.status;
- }
- }, {
- key: 'extraData',
- get: function get() {
- return this._extraData;
- },
- set: function set(data) {
- this._extraData = data;
- }
- // prototype: function onDataArrival(chunks: ArrayBuffer, byteStart: number): number
- }, {
- key: 'onDataArrival',
- get: function get() {
- return this._onDataArrival;
- },
- set: function set(callback) {
- this._onDataArrival = callback;
- }
- }, {
- key: 'onSeeked',
- get: function get() {
- return this._onSeeked;
- },
- set: function set(callback) {
- this._onSeeked = callback;
- }
- // prototype: function onError(type: number, info: {code: number, msg: string}): void
- }, {
- key: 'onError',
- get: function get() {
- return this._onError;
- },
- set: function set(callback) {
- this._onError = callback;
- }
- }, {
- key: 'onComplete',
- get: function get() {
- return this._onComplete;
- },
- set: function set(callback) {
- this._onComplete = callback;
- }
- }, {
- key: 'onRedirect',
- get: function get() {
- return this._onRedirect;
- },
- set: function set(callback) {
- this._onRedirect = callback;
- }
- }, {
- key: 'onRecoveredEarlyEof',
- get: function get() {
- return this._onRecoveredEarlyEof;
- },
- set: function set(callback) {
- this._onRecoveredEarlyEof = callback;
- }
- }, {
- key: 'currentURL',
- get: function get() {
- return this._dataSource.url;
- }
- }, {
- key: 'hasRedirect',
- get: function get() {
- return this._redirectedURL != null || this._dataSource.redirectedURL != undefined;
- }
- }, {
- key: 'currentRedirectedURL',
- get: function get() {
- return this._redirectedURL || this._dataSource.redirectedURL;
- }
- // in KB/s
- }, {
- key: 'currentSpeed',
- get: function get() {
- if (this._loaderClass === _xhrRangeLoader2.default) {
- // SpeedSampler is inaccuracy if loader is RangeLoader
- return this._loader.currentSpeed;
- }
- return this._speedSampler.lastSecondKBps;
- }
- }, {
- key: 'loaderType',
- get: function get() {
- return this._loader.type;
- }
- }]);
- return IOController;
- }();
- exports.default = IOController;
- },{"../utils/exception.js":40,"../utils/logger.js":41,"./fetch-stream-loader.js":22,"./loader.js":24,"./param-seek-handler.js":25,"./range-seek-handler.js":26,"./speed-sampler.js":27,"./websocket-loader.js":28,"./xhr-moz-chunked-loader.js":29,"./xhr-msstream-loader.js":30,"./xhr-range-loader.js":31}],24:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.BaseLoader = exports.LoaderErrors = exports.LoaderStatus = undefined;
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _exception = _dereq_('../utils/exception.js');
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var LoaderStatus = exports.LoaderStatus = {
- kIdle: 0,
- kConnecting: 1,
- kBuffering: 2,
- kError: 3,
- kComplete: 4
- };
- var LoaderErrors = exports.LoaderErrors = {
- OK: 'OK',
- EXCEPTION: 'Exception',
- HTTP_STATUS_CODE_INVALID: 'HttpStatusCodeInvalid',
- CONNECTING_TIMEOUT: 'ConnectingTimeout',
- EARLY_EOF: 'EarlyEof',
- UNRECOVERABLE_EARLY_EOF: 'UnrecoverableEarlyEof'
- };
- /* Loader has callbacks which have following prototypes:
- * function onContentLengthKnown(contentLength: number): void
- * function onURLRedirect(url: string): void
- * function onDataArrival(chunk: ArrayBuffer, byteStart: number, receivedLength: number): void
- * function onError(errorType: number, errorInfo: {code: number, msg: string}): void
- * function onComplete(rangeFrom: number, rangeTo: number): void
- */
- var BaseLoader = exports.BaseLoader = function () {
- function BaseLoader(typeName) {
- _classCallCheck(this, BaseLoader);
- this._type = typeName || 'undefined';
- this._status = LoaderStatus.kIdle;
- this._needStash = false;
- // callbacks
- this._onContentLengthKnown = null;
- this._onURLRedirect = null;
- this._onDataArrival = null;
- this._onError = null;
- this._onComplete = null;
- }
- _createClass(BaseLoader, [{
- key: 'destroy',
- value: function destroy() {
- this._status = LoaderStatus.kIdle;
- this._onContentLengthKnown = null;
- this._onURLRedirect = null;
- this._onDataArrival = null;
- this._onError = null;
- this._onComplete = null;
- }
- }, {
- key: 'isWorking',
- value: function isWorking() {
- return this._status === LoaderStatus.kConnecting || this._status === LoaderStatus.kBuffering;
- }
- }, {
- key: 'open',
- // pure virtual
- value: function open(dataSource, range) {
- throw new _exception.NotImplementedException('Unimplemented abstract function!');
- }
- }, {
- key: 'abort',
- value: function abort() {
- throw new _exception.NotImplementedException('Unimplemented abstract function!');
- }
- }, {
- key: 'type',
- get: function get() {
- return this._type;
- }
- }, {
- key: 'status',
- get: function get() {
- return this._status;
- }
- }, {
- key: 'needStashBuffer',
- get: function get() {
- return this._needStash;
- }
- }, {
- key: 'onContentLengthKnown',
- get: function get() {
- return this._onContentLengthKnown;
- },
- set: function set(callback) {
- this._onContentLengthKnown = callback;
- }
- }, {
- key: 'onURLRedirect',
- get: function get() {
- return this._onURLRedirect;
- },
- set: function set(callback) {
- this._onURLRedirect = callback;
- }
- }, {
- key: 'onDataArrival',
- get: function get() {
- return this._onDataArrival;
- },
- set: function set(callback) {
- this._onDataArrival = callback;
- }
- }, {
- key: 'onError',
- get: function get() {
- return this._onError;
- },
- set: function set(callback) {
- this._onError = callback;
- }
- }, {
- key: 'onComplete',
- get: function get() {
- return this._onComplete;
- },
- set: function set(callback) {
- this._onComplete = callback;
- }
- }]);
- return BaseLoader;
- }();
- },{"../utils/exception.js":40}],25:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var ParamSeekHandler = function () {
- function ParamSeekHandler(paramStart, paramEnd) {
- _classCallCheck(this, ParamSeekHandler);
- this._startName = paramStart;
- this._endName = paramEnd;
- }
- _createClass(ParamSeekHandler, [{
- key: 'getConfig',
- value: function getConfig(baseUrl, range) {
- var url = baseUrl;
- if (range.from !== 0 || range.to !== -1) {
- var needAnd = true;
- if (url.indexOf('?') === -1) {
- url += '?';
- needAnd = false;
- }
- if (needAnd) {
- url += '&';
- }
- url += this._startName + '=' + range.from.toString();
- if (range.to !== -1) {
- url += '&' + this._endName + '=' + range.to.toString();
- }
- }
- return {
- url: url,
- headers: {}
- };
- }
- }, {
- key: 'removeURLParameters',
- value: function removeURLParameters(seekedURL) {
- var baseURL = seekedURL.split('?')[0];
- var params = undefined;
- var queryIndex = seekedURL.indexOf('?');
- if (queryIndex !== -1) {
- params = seekedURL.substring(queryIndex + 1);
- }
- var resultParams = '';
- if (params != undefined && params.length > 0) {
- var pairs = params.split('&');
- for (var i = 0; i < pairs.length; i++) {
- var pair = pairs[i].split('=');
- var requireAnd = i > 0;
- if (pair[0] !== this._startName && pair[0] !== this._endName) {
- if (requireAnd) {
- resultParams += '&';
- }
- resultParams += pairs[i];
- }
- }
- }
- return resultParams.length === 0 ? baseURL : baseURL + '?' + resultParams;
- }
- }]);
- return ParamSeekHandler;
- }();
- exports.default = ParamSeekHandler;
- },{}],26:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var RangeSeekHandler = function () {
- function RangeSeekHandler(zeroStart) {
- _classCallCheck(this, RangeSeekHandler);
- this._zeroStart = zeroStart || false;
- }
- _createClass(RangeSeekHandler, [{
- key: 'getConfig',
- value: function getConfig(url, range) {
- var headers = {};
- if (range.from !== 0 || range.to !== -1) {
- var param = void 0;
- if (range.to !== -1) {
- param = 'bytes=' + range.from.toString() + '-' + range.to.toString();
- } else {
- param = 'bytes=' + range.from.toString() + '-';
- }
- headers['Range'] = param;
- } else if (this._zeroStart) {
- headers['Range'] = 'bytes=0-';
- }
- return {
- url: url,
- headers: headers
- };
- }
- }, {
- key: 'removeURLParameters',
- value: function removeURLParameters(seekedURL) {
- return seekedURL;
- }
- }]);
- return RangeSeekHandler;
- }();
- exports.default = RangeSeekHandler;
- },{}],27:[function(_dereq_,module,exports){
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- // Utility class to calculate realtime network I/O speed
- var SpeedSampler = function () {
- function SpeedSampler() {
- _classCallCheck(this, SpeedSampler);
- // milliseconds
- this._firstCheckpoint = 0;
- this._lastCheckpoint = 0;
- this._intervalBytes = 0;
- this._totalBytes = 0;
- this._lastSecondBytes = 0;
- // compatibility detection
- if (self.performance && self.performance.now) {
- this._now = self.performance.now.bind(self.performance);
- } else {
- this._now = Date.now;
- }
- }
- _createClass(SpeedSampler, [{
- key: "reset",
- value: function reset() {
- this._firstCheckpoint = this._lastCheckpoint = 0;
- this._totalBytes = this._intervalBytes = 0;
- this._lastSecondBytes = 0;
- }
- }, {
- key: "addBytes",
- value: function addBytes(bytes) {
- if (this._firstCheckpoint === 0) {
- this._firstCheckpoint = this._now();
- this._lastCheckpoint = this._firstCheckpoint;
- this._intervalBytes += bytes;
- this._totalBytes += bytes;
- } else if (this._now() - this._lastCheckpoint < 1000) {
- this._intervalBytes += bytes;
- this._totalBytes += bytes;
- } else {
- // duration >= 1000
- this._lastSecondBytes = this._intervalBytes;
- this._intervalBytes = bytes;
- this._totalBytes += bytes;
- this._lastCheckpoint = this._now();
- }
- }
- }, {
- key: "currentKBps",
- get: function get() {
- this.addBytes(0);
- var durationSeconds = (this._now() - this._lastCheckpoint) / 1000;
- if (durationSeconds == 0) durationSeconds = 1;
- return this._intervalBytes / durationSeconds / 1024;
- }
- }, {
- key: "lastSecondKBps",
- get: function get() {
- this.addBytes(0);
- if (this._lastSecondBytes !== 0) {
- return this._lastSecondBytes / 1024;
- } else {
- // lastSecondBytes === 0
- if (this._now() - this._lastCheckpoint >= 500) {
- // if time interval since last checkpoint has exceeded 500ms
- // the speed is nearly accurate
- return this.currentKBps;
- } else {
- // We don't know
- return 0;
- }
- }
- }
- }, {
- key: "averageKBps",
- get: function get() {
- var durationSeconds = (this._now() - this._firstCheckpoint) / 1000;
- return this._totalBytes / durationSeconds / 1024;
- }
- }]);
- return SpeedSampler;
- }();
- exports.default = SpeedSampler;
- },{}],28:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _loader = _dereq_('./loader.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
- function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- // For FLV over WebSocket live stream
- var WebSocketLoader = function (_BaseLoader) {
- _inherits(WebSocketLoader, _BaseLoader);
- _createClass(WebSocketLoader, null, [{
- key: 'isSupported',
- value: function isSupported() {
- try {
- return typeof self.WebSocket !== 'undefined';
- } catch (e) {
- return false;
- }
- }
- }]);
- function WebSocketLoader() {
- _classCallCheck(this, WebSocketLoader);
- var _this = _possibleConstructorReturn(this, (WebSocketLoader.__proto__ || Object.getPrototypeOf(WebSocketLoader)).call(this, 'websocket-loader'));
- _this.TAG = 'WebSocketLoader';
- _this._needStash = true;
- _this._ws = null;
- _this._requestAbort = false;
- _this._receivedLength = 0;
- return _this;
- }
- _createClass(WebSocketLoader, [{
- key: 'destroy',
- value: function destroy() {
- if (this._ws) {
- this.abort();
- }
- _get(WebSocketLoader.prototype.__proto__ || Object.getPrototypeOf(WebSocketLoader.prototype), 'destroy', this).call(this);
- }
- }, {
- key: 'open',
- value: function open(dataSource) {
- try {
- var ws = this._ws = new self.WebSocket(dataSource.url);
- ws.binaryType = 'arraybuffer';
- ws.onopen = this._onWebSocketOpen.bind(this);
- ws.onclose = this._onWebSocketClose.bind(this);
- ws.onmessage = this._onWebSocketMessage.bind(this);
- ws.onerror = this._onWebSocketError.bind(this);
- this._status = _loader.LoaderStatus.kConnecting;
- } catch (e) {
- this._status = _loader.LoaderStatus.kError;
- var info = { code: e.code, msg: e.message };
- if (this._onError) {
- this._onError(_loader.LoaderErrors.EXCEPTION, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }
- }, {
- key: 'abort',
- value: function abort() {
- var ws = this._ws;
- if (ws && (ws.readyState === 0 || ws.readyState === 1)) {
- // CONNECTING || OPEN
- this._requestAbort = true;
- ws.close();
- }
- this._ws = null;
- this._status = _loader.LoaderStatus.kComplete;
- }
- }, {
- key: '_onWebSocketOpen',
- value: function _onWebSocketOpen(e) {
- this._status = _loader.LoaderStatus.kBuffering;
- }
- }, {
- key: '_onWebSocketClose',
- value: function _onWebSocketClose(e) {
- if (this._requestAbort === true) {
- this._requestAbort = false;
- return;
- }
- this._status = _loader.LoaderStatus.kComplete;
- if (this._onComplete) {
- this._onComplete(0, this._receivedLength - 1);
- }
- }
- }, {
- key: '_onWebSocketMessage',
- value: function _onWebSocketMessage(e) {
- var _this2 = this;
- if (e.data instanceof ArrayBuffer) {
- this._dispatchArrayBuffer(e.data);
- } else if (e.data instanceof Blob) {
- var reader = new FileReader();
- reader.onload = function () {
- _this2._dispatchArrayBuffer(reader.result);
- };
- reader.readAsArrayBuffer(e.data);
- } else {
- this._status = _loader.LoaderStatus.kError;
- var info = { code: -1, msg: 'Unsupported WebSocket message type: ' + e.data.constructor.name };
- if (this._onError) {
- this._onError(_loader.LoaderErrors.EXCEPTION, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }
- }, {
- key: '_dispatchArrayBuffer',
- value: function _dispatchArrayBuffer(arraybuffer) {
- var chunk = arraybuffer;
- var byteStart = this._receivedLength;
- this._receivedLength += chunk.byteLength;
- if (this._onDataArrival) {
- this._onDataArrival(chunk, byteStart, this._receivedLength);
- }
- }
- }, {
- key: '_onWebSocketError',
- value: function _onWebSocketError(e) {
- this._status = _loader.LoaderStatus.kError;
- var info = {
- code: e.code,
- msg: e.message
- };
- if (this._onError) {
- this._onError(_loader.LoaderErrors.EXCEPTION, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }]);
- return WebSocketLoader;
- }(_loader.BaseLoader);
- exports.default = WebSocketLoader;
- },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],29:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _loader = _dereq_('./loader.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
- function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- // For FireFox browser which supports `xhr.responseType = 'moz-chunked-arraybuffer'`
- var MozChunkedLoader = function (_BaseLoader) {
- _inherits(MozChunkedLoader, _BaseLoader);
- _createClass(MozChunkedLoader, null, [{
- key: 'isSupported',
- value: function isSupported() {
- try {
- var xhr = new XMLHttpRequest();
- // Firefox 37- requires .open() to be called before setting responseType
- xhr.open('GET', 'https://example.com', true);
- xhr.responseType = 'moz-chunked-arraybuffer';
- return xhr.responseType === 'moz-chunked-arraybuffer';
- } catch (e) {
- _logger2.default.w('MozChunkedLoader', e.message);
- return false;
- }
- }
- }]);
- function MozChunkedLoader(seekHandler, config) {
- _classCallCheck(this, MozChunkedLoader);
- var _this = _possibleConstructorReturn(this, (MozChunkedLoader.__proto__ || Object.getPrototypeOf(MozChunkedLoader)).call(this, 'xhr-moz-chunked-loader'));
- _this.TAG = 'MozChunkedLoader';
- _this._seekHandler = seekHandler;
- _this._config = config;
- _this._needStash = true;
- _this._xhr = null;
- _this._requestAbort = false;
- _this._contentLength = null;
- _this._receivedLength = 0;
- return _this;
- }
- _createClass(MozChunkedLoader, [{
- key: 'destroy',
- value: function destroy() {
- if (this.isWorking()) {
- this.abort();
- }
- if (this._xhr) {
- this._xhr.onreadystatechange = null;
- this._xhr.onprogress = null;
- this._xhr.onloadend = null;
- this._xhr.onerror = null;
- this._xhr = null;
- }
- _get(MozChunkedLoader.prototype.__proto__ || Object.getPrototypeOf(MozChunkedLoader.prototype), 'destroy', this).call(this);
- }
- }, {
- key: 'open',
- value: function open(dataSource, range) {
- this._dataSource = dataSource;
- this._range = range;
- var sourceURL = dataSource.url;
- if (this._config.reuseRedirectedURL && dataSource.redirectedURL != undefined) {
- sourceURL = dataSource.redirectedURL;
- }
- var seekConfig = this._seekHandler.getConfig(sourceURL, range);
- this._requestURL = seekConfig.url;
- var xhr = this._xhr = new XMLHttpRequest();
- xhr.open('GET', seekConfig.url, true);
- xhr.responseType = 'moz-chunked-arraybuffer';
- xhr.onreadystatechange = this._onReadyStateChange.bind(this);
- xhr.onprogress = this._onProgress.bind(this);
- xhr.onloadend = this._onLoadEnd.bind(this);
- xhr.onerror = this._onXhrError.bind(this);
- // cors is auto detected and enabled by xhr
- // withCredentials is disabled by default
- if (dataSource.withCredentials) {
- xhr.withCredentials = true;
- }
- if (_typeof(seekConfig.headers) === 'object') {
- var headers = seekConfig.headers;
- for (var key in headers) {
- if (headers.hasOwnProperty(key)) {
- xhr.setRequestHeader(key, headers[key]);
- }
- }
- }
- this._status = _loader.LoaderStatus.kConnecting;
- xhr.send();
- }
- }, {
- key: 'abort',
- value: function abort() {
- this._requestAbort = true;
- if (this._xhr) {
- this._xhr.abort();
- }
- this._status = _loader.LoaderStatus.kComplete;
- }
- }, {
- key: '_onReadyStateChange',
- value: function _onReadyStateChange(e) {
- var xhr = e.target;
- if (xhr.readyState === 2) {
- // HEADERS_RECEIVED
- if (xhr.responseURL != undefined && xhr.responseURL !== this._requestURL) {
- if (this._onURLRedirect) {
- var redirectedURL = this._seekHandler.removeURLParameters(xhr.responseURL);
- this._onURLRedirect(redirectedURL);
- }
- }
- if (xhr.status !== 0 && (xhr.status < 200 || xhr.status > 299)) {
- this._status = _loader.LoaderStatus.kError;
- if (this._onError) {
- this._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: xhr.status, msg: xhr.statusText });
- } else {
- throw new _exception.RuntimeException('MozChunkedLoader: Http code invalid, ' + xhr.status + ' ' + xhr.statusText);
- }
- } else {
- this._status = _loader.LoaderStatus.kBuffering;
- }
- }
- }
- }, {
- key: '_onProgress',
- value: function _onProgress(e) {
- if (this._status === _loader.LoaderStatus.kError) {
- // Ignore error response
- return;
- }
- if (this._contentLength === null) {
- if (e.total !== null && e.total !== 0) {
- this._contentLength = e.total;
- if (this._onContentLengthKnown) {
- this._onContentLengthKnown(this._contentLength);
- }
- }
- }
- var chunk = e.target.response;
- var byteStart = this._range.from + this._receivedLength;
- this._receivedLength += chunk.byteLength;
- if (this._onDataArrival) {
- this._onDataArrival(chunk, byteStart, this._receivedLength);
- }
- }
- }, {
- key: '_onLoadEnd',
- value: function _onLoadEnd(e) {
- if (this._requestAbort === true) {
- this._requestAbort = false;
- return;
- } else if (this._status === _loader.LoaderStatus.kError) {
- return;
- }
- this._status = _loader.LoaderStatus.kComplete;
- if (this._onComplete) {
- this._onComplete(this._range.from, this._range.from + this._receivedLength - 1);
- }
- }
- }, {
- key: '_onXhrError',
- value: function _onXhrError(e) {
- this._status = _loader.LoaderStatus.kError;
- var type = 0;
- var info = null;
- if (this._contentLength && e.loaded < this._contentLength) {
- type = _loader.LoaderErrors.EARLY_EOF;
- info = { code: -1, msg: 'Moz-Chunked stream meet Early-Eof' };
- } else {
- type = _loader.LoaderErrors.EXCEPTION;
- info = { code: -1, msg: e.constructor.name + ' ' + e.type };
- }
- if (this._onError) {
- this._onError(type, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }]);
- return MozChunkedLoader;
- }(_loader.BaseLoader);
- exports.default = MozChunkedLoader;
- },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],30:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _loader = _dereq_('./loader.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
- function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- /* Notice: ms-stream may cause IE/Edge browser crash if seek too frequently!!!
- * The browser may crash in wininet.dll. Disable for now.
- *
- * For IE11/Edge browser by microsoft which supports `xhr.responseType = 'ms-stream'`
- * Notice that ms-stream API sucks. The buffer is always expanding along with downloading.
- *
- * We need to abort the xhr if buffer size exceeded limit size (e.g. 16 MiB), then do reconnect.
- * in order to release previous ArrayBuffer to avoid memory leak
- *
- * Otherwise, the ArrayBuffer will increase to a terrible size that equals final file size.
- */
- var MSStreamLoader = function (_BaseLoader) {
- _inherits(MSStreamLoader, _BaseLoader);
- _createClass(MSStreamLoader, null, [{
- key: 'isSupported',
- value: function isSupported() {
- try {
- if (typeof self.MSStream === 'undefined' || typeof self.MSStreamReader === 'undefined') {
- return false;
- }
- var xhr = new XMLHttpRequest();
- xhr.open('GET', 'https://example.com', true);
- xhr.responseType = 'ms-stream';
- return xhr.responseType === 'ms-stream';
- } catch (e) {
- _logger2.default.w('MSStreamLoader', e.message);
- return false;
- }
- }
- }]);
- function MSStreamLoader(seekHandler, config) {
- _classCallCheck(this, MSStreamLoader);
- var _this = _possibleConstructorReturn(this, (MSStreamLoader.__proto__ || Object.getPrototypeOf(MSStreamLoader)).call(this, 'xhr-msstream-loader'));
- _this.TAG = 'MSStreamLoader';
- _this._seekHandler = seekHandler;
- _this._config = config;
- _this._needStash = true;
- _this._xhr = null;
- _this._reader = null; // MSStreamReader
- _this._totalRange = null;
- _this._currentRange = null;
- _this._currentRequestURL = null;
- _this._currentRedirectedURL = null;
- _this._contentLength = null;
- _this._receivedLength = 0;
- _this._bufferLimit = 16 * 1024 * 1024; // 16MB
- _this._lastTimeBufferSize = 0;
- _this._isReconnecting = false;
- return _this;
- }
- _createClass(MSStreamLoader, [{
- key: 'destroy',
- value: function destroy() {
- if (this.isWorking()) {
- this.abort();
- }
- if (this._reader) {
- this._reader.onprogress = null;
- this._reader.onload = null;
- this._reader.onerror = null;
- this._reader = null;
- }
- if (this._xhr) {
- this._xhr.onreadystatechange = null;
- this._xhr = null;
- }
- _get(MSStreamLoader.prototype.__proto__ || Object.getPrototypeOf(MSStreamLoader.prototype), 'destroy', this).call(this);
- }
- }, {
- key: 'open',
- value: function open(dataSource, range) {
- this._internalOpen(dataSource, range, false);
- }
- }, {
- key: '_internalOpen',
- value: function _internalOpen(dataSource, range, isSubrange) {
- this._dataSource = dataSource;
- if (!isSubrange) {
- this._totalRange = range;
- } else {
- this._currentRange = range;
- }
- var sourceURL = dataSource.url;
- if (this._config.reuseRedirectedURL) {
- if (this._currentRedirectedURL != undefined) {
- sourceURL = this._currentRedirectedURL;
- } else if (dataSource.redirectedURL != undefined) {
- sourceURL = dataSource.redirectedURL;
- }
- }
- var seekConfig = this._seekHandler.getConfig(sourceURL, range);
- this._currentRequestURL = seekConfig.url;
- var reader = this._reader = new self.MSStreamReader();
- reader.onprogress = this._msrOnProgress.bind(this);
- reader.onload = this._msrOnLoad.bind(this);
- reader.onerror = this._msrOnError.bind(this);
- var xhr = this._xhr = new XMLHttpRequest();
- xhr.open('GET', seekConfig.url, true);
- xhr.responseType = 'ms-stream';
- xhr.onreadystatechange = this._xhrOnReadyStateChange.bind(this);
- xhr.onerror = this._xhrOnError.bind(this);
- if (dataSource.withCredentials) {
- xhr.withCredentials = true;
- }
- if (_typeof(seekConfig.headers) === 'object') {
- var headers = seekConfig.headers;
- for (var key in headers) {
- if (headers.hasOwnProperty(key)) {
- xhr.setRequestHeader(key, headers[key]);
- }
- }
- }
- if (this._isReconnecting) {
- this._isReconnecting = false;
- } else {
- this._status = _loader.LoaderStatus.kConnecting;
- }
- xhr.send();
- }
- }, {
- key: 'abort',
- value: function abort() {
- this._internalAbort();
- this._status = _loader.LoaderStatus.kComplete;
- }
- }, {
- key: '_internalAbort',
- value: function _internalAbort() {
- if (this._reader) {
- if (this._reader.readyState === 1) {
- // LOADING
- this._reader.abort();
- }
- this._reader.onprogress = null;
- this._reader.onload = null;
- this._reader.onerror = null;
- this._reader = null;
- }
- if (this._xhr) {
- this._xhr.abort();
- this._xhr.onreadystatechange = null;
- this._xhr = null;
- }
- }
- }, {
- key: '_xhrOnReadyStateChange',
- value: function _xhrOnReadyStateChange(e) {
- var xhr = e.target;
- if (xhr.readyState === 2) {
- // HEADERS_RECEIVED
- if (xhr.status >= 200 && xhr.status <= 299) {
- this._status = _loader.LoaderStatus.kBuffering;
- if (xhr.responseURL != undefined) {
- var redirectedURL = this._seekHandler.removeURLParameters(xhr.responseURL);
- if (xhr.responseURL !== this._currentRequestURL && redirectedURL !== this._currentRedirectedURL) {
- this._currentRedirectedURL = redirectedURL;
- if (this._onURLRedirect) {
- this._onURLRedirect(redirectedURL);
- }
- }
- }
- var lengthHeader = xhr.getResponseHeader('Content-Length');
- if (lengthHeader != null && this._contentLength == null) {
- var length = parseInt(lengthHeader);
- if (length > 0) {
- this._contentLength = length;
- if (this._onContentLengthKnown) {
- this._onContentLengthKnown(this._contentLength);
- }
- }
- }
- } else {
- this._status = _loader.LoaderStatus.kError;
- if (this._onError) {
- this._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: xhr.status, msg: xhr.statusText });
- } else {
- throw new _exception.RuntimeException('MSStreamLoader: Http code invalid, ' + xhr.status + ' ' + xhr.statusText);
- }
- }
- } else if (xhr.readyState === 3) {
- // LOADING
- if (xhr.status >= 200 && xhr.status <= 299) {
- this._status = _loader.LoaderStatus.kBuffering;
- var msstream = xhr.response;
- this._reader.readAsArrayBuffer(msstream);
- }
- }
- }
- }, {
- key: '_xhrOnError',
- value: function _xhrOnError(e) {
- this._status = _loader.LoaderStatus.kError;
- var type = _loader.LoaderErrors.EXCEPTION;
- var info = { code: -1, msg: e.constructor.name + ' ' + e.type };
- if (this._onError) {
- this._onError(type, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }, {
- key: '_msrOnProgress',
- value: function _msrOnProgress(e) {
- var reader = e.target;
- var bigbuffer = reader.result;
- if (bigbuffer == null) {
- // result may be null, workaround for buggy M$
- this._doReconnectIfNeeded();
- return;
- }
- var slice = bigbuffer.slice(this._lastTimeBufferSize);
- this._lastTimeBufferSize = bigbuffer.byteLength;
- var byteStart = this._totalRange.from + this._receivedLength;
- this._receivedLength += slice.byteLength;
- if (this._onDataArrival) {
- this._onDataArrival(slice, byteStart, this._receivedLength);
- }
- if (bigbuffer.byteLength >= this._bufferLimit) {
- _logger2.default.v(this.TAG, 'MSStream buffer exceeded max size near ' + (byteStart + slice.byteLength) + ', reconnecting...');
- this._doReconnectIfNeeded();
- }
- }
- }, {
- key: '_doReconnectIfNeeded',
- value: function _doReconnectIfNeeded() {
- if (this._contentLength == null || this._receivedLength < this._contentLength) {
- this._isReconnecting = true;
- this._lastTimeBufferSize = 0;
- this._internalAbort();
- var range = {
- from: this._totalRange.from + this._receivedLength,
- to: -1
- };
- this._internalOpen(this._dataSource, range, true);
- }
- }
- }, {
- key: '_msrOnLoad',
- value: function _msrOnLoad(e) {
- // actually it is onComplete event
- this._status = _loader.LoaderStatus.kComplete;
- if (this._onComplete) {
- this._onComplete(this._totalRange.from, this._totalRange.from + this._receivedLength - 1);
- }
- }
- }, {
- key: '_msrOnError',
- value: function _msrOnError(e) {
- this._status = _loader.LoaderStatus.kError;
- var type = 0;
- var info = null;
- if (this._contentLength && this._receivedLength < this._contentLength) {
- type = _loader.LoaderErrors.EARLY_EOF;
- info = { code: -1, msg: 'MSStream meet Early-Eof' };
- } else {
- type = _loader.LoaderErrors.EARLY_EOF;
- info = { code: -1, msg: e.constructor.name + ' ' + e.type };
- }
- if (this._onError) {
- this._onError(type, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }]);
- return MSStreamLoader;
- }(_loader.BaseLoader);
- exports.default = MSStreamLoader;
- },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],31:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _get = function get(object, property, receiver) { if (object === null) object = Function.prototype; var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _speedSampler = _dereq_('./speed-sampler.js');
- var _speedSampler2 = _interopRequireDefault(_speedSampler);
- var _loader = _dereq_('./loader.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
- function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- // Universal IO Loader, implemented by adding Range header in xhr's request header
- var RangeLoader = function (_BaseLoader) {
- _inherits(RangeLoader, _BaseLoader);
- _createClass(RangeLoader, null, [{
- key: 'isSupported',
- value: function isSupported() {
- try {
- var xhr = new XMLHttpRequest();
- xhr.open('GET', 'https://example.com', true);
- xhr.responseType = 'arraybuffer';
- return xhr.responseType === 'arraybuffer';
- } catch (e) {
- _logger2.default.w('RangeLoader', e.message);
- return false;
- }
- }
- }]);
- function RangeLoader(seekHandler, config) {
- _classCallCheck(this, RangeLoader);
- var _this = _possibleConstructorReturn(this, (RangeLoader.__proto__ || Object.getPrototypeOf(RangeLoader)).call(this, 'xhr-range-loader'));
- _this.TAG = 'RangeLoader';
- _this._seekHandler = seekHandler;
- _this._config = config;
- _this._needStash = false;
- _this._chunkSizeKBList = [128, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096, 5120, 6144, 7168, 8192];
- _this._currentChunkSizeKB = 384;
- _this._currentSpeedNormalized = 0;
- _this._zeroSpeedChunkCount = 0;
- _this._xhr = null;
- _this._speedSampler = new _speedSampler2.default();
- _this._requestAbort = false;
- _this._waitForTotalLength = false;
- _this._totalLengthReceived = false;
- _this._currentRequestURL = null;
- _this._currentRedirectedURL = null;
- _this._currentRequestRange = null;
- _this._totalLength = null; // size of the entire file
- _this._contentLength = null; // Content-Length of entire request range
- _this._receivedLength = 0; // total received bytes
- _this._lastTimeLoaded = 0; // received bytes of current request sub-range
- return _this;
- }
- _createClass(RangeLoader, [{
- key: 'destroy',
- value: function destroy() {
- if (this.isWorking()) {
- this.abort();
- }
- if (this._xhr) {
- this._xhr.onreadystatechange = null;
- this._xhr.onprogress = null;
- this._xhr.onload = null;
- this._xhr.onerror = null;
- this._xhr = null;
- }
- _get(RangeLoader.prototype.__proto__ || Object.getPrototypeOf(RangeLoader.prototype), 'destroy', this).call(this);
- }
- }, {
- key: 'open',
- value: function open(dataSource, range) {
- this._dataSource = dataSource;
- this._range = range;
- this._status = _loader.LoaderStatus.kConnecting;
- var useRefTotalLength = false;
- if (this._dataSource.filesize != undefined && this._dataSource.filesize !== 0) {
- useRefTotalLength = true;
- this._totalLength = this._dataSource.filesize;
- }
- if (!this._totalLengthReceived && !useRefTotalLength) {
- // We need total filesize
- this._waitForTotalLength = true;
- this._internalOpen(this._dataSource, { from: 0, to: -1 });
- } else {
- // We have filesize, start loading
- this._openSubRange();
- }
- }
- }, {
- key: '_openSubRange',
- value: function _openSubRange() {
- var chunkSize = this._currentChunkSizeKB * 1024;
- var from = this._range.from + this._receivedLength;
- var to = from + chunkSize;
- if (this._contentLength != null) {
- if (to - this._range.from >= this._contentLength) {
- to = this._range.from + this._contentLength - 1;
- }
- }
- this._currentRequestRange = { from: from, to: to };
- this._internalOpen(this._dataSource, this._currentRequestRange);
- }
- }, {
- key: '_internalOpen',
- value: function _internalOpen(dataSource, range) {
- this._lastTimeLoaded = 0;
- var sourceURL = dataSource.url;
- if (this._config.reuseRedirectedURL) {
- if (this._currentRedirectedURL != undefined) {
- sourceURL = this._currentRedirectedURL;
- } else if (dataSource.redirectedURL != undefined) {
- sourceURL = dataSource.redirectedURL;
- }
- }
- var seekConfig = this._seekHandler.getConfig(sourceURL, range);
- this._currentRequestURL = seekConfig.url;
- var xhr = this._xhr = new XMLHttpRequest();
- xhr.open('GET', seekConfig.url, true);
- xhr.responseType = 'arraybuffer';
- xhr.onreadystatechange = this._onReadyStateChange.bind(this);
- xhr.onprogress = this._onProgress.bind(this);
- xhr.onload = this._onLoad.bind(this);
- xhr.onerror = this._onXhrError.bind(this);
- if (dataSource.withCredentials) {
- xhr.withCredentials = true;
- }
- if (_typeof(seekConfig.headers) === 'object') {
- var headers = seekConfig.headers;
- for (var key in headers) {
- if (headers.hasOwnProperty(key)) {
- xhr.setRequestHeader(key, headers[key]);
- }
- }
- }
- xhr.send();
- }
- }, {
- key: 'abort',
- value: function abort() {
- this._requestAbort = true;
- this._internalAbort();
- this._status = _loader.LoaderStatus.kComplete;
- }
- }, {
- key: '_internalAbort',
- value: function _internalAbort() {
- if (this._xhr) {
- this._xhr.onreadystatechange = null;
- this._xhr.onprogress = null;
- this._xhr.onload = null;
- this._xhr.onerror = null;
- this._xhr.abort();
- this._xhr = null;
- }
- }
- }, {
- key: '_onReadyStateChange',
- value: function _onReadyStateChange(e) {
- var xhr = e.target;
- if (xhr.readyState === 2) {
- // HEADERS_RECEIVED
- if (xhr.responseURL != undefined) {
- // if the browser support this property
- var redirectedURL = this._seekHandler.removeURLParameters(xhr.responseURL);
- if (xhr.responseURL !== this._currentRequestURL && redirectedURL !== this._currentRedirectedURL) {
- this._currentRedirectedURL = redirectedURL;
- if (this._onURLRedirect) {
- this._onURLRedirect(redirectedURL);
- }
- }
- }
- if (xhr.status >= 200 && xhr.status <= 299) {
- if (this._waitForTotalLength) {
- return;
- }
- this._status = _loader.LoaderStatus.kBuffering;
- } else {
- this._status = _loader.LoaderStatus.kError;
- if (this._onError) {
- this._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: xhr.status, msg: xhr.statusText });
- } else {
- throw new _exception.RuntimeException('RangeLoader: Http code invalid, ' + xhr.status + ' ' + xhr.statusText);
- }
- }
- }
- }
- }, {
- key: '_onProgress',
- value: function _onProgress(e) {
- if (this._status === _loader.LoaderStatus.kError) {
- // Ignore error response
- return;
- }
- if (this._contentLength === null) {
- var openNextRange = false;
- if (this._waitForTotalLength) {
- this._waitForTotalLength = false;
- this._totalLengthReceived = true;
- openNextRange = true;
- var total = e.total;
- this._internalAbort();
- if (total != null & total !== 0) {
- this._totalLength = total;
- }
- }
- // calculate currrent request range's contentLength
- if (this._range.to === -1) {
- this._contentLength = this._totalLength - this._range.from;
- } else {
- // to !== -1
- this._contentLength = this._range.to - this._range.from + 1;
- }
- if (openNextRange) {
- this._openSubRange();
- return;
- }
- if (this._onContentLengthKnown) {
- this._onContentLengthKnown(this._contentLength);
- }
- }
- var delta = e.loaded - this._lastTimeLoaded;
- this._lastTimeLoaded = e.loaded;
- this._speedSampler.addBytes(delta);
- }
- }, {
- key: '_normalizeSpeed',
- value: function _normalizeSpeed(input) {
- var list = this._chunkSizeKBList;
- var last = list.length - 1;
- var mid = 0;
- var lbound = 0;
- var ubound = last;
- if (input < list[0]) {
- return list[0];
- }
- while (lbound <= ubound) {
- mid = lbound + Math.floor((ubound - lbound) / 2);
- if (mid === last || input >= list[mid] && input < list[mid + 1]) {
- return list[mid];
- } else if (list[mid] < input) {
- lbound = mid + 1;
- } else {
- ubound = mid - 1;
- }
- }
- }
- }, {
- key: '_onLoad',
- value: function _onLoad(e) {
- if (this._status === _loader.LoaderStatus.kError) {
- // Ignore error response
- return;
- }
- if (this._waitForTotalLength) {
- this._waitForTotalLength = false;
- return;
- }
- this._lastTimeLoaded = 0;
- var KBps = this._speedSampler.lastSecondKBps;
- if (KBps === 0) {
- this._zeroSpeedChunkCount++;
- if (this._zeroSpeedChunkCount >= 3) {
- // Try get currentKBps after 3 chunks
- KBps = this._speedSampler.currentKBps;
- }
- }
- if (KBps !== 0) {
- var normalized = this._normalizeSpeed(KBps);
- if (this._currentSpeedNormalized !== normalized) {
- this._currentSpeedNormalized = normalized;
- this._currentChunkSizeKB = normalized;
- }
- }
- var chunk = e.target.response;
- var byteStart = this._range.from + this._receivedLength;
- this._receivedLength += chunk.byteLength;
- var reportComplete = false;
- if (this._contentLength != null && this._receivedLength < this._contentLength) {
- // continue load next chunk
- this._openSubRange();
- } else {
- reportComplete = true;
- }
- // dispatch received chunk
- if (this._onDataArrival) {
- this._onDataArrival(chunk, byteStart, this._receivedLength);
- }
- if (reportComplete) {
- this._status = _loader.LoaderStatus.kComplete;
- if (this._onComplete) {
- this._onComplete(this._range.from, this._range.from + this._receivedLength - 1);
- }
- }
- }
- }, {
- key: '_onXhrError',
- value: function _onXhrError(e) {
- this._status = _loader.LoaderStatus.kError;
- var type = 0;
- var info = null;
- if (this._contentLength && this._receivedLength > 0 && this._receivedLength < this._contentLength) {
- type = _loader.LoaderErrors.EARLY_EOF;
- info = { code: -1, msg: 'RangeLoader meet Early-Eof' };
- } else {
- type = _loader.LoaderErrors.EXCEPTION;
- info = { code: -1, msg: e.constructor.name + ' ' + e.type };
- }
- if (this._onError) {
- this._onError(type, info);
- } else {
- throw new _exception.RuntimeException(info.msg);
- }
- }
- }, {
- key: 'currentSpeed',
- get: function get() {
- return this._speedSampler.lastSecondKBps;
- }
- }]);
- return RangeLoader;
- }(_loader.BaseLoader);
- exports.default = RangeLoader;
- },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24,"./speed-sampler.js":27}],32:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _browser = _dereq_('../utils/browser.js');
- var _browser2 = _interopRequireDefault(_browser);
- var _playerEvents = _dereq_('./player-events.js');
- var _playerEvents2 = _interopRequireDefault(_playerEvents);
- var _transmuxer = _dereq_('../core/transmuxer.js');
- var _transmuxer2 = _interopRequireDefault(_transmuxer);
- var _transmuxingEvents = _dereq_('../core/transmuxing-events.js');
- var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents);
- var _mseController = _dereq_('../core/mse-controller.js');
- var _mseController2 = _interopRequireDefault(_mseController);
- var _mseEvents = _dereq_('../core/mse-events.js');
- var _mseEvents2 = _interopRequireDefault(_mseEvents);
- var _playerErrors = _dereq_('./player-errors.js');
- var _config = _dereq_('../config.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var FlvPlayer = function () {
- function FlvPlayer(mediaDataSource, config) {
- _classCallCheck(this, FlvPlayer);
- this.TAG = 'FlvPlayer';
- this._type = 'FlvPlayer';
- this._emitter = new _events2.default();
- this._config = (0, _config.createDefaultConfig)();
- if ((typeof config === 'undefined' ? 'undefined' : _typeof(config)) === 'object') {
- Object.assign(this._config, config);
- }
- if (mediaDataSource.type.toLowerCase() !== 'flv') {
- throw new _exception.InvalidArgumentException('FlvPlayer requires an flv MediaDataSource input!');
- }
- if (mediaDataSource.isLive === true) {
- this._config.isLive = true;
- }
- this.e = {
- onvLoadedMetadata: this._onvLoadedMetadata.bind(this),
- onvSeeking: this._onvSeeking.bind(this),
- onvCanPlay: this._onvCanPlay.bind(this),
- onvStalled: this._onvStalled.bind(this),
- onvProgress: this._onvProgress.bind(this)
- };
- if (self.performance && self.performance.now) {
- this._now = self.performance.now.bind(self.performance);
- } else {
- this._now = Date.now;
- }
- this._pendingSeekTime = null; // in seconds
- this._requestSetTime = false;
- this._seekpointRecord = null;
- this._progressChecker = null;
- this._mediaDataSource = mediaDataSource;
- this._mediaElement = null;
- this._msectl = null;
- this._transmuxer = null;
- this._mseSourceOpened = false;
- this._hasPendingLoad = false;
- this._receivedCanPlay = false;
- this._mediaInfo = null;
- this._statisticsInfo = null;
- var chromeNeedIDRFix = _browser2.default.chrome && (_browser2.default.version.major < 50 || _browser2.default.version.major === 50 && _browser2.default.version.build < 2661);
- this._alwaysSeekKeyframe = chromeNeedIDRFix || _browser2.default.msedge || _browser2.default.msie ? true : false;
- if (this._alwaysSeekKeyframe) {
- this._config.accurateSeek = false;
- }
- }
- _createClass(FlvPlayer, [{
- key: 'destroy',
- value: function destroy() {
- if (this._progressChecker != null) {
- window.clearInterval(this._progressChecker);
- this._progressChecker = null;
- }
- if (this._transmuxer) {
- this.unload();
- }
- if (this._mediaElement) {
- this.detachMediaElement();
- }
- this.e = null;
- this._mediaDataSource = null;
- this._emitter.removeAllListeners();
- this._emitter = null;
- }
- }, {
- key: 'on',
- value: function on(event, listener) {
- var _this = this;
- if (event === _playerEvents2.default.MEDIA_INFO) {
- if (this._mediaInfo != null) {
- Promise.resolve().then(function () {
- _this._emitter.emit(_playerEvents2.default.MEDIA_INFO, _this.mediaInfo);
- });
- }
- } else if (event === _playerEvents2.default.STATISTICS_INFO) {
- if (this._statisticsInfo != null) {
- Promise.resolve().then(function () {
- _this._emitter.emit(_playerEvents2.default.STATISTICS_INFO, _this.statisticsInfo);
- });
- }
- }
- this._emitter.addListener(event, listener);
- }
- }, {
- key: 'off',
- value: function off(event, listener) {
- this._emitter.removeListener(event, listener);
- }
- }, {
- key: 'attachMediaElement',
- value: function attachMediaElement(mediaElement) {
- var _this2 = this;
- this._mediaElement = mediaElement;
- mediaElement.addEventListener('loadedmetadata', this.e.onvLoadedMetadata);
- mediaElement.addEventListener('seeking', this.e.onvSeeking);
- mediaElement.addEventListener('canplay', this.e.onvCanPlay);
- mediaElement.addEventListener('stalled', this.e.onvStalled);
- mediaElement.addEventListener('progress', this.e.onvProgress);
- this._msectl = new _mseController2.default(this._config);
- this._msectl.on(_mseEvents2.default.UPDATE_END, this._onmseUpdateEnd.bind(this));
- this._msectl.on(_mseEvents2.default.BUFFER_FULL, this._onmseBufferFull.bind(this));
- this._msectl.on(_mseEvents2.default.SOURCE_OPEN, function () {
- _this2._mseSourceOpened = true;
- if (_this2._hasPendingLoad) {
- _this2._hasPendingLoad = false;
- _this2.load();
- }
- });
- this._msectl.on(_mseEvents2.default.ERROR, function (info) {
- _this2._emitter.emit(_playerEvents2.default.ERROR, _playerErrors.ErrorTypes.MEDIA_ERROR, _playerErrors.ErrorDetails.MEDIA_MSE_ERROR, info);
- });
- this._msectl.attachMediaElement(mediaElement);
- if (this._pendingSeekTime != null) {
- try {
- mediaElement.currentTime = this._pendingSeekTime;
- this._pendingSeekTime = null;
- } catch (e) {
- // IE11 may throw InvalidStateError if readyState === 0
- // We can defer set currentTime operation after loadedmetadata
- }
- }
- }
- }, {
- key: 'detachMediaElement',
- value: function detachMediaElement() {
- if (this._mediaElement) {
- this._msectl.detachMediaElement();
- this._mediaElement.removeEventListener('loadedmetadata', this.e.onvLoadedMetadata);
- this._mediaElement.removeEventListener('seeking', this.e.onvSeeking);
- this._mediaElement.removeEventListener('canplay', this.e.onvCanPlay);
- this._mediaElement.removeEventListener('stalled', this.e.onvStalled);
- this._mediaElement.removeEventListener('progress', this.e.onvProgress);
- this._mediaElement = null;
- }
- if (this._msectl) {
- this._msectl.destroy();
- this._msectl = null;
- }
- }
- }, {
- key: 'load',
- value: function load() {
- var _this3 = this;
- if (!this._mediaElement) {
- throw new _exception.IllegalStateException('HTMLMediaElement must be attached before load()!');
- }
- if (this._transmuxer) {
- throw new _exception.IllegalStateException('FlvPlayer.load() has been called, please call unload() first!');
- }
- if (this._hasPendingLoad) {
- return;
- }
- if (this._config.deferLoadAfterSourceOpen && this._mseSourceOpened === false) {
- this._hasPendingLoad = true;
- return;
- }
- if (this._mediaElement.readyState > 0) {
- this._requestSetTime = true;
- // IE11 may throw InvalidStateError if readyState === 0
- this._mediaElement.currentTime = 0;
- }
- this._transmuxer = new _transmuxer2.default(this._mediaDataSource, this._config);
- this._transmuxer.on(_transmuxingEvents2.default.INIT_SEGMENT, function (type, is) {
- _this3._msectl.appendInitSegment(is);
- });
- this._transmuxer.on(_transmuxingEvents2.default.MEDIA_SEGMENT, function (type, ms) {
- _this3._msectl.appendMediaSegment(ms);
- // lazyLoad check
- if (_this3._config.lazyLoad && !_this3._config.isLive) {
- var currentTime = _this3._mediaElement.currentTime;
- if (ms.info.endDts >= (currentTime + _this3._config.lazyLoadMaxDuration) * 1000) {
- if (_this3._progressChecker == null) {
- _logger2.default.v(_this3.TAG, 'Maximum buffering duration exceeded, suspend transmuxing task');
- _this3._suspendTransmuxer();
- }
- }
- }
- });
- this._transmuxer.on(_transmuxingEvents2.default.LOADING_COMPLETE, function () {
- _this3._msectl.endOfStream();
- _this3._emitter.emit(_playerEvents2.default.LOADING_COMPLETE);
- });
- this._transmuxer.on(_transmuxingEvents2.default.RECOVERED_EARLY_EOF, function () {
- _this3._emitter.emit(_playerEvents2.default.RECOVERED_EARLY_EOF);
- });
- this._transmuxer.on(_transmuxingEvents2.default.IO_ERROR, function (detail, info) {
- _this3._emitter.emit(_playerEvents2.default.ERROR, _playerErrors.ErrorTypes.NETWORK_ERROR, detail, info);
- });
- this._transmuxer.on(_transmuxingEvents2.default.DEMUX_ERROR, function (detail, info) {
- _this3._emitter.emit(_playerEvents2.default.ERROR, _playerErrors.ErrorTypes.MEDIA_ERROR, detail, { code: -1, msg: info });
- });
- this._transmuxer.on(_transmuxingEvents2.default.MEDIA_INFO, function (mediaInfo) {
- _this3._mediaInfo = mediaInfo;
- _this3._emitter.emit(_playerEvents2.default.MEDIA_INFO, Object.assign({}, mediaInfo));
- });
- this._transmuxer.on(_transmuxingEvents2.default.STATISTICS_INFO, function (statInfo) {
- _this3._statisticsInfo = _this3._fillStatisticsInfo(statInfo);
- _this3._emitter.emit(_playerEvents2.default.STATISTICS_INFO, Object.assign({}, _this3._statisticsInfo));
- });
- this._transmuxer.on(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, function (milliseconds) {
- if (_this3._mediaElement && !_this3._config.accurateSeek) {
- _this3._requestSetTime = true;
- _this3._mediaElement.currentTime = milliseconds / 1000;
- }
- });
- this._transmuxer.open();
- }
- }, {
- key: 'unload',
- value: function unload() {
- if (this._mediaElement) {
- this._mediaElement.pause();
- }
- if (this._msectl) {
- this._msectl.seek(0);
- }
- if (this._transmuxer) {
- this._transmuxer.close();
- this._transmuxer.destroy();
- this._transmuxer = null;
- }
- }
- }, {
- key: 'play',
- value: function play() {
- return this._mediaElement.play();
- }
- }, {
- key: 'pause',
- value: function pause() {
- this._mediaElement.pause();
- }
- }, {
- key: '_fillStatisticsInfo',
- value: function _fillStatisticsInfo(statInfo) {
- statInfo.playerType = this._type;
- if (!(this._mediaElement instanceof HTMLVideoElement)) {
- return statInfo;
- }
- var hasQualityInfo = true;
- var decoded = 0;
- var dropped = 0;
- if (this._mediaElement.getVideoPlaybackQuality) {
- var quality = this._mediaElement.getVideoPlaybackQuality();
- decoded = quality.totalVideoFrames;
- dropped = quality.droppedVideoFrames;
- } else if (this._mediaElement.webkitDecodedFrameCount != undefined) {
- decoded = this._mediaElement.webkitDecodedFrameCount;
- dropped = this._mediaElement.webkitDroppedFrameCount;
- } else {
- hasQualityInfo = false;
- }
- if (hasQualityInfo) {
- statInfo.decodedFrames = decoded;
- statInfo.droppedFrames = dropped;
- }
- return statInfo;
- }
- }, {
- key: '_onmseUpdateEnd',
- value: function _onmseUpdateEnd() {
- if (!this._config.lazyLoad || this._config.isLive) {
- return;
- }
- var buffered = this._mediaElement.buffered;
- var currentTime = this._mediaElement.currentTime;
- var currentRangeStart = 0;
- var currentRangeEnd = 0;
- for (var i = 0; i < buffered.length; i++) {
- var start = buffered.start(i);
- var end = buffered.end(i);
- if (start <= currentTime && currentTime < end) {
- currentRangeStart = start;
- currentRangeEnd = end;
- break;
- }
- }
- if (currentRangeEnd >= currentTime + this._config.lazyLoadMaxDuration && this._progressChecker == null) {
- _logger2.default.v(this.TAG, 'Maximum buffering duration exceeded, suspend transmuxing task');
- this._suspendTransmuxer();
- }
- }
- }, {
- key: '_onmseBufferFull',
- value: function _onmseBufferFull() {
- _logger2.default.v(this.TAG, 'MSE SourceBuffer is full, suspend transmuxing task');
- if (this._progressChecker == null) {
- this._suspendTransmuxer();
- }
- }
- }, {
- key: '_suspendTransmuxer',
- value: function _suspendTransmuxer() {
- if (this._transmuxer) {
- this._transmuxer.pause();
- if (this._progressChecker == null) {
- this._progressChecker = window.setInterval(this._checkProgressAndResume.bind(this), 1000);
- }
- }
- }
- }, {
- key: '_checkProgressAndResume',
- value: function _checkProgressAndResume() {
- var currentTime = this._mediaElement.currentTime;
- var buffered = this._mediaElement.buffered;
- var needResume = false;
- for (var i = 0; i < buffered.length; i++) {
- var from = buffered.start(i);
- var to = buffered.end(i);
- if (currentTime >= from && currentTime < to) {
- if (currentTime >= to - this._config.lazyLoadRecoverDuration) {
- needResume = true;
- }
- break;
- }
- }
- if (needResume) {
- window.clearInterval(this._progressChecker);
- this._progressChecker = null;
- if (needResume) {
- _logger2.default.v(this.TAG, 'Continue loading from paused position');
- this._transmuxer.resume();
- }
- }
- }
- }, {
- key: '_isTimepointBuffered',
- value: function _isTimepointBuffered(seconds) {
- var buffered = this._mediaElement.buffered;
- for (var i = 0; i < buffered.length; i++) {
- var from = buffered.start(i);
- var to = buffered.end(i);
- if (seconds >= from && seconds < to) {
- return true;
- }
- }
- return false;
- }
- }, {
- key: '_internalSeek',
- value: function _internalSeek(seconds) {
- var directSeek = this._isTimepointBuffered(seconds);
- var directSeekBegin = false;
- var directSeekBeginTime = 0;
- if (seconds < 1.0 && this._mediaElement.buffered.length > 0) {
- var videoBeginTime = this._mediaElement.buffered.start(0);
- if (videoBeginTime < 1.0 && seconds < videoBeginTime || _browser2.default.safari) {
- directSeekBegin = true;
- // also workaround for Safari: Seek to 0 may cause video stuck, use 0.1 to avoid
- directSeekBeginTime = _browser2.default.safari ? 0.1 : videoBeginTime;
- }
- }
- if (directSeekBegin) {
- // seek to video begin, set currentTime directly if beginPTS buffered
- this._requestSetTime = true;
- this._mediaElement.currentTime = directSeekBeginTime;
- } else if (directSeek) {
- // buffered position
- if (!this._alwaysSeekKeyframe) {
- this._requestSetTime = true;
- this._mediaElement.currentTime = seconds;
- } else {
- var idr = this._msectl.getNearestKeyframe(Math.floor(seconds * 1000));
- this._requestSetTime = true;
- if (idr != null) {
- this._mediaElement.currentTime = idr.dts / 1000;
- } else {
- this._mediaElement.currentTime = seconds;
- }
- }
- if (this._progressChecker != null) {
- this._checkProgressAndResume();
- }
- } else {
- if (this._progressChecker != null) {
- window.clearInterval(this._progressChecker);
- this._progressChecker = null;
- }
- this._msectl.seek(seconds);
- this._transmuxer.seek(Math.floor(seconds * 1000)); // in milliseconds
- // no need to set mediaElement.currentTime if non-accurateSeek,
- // just wait for the recommend_seekpoint callback
- if (this._config.accurateSeek) {
- this._requestSetTime = true;
- this._mediaElement.currentTime = seconds;
- }
- }
- }
- }, {
- key: '_checkAndApplyUnbufferedSeekpoint',
- value: function _checkAndApplyUnbufferedSeekpoint() {
- if (this._seekpointRecord) {
- if (this._seekpointRecord.recordTime <= this._now() - 100) {
- var target = this._mediaElement.currentTime;
- this._seekpointRecord = null;
- if (!this._isTimepointBuffered(target)) {
- if (this._progressChecker != null) {
- window.clearTimeout(this._progressChecker);
- this._progressChecker = null;
- }
- // .currentTime is consists with .buffered timestamp
- // Chrome/Edge use DTS, while FireFox/Safari use PTS
- this._msectl.seek(target);
- this._transmuxer.seek(Math.floor(target * 1000));
- // set currentTime if accurateSeek, or wait for recommend_seekpoint callback
- if (this._config.accurateSeek) {
- this._requestSetTime = true;
- this._mediaElement.currentTime = target;
- }
- }
- } else {
- window.setTimeout(this._checkAndApplyUnbufferedSeekpoint.bind(this), 50);
- }
- }
- }
- }, {
- key: '_checkAndResumeStuckPlayback',
- value: function _checkAndResumeStuckPlayback(stalled) {
- var media = this._mediaElement;
- if (stalled || !this._receivedCanPlay || media.readyState < 2) {
- // HAVE_CURRENT_DATA
- var buffered = media.buffered;
- if (buffered.length > 0 && media.currentTime < buffered.start(0)) {
- _logger2.default.w(this.TAG, 'Playback seems stuck at ' + media.currentTime + ', seek to ' + buffered.start(0));
- this._requestSetTime = true;
- this._mediaElement.currentTime = buffered.start(0);
- this._mediaElement.removeEventListener('progress', this.e.onvProgress);
- }
- } else {
- // Playback didn't stuck, remove progress event listener
- this._mediaElement.removeEventListener('progress', this.e.onvProgress);
- }
- }
- }, {
- key: '_onvLoadedMetadata',
- value: function _onvLoadedMetadata(e) {
- if (this._pendingSeekTime != null) {
- this._mediaElement.currentTime = this._pendingSeekTime;
- this._pendingSeekTime = null;
- }
- }
- }, {
- key: '_onvSeeking',
- value: function _onvSeeking(e) {
- // handle seeking request from browser's progress bar
- var target = this._mediaElement.currentTime;
- var buffered = this._mediaElement.buffered;
- if (this._requestSetTime) {
- this._requestSetTime = false;
- return;
- }
- if (target < 1.0 && buffered.length > 0) {
- // seek to video begin, set currentTime directly if beginPTS buffered
- var videoBeginTime = buffered.start(0);
- if (videoBeginTime < 1.0 && target < videoBeginTime || _browser2.default.safari) {
- this._requestSetTime = true;
- // also workaround for Safari: Seek to 0 may cause video stuck, use 0.1 to avoid
- this._mediaElement.currentTime = _browser2.default.safari ? 0.1 : videoBeginTime;
- return;
- }
- }
- if (this._isTimepointBuffered(target)) {
- if (this._alwaysSeekKeyframe) {
- var idr = this._msectl.getNearestKeyframe(Math.floor(target * 1000));
- if (idr != null) {
- this._requestSetTime = true;
- this._mediaElement.currentTime = idr.dts / 1000;
- }
- }
- if (this._progressChecker != null) {
- this._checkProgressAndResume();
- }
- return;
- }
- this._seekpointRecord = {
- seekPoint: target,
- recordTime: this._now()
- };
- window.setTimeout(this._checkAndApplyUnbufferedSeekpoint.bind(this), 50);
- }
- }, {
- key: '_onvCanPlay',
- value: function _onvCanPlay(e) {
- this._receivedCanPlay = true;
- this._mediaElement.removeEventListener('canplay', this.e.onvCanPlay);
- }
- }, {
- key: '_onvStalled',
- value: function _onvStalled(e) {
- this._checkAndResumeStuckPlayback(true);
- }
- }, {
- key: '_onvProgress',
- value: function _onvProgress(e) {
- this._checkAndResumeStuckPlayback();
- }
- }, {
- key: 'type',
- get: function get() {
- return this._type;
- }
- }, {
- key: 'buffered',
- get: function get() {
- return this._mediaElement.buffered;
- }
- }, {
- key: 'duration',
- get: function get() {
- return this._mediaElement.duration;
- }
- }, {
- key: 'volume',
- get: function get() {
- return this._mediaElement.volume;
- },
- set: function set(value) {
- this._mediaElement.volume = value;
- }
- }, {
- key: 'muted',
- get: function get() {
- return this._mediaElement.muted;
- },
- set: function set(muted) {
- this._mediaElement.muted = muted;
- }
- }, {
- key: 'currentTime',
- get: function get() {
- if (this._mediaElement) {
- return this._mediaElement.currentTime;
- }
- return 0;
- },
- set: function set(seconds) {
- if (this._mediaElement) {
- this._internalSeek(seconds);
- } else {
- this._pendingSeekTime = seconds;
- }
- }
- }, {
- key: 'mediaInfo',
- get: function get() {
- return Object.assign({}, this._mediaInfo);
- }
- }, {
- key: 'statisticsInfo',
- get: function get() {
- if (this._statisticsInfo == null) {
- this._statisticsInfo = {};
- }
- this._statisticsInfo = this._fillStatisticsInfo(this._statisticsInfo);
- return Object.assign({}, this._statisticsInfo);
- }
- }]);
- return FlvPlayer;
- }();
- exports.default = FlvPlayer;
- },{"../config.js":5,"../core/mse-controller.js":9,"../core/mse-events.js":10,"../core/transmuxer.js":11,"../core/transmuxing-events.js":13,"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./player-errors.js":34,"./player-events.js":35,"events":2}],33:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- var _playerEvents = _dereq_('./player-events.js');
- var _playerEvents2 = _interopRequireDefault(_playerEvents);
- var _config = _dereq_('../config.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- // Player wrapper for browser's native player (HTMLVideoElement) without MediaSource src.
- var NativePlayer = function () {
- function NativePlayer(mediaDataSource, config) {
- _classCallCheck(this, NativePlayer);
- this.TAG = 'NativePlayer';
- this._type = 'NativePlayer';
- this._emitter = new _events2.default();
- this._config = (0, _config.createDefaultConfig)();
- if ((typeof config === 'undefined' ? 'undefined' : _typeof(config)) === 'object') {
- Object.assign(this._config, config);
- }
- if (mediaDataSource.type.toLowerCase() === 'flv') {
- throw new _exception.InvalidArgumentException('NativePlayer does\'t support flv MediaDataSource input!');
- }
- if (mediaDataSource.hasOwnProperty('segments')) {
- throw new _exception.InvalidArgumentException('NativePlayer(' + mediaDataSource.type + ') doesn\'t support multipart playback!');
- }
- this.e = {
- onvLoadedMetadata: this._onvLoadedMetadata.bind(this)
- };
- this._pendingSeekTime = null;
- this._statisticsReporter = null;
- this._mediaDataSource = mediaDataSource;
- this._mediaElement = null;
- }
- _createClass(NativePlayer, [{
- key: 'destroy',
- value: function destroy() {
- if (this._mediaElement) {
- this.unload();
- this.detachMediaElement();
- }
- this.e = null;
- this._mediaDataSource = null;
- this._emitter.removeAllListeners();
- this._emitter = null;
- }
- }, {
- key: 'on',
- value: function on(event, listener) {
- var _this = this;
- if (event === _playerEvents2.default.MEDIA_INFO) {
- if (this._mediaElement != null && this._mediaElement.readyState !== 0) {
- // HAVE_NOTHING
- Promise.resolve().then(function () {
- _this._emitter.emit(_playerEvents2.default.MEDIA_INFO, _this.mediaInfo);
- });
- }
- } else if (event === _playerEvents2.default.STATISTICS_INFO) {
- if (this._mediaElement != null && this._mediaElement.readyState !== 0) {
- Promise.resolve().then(function () {
- _this._emitter.emit(_playerEvents2.default.STATISTICS_INFO, _this.statisticsInfo);
- });
- }
- }
- this._emitter.addListener(event, listener);
- }
- }, {
- key: 'off',
- value: function off(event, listener) {
- this._emitter.removeListener(event, listener);
- }
- }, {
- key: 'attachMediaElement',
- value: function attachMediaElement(mediaElement) {
- this._mediaElement = mediaElement;
- mediaElement.addEventListener('loadedmetadata', this.e.onvLoadedMetadata);
- if (this._pendingSeekTime != null) {
- try {
- mediaElement.currentTime = this._pendingSeekTime;
- this._pendingSeekTime = null;
- } catch (e) {
- // IE11 may throw InvalidStateError if readyState === 0
- // Defer set currentTime operation after loadedmetadata
- }
- }
- }
- }, {
- key: 'detachMediaElement',
- value: function detachMediaElement() {
- if (this._mediaElement) {
- this._mediaElement.src = '';
- this._mediaElement.removeAttribute('src');
- this._mediaElement.removeEventListener('loadedmetadata', this.e.onvLoadedMetadata);
- this._mediaElement = null;
- }
- if (this._statisticsReporter != null) {
- window.clearInterval(this._statisticsReporter);
- this._statisticsReporter = null;
- }
- }
- }, {
- key: 'load',
- value: function load() {
- if (!this._mediaElement) {
- throw new _exception.IllegalStateException('HTMLMediaElement must be attached before load()!');
- }
- this._mediaElement.src = this._mediaDataSource.url;
- if (this._mediaElement.readyState > 0) {
- this._mediaElement.currentTime = 0;
- }
- this._mediaElement.preload = 'auto';
- this._mediaElement.load();
- this._statisticsReporter = window.setInterval(this._reportStatisticsInfo.bind(this), this._config.statisticsInfoReportInterval);
- }
- }, {
- key: 'unload',
- value: function unload() {
- if (this._mediaElement) {
- this._mediaElement.src = '';
- this._mediaElement.removeAttribute('src');
- }
- if (this._statisticsReporter != null) {
- window.clearInterval(this._statisticsReporter);
- this._statisticsReporter = null;
- }
- }
- }, {
- key: 'play',
- value: function play() {
- return this._mediaElement.play();
- }
- }, {
- key: 'pause',
- value: function pause() {
- this._mediaElement.pause();
- }
- }, {
- key: '_onvLoadedMetadata',
- value: function _onvLoadedMetadata(e) {
- if (this._pendingSeekTime != null) {
- this._mediaElement.currentTime = this._pendingSeekTime;
- this._pendingSeekTime = null;
- }
- this._emitter.emit(_playerEvents2.default.MEDIA_INFO, this.mediaInfo);
- }
- }, {
- key: '_reportStatisticsInfo',
- value: function _reportStatisticsInfo() {
- this._emitter.emit(_playerEvents2.default.STATISTICS_INFO, this.statisticsInfo);
- }
- }, {
- key: 'type',
- get: function get() {
- return this._type;
- }
- }, {
- key: 'buffered',
- get: function get() {
- return this._mediaElement.buffered;
- }
- }, {
- key: 'duration',
- get: function get() {
- return this._mediaElement.duration;
- }
- }, {
- key: 'volume',
- get: function get() {
- return this._mediaElement.volume;
- },
- set: function set(value) {
- this._mediaElement.volume = value;
- }
- }, {
- key: 'muted',
- get: function get() {
- return this._mediaElement.muted;
- },
- set: function set(muted) {
- this._mediaElement.muted = muted;
- }
- }, {
- key: 'currentTime',
- get: function get() {
- if (this._mediaElement) {
- return this._mediaElement.currentTime;
- }
- return 0;
- },
- set: function set(seconds) {
- if (this._mediaElement) {
- this._mediaElement.currentTime = seconds;
- } else {
- this._pendingSeekTime = seconds;
- }
- }
- }, {
- key: 'mediaInfo',
- get: function get() {
- var mediaPrefix = this._mediaElement instanceof HTMLAudioElement ? 'audio/' : 'video/';
- var info = {
- mimeType: mediaPrefix + this._mediaDataSource.type
- };
- if (this._mediaElement) {
- info.duration = Math.floor(this._mediaElement.duration * 1000);
- if (this._mediaElement instanceof HTMLVideoElement) {
- info.width = this._mediaElement.videoWidth;
- info.height = this._mediaElement.videoHeight;
- }
- }
- return info;
- }
- }, {
- key: 'statisticsInfo',
- get: function get() {
- var info = {
- playerType: this._type,
- url: this._mediaDataSource.url
- };
- if (!(this._mediaElement instanceof HTMLVideoElement)) {
- return info;
- }
- var hasQualityInfo = true;
- var decoded = 0;
- var dropped = 0;
- if (this._mediaElement.getVideoPlaybackQuality) {
- var quality = this._mediaElement.getVideoPlaybackQuality();
- decoded = quality.totalVideoFrames;
- dropped = quality.droppedVideoFrames;
- } else if (this._mediaElement.webkitDecodedFrameCount != undefined) {
- decoded = this._mediaElement.webkitDecodedFrameCount;
- dropped = this._mediaElement.webkitDroppedFrameCount;
- } else {
- hasQualityInfo = false;
- }
- if (hasQualityInfo) {
- info.decodedFrames = decoded;
- info.droppedFrames = dropped;
- }
- return info;
- }
- }]);
- return NativePlayer;
- }();
- exports.default = NativePlayer;
- },{"../config.js":5,"../utils/exception.js":40,"./player-events.js":35,"events":2}],34:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.ErrorDetails = exports.ErrorTypes = undefined;
- var _loader = _dereq_('../io/loader.js');
- var _demuxErrors = _dereq_('../demux/demux-errors.js');
- var _demuxErrors2 = _interopRequireDefault(_demuxErrors);
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var ErrorTypes = exports.ErrorTypes = {
- NETWORK_ERROR: 'NetworkError',
- MEDIA_ERROR: 'MediaError',
- OTHER_ERROR: 'OtherError'
- };
- var ErrorDetails = exports.ErrorDetails = {
- NETWORK_EXCEPTION: _loader.LoaderErrors.EXCEPTION,
- NETWORK_STATUS_CODE_INVALID: _loader.LoaderErrors.HTTP_STATUS_CODE_INVALID,
- NETWORK_TIMEOUT: _loader.LoaderErrors.CONNECTING_TIMEOUT,
- NETWORK_UNRECOVERABLE_EARLY_EOF: _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF,
- MEDIA_MSE_ERROR: 'MediaMSEError',
- MEDIA_FORMAT_ERROR: _demuxErrors2.default.FORMAT_ERROR,
- MEDIA_FORMAT_UNSUPPORTED: _demuxErrors2.default.FORMAT_UNSUPPORTED,
- MEDIA_CODEC_UNSUPPORTED: _demuxErrors2.default.CODEC_UNSUPPORTED
- };
- },{"../demux/demux-errors.js":16,"../io/loader.js":24}],35:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var PlayerEvents = {
- ERROR: 'error',
- LOADING_COMPLETE: 'loading_complete',
- RECOVERED_EARLY_EOF: 'recovered_early_eof',
- MEDIA_INFO: 'media_info',
- STATISTICS_INFO: 'statistics_info'
- };
- exports.default = PlayerEvents;
- },{}],36:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * This file is modified from dailymotion's hls.js library (hls.js/src/helper/aac.js)
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var AAC = function () {
- function AAC() {
- _classCallCheck(this, AAC);
- }
- _createClass(AAC, null, [{
- key: 'getSilentFrame',
- value: function getSilentFrame(codec, channelCount) {
- if (codec === 'mp4a.40.2') {
- // handle LC-AAC
- if (channelCount === 1) {
- return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x23, 0x80]);
- } else if (channelCount === 2) {
- return new Uint8Array([0x21, 0x00, 0x49, 0x90, 0x02, 0x19, 0x00, 0x23, 0x80]);
- } else if (channelCount === 3) {
- return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x8e]);
- } else if (channelCount === 4) {
- return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x80, 0x2c, 0x80, 0x08, 0x02, 0x38]);
- } else if (channelCount === 5) {
- return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x82, 0x30, 0x04, 0x99, 0x00, 0x21, 0x90, 0x02, 0x38]);
- } else if (channelCount === 6) {
- return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x82, 0x30, 0x04, 0x99, 0x00, 0x21, 0x90, 0x02, 0x00, 0xb2, 0x00, 0x20, 0x08, 0xe0]);
- }
- } else {
- // handle HE-AAC (mp4a.40.5 / mp4a.40.29)
- if (channelCount === 1) {
- // ffmpeg -y -f lavfi -i "aevalsrc=0:d=0.05" -c:a libfdk_aac -profile:a aac_he -b:a 4k output.aac && hexdump -v -e '16/1 "0x%x," "\n"' -v output.aac
- return new Uint8Array([0x1, 0x40, 0x22, 0x80, 0xa3, 0x4e, 0xe6, 0x80, 0xba, 0x8, 0x0, 0x0, 0x0, 0x1c, 0x6, 0xf1, 0xc1, 0xa, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5e]);
- } else if (channelCount === 2) {
- // ffmpeg -y -f lavfi -i "aevalsrc=0|0:d=0.05" -c:a libfdk_aac -profile:a aac_he_v2 -b:a 4k output.aac && hexdump -v -e '16/1 "0x%x," "\n"' -v output.aac
- return new Uint8Array([0x1, 0x40, 0x22, 0x80, 0xa3, 0x5e, 0xe6, 0x80, 0xba, 0x8, 0x0, 0x0, 0x0, 0x0, 0x95, 0x0, 0x6, 0xf1, 0xa1, 0xa, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5e]);
- } else if (channelCount === 3) {
- // ffmpeg -y -f lavfi -i "aevalsrc=0|0|0:d=0.05" -c:a libfdk_aac -profile:a aac_he_v2 -b:a 4k output.aac && hexdump -v -e '16/1 "0x%x," "\n"' -v output.aac
- return new Uint8Array([0x1, 0x40, 0x22, 0x80, 0xa3, 0x5e, 0xe6, 0x80, 0xba, 0x8, 0x0, 0x0, 0x0, 0x0, 0x95, 0x0, 0x6, 0xf1, 0xa1, 0xa, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5a, 0x5e]);
- }
- }
- return null;
- }
- }]);
- return AAC;
- }();
- exports.default = AAC;
- },{}],37:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * This file is derived from dailymotion's hls.js library (hls.js/src/remux/mp4-generator.js)
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- // MP4 boxes generator for ISO BMFF (ISO Base Media File Format, defined in ISO/IEC 14496-12)
- var MP4 = function () {
- function MP4() {
- _classCallCheck(this, MP4);
- }
- _createClass(MP4, null, [{
- key: 'init',
- value: function init() {
- MP4.types = {
- avc1: [], avcC: [], btrt: [], dinf: [],
- dref: [], esds: [], ftyp: [], hdlr: [],
- mdat: [], mdhd: [], mdia: [], mfhd: [],
- minf: [], moof: [], moov: [], mp4a: [],
- mvex: [], mvhd: [], sdtp: [], stbl: [],
- stco: [], stsc: [], stsd: [], stsz: [],
- stts: [], tfdt: [], tfhd: [], traf: [],
- trak: [], trun: [], trex: [], tkhd: [],
- vmhd: [], smhd: [], '.mp3': []
- };
- for (var name in MP4.types) {
- if (MP4.types.hasOwnProperty(name)) {
- MP4.types[name] = [name.charCodeAt(0), name.charCodeAt(1), name.charCodeAt(2), name.charCodeAt(3)];
- }
- }
- var constants = MP4.constants = {};
- constants.FTYP = new Uint8Array([0x69, 0x73, 0x6F, 0x6D, // major_brand: isom
- 0x0, 0x0, 0x0, 0x1, // minor_version: 0x01
- 0x69, 0x73, 0x6F, 0x6D, // isom
- 0x61, 0x76, 0x63, 0x31 // avc1
- ]);
- constants.STSD_PREFIX = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x01 // entry_count
- ]);
- constants.STTS = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00 // entry_count
- ]);
- constants.STSC = constants.STCO = constants.STTS;
- constants.STSZ = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00, // sample_size
- 0x00, 0x00, 0x00, 0x00 // sample_count
- ]);
- constants.HDLR_VIDEO = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00, // pre_defined
- 0x76, 0x69, 0x64, 0x65, // handler_type: 'vide'
- 0x00, 0x00, 0x00, 0x00, // reserved: 3 * 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x56, 0x69, 0x64, 0x65, 0x6F, 0x48, 0x61, 0x6E, 0x64, 0x6C, 0x65, 0x72, 0x00 // name: VideoHandler
- ]);
- constants.HDLR_AUDIO = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00, // pre_defined
- 0x73, 0x6F, 0x75, 0x6E, // handler_type: 'soun'
- 0x00, 0x00, 0x00, 0x00, // reserved: 3 * 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x53, 0x6F, 0x75, 0x6E, 0x64, 0x48, 0x61, 0x6E, 0x64, 0x6C, 0x65, 0x72, 0x00 // name: SoundHandler
- ]);
- constants.DREF = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x01, // entry_count
- 0x00, 0x00, 0x00, 0x0C, // entry_size
- 0x75, 0x72, 0x6C, 0x20, // type 'url '
- 0x00, 0x00, 0x00, 0x01 // version(0) + flags
- ]);
- // Sound media header
- constants.SMHD = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00 // balance(2) + reserved(2)
- ]);
- // video media header
- constants.VMHD = new Uint8Array([0x00, 0x00, 0x00, 0x01, // version(0) + flags
- 0x00, 0x00, // graphicsmode: 2 bytes
- 0x00, 0x00, 0x00, 0x00, // opcolor: 3 * 2 bytes
- 0x00, 0x00]);
- }
- // Generate a box
- }, {
- key: 'box',
- value: function box(type) {
- var size = 8;
- var result = null;
- var datas = Array.prototype.slice.call(arguments, 1);
- var arrayCount = datas.length;
- for (var i = 0; i < arrayCount; i++) {
- size += datas[i].byteLength;
- }
- result = new Uint8Array(size);
- result[0] = size >>> 24 & 0xFF; // size
- result[1] = size >>> 16 & 0xFF;
- result[2] = size >>> 8 & 0xFF;
- result[3] = size & 0xFF;
- result.set(type, 4); // type
- var offset = 8;
- for (var _i = 0; _i < arrayCount; _i++) {
- // data body
- result.set(datas[_i], offset);
- offset += datas[_i].byteLength;
- }
- return result;
- }
- // emit ftyp & moov
- }, {
- key: 'generateInitSegment',
- value: function generateInitSegment(meta) {
- var ftyp = MP4.box(MP4.types.ftyp, MP4.constants.FTYP);
- var moov = MP4.moov(meta);
- var result = new Uint8Array(ftyp.byteLength + moov.byteLength);
- result.set(ftyp, 0);
- result.set(moov, ftyp.byteLength);
- return result;
- }
- // Movie metadata box
- }, {
- key: 'moov',
- value: function moov(meta) {
- var mvhd = MP4.mvhd(meta.timescale, meta.duration);
- var trak = MP4.trak(meta);
- var mvex = MP4.mvex(meta);
- return MP4.box(MP4.types.moov, mvhd, trak, mvex);
- }
- // Movie header box
- }, {
- key: 'mvhd',
- value: function mvhd(timescale, duration) {
- return MP4.box(MP4.types.mvhd, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00, // creation_time
- 0x00, 0x00, 0x00, 0x00, // modification_time
- timescale >>> 24 & 0xFF, // timescale: 4 bytes
- timescale >>> 16 & 0xFF, timescale >>> 8 & 0xFF, timescale & 0xFF, duration >>> 24 & 0xFF, // duration: 4 bytes
- duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, 0x00, 0x01, 0x00, 0x00, // Preferred rate: 1.0
- 0x01, 0x00, 0x00, 0x00, // PreferredVolume(1.0, 2bytes) + reserved(2bytes)
- 0x00, 0x00, 0x00, 0x00, // reserved: 4 + 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, // ----begin composition matrix----
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, // ----end composition matrix----
- 0x00, 0x00, 0x00, 0x00, // ----begin pre_defined 6 * 4 bytes----
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ----end pre_defined 6 * 4 bytes----
- 0xFF, 0xFF, 0xFF, 0xFF // next_track_ID
- ]));
- }
- // Track box
- }, {
- key: 'trak',
- value: function trak(meta) {
- return MP4.box(MP4.types.trak, MP4.tkhd(meta), MP4.mdia(meta));
- }
- // Track header box
- }, {
- key: 'tkhd',
- value: function tkhd(meta) {
- var trackId = meta.id,
- duration = meta.duration;
- var width = meta.presentWidth,
- height = meta.presentHeight;
- return MP4.box(MP4.types.tkhd, new Uint8Array([0x00, 0x00, 0x00, 0x07, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00, // creation_time
- 0x00, 0x00, 0x00, 0x00, // modification_time
- trackId >>> 24 & 0xFF, // track_ID: 4 bytes
- trackId >>> 16 & 0xFF, trackId >>> 8 & 0xFF, trackId & 0xFF, 0x00, 0x00, 0x00, 0x00, // reserved: 4 bytes
- duration >>> 24 & 0xFF, // duration: 4 bytes
- duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, 0x00, 0x00, 0x00, 0x00, // reserved: 2 * 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // layer(2bytes) + alternate_group(2bytes)
- 0x00, 0x00, 0x00, 0x00, // volume(2bytes) + reserved(2bytes)
- 0x00, 0x01, 0x00, 0x00, // ----begin composition matrix----
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, // ----end composition matrix----
- width >>> 8 & 0xFF, // width and height
- width & 0xFF, 0x00, 0x00, height >>> 8 & 0xFF, height & 0xFF, 0x00, 0x00]));
- }
- // Media Box
- }, {
- key: 'mdia',
- value: function mdia(meta) {
- return MP4.box(MP4.types.mdia, MP4.mdhd(meta), MP4.hdlr(meta), MP4.minf(meta));
- }
- // Media header box
- }, {
- key: 'mdhd',
- value: function mdhd(meta) {
- var timescale = meta.timescale;
- var duration = meta.duration;
- return MP4.box(MP4.types.mdhd, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- 0x00, 0x00, 0x00, 0x00, // creation_time
- 0x00, 0x00, 0x00, 0x00, // modification_time
- timescale >>> 24 & 0xFF, // timescale: 4 bytes
- timescale >>> 16 & 0xFF, timescale >>> 8 & 0xFF, timescale & 0xFF, duration >>> 24 & 0xFF, // duration: 4 bytes
- duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, 0x55, 0xC4, // language: und (undetermined)
- 0x00, 0x00 // pre_defined = 0
- ]));
- }
- // Media handler reference box
- }, {
- key: 'hdlr',
- value: function hdlr(meta) {
- var data = null;
- if (meta.type === 'audio') {
- data = MP4.constants.HDLR_AUDIO;
- } else {
- data = MP4.constants.HDLR_VIDEO;
- }
- return MP4.box(MP4.types.hdlr, data);
- }
- // Media infomation box
- }, {
- key: 'minf',
- value: function minf(meta) {
- var xmhd = null;
- if (meta.type === 'audio') {
- xmhd = MP4.box(MP4.types.smhd, MP4.constants.SMHD);
- } else {
- xmhd = MP4.box(MP4.types.vmhd, MP4.constants.VMHD);
- }
- return MP4.box(MP4.types.minf, xmhd, MP4.dinf(), MP4.stbl(meta));
- }
- // Data infomation box
- }, {
- key: 'dinf',
- value: function dinf() {
- var result = MP4.box(MP4.types.dinf, MP4.box(MP4.types.dref, MP4.constants.DREF));
- return result;
- }
- // Sample table box
- }, {
- key: 'stbl',
- value: function stbl(meta) {
- var result = MP4.box(MP4.types.stbl, // type: stbl
- MP4.stsd(meta), // Sample Description Table
- MP4.box(MP4.types.stts, MP4.constants.STTS), // Time-To-Sample
- MP4.box(MP4.types.stsc, MP4.constants.STSC), // Sample-To-Chunk
- MP4.box(MP4.types.stsz, MP4.constants.STSZ), // Sample size
- MP4.box(MP4.types.stco, MP4.constants.STCO // Chunk offset
- ));
- return result;
- }
- // Sample description box
- }, {
- key: 'stsd',
- value: function stsd(meta) {
- if (meta.type === 'audio') {
- if (meta.codec === 'mp3') {
- return MP4.box(MP4.types.stsd, MP4.constants.STSD_PREFIX, MP4.mp3(meta));
- }
- // else: aac -> mp4a
- return MP4.box(MP4.types.stsd, MP4.constants.STSD_PREFIX, MP4.mp4a(meta));
- } else {
- return MP4.box(MP4.types.stsd, MP4.constants.STSD_PREFIX, MP4.avc1(meta));
- }
- }
- }, {
- key: 'mp3',
- value: function mp3(meta) {
- var channelCount = meta.channelCount;
- var sampleRate = meta.audioSampleRate;
- var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // reserved(4)
- 0x00, 0x00, 0x00, 0x01, // reserved(2) + data_reference_index(2)
- 0x00, 0x00, 0x00, 0x00, // reserved: 2 * 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, channelCount, // channelCount(2)
- 0x00, 0x10, // sampleSize(2)
- 0x00, 0x00, 0x00, 0x00, // reserved(4)
- sampleRate >>> 8 & 0xFF, // Audio sample rate
- sampleRate & 0xFF, 0x00, 0x00]);
- return MP4.box(MP4.types['.mp3'], data);
- }
- }, {
- key: 'mp4a',
- value: function mp4a(meta) {
- var channelCount = meta.channelCount;
- var sampleRate = meta.audioSampleRate;
- var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // reserved(4)
- 0x00, 0x00, 0x00, 0x01, // reserved(2) + data_reference_index(2)
- 0x00, 0x00, 0x00, 0x00, // reserved: 2 * 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, channelCount, // channelCount(2)
- 0x00, 0x10, // sampleSize(2)
- 0x00, 0x00, 0x00, 0x00, // reserved(4)
- sampleRate >>> 8 & 0xFF, // Audio sample rate
- sampleRate & 0xFF, 0x00, 0x00]);
- return MP4.box(MP4.types.mp4a, data, MP4.esds(meta));
- }
- }, {
- key: 'esds',
- value: function esds(meta) {
- var config = meta.config || [];
- var configSize = config.length;
- var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version 0 + flags
- 0x03, // descriptor_type
- 0x17 + configSize, // length3
- 0x00, 0x01, // es_id
- 0x00, // stream_priority
- 0x04, // descriptor_type
- 0x0F + configSize, // length
- 0x40, // codec: mpeg4_audio
- 0x15, // stream_type: Audio
- 0x00, 0x00, 0x00, // buffer_size
- 0x00, 0x00, 0x00, 0x00, // maxBitrate
- 0x00, 0x00, 0x00, 0x00, // avgBitrate
- 0x05 // descriptor_type
- ].concat([configSize]).concat(config).concat([0x06, 0x01, 0x02 // GASpecificConfig
- ]));
- return MP4.box(MP4.types.esds, data);
- }
- }, {
- key: 'avc1',
- value: function avc1(meta) {
- var avcc = meta.avcc;
- var width = meta.codecWidth,
- height = meta.codecHeight;
- var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // reserved(4)
- 0x00, 0x00, 0x00, 0x01, // reserved(2) + data_reference_index(2)
- 0x00, 0x00, 0x00, 0x00, // pre_defined(2) + reserved(2)
- 0x00, 0x00, 0x00, 0x00, // pre_defined: 3 * 4 bytes
- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, width >>> 8 & 0xFF, // width: 2 bytes
- width & 0xFF, height >>> 8 & 0xFF, // height: 2 bytes
- height & 0xFF, 0x00, 0x48, 0x00, 0x00, // horizresolution: 4 bytes
- 0x00, 0x48, 0x00, 0x00, // vertresolution: 4 bytes
- 0x00, 0x00, 0x00, 0x00, // reserved: 4 bytes
- 0x00, 0x01, // frame_count
- 0x0A, // strlen
- 0x78, 0x71, 0x71, 0x2F, // compressorname: 32 bytes
- 0x66, 0x6C, 0x76, 0x2E, 0x6A, 0x73, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, // depth
- 0xFF, 0xFF // pre_defined = -1
- ]);
- return MP4.box(MP4.types.avc1, data, MP4.box(MP4.types.avcC, avcc));
- }
- // Movie Extends box
- }, {
- key: 'mvex',
- value: function mvex(meta) {
- return MP4.box(MP4.types.mvex, MP4.trex(meta));
- }
- // Track Extends box
- }, {
- key: 'trex',
- value: function trex(meta) {
- var trackId = meta.id;
- var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags
- trackId >>> 24 & 0xFF, // track_ID
- trackId >>> 16 & 0xFF, trackId >>> 8 & 0xFF, trackId & 0xFF, 0x00, 0x00, 0x00, 0x01, // default_sample_description_index
- 0x00, 0x00, 0x00, 0x00, // default_sample_duration
- 0x00, 0x00, 0x00, 0x00, // default_sample_size
- 0x00, 0x01, 0x00, 0x01 // default_sample_flags
- ]);
- return MP4.box(MP4.types.trex, data);
- }
- // Movie fragment box
- }, {
- key: 'moof',
- value: function moof(track, baseMediaDecodeTime) {
- return MP4.box(MP4.types.moof, MP4.mfhd(track.sequenceNumber), MP4.traf(track, baseMediaDecodeTime));
- }
- }, {
- key: 'mfhd',
- value: function mfhd(sequenceNumber) {
- var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, sequenceNumber >>> 24 & 0xFF, // sequence_number: int32
- sequenceNumber >>> 16 & 0xFF, sequenceNumber >>> 8 & 0xFF, sequenceNumber & 0xFF]);
- return MP4.box(MP4.types.mfhd, data);
- }
- // Track fragment box
- }, {
- key: 'traf',
- value: function traf(track, baseMediaDecodeTime) {
- var trackId = track.id;
- // Track fragment header box
- var tfhd = MP4.box(MP4.types.tfhd, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) & flags
- trackId >>> 24 & 0xFF, // track_ID
- trackId >>> 16 & 0xFF, trackId >>> 8 & 0xFF, trackId & 0xFF]));
- // Track Fragment Decode Time
- var tfdt = MP4.box(MP4.types.tfdt, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) & flags
- baseMediaDecodeTime >>> 24 & 0xFF, // baseMediaDecodeTime: int32
- baseMediaDecodeTime >>> 16 & 0xFF, baseMediaDecodeTime >>> 8 & 0xFF, baseMediaDecodeTime & 0xFF]));
- var sdtp = MP4.sdtp(track);
- var trun = MP4.trun(track, sdtp.byteLength + 16 + 16 + 8 + 16 + 8 + 8);
- return MP4.box(MP4.types.traf, tfhd, tfdt, trun, sdtp);
- }
- // Sample Dependency Type box
- }, {
- key: 'sdtp',
- value: function sdtp(track) {
- var samples = track.samples || [];
- var sampleCount = samples.length;
- var data = new Uint8Array(4 + sampleCount);
- // 0~4 bytes: version(0) & flags
- for (var i = 0; i < sampleCount; i++) {
- var flags = samples[i].flags;
- data[i + 4] = flags.isLeading << 6 | // is_leading: 2 (bit)
- flags.dependsOn << 4 // sample_depends_on
- | flags.isDependedOn << 2 // sample_is_depended_on
- | flags.hasRedundancy; // sample_has_redundancy
- }
- return MP4.box(MP4.types.sdtp, data);
- }
- // Track fragment run box
- }, {
- key: 'trun',
- value: function trun(track, offset) {
- var samples = track.samples || [];
- var sampleCount = samples.length;
- var dataSize = 12 + 16 * sampleCount;
- var data = new Uint8Array(dataSize);
- offset += 8 + dataSize;
- data.set([0x00, 0x00, 0x0F, 0x01, // version(0) & flags
- sampleCount >>> 24 & 0xFF, // sample_count
- sampleCount >>> 16 & 0xFF, sampleCount >>> 8 & 0xFF, sampleCount & 0xFF, offset >>> 24 & 0xFF, // data_offset
- offset >>> 16 & 0xFF, offset >>> 8 & 0xFF, offset & 0xFF], 0);
- for (var i = 0; i < sampleCount; i++) {
- var duration = samples[i].duration;
- var size = samples[i].size;
- var flags = samples[i].flags;
- var cts = samples[i].cts;
- data.set([duration >>> 24 & 0xFF, // sample_duration
- duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, size >>> 24 & 0xFF, // sample_size
- size >>> 16 & 0xFF, size >>> 8 & 0xFF, size & 0xFF, flags.isLeading << 2 | flags.dependsOn, // sample_flags
- flags.isDependedOn << 6 | flags.hasRedundancy << 4 | flags.isNonSync, 0x00, 0x00, // sample_degradation_priority
- cts >>> 24 & 0xFF, // sample_composition_time_offset
- cts >>> 16 & 0xFF, cts >>> 8 & 0xFF, cts & 0xFF], 12 + 16 * i);
- }
- return MP4.box(MP4.types.trun, data);
- }
- }, {
- key: 'mdat',
- value: function mdat(data) {
- return MP4.box(MP4.types.mdat, data);
- }
- }]);
- return MP4;
- }();
- MP4.init();
- exports.default = MP4;
- },{}],38:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _logger = _dereq_('../utils/logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- var _mp4Generator = _dereq_('./mp4-generator.js');
- var _mp4Generator2 = _interopRequireDefault(_mp4Generator);
- var _aacSilent = _dereq_('./aac-silent.js');
- var _aacSilent2 = _interopRequireDefault(_aacSilent);
- var _browser = _dereq_('../utils/browser.js');
- var _browser2 = _interopRequireDefault(_browser);
- var _mediaSegmentInfo = _dereq_('../core/media-segment-info.js');
- var _exception = _dereq_('../utils/exception.js');
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- // Fragmented mp4 remuxer
- var MP4Remuxer = function () {
- function MP4Remuxer(config) {
- _classCallCheck(this, MP4Remuxer);
- this.TAG = 'MP4Remuxer';
- this._config = config;
- this._isLive = config.isLive === true ? true : false;
- this._dtsBase = -1;
- this._dtsBaseInited = false;
- this._audioDtsBase = Infinity;
- this._videoDtsBase = Infinity;
- this._audioNextDts = undefined;
- this._videoNextDts = undefined;
- this._audioStashedLastSample = null;
- this._videoStashedLastSample = null;
- this._audioMeta = null;
- this._videoMeta = null;
- this._audioSegmentInfoList = new _mediaSegmentInfo.MediaSegmentInfoList('audio');
- this._videoSegmentInfoList = new _mediaSegmentInfo.MediaSegmentInfoList('video');
- this._onInitSegment = null;
- this._onMediaSegment = null;
- // Workaround for chrome < 50: Always force first sample as a Random Access Point in media segment
- // see https://bugs.chromium.org/p/chromium/issues/detail?id=229412
- this._forceFirstIDR = _browser2.default.chrome && (_browser2.default.version.major < 50 || _browser2.default.version.major === 50 && _browser2.default.version.build < 2661) ? true : false;
- // Workaround for IE11/Edge: Fill silent aac frame after keyframe-seeking
- // Make audio beginDts equals with video beginDts, in order to fix seek freeze
- this._fillSilentAfterSeek = _browser2.default.msedge || _browser2.default.msie;
- // While only FireFox supports 'audio/mp4, codecs="mp3"', use 'audio/mpeg' for chrome, safari, ...
- this._mp3UseMpegAudio = !_browser2.default.firefox;
- this._fillAudioTimestampGap = this._config.fixAudioTimestampGap;
- }
- _createClass(MP4Remuxer, [{
- key: 'destroy',
- value: function destroy() {
- this._dtsBase = -1;
- this._dtsBaseInited = false;
- this._audioMeta = null;
- this._videoMeta = null;
- this._audioSegmentInfoList.clear();
- this._audioSegmentInfoList = null;
- this._videoSegmentInfoList.clear();
- this._videoSegmentInfoList = null;
- this._onInitSegment = null;
- this._onMediaSegment = null;
- }
- }, {
- key: 'bindDataSource',
- value: function bindDataSource(producer) {
- producer.onDataAvailable = this.remux.bind(this);
- producer.onTrackMetadata = this._onTrackMetadataReceived.bind(this);
- return this;
- }
- /* prototype: function onInitSegment(type: string, initSegment: ArrayBuffer): void
- InitSegment: {
- type: string,
- data: ArrayBuffer,
- codec: string,
- container: string
- }
- */
- }, {
- key: 'insertDiscontinuity',
- value: function insertDiscontinuity() {
- this._audioNextDts = this._videoNextDts = undefined;
- }
- }, {
- key: 'seek',
- value: function seek(originalDts) {
- this._audioStashedLastSample = null;
- this._videoStashedLastSample = null;
- this._videoSegmentInfoList.clear();
- this._audioSegmentInfoList.clear();
- }
- }, {
- key: 'remux',
- value: function remux(audioTrack, videoTrack) {
- if (!this._onMediaSegment) {
- throw new _exception.IllegalStateException('MP4Remuxer: onMediaSegment callback must be specificed!');
- }
- if (!this._dtsBaseInited) {
- this._calculateDtsBase(audioTrack, videoTrack);
- }
- this._remuxVideo(videoTrack);
- this._remuxAudio(audioTrack);
- }
- }, {
- key: '_onTrackMetadataReceived',
- value: function _onTrackMetadataReceived(type, metadata) {
- var metabox = null;
- var container = 'mp4';
- var codec = metadata.codec;
- if (type === 'audio') {
- this._audioMeta = metadata;
- if (metadata.codec === 'mp3' && this._mp3UseMpegAudio) {
- // 'audio/mpeg' for MP3 audio track
- container = 'mpeg';
- codec = '';
- metabox = new Uint8Array();
- } else {
- // 'audio/mp4, codecs="codec"'
- metabox = _mp4Generator2.default.generateInitSegment(metadata);
- }
- } else if (type === 'video') {
- this._videoMeta = metadata;
- metabox = _mp4Generator2.default.generateInitSegment(metadata);
- } else {
- return;
- }
- // dispatch metabox (Initialization Segment)
- if (!this._onInitSegment) {
- throw new _exception.IllegalStateException('MP4Remuxer: onInitSegment callback must be specified!');
- }
- this._onInitSegment(type, {
- type: type,
- data: metabox.buffer,
- codec: codec,
- container: type + '/' + container,
- mediaDuration: metadata.duration // in timescale 1000 (milliseconds)
- });
- }
- }, {
- key: '_calculateDtsBase',
- value: function _calculateDtsBase(audioTrack, videoTrack) {
- if (this._dtsBaseInited) {
- return;
- }
- if (audioTrack.samples && audioTrack.samples.length) {
- this._audioDtsBase = audioTrack.samples[0].dts;
- }
- if (videoTrack.samples && videoTrack.samples.length) {
- this._videoDtsBase = videoTrack.samples[0].dts;
- }
- this._dtsBase = Math.min(this._audioDtsBase, this._videoDtsBase);
- this._dtsBaseInited = true;
- }
- }, {
- key: 'flushStashedSamples',
- value: function flushStashedSamples() {
- var videoSample = this._videoStashedLastSample;
- var audioSample = this._audioStashedLastSample;
- var videoTrack = {
- type: 'video',
- id: 1,
- sequenceNumber: 0,
- samples: [],
- length: 0
- };
- if (videoSample != null) {
- videoTrack.samples.push(videoSample);
- videoTrack.length = videoSample.length;
- }
- var audioTrack = {
- type: 'audio',
- id: 2,
- sequenceNumber: 0,
- samples: [],
- length: 0
- };
- if (audioSample != null) {
- audioTrack.samples.push(audioSample);
- audioTrack.length = audioSample.length;
- }
- this._videoStashedLastSample = null;
- this._audioStashedLastSample = null;
- this._remuxVideo(videoTrack, true);
- this._remuxAudio(audioTrack, true);
- }
- }, {
- key: '_remuxAudio',
- value: function _remuxAudio(audioTrack, force) {
- if (this._audioMeta == null) {
- return;
- }
- var track = audioTrack;
- var samples = track.samples;
- var dtsCorrection = undefined;
- var firstDts = -1,
- lastDts = -1,
- lastPts = -1;
- var refSampleDuration = this._audioMeta.refSampleDuration;
- var mpegRawTrack = this._audioMeta.codec === 'mp3' && this._mp3UseMpegAudio;
- var firstSegmentAfterSeek = this._dtsBaseInited && this._audioNextDts === undefined;
- var insertPrefixSilentFrame = false;
- if (!samples || samples.length === 0) {
- return;
- }
- if (samples.length === 1 && !force) {
- // If [sample count in current batch] === 1 && (force != true)
- // Ignore and keep in demuxer's queue
- return;
- } // else if (force === true) do remux
- var offset = 0;
- var mdatbox = null;
- var mdatBytes = 0;
- // calculate initial mdat size
- if (mpegRawTrack) {
- // for raw mpeg buffer
- offset = 0;
- mdatBytes = track.length;
- } else {
- // for fmp4 mdat box
- offset = 8; // size + type
- mdatBytes = 8 + track.length;
- }
- var lastSample = null;
- // Pop the lastSample and waiting for stash
- if (samples.length > 1) {
- lastSample = samples.pop();
- mdatBytes -= lastSample.length;
- }
- // Insert [stashed lastSample in the previous batch] to the front
- if (this._audioStashedLastSample != null) {
- var sample = this._audioStashedLastSample;
- this._audioStashedLastSample = null;
- samples.unshift(sample);
- mdatBytes += sample.length;
- }
- // Stash the lastSample of current batch, waiting for next batch
- if (lastSample != null) {
- this._audioStashedLastSample = lastSample;
- }
- var firstSampleOriginalDts = samples[0].dts - this._dtsBase;
- // calculate dtsCorrection
- if (this._audioNextDts) {
- dtsCorrection = firstSampleOriginalDts - this._audioNextDts;
- } else {
- // this._audioNextDts == undefined
- if (this._audioSegmentInfoList.isEmpty()) {
- dtsCorrection = 0;
- if (this._fillSilentAfterSeek && !this._videoSegmentInfoList.isEmpty()) {
- if (this._audioMeta.originalCodec !== 'mp3') {
- insertPrefixSilentFrame = true;
- }
- }
- } else {
- var _lastSample = this._audioSegmentInfoList.getLastSampleBefore(firstSampleOriginalDts);
- if (_lastSample != null) {
- var distance = firstSampleOriginalDts - (_lastSample.originalDts + _lastSample.duration);
- if (distance <= 3) {
- distance = 0;
- }
- var expectedDts = _lastSample.dts + _lastSample.duration + distance;
- dtsCorrection = firstSampleOriginalDts - expectedDts;
- } else {
- // lastSample == null, cannot found
- dtsCorrection = 0;
- }
- }
- }
- if (insertPrefixSilentFrame) {
- // align audio segment beginDts to match with current video segment's beginDts
- var firstSampleDts = firstSampleOriginalDts - dtsCorrection;
- var videoSegment = this._videoSegmentInfoList.getLastSegmentBefore(firstSampleOriginalDts);
- if (videoSegment != null && videoSegment.beginDts < firstSampleDts) {
- var silentUnit = _aacSilent2.default.getSilentFrame(this._audioMeta.originalCodec, this._audioMeta.channelCount);
- if (silentUnit) {
- var dts = videoSegment.beginDts;
- var silentFrameDuration = firstSampleDts - videoSegment.beginDts;
- _logger2.default.v(this.TAG, 'InsertPrefixSilentAudio: dts: ' + dts + ', duration: ' + silentFrameDuration);
- samples.unshift({ unit: silentUnit, dts: dts, pts: dts });
- mdatBytes += silentUnit.byteLength;
- } // silentUnit == null: Cannot generate, skip
- } else {
- insertPrefixSilentFrame = false;
- }
- }
- var mp4Samples = [];
- // Correct dts for each sample, and calculate sample duration. Then output to mp4Samples
- for (var i = 0; i < samples.length; i++) {
- var _sample = samples[i];
- var unit = _sample.unit;
- var originalDts = _sample.dts - this._dtsBase;
- var _dts = originalDts - dtsCorrection;
- if (firstDts === -1) {
- firstDts = _dts;
- }
- var sampleDuration = 0;
- if (i !== samples.length - 1) {
- var nextDts = samples[i + 1].dts - this._dtsBase - dtsCorrection;
- sampleDuration = nextDts - _dts;
- } else {
- // the last sample
- if (lastSample != null) {
- // use stashed sample's dts to calculate sample duration
- var _nextDts = lastSample.dts - this._dtsBase - dtsCorrection;
- sampleDuration = _nextDts - _dts;
- } else if (mp4Samples.length >= 1) {
- // use second last sample duration
- sampleDuration = mp4Samples[mp4Samples.length - 1].duration;
- } else {
- // the only one sample, use reference sample duration
- sampleDuration = Math.floor(refSampleDuration);
- }
- }
- var needFillSilentFrames = false;
- var silentFrames = null;
- // Silent frame generation, if large timestamp gap detected && config.fixAudioTimestampGap
- if (sampleDuration > refSampleDuration * 1.5 && this._audioMeta.codec !== 'mp3' && this._fillAudioTimestampGap && !_browser2.default.safari) {
- // We need to insert silent frames to fill timestamp gap
- needFillSilentFrames = true;
- var delta = Math.abs(sampleDuration - refSampleDuration);
- var frameCount = Math.ceil(delta / refSampleDuration);
- var currentDts = _dts + refSampleDuration; // Notice: in float
- _logger2.default.w(this.TAG, 'Large audio timestamp gap detected, may cause AV sync to drift. ' + 'Silent frames will be generated to avoid unsync.\n' + ('dts: ' + (_dts + sampleDuration) + ' ms, expected: ' + (_dts + Math.round(refSampleDuration)) + ' ms, ') + ('delta: ' + Math.round(delta) + ' ms, generate: ' + frameCount + ' frames'));
- var _silentUnit = _aacSilent2.default.getSilentFrame(this._audioMeta.originalCodec, this._audioMeta.channelCount);
- if (_silentUnit == null) {
- _logger2.default.w(this.TAG, 'Unable to generate silent frame for ' + (this._audioMeta.originalCodec + ' with ' + this._audioMeta.channelCount + ' channels, repeat last frame'));
- // Repeat last frame
- _silentUnit = unit;
- }
- silentFrames = [];
- for (var j = 0; j < frameCount; j++) {
- var intDts = Math.round(currentDts); // round to integer
- if (silentFrames.length > 0) {
- // Set previous frame sample duration
- var previousFrame = silentFrames[silentFrames.length - 1];
- previousFrame.duration = intDts - previousFrame.dts;
- }
- var frame = {
- dts: intDts,
- pts: intDts,
- cts: 0,
- unit: _silentUnit,
- size: _silentUnit.byteLength,
- duration: 0, // wait for next sample
- originalDts: originalDts,
- flags: {
- isLeading: 0,
- dependsOn: 1,
- isDependedOn: 0,
- hasRedundancy: 0
- }
- };
- silentFrames.push(frame);
- mdatBytes += unit.byteLength;
- currentDts += refSampleDuration;
- }
- // last frame: align end time to next frame dts
- var lastFrame = silentFrames[silentFrames.length - 1];
- lastFrame.duration = _dts + sampleDuration - lastFrame.dts;
- // silentFrames.forEach((frame) => {
- // Log.w(this.TAG, `SilentAudio: dts: ${frame.dts}, duration: ${frame.duration}`);
- // });
- // Set correct sample duration for current frame
- sampleDuration = Math.round(refSampleDuration);
- }
- mp4Samples.push({
- dts: _dts,
- pts: _dts,
- cts: 0,
- unit: _sample.unit,
- size: _sample.unit.byteLength,
- duration: sampleDuration,
- originalDts: originalDts,
- flags: {
- isLeading: 0,
- dependsOn: 1,
- isDependedOn: 0,
- hasRedundancy: 0
- }
- });
- if (needFillSilentFrames) {
- // Silent frames should be inserted after wrong-duration frame
- mp4Samples.push.apply(mp4Samples, silentFrames);
- }
- }
- // allocate mdatbox
- if (mpegRawTrack) {
- // allocate for raw mpeg buffer
- mdatbox = new Uint8Array(mdatBytes);
- } else {
- // allocate for fmp4 mdat box
- mdatbox = new Uint8Array(mdatBytes);
- // size field
- mdatbox[0] = mdatBytes >>> 24 & 0xFF;
- mdatbox[1] = mdatBytes >>> 16 & 0xFF;
- mdatbox[2] = mdatBytes >>> 8 & 0xFF;
- mdatbox[3] = mdatBytes & 0xFF;
- // type field (fourCC)
- mdatbox.set(_mp4Generator2.default.types.mdat, 4);
- }
- // Write samples into mdatbox
- for (var _i = 0; _i < mp4Samples.length; _i++) {
- var _unit = mp4Samples[_i].unit;
- mdatbox.set(_unit, offset);
- offset += _unit.byteLength;
- }
- var latest = mp4Samples[mp4Samples.length - 1];
- lastDts = latest.dts + latest.duration;
- this._audioNextDts = lastDts;
- // fill media segment info & add to info list
- var info = new _mediaSegmentInfo.MediaSegmentInfo();
- info.beginDts = firstDts;
- info.endDts = lastDts;
- info.beginPts = firstDts;
- info.endPts = lastDts;
- info.originalBeginDts = mp4Samples[0].originalDts;
- info.originalEndDts = latest.originalDts + latest.duration;
- info.firstSample = new _mediaSegmentInfo.SampleInfo(mp4Samples[0].dts, mp4Samples[0].pts, mp4Samples[0].duration, mp4Samples[0].originalDts, false);
- info.lastSample = new _mediaSegmentInfo.SampleInfo(latest.dts, latest.pts, latest.duration, latest.originalDts, false);
- if (!this._isLive) {
- this._audioSegmentInfoList.append(info);
- }
- track.samples = mp4Samples;
- track.sequenceNumber++;
- var moofbox = null;
- if (mpegRawTrack) {
- // Generate empty buffer, because useless for raw mpeg
- moofbox = new Uint8Array();
- } else {
- // Generate moof for fmp4 segment
- moofbox = _mp4Generator2.default.moof(track, firstDts);
- }
- track.samples = [];
- track.length = 0;
- var segment = {
- type: 'audio',
- data: this._mergeBoxes(moofbox, mdatbox).buffer,
- sampleCount: mp4Samples.length,
- info: info
- };
- if (mpegRawTrack && firstSegmentAfterSeek) {
- // For MPEG audio stream in MSE, if seeking occurred, before appending new buffer
- // We need explicitly set timestampOffset to the desired point in timeline for mpeg SourceBuffer.
- segment.timestampOffset = firstDts;
- }
- this._onMediaSegment('audio', segment);
- }
- }, {
- key: '_remuxVideo',
- value: function _remuxVideo(videoTrack, force) {
- if (this._videoMeta == null) {
- return;
- }
- var track = videoTrack;
- var samples = track.samples;
- var dtsCorrection = undefined;
- var firstDts = -1,
- lastDts = -1;
- var firstPts = -1,
- lastPts = -1;
- if (!samples || samples.length === 0) {
- return;
- }
- if (samples.length === 1 && !force) {
- // If [sample count in current batch] === 1 && (force != true)
- // Ignore and keep in demuxer's queue
- return;
- } // else if (force === true) do remux
- var offset = 8;
- var mdatbox = null;
- var mdatBytes = 8 + videoTrack.length;
- var lastSample = null;
- // Pop the lastSample and waiting for stash
- if (samples.length > 1) {
- lastSample = samples.pop();
- mdatBytes -= lastSample.length;
- }
- // Insert [stashed lastSample in the previous batch] to the front
- if (this._videoStashedLastSample != null) {
- var sample = this._videoStashedLastSample;
- this._videoStashedLastSample = null;
- samples.unshift(sample);
- mdatBytes += sample.length;
- }
- // Stash the lastSample of current batch, waiting for next batch
- if (lastSample != null) {
- this._videoStashedLastSample = lastSample;
- }
- var firstSampleOriginalDts = samples[0].dts - this._dtsBase;
- // calculate dtsCorrection
- if (this._videoNextDts) {
- dtsCorrection = firstSampleOriginalDts - this._videoNextDts;
- } else {
- // this._videoNextDts == undefined
- if (this._videoSegmentInfoList.isEmpty()) {
- dtsCorrection = 0;
- } else {
- var _lastSample2 = this._videoSegmentInfoList.getLastSampleBefore(firstSampleOriginalDts);
- if (_lastSample2 != null) {
- var distance = firstSampleOriginalDts - (_lastSample2.originalDts + _lastSample2.duration);
- if (distance <= 3) {
- distance = 0;
- }
- var expectedDts = _lastSample2.dts + _lastSample2.duration + distance;
- dtsCorrection = firstSampleOriginalDts - expectedDts;
- } else {
- // lastSample == null, cannot found
- dtsCorrection = 0;
- }
- }
- }
- var info = new _mediaSegmentInfo.MediaSegmentInfo();
- var mp4Samples = [];
- // Correct dts for each sample, and calculate sample duration. Then output to mp4Samples
- for (var i = 0; i < samples.length; i++) {
- var _sample2 = samples[i];
- var originalDts = _sample2.dts - this._dtsBase;
- var isKeyframe = _sample2.isKeyframe;
- var dts = originalDts - dtsCorrection;
- var cts = _sample2.cts;
- var pts = dts + cts;
- if (firstDts === -1) {
- firstDts = dts;
- firstPts = pts;
- }
- var sampleDuration = 0;
- if (i !== samples.length - 1) {
- var nextDts = samples[i + 1].dts - this._dtsBase - dtsCorrection;
- sampleDuration = nextDts - dts;
- } else {
- // the last sample
- if (lastSample != null) {
- // use stashed sample's dts to calculate sample duration
- var _nextDts2 = lastSample.dts - this._dtsBase - dtsCorrection;
- sampleDuration = _nextDts2 - dts;
- } else if (mp4Samples.length >= 1) {
- // use second last sample duration
- sampleDuration = mp4Samples[mp4Samples.length - 1].duration;
- } else {
- // the only one sample, use reference sample duration
- sampleDuration = Math.floor(this._videoMeta.refSampleDuration);
- }
- }
- if (isKeyframe) {
- var syncPoint = new _mediaSegmentInfo.SampleInfo(dts, pts, sampleDuration, _sample2.dts, true);
- syncPoint.fileposition = _sample2.fileposition;
- info.appendSyncPoint(syncPoint);
- }
- mp4Samples.push({
- dts: dts,
- pts: pts,
- cts: cts,
- units: _sample2.units,
- size: _sample2.length,
- isKeyframe: isKeyframe,
- duration: sampleDuration,
- originalDts: originalDts,
- flags: {
- isLeading: 0,
- dependsOn: isKeyframe ? 2 : 1,
- isDependedOn: isKeyframe ? 1 : 0,
- hasRedundancy: 0,
- isNonSync: isKeyframe ? 0 : 1
- }
- });
- }
- // allocate mdatbox
- mdatbox = new Uint8Array(mdatBytes);
- mdatbox[0] = mdatBytes >>> 24 & 0xFF;
- mdatbox[1] = mdatBytes >>> 16 & 0xFF;
- mdatbox[2] = mdatBytes >>> 8 & 0xFF;
- mdatbox[3] = mdatBytes & 0xFF;
- mdatbox.set(_mp4Generator2.default.types.mdat, 4);
- // Write samples into mdatbox
- for (var _i2 = 0; _i2 < mp4Samples.length; _i2++) {
- var units = mp4Samples[_i2].units;
- while (units.length) {
- var unit = units.shift();
- var data = unit.data;
- mdatbox.set(data, offset);
- offset += data.byteLength;
- }
- }
- var latest = mp4Samples[mp4Samples.length - 1];
- lastDts = latest.dts + latest.duration;
- lastPts = latest.pts + latest.duration;
- this._videoNextDts = lastDts;
- // fill media segment info & add to info list
- info.beginDts = firstDts;
- info.endDts = lastDts;
- info.beginPts = firstPts;
- info.endPts = lastPts;
- info.originalBeginDts = mp4Samples[0].originalDts;
- info.originalEndDts = latest.originalDts + latest.duration;
- info.firstSample = new _mediaSegmentInfo.SampleInfo(mp4Samples[0].dts, mp4Samples[0].pts, mp4Samples[0].duration, mp4Samples[0].originalDts, mp4Samples[0].isKeyframe);
- info.lastSample = new _mediaSegmentInfo.SampleInfo(latest.dts, latest.pts, latest.duration, latest.originalDts, latest.isKeyframe);
- if (!this._isLive) {
- this._videoSegmentInfoList.append(info);
- }
- track.samples = mp4Samples;
- track.sequenceNumber++;
- // workaround for chrome < 50: force first sample as a random access point
- // see https://bugs.chromium.org/p/chromium/issues/detail?id=229412
- if (this._forceFirstIDR) {
- var flags = mp4Samples[0].flags;
- flags.dependsOn = 2;
- flags.isNonSync = 0;
- }
- var moofbox = _mp4Generator2.default.moof(track, firstDts);
- track.samples = [];
- track.length = 0;
- this._onMediaSegment('video', {
- type: 'video',
- data: this._mergeBoxes(moofbox, mdatbox).buffer,
- sampleCount: mp4Samples.length,
- info: info
- });
- }
- }, {
- key: '_mergeBoxes',
- value: function _mergeBoxes(moof, mdat) {
- var result = new Uint8Array(moof.byteLength + mdat.byteLength);
- result.set(moof, 0);
- result.set(mdat, moof.byteLength);
- return result;
- }
- }, {
- key: 'onInitSegment',
- get: function get() {
- return this._onInitSegment;
- },
- set: function set(callback) {
- this._onInitSegment = callback;
- }
- /* prototype: function onMediaSegment(type: string, mediaSegment: MediaSegment): void
- MediaSegment: {
- type: string,
- data: ArrayBuffer,
- sampleCount: int32
- info: MediaSegmentInfo
- }
- */
- }, {
- key: 'onMediaSegment',
- get: function get() {
- return this._onMediaSegment;
- },
- set: function set(callback) {
- this._onMediaSegment = callback;
- }
- }]);
- return MP4Remuxer;
- }();
- exports.default = MP4Remuxer;
- },{"../core/media-segment-info.js":8,"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./aac-silent.js":36,"./mp4-generator.js":37}],39:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var Browser = {};
- function detect() {
- // modified from jquery-browser-plugin
- var ua = self.navigator.userAgent.toLowerCase();
- var match = /(edge)\/([\w.]+)/.exec(ua) || /(opr)[\/]([\w.]+)/.exec(ua) || /(chrome)[ \/]([\w.]+)/.exec(ua) || /(iemobile)[\/]([\w.]+)/.exec(ua) || /(version)(applewebkit)[ \/]([\w.]+).*(safari)[ \/]([\w.]+)/.exec(ua) || /(webkit)[ \/]([\w.]+).*(version)[ \/]([\w.]+).*(safari)[ \/]([\w.]+)/.exec(ua) || /(webkit)[ \/]([\w.]+)/.exec(ua) || /(opera)(?:.*version|)[ \/]([\w.]+)/.exec(ua) || /(msie) ([\w.]+)/.exec(ua) || ua.indexOf('trident') >= 0 && /(rv)(?::| )([\w.]+)/.exec(ua) || ua.indexOf('compatible') < 0 && /(firefox)[ \/]([\w.]+)/.exec(ua) || [];
- var platform_match = /(ipad)/.exec(ua) || /(ipod)/.exec(ua) || /(windows phone)/.exec(ua) || /(iphone)/.exec(ua) || /(kindle)/.exec(ua) || /(android)/.exec(ua) || /(windows)/.exec(ua) || /(mac)/.exec(ua) || /(linux)/.exec(ua) || /(cros)/.exec(ua) || [];
- var matched = {
- browser: match[5] || match[3] || match[1] || '',
- version: match[2] || match[4] || '0',
- majorVersion: match[4] || match[2] || '0',
- platform: platform_match[0] || ''
- };
- var browser = {};
- if (matched.browser) {
- browser[matched.browser] = true;
- var versionArray = matched.majorVersion.split('.');
- browser.version = {
- major: parseInt(matched.majorVersion, 10),
- string: matched.version
- };
- if (versionArray.length > 1) {
- browser.version.minor = parseInt(versionArray[1], 10);
- }
- if (versionArray.length > 2) {
- browser.version.build = parseInt(versionArray[2], 10);
- }
- }
- if (matched.platform) {
- browser[matched.platform] = true;
- }
- if (browser.chrome || browser.opr || browser.safari) {
- browser.webkit = true;
- }
- // MSIE. IE11 has 'rv' identifer
- if (browser.rv || browser.iemobile) {
- if (browser.rv) {
- delete browser.rv;
- }
- var msie = 'msie';
- matched.browser = msie;
- browser[msie] = true;
- }
- // Microsoft Edge
- if (browser.edge) {
- delete browser.edge;
- var msedge = 'msedge';
- matched.browser = msedge;
- browser[msedge] = true;
- }
- // Opera 15+
- if (browser.opr) {
- var opera = 'opera';
- matched.browser = opera;
- browser[opera] = true;
- }
- // Stock android browsers are marked as Safari
- if (browser.safari && browser.android) {
- var android = 'android';
- matched.browser = android;
- browser[android] = true;
- }
- browser.name = matched.browser;
- browser.platform = matched.platform;
- for (var key in Browser) {
- if (Browser.hasOwnProperty(key)) {
- delete Browser[key];
- }
- }
- Object.assign(Browser, browser);
- }
- detect();
- exports.default = Browser;
- },{}],40:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
- function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var RuntimeException = exports.RuntimeException = function () {
- function RuntimeException(message) {
- _classCallCheck(this, RuntimeException);
- this._message = message;
- }
- _createClass(RuntimeException, [{
- key: 'toString',
- value: function toString() {
- return this.name + ': ' + this.message;
- }
- }, {
- key: 'name',
- get: function get() {
- return 'RuntimeException';
- }
- }, {
- key: 'message',
- get: function get() {
- return this._message;
- }
- }]);
- return RuntimeException;
- }();
- var IllegalStateException = exports.IllegalStateException = function (_RuntimeException) {
- _inherits(IllegalStateException, _RuntimeException);
- function IllegalStateException(message) {
- _classCallCheck(this, IllegalStateException);
- return _possibleConstructorReturn(this, (IllegalStateException.__proto__ || Object.getPrototypeOf(IllegalStateException)).call(this, message));
- }
- _createClass(IllegalStateException, [{
- key: 'name',
- get: function get() {
- return 'IllegalStateException';
- }
- }]);
- return IllegalStateException;
- }(RuntimeException);
- var InvalidArgumentException = exports.InvalidArgumentException = function (_RuntimeException2) {
- _inherits(InvalidArgumentException, _RuntimeException2);
- function InvalidArgumentException(message) {
- _classCallCheck(this, InvalidArgumentException);
- return _possibleConstructorReturn(this, (InvalidArgumentException.__proto__ || Object.getPrototypeOf(InvalidArgumentException)).call(this, message));
- }
- _createClass(InvalidArgumentException, [{
- key: 'name',
- get: function get() {
- return 'InvalidArgumentException';
- }
- }]);
- return InvalidArgumentException;
- }(RuntimeException);
- var NotImplementedException = exports.NotImplementedException = function (_RuntimeException3) {
- _inherits(NotImplementedException, _RuntimeException3);
- function NotImplementedException(message) {
- _classCallCheck(this, NotImplementedException);
- return _possibleConstructorReturn(this, (NotImplementedException.__proto__ || Object.getPrototypeOf(NotImplementedException)).call(this, message));
- }
- _createClass(NotImplementedException, [{
- key: 'name',
- get: function get() {
- return 'NotImplementedException';
- }
- }]);
- return NotImplementedException;
- }(RuntimeException);
- },{}],41:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var Log = function () {
- function Log() {
- _classCallCheck(this, Log);
- }
- _createClass(Log, null, [{
- key: 'e',
- value: function e(tag, msg) {
- if (!tag || Log.FORCE_GLOBAL_TAG) tag = Log.GLOBAL_TAG;
- var str = '[' + tag + '] > ' + msg;
- if (Log.ENABLE_CALLBACK) {
- Log.emitter.emit('log', 'error', str);
- }
- if (!Log.ENABLE_ERROR) {
- return;
- }
- if (console.error) {
- console.error(str);
- } else if (console.warn) {
- console.warn(str);
- } else {
- console.log(str);
- }
- }
- }, {
- key: 'i',
- value: function i(tag, msg) {
- if (!tag || Log.FORCE_GLOBAL_TAG) tag = Log.GLOBAL_TAG;
- var str = '[' + tag + '] > ' + msg;
- if (Log.ENABLE_CALLBACK) {
- Log.emitter.emit('log', 'info', str);
- }
- if (!Log.ENABLE_INFO) {
- return;
- }
- if (console.info) {
- console.info(str);
- } else {
- console.log(str);
- }
- }
- }, {
- key: 'w',
- value: function w(tag, msg) {
- if (!tag || Log.FORCE_GLOBAL_TAG) tag = Log.GLOBAL_TAG;
- var str = '[' + tag + '] > ' + msg;
- if (Log.ENABLE_CALLBACK) {
- Log.emitter.emit('log', 'warn', str);
- }
- if (!Log.ENABLE_WARN) {
- return;
- }
- if (console.warn) {
- console.warn(str);
- } else {
- console.log(str);
- }
- }
- }, {
- key: 'd',
- value: function d(tag, msg) {
- if (!tag || Log.FORCE_GLOBAL_TAG) tag = Log.GLOBAL_TAG;
- var str = '[' + tag + '] > ' + msg;
- if (Log.ENABLE_CALLBACK) {
- Log.emitter.emit('log', 'debug', str);
- }
- if (!Log.ENABLE_DEBUG) {
- return;
- }
- if (console.debug) {
- console.debug(str);
- } else {
- console.log(str);
- }
- }
- }, {
- key: 'v',
- value: function v(tag, msg) {
- if (!tag || Log.FORCE_GLOBAL_TAG) tag = Log.GLOBAL_TAG;
- var str = '[' + tag + '] > ' + msg;
- if (Log.ENABLE_CALLBACK) {
- Log.emitter.emit('log', 'verbose', str);
- }
- if (!Log.ENABLE_VERBOSE) {
- return;
- }
- console.log(str);
- }
- }]);
- return Log;
- }();
- Log.GLOBAL_TAG = 'flv.js';
- Log.FORCE_GLOBAL_TAG = false;
- Log.ENABLE_ERROR = true;
- Log.ENABLE_INFO = true;
- Log.ENABLE_WARN = true;
- Log.ENABLE_DEBUG = true;
- Log.ENABLE_VERBOSE = true;
- Log.ENABLE_CALLBACK = false;
- Log.emitter = new _events2.default();
- exports.default = Log;
- },{"events":2}],42:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var _events = _dereq_('events');
- var _events2 = _interopRequireDefault(_events);
- var _logger = _dereq_('./logger.js');
- var _logger2 = _interopRequireDefault(_logger);
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- var LoggingControl = function () {
- function LoggingControl() {
- _classCallCheck(this, LoggingControl);
- }
- _createClass(LoggingControl, null, [{
- key: 'getConfig',
- value: function getConfig() {
- return {
- globalTag: _logger2.default.GLOBAL_TAG,
- forceGlobalTag: _logger2.default.FORCE_GLOBAL_TAG,
- enableVerbose: _logger2.default.ENABLE_VERBOSE,
- enableDebug: _logger2.default.ENABLE_DEBUG,
- enableInfo: _logger2.default.ENABLE_INFO,
- enableWarn: _logger2.default.ENABLE_WARN,
- enableError: _logger2.default.ENABLE_ERROR,
- enableCallback: _logger2.default.ENABLE_CALLBACK
- };
- }
- }, {
- key: 'applyConfig',
- value: function applyConfig(config) {
- _logger2.default.GLOBAL_TAG = config.globalTag;
- _logger2.default.FORCE_GLOBAL_TAG = config.forceGlobalTag;
- _logger2.default.ENABLE_VERBOSE = config.enableVerbose;
- _logger2.default.ENABLE_DEBUG = config.enableDebug;
- _logger2.default.ENABLE_INFO = config.enableInfo;
- _logger2.default.ENABLE_WARN = config.enableWarn;
- _logger2.default.ENABLE_ERROR = config.enableError;
- _logger2.default.ENABLE_CALLBACK = config.enableCallback;
- }
- }, {
- key: '_notifyChange',
- value: function _notifyChange() {
- var emitter = LoggingControl.emitter;
- if (emitter.listenerCount('change') > 0) {
- var config = LoggingControl.getConfig();
- emitter.emit('change', config);
- }
- }
- }, {
- key: 'registerListener',
- value: function registerListener(listener) {
- LoggingControl.emitter.addListener('change', listener);
- }
- }, {
- key: 'removeListener',
- value: function removeListener(listener) {
- LoggingControl.emitter.removeListener('change', listener);
- }
- }, {
- key: 'addLogListener',
- value: function addLogListener(listener) {
- _logger2.default.emitter.addListener('log', listener);
- if (_logger2.default.emitter.listenerCount('log') > 0) {
- _logger2.default.ENABLE_CALLBACK = true;
- LoggingControl._notifyChange();
- }
- }
- }, {
- key: 'removeLogListener',
- value: function removeLogListener(listener) {
- _logger2.default.emitter.removeListener('log', listener);
- if (_logger2.default.emitter.listenerCount('log') === 0) {
- _logger2.default.ENABLE_CALLBACK = false;
- LoggingControl._notifyChange();
- }
- }
- }, {
- key: 'forceGlobalTag',
- get: function get() {
- return _logger2.default.FORCE_GLOBAL_TAG;
- },
- set: function set(enable) {
- _logger2.default.FORCE_GLOBAL_TAG = enable;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'globalTag',
- get: function get() {
- return _logger2.default.GLOBAL_TAG;
- },
- set: function set(tag) {
- _logger2.default.GLOBAL_TAG = tag;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'enableAll',
- get: function get() {
- return _logger2.default.ENABLE_VERBOSE && _logger2.default.ENABLE_DEBUG && _logger2.default.ENABLE_INFO && _logger2.default.ENABLE_WARN && _logger2.default.ENABLE_ERROR;
- },
- set: function set(enable) {
- _logger2.default.ENABLE_VERBOSE = enable;
- _logger2.default.ENABLE_DEBUG = enable;
- _logger2.default.ENABLE_INFO = enable;
- _logger2.default.ENABLE_WARN = enable;
- _logger2.default.ENABLE_ERROR = enable;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'enableDebug',
- get: function get() {
- return _logger2.default.ENABLE_DEBUG;
- },
- set: function set(enable) {
- _logger2.default.ENABLE_DEBUG = enable;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'enableVerbose',
- get: function get() {
- return _logger2.default.ENABLE_VERBOSE;
- },
- set: function set(enable) {
- _logger2.default.ENABLE_VERBOSE = enable;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'enableInfo',
- get: function get() {
- return _logger2.default.ENABLE_INFO;
- },
- set: function set(enable) {
- _logger2.default.ENABLE_INFO = enable;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'enableWarn',
- get: function get() {
- return _logger2.default.ENABLE_WARN;
- },
- set: function set(enable) {
- _logger2.default.ENABLE_WARN = enable;
- LoggingControl._notifyChange();
- }
- }, {
- key: 'enableError',
- get: function get() {
- return _logger2.default.ENABLE_ERROR;
- },
- set: function set(enable) {
- _logger2.default.ENABLE_ERROR = enable;
- LoggingControl._notifyChange();
- }
- }]);
- return LoggingControl;
- }();
- LoggingControl.emitter = new _events2.default();
- exports.default = LoggingControl;
- },{"./logger.js":41,"events":2}],43:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
- function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- var Polyfill = function () {
- function Polyfill() {
- _classCallCheck(this, Polyfill);
- }
- _createClass(Polyfill, null, [{
- key: 'install',
- value: function install() {
- // ES6 Object.setPrototypeOf
- Object.setPrototypeOf = Object.setPrototypeOf || function (obj, proto) {
- obj.__proto__ = proto;
- return obj;
- };
- // ES6 Object.assign
- Object.assign = Object.assign || function (target) {
- if (target === undefined || target === null) {
- throw new TypeError('Cannot convert undefined or null to object');
- }
- var output = Object(target);
- for (var i = 1; i < arguments.length; i++) {
- var source = arguments[i];
- if (source !== undefined && source !== null) {
- for (var key in source) {
- if (source.hasOwnProperty(key)) {
- output[key] = source[key];
- }
- }
- }
- }
- return output;
- };
- // ES6 Promise (missing support in IE11)
- if (typeof self.Promise !== 'function') {
- _dereq_('es6-promise').polyfill();
- }
- }
- }]);
- return Polyfill;
- }();
- Polyfill.install();
- exports.default = Polyfill;
- },{"es6-promise":1}],44:[function(_dereq_,module,exports){
- 'use strict';
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /*
- * Copyright (C) 2016 Bilibili. All Rights Reserved.
- *
- * This file is derived from C++ project libWinTF8 (https://github.com/m13253/libWinTF8)
- * @author zheng qian <xqq@xqq.im>
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- function checkContinuation(uint8array, start, checkLength) {
- var array = uint8array;
- if (start + checkLength < array.length) {
- while (checkLength--) {
- if ((array[++start] & 0xC0) !== 0x80) return false;
- }
- return true;
- } else {
- return false;
- }
- }
- function decodeUTF8(uint8array) {
- var out = [];
- var input = uint8array;
- var i = 0;
- var length = uint8array.length;
- while (i < length) {
- if (input[i] < 0x80) {
- out.push(String.fromCharCode(input[i]));
- ++i;
- continue;
- } else if (input[i] < 0xC0) {
- // fallthrough
- } else if (input[i] < 0xE0) {
- if (checkContinuation(input, i, 1)) {
- var ucs4 = (input[i] & 0x1F) << 6 | input[i + 1] & 0x3F;
- if (ucs4 >= 0x80) {
- out.push(String.fromCharCode(ucs4 & 0xFFFF));
- i += 2;
- continue;
- }
- }
- } else if (input[i] < 0xF0) {
- if (checkContinuation(input, i, 2)) {
- var _ucs = (input[i] & 0xF) << 12 | (input[i + 1] & 0x3F) << 6 | input[i + 2] & 0x3F;
- if (_ucs >= 0x800 && (_ucs & 0xF800) !== 0xD800) {
- out.push(String.fromCharCode(_ucs & 0xFFFF));
- i += 3;
- continue;
- }
- }
- } else if (input[i] < 0xF8) {
- if (checkContinuation(input, i, 3)) {
- var _ucs2 = (input[i] & 0x7) << 18 | (input[i + 1] & 0x3F) << 12 | (input[i + 2] & 0x3F) << 6 | input[i + 3] & 0x3F;
- if (_ucs2 > 0x10000 && _ucs2 < 0x110000) {
- _ucs2 -= 0x10000;
- out.push(String.fromCharCode(_ucs2 >>> 10 | 0xD800));
- out.push(String.fromCharCode(_ucs2 & 0x3FF | 0xDC00));
- i += 4;
- continue;
- }
- }
- }
- out.push(String.fromCharCode(0xFFFD));
- ++i;
- }
- return out.join('');
- }
- exports.default = decodeUTF8;
- },{}]},{},[21])(21)
- });
- //# sourceMappingURL=flv.js.map
|