excel.js 1.1 MB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690769176927693769476957696769776987699770077017702770377047705770677077708770977107711771277137714771577167717771877197720772177227723772477257726772777287729773077317732773377347735773677377738773977407741774277437744774577467747774877497750775177527753775477557756775777587759776077617762776377647765776677677768776977707771777277737774777577767777777877797780778177827783778477857786778777887789779077917792779377947795779677977798779978007801780278037804780578067807780878097810781178127813781478157816781778187819782078217822782378247825782678277828782978307831783278337834783578367837783878397840784178427843784478457846784778487849785078517852785378547855785678577858785978607861786278637864786578667867786878697870787178727873787478757876787778787879788078817882788378847885788678877888788978907891789278937894789578967897789878997900790179027903790479057906790779087909791079117912791379147915791679177918791979207921792279237924792579267927792879297930793179327933793479357936793779387939794079417942794379447945794679477948794979507951795279537954795579567957795879597960796179627963796479657966796779687969797079717972797379747975797679777978797979807981798279837984798579867987798879897990799179927993799479957996799779987999800080018002800380048005800680078008800980108011801280138014801580168017801880198020802180228023802480258026802780288029803080318032803380348035803680378038803980408041804280438044804580468047804880498050805180528053805480558056805780588059806080618062806380648065806680678068806980708071807280738074807580768077807880798080808180828083808480858086808780888089809080918092809380948095809680978098809981008101810281038104810581068107810881098110811181128113811481158116811781188119812081218122812381248125812681278128812981308131813281338134813581368137813881398140814181428143814481458146814781488149815081518152815381548155815681578158815981608161816281638164816581668167816881698170817181728173817481758176817781788179818081818182818381848185818681878188818981908191819281938194819581968197819881998200820182028203820482058206820782088209821082118212821382148215821682178218821982208221822282238224822582268227822882298230823182328233823482358236823782388239824082418242824382448245824682478248824982508251825282538254825582568257825882598260826182628263826482658266826782688269827082718272827382748275827682778278827982808281828282838284828582868287828882898290829182928293829482958296829782988299830083018302830383048305830683078308830983108311831283138314831583168317831883198320832183228323832483258326832783288329833083318332833383348335833683378338833983408341834283438344834583468347834883498350835183528353835483558356835783588359836083618362836383648365836683678368836983708371837283738374837583768377837883798380838183828383838483858386838783888389839083918392839383948395839683978398839984008401840284038404840584068407840884098410841184128413841484158416841784188419842084218422842384248425842684278428842984308431843284338434843584368437843884398440844184428443844484458446844784488449845084518452845384548455845684578458845984608461846284638464846584668467846884698470847184728473847484758476847784788479848084818482848384848485848684878488848984908491849284938494849584968497849884998500850185028503850485058506850785088509851085118512851385148515851685178518851985208521852285238524852585268527852885298530853185328533853485358536853785388539854085418542854385448545854685478548854985508551855285538554855585568557855885598560856185628563856485658566856785688569857085718572857385748575857685778578857985808581858285838584858585868587858885898590859185928593859485958596859785988599860086018602860386048605860686078608860986108611861286138614861586168617861886198620862186228623862486258626862786288629863086318632863386348635863686378638863986408641864286438644864586468647864886498650865186528653865486558656865786588659866086618662866386648665866686678668866986708671867286738674867586768677867886798680868186828683868486858686868786888689869086918692869386948695869686978698869987008701870287038704870587068707870887098710871187128713871487158716871787188719872087218722872387248725872687278728872987308731873287338734873587368737873887398740874187428743874487458746874787488749875087518752875387548755875687578758875987608761876287638764876587668767876887698770877187728773877487758776877787788779878087818782878387848785878687878788878987908791879287938794879587968797879887998800880188028803880488058806880788088809881088118812881388148815881688178818881988208821882288238824882588268827882888298830883188328833883488358836883788388839884088418842884388448845884688478848884988508851885288538854885588568857885888598860886188628863886488658866886788688869887088718872887388748875887688778878887988808881888288838884888588868887888888898890889188928893889488958896889788988899890089018902890389048905890689078908890989108911891289138914891589168917891889198920892189228923892489258926892789288929893089318932893389348935893689378938893989408941894289438944894589468947894889498950895189528953895489558956895789588959896089618962896389648965896689678968896989708971897289738974897589768977897889798980898189828983898489858986898789888989899089918992899389948995899689978998899990009001900290039004900590069007900890099010901190129013901490159016901790189019902090219022902390249025902690279028902990309031903290339034903590369037903890399040904190429043904490459046904790489049905090519052905390549055905690579058905990609061906290639064906590669067906890699070907190729073907490759076907790789079908090819082908390849085908690879088908990909091909290939094909590969097909890999100910191029103910491059106910791089109911091119112911391149115911691179118911991209121912291239124912591269127912891299130913191329133913491359136913791389139914091419142914391449145914691479148914991509151915291539154915591569157915891599160916191629163916491659166916791689169917091719172917391749175917691779178917991809181918291839184918591869187918891899190919191929193919491959196919791989199920092019202920392049205920692079208920992109211921292139214921592169217921892199220922192229223922492259226922792289229923092319232923392349235923692379238923992409241924292439244924592469247924892499250925192529253925492559256925792589259926092619262926392649265926692679268926992709271927292739274927592769277927892799280928192829283928492859286928792889289929092919292929392949295929692979298929993009301930293039304930593069307930893099310931193129313931493159316931793189319932093219322932393249325932693279328932993309331933293339334933593369337933893399340934193429343934493459346934793489349935093519352935393549355935693579358935993609361936293639364936593669367936893699370937193729373937493759376937793789379938093819382938393849385938693879388938993909391939293939394939593969397939893999400940194029403940494059406940794089409941094119412941394149415941694179418941994209421942294239424942594269427942894299430943194329433943494359436943794389439944094419442944394449445944694479448944994509451945294539454945594569457945894599460946194629463946494659466946794689469947094719472947394749475947694779478947994809481948294839484948594869487948894899490949194929493949494959496949794989499950095019502950395049505950695079508950995109511951295139514951595169517951895199520952195229523952495259526952795289529953095319532953395349535953695379538953995409541954295439544954595469547954895499550955195529553955495559556955795589559956095619562956395649565956695679568956995709571957295739574957595769577957895799580958195829583958495859586958795889589959095919592959395949595959695979598959996009601960296039604960596069607960896099610961196129613961496159616961796189619962096219622962396249625962696279628962996309631963296339634963596369637963896399640964196429643964496459646964796489649965096519652965396549655965696579658965996609661966296639664966596669667966896699670967196729673967496759676967796789679968096819682968396849685968696879688968996909691969296939694969596969697969896999700970197029703970497059706970797089709971097119712971397149715971697179718971997209721972297239724972597269727972897299730973197329733973497359736973797389739974097419742974397449745974697479748974997509751975297539754975597569757975897599760976197629763976497659766976797689769977097719772977397749775977697779778977997809781978297839784978597869787978897899790979197929793979497959796979797989799980098019802980398049805980698079808980998109811981298139814981598169817981898199820982198229823982498259826982798289829983098319832983398349835983698379838983998409841984298439844984598469847984898499850985198529853985498559856985798589859986098619862986398649865986698679868986998709871987298739874987598769877987898799880988198829883988498859886988798889889989098919892989398949895989698979898989999009901990299039904990599069907990899099910991199129913991499159916991799189919992099219922992399249925992699279928992999309931993299339934993599369937993899399940994199429943994499459946994799489949995099519952995399549955995699579958995999609961996299639964996599669967996899699970997199729973997499759976997799789979998099819982998399849985998699879988998999909991999299939994999599969997999899991000010001100021000310004100051000610007100081000910010100111001210013100141001510016100171001810019100201002110022100231002410025100261002710028100291003010031100321003310034100351003610037100381003910040100411004210043100441004510046100471004810049100501005110052100531005410055100561005710058100591006010061100621006310064100651006610067100681006910070100711007210073100741007510076100771007810079100801008110082100831008410085100861008710088100891009010091100921009310094100951009610097100981009910100101011010210103101041010510106101071010810109101101011110112101131011410115101161011710118101191012010121101221012310124101251012610127101281012910130101311013210133101341013510136101371013810139101401014110142101431014410145101461014710148101491015010151101521015310154101551015610157101581015910160101611016210163101641016510166101671016810169101701017110172101731017410175101761017710178101791018010181101821018310184101851018610187101881018910190101911019210193101941019510196101971019810199102001020110202102031020410205102061020710208102091021010211102121021310214102151021610217102181021910220102211022210223102241022510226102271022810229102301023110232102331023410235102361023710238102391024010241102421024310244102451024610247102481024910250102511025210253102541025510256102571025810259102601026110262102631026410265102661026710268102691027010271102721027310274102751027610277102781027910280102811028210283102841028510286102871028810289102901029110292102931029410295102961029710298102991030010301103021030310304103051030610307103081030910310103111031210313103141031510316103171031810319103201032110322103231032410325103261032710328103291033010331103321033310334103351033610337103381033910340103411034210343103441034510346103471034810349103501035110352103531035410355103561035710358103591036010361103621036310364103651036610367103681036910370103711037210373103741037510376103771037810379103801038110382103831038410385103861038710388103891039010391103921039310394103951039610397103981039910400104011040210403104041040510406104071040810409104101041110412104131041410415104161041710418104191042010421104221042310424104251042610427104281042910430104311043210433104341043510436104371043810439104401044110442104431044410445104461044710448104491045010451104521045310454104551045610457104581045910460104611046210463104641046510466104671046810469104701047110472104731047410475104761047710478104791048010481104821048310484104851048610487104881048910490104911049210493104941049510496104971049810499105001050110502105031050410505105061050710508105091051010511105121051310514105151051610517105181051910520105211052210523105241052510526105271052810529105301053110532105331053410535105361053710538105391054010541105421054310544105451054610547105481054910550105511055210553105541055510556105571055810559105601056110562105631056410565105661056710568105691057010571105721057310574105751057610577105781057910580105811058210583105841058510586105871058810589105901059110592105931059410595105961059710598105991060010601106021060310604106051060610607106081060910610106111061210613106141061510616106171061810619106201062110622106231062410625106261062710628106291063010631106321063310634106351063610637106381063910640106411064210643106441064510646106471064810649106501065110652106531065410655106561065710658106591066010661106621066310664106651066610667106681066910670106711067210673106741067510676106771067810679106801068110682106831068410685106861068710688106891069010691106921069310694106951069610697106981069910700107011070210703107041070510706107071070810709107101071110712107131071410715107161071710718107191072010721107221072310724107251072610727107281072910730107311073210733107341073510736107371073810739107401074110742107431074410745107461074710748107491075010751107521075310754107551075610757107581075910760107611076210763107641076510766107671076810769107701077110772107731077410775107761077710778107791078010781107821078310784107851078610787107881078910790107911079210793107941079510796107971079810799108001080110802108031080410805108061080710808108091081010811108121081310814108151081610817108181081910820108211082210823108241082510826108271082810829108301083110832108331083410835108361083710838108391084010841108421084310844108451084610847108481084910850108511085210853108541085510856108571085810859108601086110862108631086410865108661086710868108691087010871108721087310874108751087610877108781087910880108811088210883108841088510886108871088810889108901089110892108931089410895108961089710898108991090010901109021090310904109051090610907109081090910910109111091210913109141091510916109171091810919109201092110922109231092410925109261092710928109291093010931109321093310934109351093610937109381093910940109411094210943109441094510946109471094810949109501095110952109531095410955109561095710958109591096010961109621096310964109651096610967109681096910970109711097210973109741097510976109771097810979109801098110982109831098410985109861098710988109891099010991109921099310994109951099610997109981099911000110011100211003110041100511006110071100811009110101101111012110131101411015110161101711018110191102011021110221102311024110251102611027110281102911030110311103211033110341103511036110371103811039110401104111042110431104411045110461104711048110491105011051110521105311054110551105611057110581105911060110611106211063110641106511066110671106811069110701107111072110731107411075110761107711078110791108011081110821108311084110851108611087110881108911090110911109211093110941109511096110971109811099111001110111102111031110411105111061110711108111091111011111111121111311114111151111611117111181111911120111211112211123111241112511126111271112811129111301113111132111331113411135111361113711138111391114011141111421114311144111451114611147111481114911150111511115211153111541115511156111571115811159111601116111162111631116411165111661116711168111691117011171111721117311174111751117611177111781117911180111811118211183111841118511186111871118811189111901119111192111931119411195111961119711198111991120011201112021120311204112051120611207112081120911210112111121211213112141121511216112171121811219112201122111222112231122411225112261122711228112291123011231112321123311234112351123611237112381123911240112411124211243112441124511246112471124811249112501125111252112531125411255112561125711258112591126011261112621126311264112651126611267112681126911270112711127211273112741127511276112771127811279112801128111282112831128411285112861128711288112891129011291112921129311294112951129611297112981129911300113011130211303113041130511306113071130811309113101131111312113131131411315113161131711318113191132011321113221132311324113251132611327113281132911330113311133211333113341133511336113371133811339113401134111342113431134411345113461134711348113491135011351113521135311354113551135611357113581135911360113611136211363113641136511366113671136811369113701137111372113731137411375113761137711378113791138011381113821138311384113851138611387113881138911390113911139211393113941139511396113971139811399114001140111402114031140411405114061140711408114091141011411114121141311414114151141611417114181141911420114211142211423114241142511426114271142811429114301143111432114331143411435114361143711438114391144011441114421144311444114451144611447114481144911450114511145211453114541145511456114571145811459114601146111462114631146411465114661146711468114691147011471114721147311474114751147611477114781147911480114811148211483114841148511486114871148811489114901149111492114931149411495114961149711498114991150011501115021150311504115051150611507115081150911510115111151211513115141151511516115171151811519115201152111522115231152411525115261152711528115291153011531115321153311534115351153611537115381153911540115411154211543115441154511546115471154811549115501155111552115531155411555115561155711558115591156011561115621156311564115651156611567115681156911570115711157211573115741157511576115771157811579115801158111582115831158411585115861158711588115891159011591115921159311594115951159611597115981159911600116011160211603116041160511606116071160811609116101161111612116131161411615116161161711618116191162011621116221162311624116251162611627116281162911630116311163211633116341163511636116371163811639116401164111642116431164411645116461164711648116491165011651116521165311654116551165611657116581165911660116611166211663116641166511666116671166811669116701167111672116731167411675116761167711678116791168011681116821168311684116851168611687116881168911690116911169211693116941169511696116971169811699117001170111702117031170411705117061170711708117091171011711117121171311714117151171611717117181171911720117211172211723117241172511726117271172811729117301173111732117331173411735117361173711738117391174011741117421174311744117451174611747117481174911750117511175211753117541175511756117571175811759117601176111762117631176411765117661176711768117691177011771117721177311774117751177611777117781177911780117811178211783117841178511786117871178811789117901179111792117931179411795117961179711798117991180011801118021180311804118051180611807118081180911810118111181211813118141181511816118171181811819118201182111822118231182411825118261182711828118291183011831118321183311834118351183611837118381183911840118411184211843118441184511846118471184811849118501185111852118531185411855118561185711858118591186011861118621186311864118651186611867118681186911870118711187211873118741187511876118771187811879118801188111882118831188411885118861188711888118891189011891118921189311894118951189611897118981189911900119011190211903119041190511906119071190811909119101191111912119131191411915119161191711918119191192011921119221192311924119251192611927119281192911930119311193211933119341193511936119371193811939119401194111942119431194411945119461194711948119491195011951119521195311954119551195611957119581195911960119611196211963119641196511966119671196811969119701197111972119731197411975119761197711978119791198011981119821198311984119851198611987119881198911990119911199211993119941199511996119971199811999120001200112002120031200412005120061200712008120091201012011120121201312014120151201612017120181201912020120211202212023120241202512026120271202812029120301203112032120331203412035120361203712038120391204012041120421204312044120451204612047120481204912050120511205212053120541205512056120571205812059120601206112062120631206412065120661206712068120691207012071120721207312074120751207612077120781207912080120811208212083120841208512086120871208812089120901209112092120931209412095120961209712098120991210012101121021210312104121051210612107121081210912110121111211212113121141211512116121171211812119121201212112122121231212412125121261212712128121291213012131121321213312134121351213612137121381213912140121411214212143121441214512146121471214812149121501215112152121531215412155121561215712158121591216012161121621216312164121651216612167121681216912170121711217212173121741217512176121771217812179121801218112182121831218412185121861218712188121891219012191121921219312194121951219612197121981219912200122011220212203122041220512206122071220812209122101221112212122131221412215122161221712218122191222012221122221222312224122251222612227122281222912230122311223212233122341223512236122371223812239122401224112242122431224412245122461224712248122491225012251122521225312254122551225612257122581225912260122611226212263122641226512266122671226812269122701227112272122731227412275122761227712278122791228012281122821228312284122851228612287122881228912290122911229212293122941229512296122971229812299123001230112302123031230412305123061230712308123091231012311123121231312314123151231612317123181231912320123211232212323123241232512326123271232812329123301233112332123331233412335123361233712338123391234012341123421234312344123451234612347123481234912350123511235212353123541235512356123571235812359123601236112362123631236412365123661236712368123691237012371123721237312374123751237612377123781237912380123811238212383123841238512386123871238812389123901239112392123931239412395123961239712398123991240012401124021240312404124051240612407124081240912410124111241212413124141241512416124171241812419124201242112422124231242412425124261242712428124291243012431124321243312434124351243612437124381243912440124411244212443124441244512446124471244812449124501245112452124531245412455124561245712458124591246012461124621246312464124651246612467124681246912470124711247212473124741247512476124771247812479124801248112482124831248412485124861248712488124891249012491124921249312494124951249612497124981249912500125011250212503125041250512506125071250812509125101251112512125131251412515125161251712518125191252012521125221252312524125251252612527125281252912530125311253212533125341253512536125371253812539125401254112542125431254412545125461254712548125491255012551125521255312554125551255612557125581255912560125611256212563125641256512566125671256812569125701257112572125731257412575125761257712578125791258012581125821258312584125851258612587125881258912590125911259212593125941259512596125971259812599126001260112602126031260412605126061260712608126091261012611126121261312614126151261612617126181261912620126211262212623126241262512626126271262812629126301263112632126331263412635126361263712638126391264012641126421264312644126451264612647126481264912650126511265212653126541265512656126571265812659126601266112662126631266412665126661266712668126691267012671126721267312674126751267612677126781267912680126811268212683126841268512686126871268812689126901269112692126931269412695126961269712698126991270012701127021270312704127051270612707127081270912710127111271212713127141271512716127171271812719127201272112722127231272412725127261272712728127291273012731127321273312734127351273612737127381273912740127411274212743127441274512746127471274812749127501275112752127531275412755127561275712758127591276012761127621276312764127651276612767127681276912770127711277212773127741277512776127771277812779127801278112782127831278412785127861278712788127891279012791127921279312794127951279612797127981279912800128011280212803128041280512806128071280812809128101281112812128131281412815128161281712818128191282012821128221282312824128251282612827128281282912830128311283212833128341283512836128371283812839128401284112842128431284412845128461284712848128491285012851128521285312854128551285612857128581285912860128611286212863128641286512866128671286812869128701287112872128731287412875128761287712878128791288012881128821288312884128851288612887128881288912890128911289212893128941289512896128971289812899129001290112902129031290412905129061290712908129091291012911129121291312914129151291612917129181291912920129211292212923129241292512926129271292812929129301293112932129331293412935129361293712938129391294012941129421294312944129451294612947129481294912950129511295212953129541295512956129571295812959129601296112962129631296412965129661296712968129691297012971129721297312974129751297612977129781297912980129811298212983129841298512986129871298812989129901299112992129931299412995129961299712998129991300013001130021300313004130051300613007130081300913010130111301213013130141301513016130171301813019130201302113022130231302413025130261302713028130291303013031130321303313034130351303613037130381303913040130411304213043130441304513046130471304813049130501305113052130531305413055130561305713058130591306013061130621306313064130651306613067130681306913070130711307213073130741307513076130771307813079130801308113082130831308413085130861308713088130891309013091130921309313094130951309613097130981309913100131011310213103131041310513106131071310813109131101311113112131131311413115131161311713118131191312013121131221312313124131251312613127131281312913130131311313213133131341313513136131371313813139131401314113142131431314413145131461314713148131491315013151131521315313154131551315613157131581315913160131611316213163131641316513166131671316813169131701317113172131731317413175131761317713178131791318013181131821318313184131851318613187131881318913190131911319213193131941319513196131971319813199132001320113202132031320413205132061320713208132091321013211132121321313214132151321613217132181321913220132211322213223132241322513226132271322813229132301323113232132331323413235132361323713238132391324013241132421324313244132451324613247132481324913250132511325213253132541325513256132571325813259132601326113262132631326413265132661326713268132691327013271132721327313274132751327613277132781327913280132811328213283132841328513286132871328813289132901329113292132931329413295132961329713298132991330013301133021330313304133051330613307133081330913310133111331213313133141331513316133171331813319133201332113322133231332413325133261332713328133291333013331133321333313334133351333613337133381333913340133411334213343133441334513346133471334813349133501335113352133531335413355133561335713358133591336013361133621336313364133651336613367133681336913370133711337213373133741337513376133771337813379133801338113382133831338413385133861338713388133891339013391133921339313394133951339613397133981339913400134011340213403134041340513406134071340813409134101341113412134131341413415134161341713418134191342013421134221342313424134251342613427134281342913430134311343213433134341343513436134371343813439134401344113442134431344413445134461344713448134491345013451134521345313454134551345613457134581345913460134611346213463134641346513466134671346813469134701347113472134731347413475134761347713478134791348013481134821348313484134851348613487134881348913490134911349213493134941349513496134971349813499135001350113502135031350413505135061350713508135091351013511135121351313514135151351613517135181351913520135211352213523135241352513526135271352813529135301353113532135331353413535135361353713538135391354013541135421354313544135451354613547135481354913550135511355213553135541355513556135571355813559135601356113562135631356413565135661356713568135691357013571135721357313574135751357613577135781357913580135811358213583135841358513586135871358813589135901359113592135931359413595135961359713598135991360013601136021360313604136051360613607136081360913610136111361213613136141361513616136171361813619136201362113622136231362413625136261362713628136291363013631136321363313634136351363613637136381363913640136411364213643136441364513646136471364813649136501365113652136531365413655136561365713658136591366013661136621366313664136651366613667136681366913670136711367213673136741367513676136771367813679136801368113682136831368413685136861368713688136891369013691136921369313694136951369613697136981369913700137011370213703137041370513706137071370813709137101371113712137131371413715137161371713718137191372013721137221372313724137251372613727137281372913730137311373213733137341373513736137371373813739137401374113742137431374413745137461374713748137491375013751137521375313754137551375613757137581375913760137611376213763137641376513766137671376813769137701377113772137731377413775137761377713778137791378013781137821378313784137851378613787137881378913790137911379213793137941379513796137971379813799138001380113802138031380413805138061380713808138091381013811138121381313814138151381613817138181381913820138211382213823138241382513826138271382813829138301383113832138331383413835138361383713838138391384013841138421384313844138451384613847138481384913850138511385213853138541385513856138571385813859138601386113862138631386413865138661386713868138691387013871138721387313874138751387613877138781387913880138811388213883138841388513886138871388813889138901389113892138931389413895138961389713898138991390013901139021390313904139051390613907139081390913910139111391213913139141391513916139171391813919139201392113922139231392413925139261392713928139291393013931139321393313934139351393613937139381393913940139411394213943139441394513946139471394813949139501395113952139531395413955139561395713958139591396013961139621396313964139651396613967139681396913970139711397213973139741397513976139771397813979139801398113982139831398413985139861398713988139891399013991139921399313994139951399613997139981399914000140011400214003140041400514006140071400814009140101401114012140131401414015140161401714018140191402014021140221402314024140251402614027140281402914030140311403214033140341403514036140371403814039140401404114042140431404414045140461404714048140491405014051140521405314054140551405614057140581405914060140611406214063140641406514066140671406814069140701407114072140731407414075140761407714078140791408014081140821408314084140851408614087140881408914090140911409214093140941409514096140971409814099141001410114102141031410414105141061410714108141091411014111141121411314114141151411614117141181411914120141211412214123141241412514126141271412814129141301413114132141331413414135141361413714138141391414014141141421414314144141451414614147141481414914150141511415214153141541415514156141571415814159141601416114162141631416414165141661416714168141691417014171141721417314174141751417614177141781417914180141811418214183141841418514186141871418814189141901419114192141931419414195141961419714198141991420014201142021420314204142051420614207142081420914210142111421214213142141421514216142171421814219142201422114222142231422414225142261422714228142291423014231142321423314234142351423614237142381423914240142411424214243142441424514246142471424814249142501425114252142531425414255142561425714258142591426014261142621426314264142651426614267142681426914270142711427214273142741427514276142771427814279142801428114282142831428414285142861428714288142891429014291142921429314294142951429614297142981429914300143011430214303143041430514306143071430814309143101431114312143131431414315143161431714318143191432014321143221432314324143251432614327143281432914330143311433214333143341433514336143371433814339143401434114342143431434414345143461434714348143491435014351143521435314354143551435614357143581435914360143611436214363143641436514366143671436814369143701437114372143731437414375143761437714378143791438014381143821438314384143851438614387143881438914390143911439214393143941439514396143971439814399144001440114402144031440414405144061440714408144091441014411144121441314414144151441614417144181441914420144211442214423144241442514426144271442814429144301443114432144331443414435144361443714438144391444014441144421444314444144451444614447144481444914450144511445214453144541445514456144571445814459144601446114462144631446414465144661446714468144691447014471144721447314474144751447614477144781447914480144811448214483144841448514486144871448814489144901449114492144931449414495144961449714498144991450014501145021450314504145051450614507145081450914510145111451214513145141451514516145171451814519145201452114522145231452414525145261452714528145291453014531145321453314534145351453614537145381453914540145411454214543145441454514546145471454814549145501455114552145531455414555145561455714558145591456014561145621456314564145651456614567145681456914570145711457214573145741457514576145771457814579145801458114582145831458414585145861458714588145891459014591145921459314594145951459614597145981459914600146011460214603146041460514606146071460814609146101461114612146131461414615146161461714618146191462014621146221462314624146251462614627146281462914630146311463214633146341463514636146371463814639146401464114642146431464414645146461464714648146491465014651146521465314654146551465614657146581465914660146611466214663146641466514666146671466814669146701467114672146731467414675146761467714678146791468014681146821468314684146851468614687146881468914690146911469214693146941469514696146971469814699147001470114702147031470414705147061470714708147091471014711147121471314714147151471614717147181471914720147211472214723147241472514726147271472814729147301473114732147331473414735147361473714738147391474014741147421474314744147451474614747147481474914750147511475214753147541475514756147571475814759147601476114762147631476414765147661476714768147691477014771147721477314774147751477614777147781477914780147811478214783147841478514786147871478814789147901479114792147931479414795147961479714798147991480014801148021480314804148051480614807148081480914810148111481214813148141481514816148171481814819148201482114822148231482414825148261482714828148291483014831148321483314834148351483614837148381483914840148411484214843148441484514846148471484814849148501485114852148531485414855148561485714858148591486014861148621486314864148651486614867148681486914870148711487214873148741487514876148771487814879148801488114882148831488414885148861488714888148891489014891148921489314894148951489614897148981489914900149011490214903149041490514906149071490814909149101491114912149131491414915149161491714918149191492014921149221492314924149251492614927149281492914930149311493214933149341493514936149371493814939149401494114942149431494414945149461494714948149491495014951149521495314954149551495614957149581495914960149611496214963149641496514966149671496814969149701497114972149731497414975149761497714978149791498014981149821498314984149851498614987149881498914990149911499214993149941499514996149971499814999150001500115002150031500415005150061500715008150091501015011150121501315014150151501615017150181501915020150211502215023150241502515026150271502815029150301503115032150331503415035150361503715038150391504015041150421504315044150451504615047150481504915050150511505215053150541505515056150571505815059150601506115062150631506415065150661506715068150691507015071150721507315074150751507615077150781507915080150811508215083150841508515086150871508815089150901509115092150931509415095150961509715098150991510015101151021510315104151051510615107151081510915110151111511215113151141511515116151171511815119151201512115122151231512415125151261512715128151291513015131151321513315134151351513615137151381513915140151411514215143151441514515146151471514815149151501515115152151531515415155151561515715158151591516015161151621516315164151651516615167151681516915170151711517215173151741517515176151771517815179151801518115182151831518415185151861518715188151891519015191151921519315194151951519615197151981519915200152011520215203152041520515206152071520815209152101521115212152131521415215152161521715218152191522015221152221522315224152251522615227152281522915230152311523215233152341523515236152371523815239152401524115242152431524415245152461524715248152491525015251152521525315254152551525615257152581525915260152611526215263152641526515266152671526815269152701527115272152731527415275152761527715278152791528015281152821528315284152851528615287152881528915290152911529215293152941529515296152971529815299153001530115302153031530415305153061530715308153091531015311153121531315314153151531615317153181531915320153211532215323153241532515326153271532815329153301533115332153331533415335153361533715338153391534015341153421534315344153451534615347153481534915350153511535215353153541535515356153571535815359153601536115362153631536415365153661536715368153691537015371153721537315374153751537615377153781537915380153811538215383153841538515386153871538815389153901539115392153931539415395153961539715398153991540015401154021540315404154051540615407154081540915410154111541215413154141541515416154171541815419154201542115422154231542415425154261542715428154291543015431154321543315434154351543615437154381543915440154411544215443154441544515446154471544815449154501545115452154531545415455154561545715458154591546015461154621546315464154651546615467154681546915470154711547215473154741547515476154771547815479154801548115482154831548415485154861548715488154891549015491154921549315494154951549615497154981549915500155011550215503155041550515506155071550815509155101551115512155131551415515155161551715518155191552015521155221552315524155251552615527155281552915530155311553215533155341553515536155371553815539155401554115542155431554415545155461554715548155491555015551155521555315554155551555615557155581555915560155611556215563155641556515566155671556815569155701557115572155731557415575155761557715578155791558015581155821558315584155851558615587155881558915590155911559215593155941559515596155971559815599156001560115602156031560415605156061560715608156091561015611156121561315614156151561615617156181561915620156211562215623156241562515626156271562815629156301563115632156331563415635156361563715638156391564015641156421564315644156451564615647156481564915650156511565215653156541565515656156571565815659156601566115662156631566415665156661566715668156691567015671156721567315674156751567615677156781567915680156811568215683156841568515686156871568815689156901569115692156931569415695156961569715698156991570015701157021570315704157051570615707157081570915710157111571215713157141571515716157171571815719157201572115722157231572415725157261572715728157291573015731157321573315734157351573615737157381573915740157411574215743157441574515746157471574815749157501575115752157531575415755157561575715758157591576015761157621576315764157651576615767157681576915770157711577215773157741577515776157771577815779157801578115782157831578415785157861578715788157891579015791157921579315794157951579615797157981579915800158011580215803158041580515806158071580815809158101581115812158131581415815158161581715818158191582015821158221582315824158251582615827158281582915830158311583215833158341583515836158371583815839158401584115842158431584415845158461584715848158491585015851158521585315854158551585615857158581585915860158611586215863158641586515866158671586815869158701587115872158731587415875158761587715878158791588015881158821588315884158851588615887158881588915890158911589215893158941589515896158971589815899159001590115902159031590415905159061590715908159091591015911159121591315914159151591615917159181591915920159211592215923159241592515926159271592815929159301593115932159331593415935159361593715938159391594015941159421594315944159451594615947159481594915950159511595215953159541595515956159571595815959159601596115962159631596415965159661596715968159691597015971159721597315974159751597615977159781597915980159811598215983159841598515986159871598815989159901599115992159931599415995159961599715998159991600016001160021600316004160051600616007160081600916010160111601216013160141601516016160171601816019160201602116022160231602416025160261602716028160291603016031160321603316034160351603616037160381603916040160411604216043160441604516046160471604816049160501605116052160531605416055160561605716058160591606016061160621606316064160651606616067160681606916070160711607216073160741607516076160771607816079160801608116082160831608416085160861608716088160891609016091160921609316094160951609616097160981609916100161011610216103161041610516106161071610816109161101611116112161131611416115161161611716118161191612016121161221612316124161251612616127161281612916130161311613216133161341613516136161371613816139161401614116142161431614416145161461614716148161491615016151161521615316154161551615616157161581615916160161611616216163161641616516166161671616816169161701617116172161731617416175161761617716178161791618016181161821618316184161851618616187161881618916190161911619216193161941619516196161971619816199162001620116202162031620416205162061620716208162091621016211162121621316214162151621616217162181621916220162211622216223162241622516226162271622816229162301623116232162331623416235162361623716238162391624016241162421624316244162451624616247162481624916250162511625216253162541625516256162571625816259162601626116262162631626416265162661626716268162691627016271162721627316274162751627616277162781627916280162811628216283162841628516286162871628816289162901629116292162931629416295162961629716298162991630016301163021630316304163051630616307163081630916310163111631216313163141631516316163171631816319163201632116322163231632416325163261632716328163291633016331163321633316334163351633616337163381633916340163411634216343163441634516346163471634816349163501635116352163531635416355163561635716358163591636016361163621636316364163651636616367163681636916370163711637216373163741637516376163771637816379163801638116382163831638416385163861638716388163891639016391163921639316394163951639616397163981639916400164011640216403164041640516406164071640816409164101641116412164131641416415164161641716418164191642016421164221642316424164251642616427164281642916430164311643216433164341643516436164371643816439164401644116442164431644416445164461644716448164491645016451164521645316454164551645616457164581645916460164611646216463164641646516466164671646816469164701647116472164731647416475164761647716478164791648016481164821648316484164851648616487164881648916490164911649216493164941649516496164971649816499165001650116502165031650416505165061650716508165091651016511165121651316514165151651616517165181651916520165211652216523165241652516526165271652816529165301653116532165331653416535165361653716538165391654016541165421654316544165451654616547165481654916550165511655216553165541655516556165571655816559165601656116562165631656416565165661656716568165691657016571165721657316574165751657616577165781657916580165811658216583165841658516586165871658816589165901659116592165931659416595165961659716598165991660016601166021660316604166051660616607166081660916610166111661216613166141661516616166171661816619166201662116622166231662416625166261662716628166291663016631166321663316634166351663616637166381663916640166411664216643166441664516646166471664816649166501665116652166531665416655166561665716658166591666016661166621666316664166651666616667166681666916670166711667216673166741667516676166771667816679166801668116682166831668416685166861668716688166891669016691166921669316694166951669616697166981669916700167011670216703167041670516706167071670816709167101671116712167131671416715167161671716718167191672016721167221672316724167251672616727167281672916730167311673216733167341673516736167371673816739167401674116742167431674416745167461674716748167491675016751167521675316754167551675616757167581675916760167611676216763167641676516766167671676816769167701677116772167731677416775167761677716778167791678016781167821678316784167851678616787167881678916790167911679216793167941679516796167971679816799168001680116802168031680416805168061680716808168091681016811168121681316814168151681616817168181681916820168211682216823168241682516826168271682816829168301683116832168331683416835168361683716838168391684016841168421684316844168451684616847168481684916850168511685216853168541685516856168571685816859168601686116862168631686416865168661686716868168691687016871168721687316874168751687616877168781687916880168811688216883168841688516886168871688816889168901689116892168931689416895168961689716898168991690016901169021690316904169051690616907169081690916910169111691216913169141691516916169171691816919169201692116922169231692416925169261692716928169291693016931169321693316934169351693616937169381693916940169411694216943169441694516946169471694816949169501695116952169531695416955169561695716958169591696016961169621696316964169651696616967169681696916970169711697216973169741697516976169771697816979169801698116982169831698416985169861698716988169891699016991169921699316994169951699616997169981699917000170011700217003170041700517006170071700817009170101701117012170131701417015170161701717018170191702017021170221702317024170251702617027170281702917030170311703217033170341703517036170371703817039170401704117042170431704417045170461704717048170491705017051170521705317054170551705617057170581705917060170611706217063170641706517066170671706817069170701707117072170731707417075170761707717078170791708017081170821708317084170851708617087170881708917090170911709217093170941709517096170971709817099171001710117102171031710417105171061710717108171091711017111171121711317114171151711617117171181711917120171211712217123171241712517126171271712817129171301713117132171331713417135171361713717138171391714017141171421714317144171451714617147171481714917150171511715217153171541715517156171571715817159171601716117162171631716417165171661716717168171691717017171171721717317174171751717617177171781717917180171811718217183171841718517186171871718817189171901719117192171931719417195171961719717198171991720017201172021720317204172051720617207172081720917210172111721217213172141721517216172171721817219172201722117222172231722417225172261722717228172291723017231172321723317234172351723617237172381723917240172411724217243172441724517246172471724817249172501725117252172531725417255172561725717258172591726017261172621726317264172651726617267172681726917270172711727217273172741727517276172771727817279172801728117282172831728417285172861728717288172891729017291172921729317294172951729617297172981729917300173011730217303173041730517306173071730817309173101731117312173131731417315173161731717318173191732017321173221732317324173251732617327173281732917330173311733217333173341733517336173371733817339173401734117342173431734417345173461734717348173491735017351173521735317354173551735617357173581735917360173611736217363173641736517366173671736817369173701737117372173731737417375173761737717378173791738017381173821738317384173851738617387173881738917390173911739217393173941739517396173971739817399174001740117402174031740417405174061740717408174091741017411174121741317414174151741617417174181741917420174211742217423174241742517426174271742817429174301743117432174331743417435174361743717438174391744017441174421744317444174451744617447174481744917450174511745217453174541745517456174571745817459174601746117462174631746417465174661746717468174691747017471174721747317474174751747617477174781747917480174811748217483174841748517486174871748817489174901749117492174931749417495174961749717498174991750017501175021750317504175051750617507175081750917510175111751217513175141751517516175171751817519175201752117522175231752417525175261752717528175291753017531175321753317534175351753617537175381753917540175411754217543175441754517546175471754817549175501755117552175531755417555175561755717558175591756017561175621756317564175651756617567175681756917570175711757217573175741757517576175771757817579175801758117582175831758417585175861758717588175891759017591175921759317594175951759617597175981759917600176011760217603176041760517606176071760817609176101761117612176131761417615176161761717618176191762017621176221762317624176251762617627176281762917630176311763217633176341763517636176371763817639176401764117642176431764417645176461764717648176491765017651176521765317654176551765617657176581765917660176611766217663176641766517666176671766817669176701767117672176731767417675176761767717678176791768017681176821768317684176851768617687176881768917690176911769217693176941769517696176971769817699177001770117702177031770417705177061770717708177091771017711177121771317714177151771617717177181771917720177211772217723177241772517726177271772817729177301773117732177331773417735177361773717738177391774017741177421774317744177451774617747177481774917750177511775217753177541775517756177571775817759177601776117762177631776417765177661776717768177691777017771177721777317774177751777617777177781777917780177811778217783177841778517786177871778817789177901779117792177931779417795177961779717798177991780017801178021780317804178051780617807178081780917810178111781217813178141781517816178171781817819178201782117822178231782417825178261782717828178291783017831178321783317834178351783617837178381783917840178411784217843178441784517846178471784817849178501785117852178531785417855178561785717858178591786017861178621786317864178651786617867178681786917870178711787217873178741787517876178771787817879178801788117882178831788417885178861788717888178891789017891178921789317894178951789617897178981789917900179011790217903179041790517906179071790817909179101791117912179131791417915179161791717918179191792017921179221792317924179251792617927179281792917930179311793217933179341793517936179371793817939179401794117942179431794417945179461794717948179491795017951179521795317954179551795617957179581795917960179611796217963179641796517966179671796817969179701797117972179731797417975179761797717978179791798017981179821798317984179851798617987179881798917990179911799217993179941799517996179971799817999180001800118002180031800418005180061800718008180091801018011180121801318014180151801618017180181801918020180211802218023180241802518026180271802818029180301803118032180331803418035180361803718038180391804018041180421804318044180451804618047180481804918050180511805218053180541805518056180571805818059180601806118062180631806418065180661806718068180691807018071180721807318074180751807618077180781807918080180811808218083180841808518086180871808818089180901809118092180931809418095180961809718098180991810018101181021810318104181051810618107181081810918110181111811218113181141811518116181171811818119181201812118122181231812418125181261812718128181291813018131181321813318134181351813618137181381813918140181411814218143181441814518146181471814818149181501815118152181531815418155181561815718158181591816018161181621816318164181651816618167181681816918170181711817218173181741817518176181771817818179181801818118182181831818418185181861818718188181891819018191181921819318194181951819618197181981819918200182011820218203182041820518206182071820818209182101821118212182131821418215182161821718218182191822018221182221822318224182251822618227182281822918230182311823218233182341823518236182371823818239182401824118242182431824418245182461824718248182491825018251182521825318254182551825618257182581825918260182611826218263182641826518266182671826818269182701827118272182731827418275182761827718278182791828018281182821828318284182851828618287182881828918290182911829218293182941829518296182971829818299183001830118302183031830418305183061830718308183091831018311183121831318314183151831618317183181831918320183211832218323183241832518326183271832818329183301833118332183331833418335183361833718338183391834018341183421834318344183451834618347183481834918350183511835218353183541835518356183571835818359183601836118362183631836418365183661836718368183691837018371183721837318374183751837618377183781837918380183811838218383183841838518386183871838818389183901839118392183931839418395183961839718398183991840018401184021840318404184051840618407184081840918410184111841218413184141841518416184171841818419184201842118422184231842418425184261842718428184291843018431184321843318434184351843618437184381843918440184411844218443184441844518446184471844818449184501845118452184531845418455184561845718458184591846018461184621846318464184651846618467184681846918470184711847218473184741847518476184771847818479184801848118482184831848418485184861848718488184891849018491184921849318494184951849618497184981849918500185011850218503185041850518506185071850818509185101851118512185131851418515185161851718518185191852018521185221852318524185251852618527185281852918530185311853218533185341853518536185371853818539185401854118542185431854418545185461854718548185491855018551185521855318554185551855618557185581855918560185611856218563185641856518566185671856818569185701857118572185731857418575185761857718578185791858018581185821858318584185851858618587185881858918590185911859218593185941859518596185971859818599186001860118602186031860418605186061860718608186091861018611186121861318614186151861618617186181861918620186211862218623186241862518626186271862818629186301863118632186331863418635186361863718638186391864018641186421864318644186451864618647186481864918650186511865218653186541865518656186571865818659186601866118662186631866418665186661866718668186691867018671186721867318674186751867618677186781867918680186811868218683186841868518686186871868818689186901869118692186931869418695186961869718698186991870018701187021870318704187051870618707187081870918710187111871218713187141871518716187171871818719187201872118722187231872418725187261872718728187291873018731187321873318734187351873618737187381873918740187411874218743187441874518746187471874818749187501875118752187531875418755187561875718758187591876018761187621876318764187651876618767187681876918770187711877218773187741877518776187771877818779187801878118782187831878418785187861878718788187891879018791187921879318794187951879618797187981879918800188011880218803188041880518806188071880818809188101881118812188131881418815188161881718818188191882018821188221882318824188251882618827188281882918830188311883218833188341883518836188371883818839188401884118842188431884418845188461884718848188491885018851188521885318854188551885618857188581885918860188611886218863188641886518866188671886818869188701887118872188731887418875188761887718878188791888018881188821888318884188851888618887188881888918890188911889218893188941889518896188971889818899189001890118902189031890418905189061890718908189091891018911189121891318914189151891618917189181891918920189211892218923189241892518926189271892818929189301893118932189331893418935189361893718938189391894018941189421894318944189451894618947189481894918950189511895218953189541895518956189571895818959189601896118962189631896418965189661896718968189691897018971189721897318974189751897618977189781897918980189811898218983189841898518986189871898818989189901899118992189931899418995189961899718998189991900019001190021900319004190051900619007190081900919010190111901219013190141901519016190171901819019190201902119022190231902419025190261902719028190291903019031190321903319034190351903619037190381903919040190411904219043190441904519046190471904819049190501905119052190531905419055190561905719058190591906019061190621906319064190651906619067190681906919070190711907219073190741907519076190771907819079190801908119082190831908419085190861908719088190891909019091190921909319094190951909619097190981909919100191011910219103191041910519106191071910819109191101911119112191131911419115191161911719118191191912019121191221912319124191251912619127191281912919130191311913219133191341913519136191371913819139191401914119142191431914419145191461914719148191491915019151191521915319154191551915619157191581915919160191611916219163191641916519166191671916819169191701917119172191731917419175191761917719178191791918019181191821918319184191851918619187191881918919190191911919219193191941919519196191971919819199192001920119202192031920419205192061920719208192091921019211192121921319214192151921619217192181921919220192211922219223192241922519226192271922819229192301923119232192331923419235192361923719238192391924019241192421924319244192451924619247192481924919250192511925219253192541925519256192571925819259192601926119262192631926419265192661926719268192691927019271192721927319274192751927619277192781927919280192811928219283192841928519286192871928819289192901929119292192931929419295192961929719298192991930019301193021930319304193051930619307193081930919310193111931219313193141931519316193171931819319193201932119322193231932419325193261932719328193291933019331193321933319334193351933619337193381933919340193411934219343193441934519346193471934819349193501935119352193531935419355193561935719358193591936019361193621936319364193651936619367193681936919370193711937219373193741937519376193771937819379193801938119382193831938419385193861938719388193891939019391193921939319394193951939619397193981939919400194011940219403194041940519406194071940819409194101941119412194131941419415194161941719418194191942019421194221942319424194251942619427194281942919430194311943219433194341943519436194371943819439194401944119442194431944419445194461944719448194491945019451194521945319454194551945619457194581945919460194611946219463194641946519466194671946819469194701947119472194731947419475194761947719478194791948019481194821948319484194851948619487194881948919490194911949219493194941949519496194971949819499195001950119502195031950419505195061950719508195091951019511195121951319514195151951619517195181951919520195211952219523195241952519526195271952819529195301953119532195331953419535195361953719538195391954019541195421954319544195451954619547195481954919550195511955219553195541955519556195571955819559195601956119562195631956419565195661956719568195691957019571195721957319574195751957619577195781957919580195811958219583195841958519586195871958819589195901959119592195931959419595195961959719598195991960019601196021960319604196051960619607196081960919610196111961219613196141961519616196171961819619196201962119622196231962419625196261962719628196291963019631196321963319634196351963619637196381963919640196411964219643196441964519646196471964819649196501965119652196531965419655196561965719658196591966019661196621966319664196651966619667196681966919670196711967219673196741967519676196771967819679196801968119682196831968419685196861968719688196891969019691196921969319694196951969619697196981969919700197011970219703197041970519706197071970819709197101971119712197131971419715197161971719718197191972019721197221972319724197251972619727197281972919730197311973219733197341973519736197371973819739197401974119742197431974419745197461974719748197491975019751197521975319754197551975619757197581975919760197611976219763197641976519766197671976819769197701977119772197731977419775197761977719778197791978019781197821978319784197851978619787197881978919790197911979219793197941979519796197971979819799198001980119802198031980419805198061980719808198091981019811198121981319814198151981619817198181981919820198211982219823198241982519826198271982819829198301983119832198331983419835198361983719838198391984019841198421984319844198451984619847198481984919850198511985219853198541985519856198571985819859198601986119862198631986419865198661986719868198691987019871198721987319874198751987619877198781987919880198811988219883198841988519886198871988819889198901989119892198931989419895198961989719898198991990019901199021990319904199051990619907199081990919910199111991219913199141991519916199171991819919199201992119922199231992419925199261992719928199291993019931199321993319934199351993619937199381993919940199411994219943199441994519946199471994819949199501995119952199531995419955199561995719958199591996019961199621996319964199651996619967199681996919970199711997219973199741997519976199771997819979199801998119982199831998419985199861998719988199891999019991199921999319994199951999619997199981999920000200012000220003200042000520006200072000820009200102001120012200132001420015200162001720018200192002020021200222002320024200252002620027200282002920030200312003220033200342003520036200372003820039200402004120042200432004420045200462004720048200492005020051200522005320054200552005620057200582005920060200612006220063200642006520066200672006820069200702007120072200732007420075200762007720078200792008020081200822008320084200852008620087200882008920090200912009220093200942009520096200972009820099201002010120102201032010420105201062010720108201092011020111201122011320114201152011620117201182011920120201212012220123201242012520126201272012820129201302013120132201332013420135201362013720138201392014020141201422014320144201452014620147201482014920150201512015220153201542015520156201572015820159201602016120162201632016420165201662016720168201692017020171201722017320174201752017620177201782017920180201812018220183201842018520186201872018820189201902019120192201932019420195201962019720198201992020020201202022020320204202052020620207202082020920210202112021220213202142021520216202172021820219202202022120222202232022420225202262022720228202292023020231202322023320234202352023620237202382023920240202412024220243202442024520246202472024820249202502025120252202532025420255202562025720258202592026020261202622026320264202652026620267202682026920270202712027220273202742027520276202772027820279202802028120282202832028420285202862028720288202892029020291202922029320294202952029620297202982029920300203012030220303203042030520306203072030820309203102031120312203132031420315203162031720318203192032020321203222032320324203252032620327203282032920330203312033220333203342033520336203372033820339203402034120342203432034420345203462034720348203492035020351203522035320354203552035620357203582035920360203612036220363203642036520366203672036820369203702037120372203732037420375203762037720378203792038020381203822038320384203852038620387203882038920390203912039220393203942039520396203972039820399204002040120402204032040420405204062040720408204092041020411204122041320414204152041620417204182041920420204212042220423204242042520426204272042820429204302043120432204332043420435204362043720438204392044020441204422044320444204452044620447204482044920450204512045220453204542045520456204572045820459204602046120462204632046420465204662046720468204692047020471204722047320474204752047620477204782047920480204812048220483204842048520486204872048820489204902049120492204932049420495204962049720498204992050020501205022050320504205052050620507205082050920510205112051220513205142051520516205172051820519205202052120522205232052420525205262052720528205292053020531205322053320534205352053620537205382053920540205412054220543205442054520546205472054820549205502055120552205532055420555205562055720558205592056020561205622056320564205652056620567205682056920570205712057220573205742057520576205772057820579205802058120582205832058420585205862058720588205892059020591205922059320594205952059620597205982059920600206012060220603206042060520606206072060820609206102061120612206132061420615206162061720618206192062020621206222062320624206252062620627206282062920630206312063220633206342063520636206372063820639206402064120642206432064420645206462064720648206492065020651206522065320654206552065620657206582065920660206612066220663206642066520666206672066820669206702067120672206732067420675206762067720678206792068020681206822068320684206852068620687206882068920690206912069220693206942069520696206972069820699207002070120702207032070420705207062070720708207092071020711207122071320714207152071620717207182071920720207212072220723207242072520726207272072820729207302073120732207332073420735207362073720738207392074020741207422074320744207452074620747207482074920750207512075220753207542075520756207572075820759207602076120762207632076420765207662076720768207692077020771207722077320774207752077620777207782077920780207812078220783207842078520786207872078820789207902079120792207932079420795207962079720798207992080020801208022080320804208052080620807208082080920810208112081220813208142081520816208172081820819208202082120822208232082420825208262082720828208292083020831208322083320834208352083620837208382083920840208412084220843208442084520846208472084820849208502085120852208532085420855208562085720858208592086020861208622086320864208652086620867208682086920870208712087220873208742087520876208772087820879208802088120882208832088420885208862088720888208892089020891208922089320894208952089620897208982089920900209012090220903209042090520906209072090820909209102091120912209132091420915209162091720918209192092020921209222092320924209252092620927209282092920930209312093220933209342093520936209372093820939209402094120942209432094420945209462094720948209492095020951209522095320954209552095620957209582095920960209612096220963209642096520966209672096820969209702097120972209732097420975209762097720978209792098020981209822098320984209852098620987209882098920990209912099220993209942099520996209972099820999210002100121002210032100421005210062100721008210092101021011210122101321014210152101621017210182101921020210212102221023210242102521026210272102821029210302103121032210332103421035210362103721038210392104021041210422104321044210452104621047210482104921050210512105221053210542105521056210572105821059210602106121062210632106421065210662106721068210692107021071210722107321074210752107621077210782107921080210812108221083210842108521086210872108821089210902109121092210932109421095210962109721098210992110021101211022110321104211052110621107211082110921110211112111221113211142111521116211172111821119211202112121122211232112421125211262112721128211292113021131211322113321134211352113621137211382113921140211412114221143211442114521146211472114821149211502115121152211532115421155211562115721158211592116021161211622116321164211652116621167211682116921170211712117221173211742117521176211772117821179211802118121182211832118421185211862118721188211892119021191211922119321194211952119621197211982119921200212012120221203212042120521206212072120821209212102121121212212132121421215212162121721218212192122021221212222122321224212252122621227212282122921230212312123221233212342123521236212372123821239212402124121242212432124421245212462124721248212492125021251212522125321254212552125621257212582125921260212612126221263212642126521266212672126821269212702127121272212732127421275212762127721278212792128021281212822128321284212852128621287212882128921290212912129221293212942129521296212972129821299213002130121302213032130421305213062130721308213092131021311213122131321314213152131621317213182131921320213212132221323213242132521326213272132821329213302133121332213332133421335213362133721338213392134021341213422134321344213452134621347213482134921350213512135221353213542135521356213572135821359213602136121362213632136421365213662136721368213692137021371213722137321374213752137621377213782137921380213812138221383213842138521386213872138821389213902139121392213932139421395213962139721398213992140021401214022140321404214052140621407214082140921410214112141221413214142141521416214172141821419214202142121422214232142421425214262142721428214292143021431214322143321434214352143621437214382143921440214412144221443214442144521446214472144821449214502145121452214532145421455214562145721458214592146021461214622146321464214652146621467214682146921470214712147221473214742147521476214772147821479214802148121482214832148421485214862148721488214892149021491214922149321494214952149621497214982149921500215012150221503215042150521506215072150821509215102151121512215132151421515215162151721518215192152021521215222152321524215252152621527215282152921530215312153221533215342153521536215372153821539215402154121542215432154421545215462154721548215492155021551215522155321554215552155621557215582155921560215612156221563215642156521566215672156821569215702157121572215732157421575215762157721578215792158021581215822158321584215852158621587215882158921590215912159221593215942159521596215972159821599216002160121602216032160421605216062160721608216092161021611216122161321614216152161621617216182161921620216212162221623216242162521626216272162821629216302163121632216332163421635216362163721638216392164021641216422164321644216452164621647216482164921650216512165221653216542165521656216572165821659216602166121662216632166421665216662166721668216692167021671216722167321674216752167621677216782167921680216812168221683216842168521686216872168821689216902169121692216932169421695216962169721698216992170021701217022170321704217052170621707217082170921710217112171221713217142171521716217172171821719217202172121722217232172421725217262172721728217292173021731217322173321734217352173621737217382173921740217412174221743217442174521746217472174821749217502175121752217532175421755217562175721758217592176021761217622176321764217652176621767217682176921770217712177221773217742177521776217772177821779217802178121782217832178421785217862178721788217892179021791217922179321794217952179621797217982179921800218012180221803218042180521806218072180821809218102181121812218132181421815218162181721818218192182021821218222182321824218252182621827218282182921830218312183221833218342183521836218372183821839218402184121842218432184421845218462184721848218492185021851218522185321854218552185621857218582185921860218612186221863218642186521866218672186821869218702187121872218732187421875218762187721878218792188021881218822188321884218852188621887218882188921890218912189221893218942189521896218972189821899219002190121902219032190421905219062190721908219092191021911219122191321914219152191621917219182191921920219212192221923219242192521926219272192821929219302193121932219332193421935219362193721938219392194021941219422194321944219452194621947219482194921950219512195221953219542195521956219572195821959219602196121962219632196421965219662196721968219692197021971219722197321974219752197621977219782197921980219812198221983219842198521986219872198821989219902199121992219932199421995219962199721998219992200022001220022200322004220052200622007220082200922010220112201222013220142201522016220172201822019220202202122022220232202422025220262202722028220292203022031220322203322034220352203622037220382203922040220412204222043220442204522046220472204822049220502205122052220532205422055220562205722058220592206022061220622206322064220652206622067220682206922070220712207222073220742207522076220772207822079220802208122082220832208422085220862208722088220892209022091220922209322094220952209622097220982209922100221012210222103221042210522106221072210822109221102211122112221132211422115221162211722118221192212022121221222212322124221252212622127221282212922130221312213222133221342213522136221372213822139221402214122142221432214422145221462214722148221492215022151221522215322154221552215622157221582215922160221612216222163221642216522166221672216822169221702217122172221732217422175221762217722178221792218022181221822218322184221852218622187221882218922190221912219222193221942219522196221972219822199222002220122202222032220422205222062220722208222092221022211222122221322214222152221622217222182221922220222212222222223222242222522226222272222822229222302223122232222332223422235222362223722238222392224022241222422224322244222452224622247222482224922250222512225222253222542225522256222572225822259222602226122262222632226422265222662226722268222692227022271222722227322274222752227622277222782227922280222812228222283222842228522286222872228822289222902229122292222932229422295222962229722298222992230022301223022230322304223052230622307223082230922310223112231222313223142231522316223172231822319223202232122322223232232422325223262232722328223292233022331223322233322334223352233622337223382233922340223412234222343223442234522346223472234822349223502235122352223532235422355223562235722358223592236022361223622236322364223652236622367223682236922370223712237222373223742237522376223772237822379223802238122382223832238422385223862238722388223892239022391223922239322394223952239622397223982239922400224012240222403224042240522406224072240822409224102241122412224132241422415224162241722418224192242022421224222242322424224252242622427224282242922430224312243222433224342243522436224372243822439224402244122442224432244422445224462244722448224492245022451224522245322454224552245622457224582245922460224612246222463224642246522466224672246822469224702247122472224732247422475224762247722478224792248022481224822248322484224852248622487224882248922490224912249222493224942249522496224972249822499225002250122502225032250422505225062250722508225092251022511225122251322514225152251622517225182251922520225212252222523225242252522526225272252822529225302253122532225332253422535225362253722538225392254022541225422254322544225452254622547225482254922550225512255222553225542255522556225572255822559225602256122562225632256422565225662256722568225692257022571225722257322574225752257622577225782257922580225812258222583225842258522586225872258822589225902259122592225932259422595225962259722598225992260022601226022260322604226052260622607226082260922610226112261222613226142261522616226172261822619226202262122622226232262422625226262262722628226292263022631226322263322634226352263622637226382263922640226412264222643226442264522646226472264822649226502265122652226532265422655226562265722658226592266022661226622266322664226652266622667226682266922670226712267222673226742267522676226772267822679226802268122682226832268422685226862268722688226892269022691226922269322694226952269622697226982269922700227012270222703227042270522706227072270822709227102271122712227132271422715227162271722718227192272022721227222272322724227252272622727227282272922730227312273222733227342273522736227372273822739227402274122742227432274422745227462274722748227492275022751227522275322754227552275622757227582275922760227612276222763227642276522766227672276822769227702277122772227732277422775227762277722778227792278022781227822278322784227852278622787227882278922790227912279222793227942279522796227972279822799228002280122802228032280422805228062280722808228092281022811228122281322814228152281622817228182281922820228212282222823228242282522826228272282822829228302283122832228332283422835228362283722838228392284022841228422284322844228452284622847228482284922850228512285222853228542285522856228572285822859228602286122862228632286422865228662286722868228692287022871228722287322874228752287622877228782287922880228812288222883228842288522886228872288822889228902289122892228932289422895228962289722898228992290022901229022290322904229052290622907229082290922910229112291222913229142291522916229172291822919229202292122922229232292422925229262292722928229292293022931229322293322934229352293622937229382293922940229412294222943229442294522946229472294822949229502295122952229532295422955229562295722958229592296022961229622296322964229652296622967229682296922970229712297222973229742297522976229772297822979229802298122982229832298422985229862298722988229892299022991229922299322994229952299622997229982299923000230012300223003230042300523006230072300823009230102301123012230132301423015230162301723018230192302023021230222302323024230252302623027230282302923030230312303223033230342303523036230372303823039230402304123042230432304423045230462304723048230492305023051230522305323054230552305623057230582305923060230612306223063230642306523066230672306823069230702307123072230732307423075230762307723078230792308023081230822308323084230852308623087230882308923090230912309223093230942309523096230972309823099231002310123102231032310423105231062310723108231092311023111231122311323114231152311623117231182311923120231212312223123231242312523126231272312823129231302313123132231332313423135231362313723138231392314023141231422314323144231452314623147231482314923150231512315223153231542315523156231572315823159231602316123162231632316423165231662316723168231692317023171231722317323174231752317623177231782317923180231812318223183231842318523186231872318823189231902319123192231932319423195231962319723198231992320023201232022320323204232052320623207232082320923210232112321223213232142321523216232172321823219232202322123222232232322423225232262322723228232292323023231232322323323234232352323623237232382323923240232412324223243232442324523246232472324823249232502325123252232532325423255232562325723258232592326023261232622326323264232652326623267232682326923270232712327223273232742327523276232772327823279232802328123282232832328423285232862328723288232892329023291232922329323294232952329623297232982329923300233012330223303233042330523306233072330823309233102331123312233132331423315233162331723318233192332023321233222332323324233252332623327233282332923330233312333223333233342333523336233372333823339233402334123342233432334423345233462334723348233492335023351233522335323354233552335623357233582335923360233612336223363233642336523366233672336823369233702337123372233732337423375233762337723378233792338023381233822338323384233852338623387233882338923390233912339223393233942339523396233972339823399234002340123402234032340423405234062340723408234092341023411234122341323414234152341623417234182341923420234212342223423234242342523426234272342823429234302343123432234332343423435234362343723438234392344023441234422344323444234452344623447234482344923450234512345223453234542345523456234572345823459234602346123462234632346423465234662346723468234692347023471234722347323474234752347623477234782347923480234812348223483234842348523486234872348823489234902349123492234932349423495234962349723498234992350023501235022350323504235052350623507235082350923510235112351223513235142351523516235172351823519235202352123522235232352423525235262352723528235292353023531235322353323534235352353623537235382353923540235412354223543235442354523546235472354823549235502355123552235532355423555235562355723558235592356023561235622356323564235652356623567235682356923570235712357223573235742357523576235772357823579235802358123582235832358423585235862358723588235892359023591235922359323594235952359623597235982359923600236012360223603236042360523606236072360823609236102361123612236132361423615236162361723618236192362023621236222362323624236252362623627236282362923630236312363223633236342363523636236372363823639236402364123642236432364423645236462364723648236492365023651236522365323654236552365623657236582365923660236612366223663236642366523666236672366823669236702367123672236732367423675236762367723678236792368023681236822368323684236852368623687236882368923690236912369223693236942369523696236972369823699237002370123702237032370423705237062370723708237092371023711237122371323714237152371623717237182371923720237212372223723237242372523726237272372823729237302373123732237332373423735237362373723738237392374023741237422374323744237452374623747237482374923750237512375223753237542375523756237572375823759237602376123762237632376423765237662376723768237692377023771237722377323774237752377623777237782377923780237812378223783237842378523786237872378823789237902379123792237932379423795237962379723798237992380023801238022380323804238052380623807238082380923810238112381223813238142381523816238172381823819238202382123822238232382423825238262382723828238292383023831238322383323834238352383623837238382383923840238412384223843238442384523846238472384823849238502385123852238532385423855238562385723858238592386023861238622386323864238652386623867238682386923870238712387223873238742387523876238772387823879238802388123882238832388423885238862388723888238892389023891238922389323894238952389623897238982389923900239012390223903239042390523906239072390823909239102391123912239132391423915239162391723918239192392023921239222392323924239252392623927239282392923930239312393223933239342393523936239372393823939239402394123942239432394423945239462394723948239492395023951239522395323954239552395623957239582395923960239612396223963239642396523966239672396823969239702397123972239732397423975239762397723978239792398023981239822398323984239852398623987239882398923990239912399223993239942399523996239972399823999240002400124002240032400424005240062400724008240092401024011240122401324014240152401624017240182401924020240212402224023240242402524026240272402824029240302403124032240332403424035240362403724038240392404024041240422404324044240452404624047240482404924050240512405224053240542405524056240572405824059240602406124062240632406424065240662406724068240692407024071240722407324074240752407624077240782407924080240812408224083240842408524086240872408824089240902409124092240932409424095240962409724098240992410024101241022410324104241052410624107241082410924110241112411224113241142411524116241172411824119241202412124122241232412424125241262412724128241292413024131241322413324134241352413624137241382413924140241412414224143241442414524146241472414824149241502415124152241532415424155241562415724158241592416024161241622416324164241652416624167241682416924170241712417224173241742417524176241772417824179241802418124182241832418424185241862418724188241892419024191241922419324194241952419624197241982419924200242012420224203242042420524206242072420824209242102421124212242132421424215242162421724218242192422024221242222422324224242252422624227242282422924230242312423224233242342423524236242372423824239242402424124242242432424424245242462424724248242492425024251242522425324254242552425624257242582425924260242612426224263242642426524266242672426824269242702427124272242732427424275242762427724278242792428024281242822428324284242852428624287242882428924290242912429224293242942429524296242972429824299243002430124302243032430424305243062430724308243092431024311243122431324314243152431624317243182431924320243212432224323243242432524326243272432824329243302433124332243332433424335243362433724338243392434024341243422434324344243452434624347243482434924350243512435224353243542435524356243572435824359243602436124362243632436424365243662436724368243692437024371243722437324374243752437624377243782437924380243812438224383243842438524386243872438824389243902439124392243932439424395243962439724398243992440024401244022440324404244052440624407244082440924410244112441224413244142441524416244172441824419244202442124422244232442424425244262442724428244292443024431244322443324434244352443624437244382443924440244412444224443244442444524446244472444824449244502445124452244532445424455244562445724458244592446024461244622446324464244652446624467244682446924470244712447224473244742447524476244772447824479244802448124482244832448424485244862448724488244892449024491244922449324494244952449624497244982449924500245012450224503245042450524506245072450824509245102451124512245132451424515245162451724518245192452024521245222452324524245252452624527245282452924530245312453224533245342453524536245372453824539245402454124542245432454424545245462454724548245492455024551245522455324554245552455624557245582455924560245612456224563245642456524566245672456824569245702457124572245732457424575245762457724578245792458024581245822458324584245852458624587245882458924590245912459224593245942459524596245972459824599246002460124602246032460424605246062460724608246092461024611246122461324614246152461624617246182461924620246212462224623246242462524626246272462824629246302463124632246332463424635246362463724638246392464024641246422464324644246452464624647246482464924650246512465224653246542465524656246572465824659246602466124662246632466424665246662466724668246692467024671246722467324674246752467624677246782467924680246812468224683246842468524686246872468824689246902469124692246932469424695246962469724698246992470024701247022470324704247052470624707247082470924710247112471224713247142471524716247172471824719247202472124722247232472424725247262472724728247292473024731247322473324734247352473624737247382473924740247412474224743247442474524746247472474824749247502475124752247532475424755247562475724758247592476024761247622476324764247652476624767247682476924770247712477224773247742477524776247772477824779247802478124782247832478424785247862478724788247892479024791247922479324794247952479624797247982479924800248012480224803248042480524806248072480824809248102481124812248132481424815248162481724818248192482024821248222482324824248252482624827248282482924830248312483224833248342483524836248372483824839248402484124842248432484424845248462484724848248492485024851248522485324854248552485624857248582485924860248612486224863248642486524866248672486824869248702487124872248732487424875248762487724878248792488024881248822488324884248852488624887248882488924890248912489224893248942489524896248972489824899249002490124902249032490424905249062490724908249092491024911249122491324914249152491624917249182491924920249212492224923249242492524926249272492824929249302493124932249332493424935249362493724938249392494024941249422494324944249452494624947249482494924950249512495224953249542495524956249572495824959249602496124962249632496424965249662496724968249692497024971249722497324974249752497624977249782497924980249812498224983249842498524986249872498824989249902499124992249932499424995249962499724998249992500025001250022500325004250052500625007250082500925010250112501225013250142501525016250172501825019250202502125022250232502425025250262502725028250292503025031250322503325034250352503625037250382503925040250412504225043250442504525046250472504825049250502505125052250532505425055250562505725058250592506025061250622506325064250652506625067250682506925070250712507225073250742507525076250772507825079250802508125082250832508425085250862508725088250892509025091250922509325094250952509625097250982509925100251012510225103251042510525106251072510825109251102511125112251132511425115251162511725118251192512025121251222512325124251252512625127251282512925130251312513225133251342513525136251372513825139251402514125142251432514425145251462514725148251492515025151251522515325154251552515625157251582515925160251612516225163251642516525166251672516825169251702517125172251732517425175251762517725178251792518025181251822518325184251852518625187251882518925190251912519225193251942519525196251972519825199252002520125202252032520425205252062520725208252092521025211252122521325214252152521625217252182521925220252212522225223252242522525226252272522825229252302523125232252332523425235252362523725238252392524025241252422524325244252452524625247252482524925250252512525225253252542525525256252572525825259252602526125262252632526425265252662526725268252692527025271252722527325274252752527625277252782527925280252812528225283252842528525286252872528825289252902529125292252932529425295252962529725298252992530025301253022530325304253052530625307253082530925310253112531225313253142531525316253172531825319253202532125322253232532425325253262532725328253292533025331253322533325334253352533625337253382533925340253412534225343253442534525346253472534825349253502535125352253532535425355253562535725358253592536025361253622536325364253652536625367253682536925370253712537225373253742537525376253772537825379253802538125382253832538425385253862538725388253892539025391253922539325394253952539625397253982539925400254012540225403254042540525406254072540825409254102541125412254132541425415254162541725418254192542025421254222542325424254252542625427254282542925430254312543225433254342543525436254372543825439254402544125442254432544425445254462544725448254492545025451254522545325454254552545625457254582545925460254612546225463254642546525466254672546825469254702547125472254732547425475254762547725478254792548025481254822548325484254852548625487254882548925490254912549225493254942549525496254972549825499255002550125502255032550425505255062550725508255092551025511255122551325514255152551625517255182551925520255212552225523255242552525526255272552825529255302553125532255332553425535255362553725538255392554025541255422554325544255452554625547255482554925550255512555225553255542555525556255572555825559255602556125562255632556425565255662556725568255692557025571255722557325574255752557625577255782557925580255812558225583255842558525586255872558825589255902559125592255932559425595255962559725598255992560025601256022560325604256052560625607256082560925610256112561225613256142561525616256172561825619256202562125622256232562425625256262562725628256292563025631256322563325634256352563625637256382563925640256412564225643256442564525646256472564825649256502565125652256532565425655256562565725658256592566025661256622566325664256652566625667256682566925670256712567225673256742567525676256772567825679256802568125682256832568425685256862568725688256892569025691256922569325694256952569625697256982569925700257012570225703257042570525706257072570825709257102571125712257132571425715257162571725718257192572025721257222572325724257252572625727257282572925730257312573225733257342573525736257372573825739257402574125742257432574425745257462574725748257492575025751257522575325754257552575625757257582575925760257612576225763257642576525766257672576825769257702577125772257732577425775257762577725778257792578025781257822578325784257852578625787257882578925790257912579225793257942579525796257972579825799258002580125802258032580425805258062580725808258092581025811258122581325814258152581625817258182581925820258212582225823258242582525826258272582825829258302583125832258332583425835258362583725838258392584025841258422584325844258452584625847258482584925850258512585225853258542585525856258572585825859258602586125862258632586425865258662586725868258692587025871258722587325874258752587625877258782587925880258812588225883258842588525886258872588825889258902589125892258932589425895258962589725898258992590025901259022590325904259052590625907259082590925910259112591225913259142591525916259172591825919259202592125922259232592425925259262592725928259292593025931259322593325934259352593625937259382593925940259412594225943259442594525946259472594825949259502595125952259532595425955259562595725958259592596025961259622596325964259652596625967259682596925970259712597225973259742597525976259772597825979259802598125982259832598425985259862598725988259892599025991259922599325994259952599625997259982599926000260012600226003260042600526006260072600826009260102601126012260132601426015260162601726018260192602026021260222602326024260252602626027260282602926030260312603226033260342603526036260372603826039260402604126042260432604426045260462604726048260492605026051260522605326054260552605626057260582605926060260612606226063260642606526066260672606826069260702607126072260732607426075260762607726078260792608026081260822608326084260852608626087260882608926090260912609226093260942609526096260972609826099261002610126102261032610426105261062610726108261092611026111261122611326114261152611626117261182611926120261212612226123261242612526126261272612826129261302613126132261332613426135261362613726138261392614026141261422614326144261452614626147261482614926150261512615226153261542615526156261572615826159261602616126162261632616426165261662616726168261692617026171261722617326174261752617626177261782617926180261812618226183261842618526186261872618826189261902619126192261932619426195261962619726198261992620026201262022620326204262052620626207262082620926210262112621226213262142621526216262172621826219262202622126222262232622426225262262622726228262292623026231262322623326234262352623626237262382623926240262412624226243262442624526246262472624826249262502625126252262532625426255262562625726258262592626026261262622626326264262652626626267262682626926270262712627226273262742627526276262772627826279262802628126282262832628426285262862628726288262892629026291262922629326294262952629626297262982629926300263012630226303263042630526306263072630826309263102631126312263132631426315263162631726318263192632026321263222632326324263252632626327263282632926330263312633226333263342633526336263372633826339263402634126342263432634426345263462634726348263492635026351263522635326354263552635626357263582635926360263612636226363263642636526366263672636826369263702637126372263732637426375263762637726378263792638026381263822638326384263852638626387263882638926390263912639226393263942639526396263972639826399264002640126402264032640426405264062640726408264092641026411264122641326414264152641626417264182641926420264212642226423264242642526426264272642826429264302643126432264332643426435264362643726438264392644026441264422644326444264452644626447264482644926450264512645226453264542645526456264572645826459264602646126462264632646426465264662646726468264692647026471264722647326474264752647626477264782647926480264812648226483264842648526486264872648826489264902649126492264932649426495264962649726498264992650026501265022650326504265052650626507265082650926510265112651226513265142651526516265172651826519265202652126522265232652426525265262652726528265292653026531265322653326534265352653626537265382653926540265412654226543265442654526546265472654826549265502655126552265532655426555265562655726558265592656026561265622656326564265652656626567265682656926570265712657226573265742657526576265772657826579265802658126582265832658426585265862658726588265892659026591265922659326594265952659626597265982659926600266012660226603266042660526606266072660826609266102661126612266132661426615266162661726618266192662026621266222662326624266252662626627266282662926630266312663226633266342663526636266372663826639266402664126642266432664426645266462664726648266492665026651266522665326654266552665626657266582665926660266612666226663266642666526666266672666826669266702667126672266732667426675266762667726678266792668026681266822668326684266852668626687266882668926690266912669226693266942669526696266972669826699267002670126702267032670426705267062670726708267092671026711267122671326714267152671626717267182671926720267212672226723267242672526726267272672826729267302673126732267332673426735267362673726738267392674026741267422674326744267452674626747267482674926750267512675226753267542675526756267572675826759267602676126762267632676426765267662676726768267692677026771267722677326774267752677626777267782677926780267812678226783267842678526786267872678826789267902679126792267932679426795267962679726798267992680026801268022680326804268052680626807268082680926810268112681226813268142681526816268172681826819268202682126822268232682426825268262682726828268292683026831268322683326834268352683626837268382683926840268412684226843268442684526846268472684826849268502685126852268532685426855268562685726858268592686026861268622686326864268652686626867268682686926870268712687226873268742687526876268772687826879268802688126882268832688426885268862688726888268892689026891268922689326894268952689626897268982689926900269012690226903269042690526906269072690826909269102691126912269132691426915269162691726918269192692026921269222692326924269252692626927269282692926930269312693226933269342693526936269372693826939269402694126942269432694426945269462694726948269492695026951269522695326954269552695626957269582695926960269612696226963269642696526966269672696826969269702697126972269732697426975269762697726978269792698026981269822698326984269852698626987269882698926990269912699226993269942699526996269972699826999270002700127002270032700427005270062700727008270092701027011270122701327014270152701627017270182701927020270212702227023270242702527026270272702827029270302703127032270332703427035270362703727038270392704027041270422704327044270452704627047270482704927050270512705227053270542705527056270572705827059270602706127062270632706427065270662706727068270692707027071270722707327074270752707627077270782707927080270812708227083270842708527086270872708827089270902709127092270932709427095270962709727098270992710027101271022710327104271052710627107271082710927110271112711227113271142711527116271172711827119271202712127122271232712427125271262712727128271292713027131271322713327134271352713627137271382713927140271412714227143271442714527146271472714827149271502715127152271532715427155271562715727158271592716027161271622716327164271652716627167271682716927170271712717227173271742717527176271772717827179271802718127182271832718427185271862718727188271892719027191271922719327194271952719627197271982719927200272012720227203272042720527206272072720827209272102721127212272132721427215272162721727218272192722027221272222722327224272252722627227272282722927230272312723227233272342723527236272372723827239272402724127242272432724427245272462724727248272492725027251272522725327254272552725627257272582725927260272612726227263272642726527266272672726827269272702727127272272732727427275272762727727278272792728027281272822728327284272852728627287272882728927290272912729227293272942729527296272972729827299273002730127302273032730427305273062730727308273092731027311273122731327314273152731627317273182731927320273212732227323273242732527326273272732827329273302733127332273332733427335273362733727338273392734027341273422734327344273452734627347273482734927350273512735227353273542735527356273572735827359273602736127362273632736427365273662736727368273692737027371273722737327374273752737627377273782737927380273812738227383273842738527386273872738827389273902739127392273932739427395273962739727398273992740027401274022740327404274052740627407274082740927410274112741227413274142741527416274172741827419274202742127422274232742427425274262742727428274292743027431274322743327434274352743627437274382743927440274412744227443274442744527446274472744827449274502745127452274532745427455274562745727458274592746027461274622746327464274652746627467274682746927470274712747227473274742747527476274772747827479274802748127482274832748427485274862748727488274892749027491274922749327494274952749627497274982749927500275012750227503275042750527506275072750827509275102751127512275132751427515275162751727518275192752027521275222752327524275252752627527275282752927530275312753227533275342753527536275372753827539275402754127542275432754427545275462754727548275492755027551275522755327554275552755627557275582755927560275612756227563275642756527566275672756827569275702757127572275732757427575275762757727578275792758027581275822758327584275852758627587275882758927590275912759227593275942759527596275972759827599276002760127602276032760427605276062760727608276092761027611276122761327614276152761627617276182761927620276212762227623276242762527626276272762827629276302763127632276332763427635276362763727638276392764027641276422764327644276452764627647276482764927650276512765227653276542765527656276572765827659276602766127662276632766427665276662766727668276692767027671276722767327674276752767627677276782767927680276812768227683276842768527686276872768827689276902769127692276932769427695276962769727698276992770027701277022770327704277052770627707277082770927710277112771227713277142771527716277172771827719277202772127722277232772427725277262772727728277292773027731277322773327734277352773627737277382773927740277412774227743277442774527746277472774827749277502775127752277532775427755277562775727758277592776027761277622776327764277652776627767277682776927770277712777227773277742777527776277772777827779277802778127782277832778427785277862778727788277892779027791277922779327794277952779627797277982779927800278012780227803278042780527806278072780827809278102781127812278132781427815278162781727818278192782027821278222782327824278252782627827278282782927830278312783227833278342783527836278372783827839278402784127842278432784427845278462784727848278492785027851278522785327854278552785627857278582785927860278612786227863278642786527866278672786827869278702787127872278732787427875278762787727878278792788027881278822788327884278852788627887278882788927890278912789227893278942789527896278972789827899279002790127902279032790427905279062790727908279092791027911279122791327914279152791627917279182791927920279212792227923279242792527926279272792827929279302793127932279332793427935279362793727938279392794027941279422794327944279452794627947279482794927950279512795227953279542795527956279572795827959279602796127962279632796427965279662796727968279692797027971279722797327974279752797627977279782797927980279812798227983279842798527986279872798827989279902799127992279932799427995279962799727998279992800028001280022800328004280052800628007280082800928010280112801228013280142801528016280172801828019280202802128022280232802428025280262802728028280292803028031280322803328034280352803628037280382803928040280412804228043280442804528046280472804828049280502805128052280532805428055280562805728058280592806028061280622806328064280652806628067280682806928070280712807228073280742807528076280772807828079280802808128082280832808428085280862808728088280892809028091280922809328094280952809628097280982809928100281012810228103281042810528106281072810828109281102811128112281132811428115281162811728118281192812028121281222812328124281252812628127281282812928130281312813228133281342813528136281372813828139281402814128142281432814428145281462814728148281492815028151281522815328154281552815628157281582815928160281612816228163281642816528166281672816828169281702817128172281732817428175281762817728178281792818028181281822818328184281852818628187281882818928190281912819228193281942819528196281972819828199282002820128202282032820428205282062820728208282092821028211282122821328214282152821628217282182821928220282212822228223282242822528226282272822828229282302823128232282332823428235282362823728238282392824028241282422824328244282452824628247282482824928250282512825228253282542825528256282572825828259282602826128262282632826428265282662826728268282692827028271282722827328274282752827628277282782827928280282812828228283282842828528286282872828828289282902829128292282932829428295282962829728298282992830028301283022830328304283052830628307283082830928310283112831228313283142831528316283172831828319283202832128322283232832428325283262832728328283292833028331283322833328334283352833628337283382833928340283412834228343283442834528346283472834828349283502835128352283532835428355283562835728358283592836028361283622836328364283652836628367283682836928370283712837228373283742837528376283772837828379283802838128382283832838428385283862838728388283892839028391283922839328394283952839628397283982839928400284012840228403284042840528406284072840828409284102841128412284132841428415284162841728418284192842028421284222842328424284252842628427284282842928430284312843228433284342843528436284372843828439284402844128442284432844428445284462844728448284492845028451284522845328454284552845628457284582845928460284612846228463284642846528466284672846828469284702847128472284732847428475284762847728478284792848028481284822848328484284852848628487284882848928490284912849228493284942849528496284972849828499285002850128502285032850428505285062850728508285092851028511285122851328514285152851628517285182851928520285212852228523285242852528526285272852828529285302853128532285332853428535285362853728538285392854028541285422854328544285452854628547285482854928550285512855228553285542855528556285572855828559285602856128562285632856428565285662856728568285692857028571285722857328574285752857628577285782857928580285812858228583285842858528586285872858828589285902859128592285932859428595285962859728598285992860028601286022860328604286052860628607286082860928610286112861228613286142861528616286172861828619286202862128622286232862428625286262862728628286292863028631286322863328634286352863628637286382863928640286412864228643286442864528646286472864828649286502865128652286532865428655286562865728658286592866028661286622866328664286652866628667286682866928670286712867228673286742867528676286772867828679286802868128682286832868428685286862868728688286892869028691286922869328694286952869628697286982869928700287012870228703287042870528706287072870828709287102871128712287132871428715287162871728718287192872028721287222872328724287252872628727287282872928730287312873228733287342873528736287372873828739287402874128742287432874428745287462874728748287492875028751287522875328754287552875628757287582875928760287612876228763287642876528766287672876828769287702877128772287732877428775287762877728778287792878028781287822878328784287852878628787287882878928790287912879228793287942879528796287972879828799288002880128802288032880428805288062880728808288092881028811288122881328814288152881628817288182881928820288212882228823288242882528826288272882828829288302883128832288332883428835288362883728838288392884028841288422884328844288452884628847288482884928850288512885228853288542885528856288572885828859288602886128862288632886428865288662886728868288692887028871288722887328874288752887628877288782887928880288812888228883288842888528886288872888828889288902889128892288932889428895288962889728898288992890028901289022890328904289052890628907289082890928910289112891228913289142891528916289172891828919289202892128922289232892428925289262892728928289292893028931289322893328934289352893628937289382893928940289412894228943289442894528946289472894828949289502895128952289532895428955289562895728958289592896028961289622896328964289652896628967289682896928970289712897228973289742897528976289772897828979289802898128982289832898428985289862898728988289892899028991289922899328994289952899628997289982899929000290012900229003290042900529006290072900829009290102901129012290132901429015290162901729018290192902029021290222902329024290252902629027290282902929030290312903229033290342903529036290372903829039290402904129042290432904429045290462904729048290492905029051290522905329054290552905629057290582905929060290612906229063290642906529066290672906829069290702907129072290732907429075290762907729078290792908029081290822908329084290852908629087290882908929090290912909229093290942909529096290972909829099291002910129102291032910429105291062910729108291092911029111291122911329114291152911629117291182911929120291212912229123291242912529126291272912829129291302913129132291332913429135291362913729138291392914029141291422914329144291452914629147291482914929150291512915229153291542915529156291572915829159291602916129162291632916429165291662916729168291692917029171291722917329174291752917629177291782917929180291812918229183291842918529186291872918829189291902919129192291932919429195291962919729198291992920029201292022920329204292052920629207292082920929210292112921229213292142921529216292172921829219292202922129222292232922429225292262922729228292292923029231292322923329234292352923629237292382923929240292412924229243292442924529246292472924829249292502925129252292532925429255292562925729258292592926029261292622926329264292652926629267292682926929270292712927229273292742927529276292772927829279292802928129282292832928429285292862928729288292892929029291292922929329294292952929629297292982929929300293012930229303293042930529306293072930829309293102931129312293132931429315293162931729318293192932029321293222932329324293252932629327293282932929330293312933229333293342933529336293372933829339293402934129342293432934429345293462934729348293492935029351293522935329354293552935629357293582935929360293612936229363293642936529366293672936829369293702937129372293732937429375293762937729378293792938029381293822938329384293852938629387293882938929390293912939229393293942939529396293972939829399294002940129402294032940429405294062940729408294092941029411294122941329414294152941629417294182941929420294212942229423294242942529426294272942829429294302943129432294332943429435294362943729438294392944029441294422944329444294452944629447294482944929450294512945229453294542945529456294572945829459294602946129462294632946429465294662946729468294692947029471294722947329474294752947629477294782947929480294812948229483294842948529486294872948829489294902949129492294932949429495294962949729498294992950029501295022950329504295052950629507295082950929510295112951229513295142951529516295172951829519295202952129522295232952429525295262952729528295292953029531295322953329534295352953629537295382953929540295412954229543295442954529546295472954829549295502955129552295532955429555295562955729558295592956029561295622956329564295652956629567295682956929570295712957229573295742957529576295772957829579295802958129582295832958429585295862958729588295892959029591295922959329594295952959629597295982959929600296012960229603296042960529606296072960829609296102961129612296132961429615296162961729618296192962029621296222962329624296252962629627296282962929630296312963229633296342963529636296372963829639296402964129642296432964429645296462964729648296492965029651296522965329654296552965629657296582965929660296612966229663296642966529666296672966829669296702967129672296732967429675296762967729678296792968029681296822968329684296852968629687296882968929690296912969229693296942969529696296972969829699297002970129702297032970429705297062970729708297092971029711297122971329714297152971629717297182971929720297212972229723297242972529726297272972829729297302973129732297332973429735297362973729738297392974029741297422974329744297452974629747297482974929750297512975229753297542975529756297572975829759297602976129762297632976429765297662976729768297692977029771297722977329774297752977629777297782977929780297812978229783297842978529786297872978829789297902979129792297932979429795297962979729798297992980029801298022980329804298052980629807298082980929810298112981229813298142981529816298172981829819298202982129822298232982429825298262982729828298292983029831298322983329834298352983629837298382983929840298412984229843298442984529846298472984829849298502985129852298532985429855298562985729858298592986029861298622986329864298652986629867298682986929870298712987229873298742987529876298772987829879298802988129882298832988429885298862988729888298892989029891298922989329894298952989629897298982989929900299012990229903299042990529906299072990829909299102991129912299132991429915299162991729918299192992029921299222992329924299252992629927299282992929930299312993229933299342993529936299372993829939299402994129942299432994429945299462994729948299492995029951299522995329954299552995629957299582995929960299612996229963299642996529966299672996829969299702997129972299732997429975299762997729978299792998029981299822998329984299852998629987299882998929990299912999229993299942999529996299972999829999300003000130002300033000430005300063000730008300093001030011300123001330014300153001630017300183001930020300213002230023300243002530026300273002830029300303003130032300333003430035300363003730038300393004030041300423004330044300453004630047300483004930050300513005230053300543005530056300573005830059300603006130062300633006430065300663006730068300693007030071300723007330074300753007630077300783007930080300813008230083300843008530086300873008830089300903009130092300933009430095300963009730098300993010030101301023010330104301053010630107301083010930110301113011230113301143011530116301173011830119301203012130122301233012430125301263012730128301293013030131301323013330134301353013630137301383013930140301413014230143301443014530146301473014830149301503015130152301533015430155301563015730158301593016030161301623016330164301653016630167301683016930170301713017230173301743017530176301773017830179301803018130182301833018430185301863018730188301893019030191301923019330194301953019630197301983019930200302013020230203302043020530206302073020830209302103021130212302133021430215302163021730218302193022030221302223022330224302253022630227302283022930230302313023230233302343023530236302373023830239302403024130242302433024430245302463024730248302493025030251302523025330254302553025630257302583025930260302613026230263302643026530266302673026830269302703027130272302733027430275302763027730278302793028030281302823028330284302853028630287302883028930290302913029230293302943029530296302973029830299303003030130302303033030430305303063030730308303093031030311303123031330314303153031630317303183031930320303213032230323303243032530326303273032830329303303033130332303333033430335303363033730338303393034030341303423034330344303453034630347303483034930350303513035230353303543035530356303573035830359303603036130362303633036430365303663036730368303693037030371303723037330374303753037630377303783037930380303813038230383303843038530386303873038830389303903039130392303933039430395303963039730398303993040030401304023040330404304053040630407304083040930410304113041230413304143041530416304173041830419304203042130422304233042430425304263042730428304293043030431304323043330434304353043630437304383043930440304413044230443304443044530446304473044830449304503045130452304533045430455304563045730458304593046030461304623046330464304653046630467304683046930470304713047230473304743047530476304773047830479304803048130482304833048430485304863048730488304893049030491304923049330494304953049630497304983049930500305013050230503305043050530506305073050830509305103051130512305133051430515305163051730518305193052030521305223052330524305253052630527305283052930530305313053230533305343053530536305373053830539305403054130542305433054430545305463054730548305493055030551305523055330554305553055630557305583055930560305613056230563305643056530566305673056830569305703057130572305733057430575305763057730578305793058030581305823058330584305853058630587305883058930590305913059230593305943059530596305973059830599306003060130602306033060430605306063060730608306093061030611306123061330614306153061630617306183061930620306213062230623306243062530626306273062830629306303063130632306333063430635306363063730638306393064030641306423064330644306453064630647306483064930650306513065230653306543065530656306573065830659306603066130662306633066430665306663066730668306693067030671306723067330674306753067630677306783067930680306813068230683306843068530686306873068830689306903069130692306933069430695306963069730698306993070030701307023070330704307053070630707307083070930710307113071230713307143071530716307173071830719307203072130722307233072430725307263072730728307293073030731307323073330734307353073630737307383073930740307413074230743307443074530746307473074830749307503075130752307533075430755307563075730758307593076030761307623076330764307653076630767307683076930770307713077230773307743077530776307773077830779307803078130782307833078430785307863078730788307893079030791307923079330794307953079630797307983079930800308013080230803308043080530806308073080830809308103081130812308133081430815308163081730818308193082030821308223082330824308253082630827308283082930830308313083230833308343083530836308373083830839308403084130842308433084430845308463084730848308493085030851308523085330854308553085630857308583085930860308613086230863308643086530866308673086830869308703087130872308733087430875308763087730878308793088030881308823088330884308853088630887308883088930890308913089230893308943089530896308973089830899309003090130902309033090430905309063090730908309093091030911309123091330914309153091630917309183091930920309213092230923309243092530926309273092830929309303093130932309333093430935309363093730938309393094030941309423094330944309453094630947309483094930950309513095230953309543095530956309573095830959309603096130962309633096430965309663096730968309693097030971309723097330974309753097630977309783097930980309813098230983309843098530986309873098830989309903099130992309933099430995309963099730998309993100031001310023100331004310053100631007310083100931010310113101231013310143101531016310173101831019310203102131022310233102431025310263102731028310293103031031310323103331034310353103631037310383103931040310413104231043310443104531046310473104831049310503105131052310533105431055310563105731058310593106031061310623106331064310653106631067310683106931070310713107231073310743107531076310773107831079310803108131082310833108431085310863108731088310893109031091310923109331094310953109631097310983109931100311013110231103311043110531106311073110831109311103111131112311133111431115311163111731118311193112031121311223112331124311253112631127311283112931130311313113231133311343113531136311373113831139311403114131142311433114431145311463114731148311493115031151311523115331154311553115631157311583115931160311613116231163311643116531166311673116831169311703117131172311733117431175311763117731178311793118031181311823118331184311853118631187311883118931190311913119231193311943119531196311973119831199312003120131202312033120431205312063120731208312093121031211312123121331214312153121631217312183121931220312213122231223312243122531226312273122831229312303123131232312333123431235312363123731238312393124031241312423124331244312453124631247312483124931250312513125231253312543125531256312573125831259312603126131262312633126431265312663126731268312693127031271312723127331274312753127631277312783127931280312813128231283312843128531286312873128831289312903129131292312933129431295312963129731298312993130031301313023130331304313053130631307313083130931310313113131231313313143131531316313173131831319313203132131322313233132431325313263132731328313293133031331313323133331334313353133631337313383133931340313413134231343313443134531346313473134831349313503135131352313533135431355313563135731358313593136031361313623136331364313653136631367313683136931370313713137231373313743137531376313773137831379313803138131382313833138431385313863138731388313893139031391313923139331394313953139631397313983139931400314013140231403314043140531406314073140831409314103141131412314133141431415314163141731418314193142031421314223142331424314253142631427314283142931430314313143231433314343143531436314373143831439314403144131442314433144431445314463144731448314493145031451314523145331454314553145631457314583145931460314613146231463314643146531466314673146831469314703147131472314733147431475314763147731478314793148031481314823148331484314853148631487314883148931490314913149231493314943149531496314973149831499315003150131502315033150431505315063150731508315093151031511315123151331514315153151631517315183151931520315213152231523315243152531526315273152831529315303153131532315333153431535315363153731538315393154031541315423154331544315453154631547315483154931550315513155231553315543155531556315573155831559315603156131562315633156431565315663156731568315693157031571315723157331574315753157631577315783157931580315813158231583315843158531586315873158831589315903159131592315933159431595315963159731598315993160031601316023160331604316053160631607316083160931610316113161231613316143161531616316173161831619316203162131622316233162431625316263162731628316293163031631316323163331634316353163631637316383163931640316413164231643316443164531646316473164831649316503165131652316533165431655316563165731658316593166031661316623166331664316653166631667316683166931670316713167231673316743167531676316773167831679316803168131682316833168431685316863168731688316893169031691316923169331694316953169631697316983169931700317013170231703317043170531706317073170831709317103171131712317133171431715317163171731718317193172031721317223172331724317253172631727317283172931730317313173231733317343173531736317373173831739317403174131742317433174431745317463174731748317493175031751317523175331754317553175631757317583175931760317613176231763317643176531766317673176831769317703177131772317733177431775317763177731778317793178031781317823178331784317853178631787317883178931790317913179231793317943179531796317973179831799318003180131802318033180431805318063180731808318093181031811318123181331814318153181631817318183181931820318213182231823318243182531826318273182831829318303183131832318333183431835318363183731838318393184031841318423184331844318453184631847318483184931850318513185231853318543185531856318573185831859318603186131862318633186431865318663186731868318693187031871318723187331874318753187631877318783187931880318813188231883318843188531886318873188831889318903189131892318933189431895318963189731898318993190031901319023190331904319053190631907319083190931910319113191231913319143191531916319173191831919319203192131922319233192431925319263192731928319293193031931319323193331934319353193631937319383193931940319413194231943319443194531946319473194831949319503195131952319533195431955319563195731958319593196031961319623196331964319653196631967319683196931970319713197231973319743197531976319773197831979319803198131982319833198431985319863198731988319893199031991319923199331994319953199631997319983199932000320013200232003320043200532006320073200832009320103201132012320133201432015320163201732018320193202032021320223202332024320253202632027320283202932030320313203232033320343203532036320373203832039320403204132042320433204432045320463204732048320493205032051320523205332054320553205632057320583205932060320613206232063320643206532066320673206832069320703207132072320733207432075320763207732078320793208032081320823208332084320853208632087320883208932090320913209232093320943209532096320973209832099321003210132102321033210432105321063210732108321093211032111321123211332114321153211632117321183211932120321213212232123321243212532126321273212832129321303213132132321333213432135321363213732138321393214032141321423214332144321453214632147321483214932150321513215232153321543215532156321573215832159321603216132162321633216432165321663216732168321693217032171321723217332174321753217632177321783217932180321813218232183321843218532186321873218832189321903219132192321933219432195321963219732198321993220032201322023220332204322053220632207322083220932210322113221232213322143221532216322173221832219322203222132222322233222432225322263222732228322293223032231322323223332234322353223632237322383223932240322413224232243322443224532246322473224832249322503225132252322533225432255322563225732258322593226032261322623226332264322653226632267322683226932270322713227232273322743227532276322773227832279322803228132282322833228432285322863228732288322893229032291322923229332294322953229632297322983229932300323013230232303323043230532306323073230832309323103231132312323133231432315323163231732318323193232032321323223232332324323253232632327323283232932330323313233232333323343233532336323373233832339323403234132342323433234432345323463234732348323493235032351323523235332354323553235632357323583235932360323613236232363323643236532366323673236832369323703237132372323733237432375323763237732378323793238032381323823238332384323853238632387323883238932390323913239232393323943239532396323973239832399324003240132402324033240432405324063240732408324093241032411324123241332414324153241632417324183241932420324213242232423324243242532426324273242832429324303243132432324333243432435324363243732438324393244032441324423244332444324453244632447324483244932450324513245232453324543245532456324573245832459324603246132462324633246432465324663246732468324693247032471324723247332474324753247632477324783247932480324813248232483324843248532486324873248832489324903249132492324933249432495324963249732498324993250032501325023250332504325053250632507325083250932510325113251232513325143251532516325173251832519325203252132522325233252432525325263252732528325293253032531325323253332534325353253632537325383253932540325413254232543325443254532546325473254832549325503255132552325533255432555325563255732558325593256032561325623256332564325653256632567325683256932570325713257232573325743257532576325773257832579325803258132582325833258432585325863258732588325893259032591325923259332594325953259632597325983259932600326013260232603326043260532606326073260832609326103261132612326133261432615326163261732618326193262032621326223262332624326253262632627326283262932630326313263232633326343263532636326373263832639326403264132642326433264432645326463264732648326493265032651326523265332654326553265632657326583265932660326613266232663326643266532666326673266832669326703267132672326733267432675326763267732678326793268032681326823268332684326853268632687326883268932690326913269232693326943269532696326973269832699327003270132702327033270432705327063270732708327093271032711327123271332714327153271632717327183271932720327213272232723327243272532726327273272832729327303273132732327333273432735327363273732738327393274032741327423274332744327453274632747327483274932750327513275232753327543275532756327573275832759327603276132762327633276432765327663276732768327693277032771327723277332774327753277632777327783277932780327813278232783327843278532786327873278832789327903279132792327933279432795327963279732798327993280032801328023280332804328053280632807328083280932810328113281232813328143281532816
  1. /*
  2. * @Author: Jeffrey Wang
  3. * @Desc: 整理强大的 SheetJS 功能,依赖 XLSX.js 和 FileSaver
  4. * @Version: v1.6
  5. * @Date: 2018-03-24 09:54:17
  6. * @Last Modified by: Jeffrey Wang
  7. * @Last Modified ~: 2019-10-03 23:12:00
  8. */
  9. if (typeof layui === 'undefined' && typeof jQuery === 'undefined') {
  10. console.error('非layui调用请先加载jQuery')
  11. }
  12. if (typeof jQuery !== 'undefined') {
  13. $ = jQuery
  14. }
  15. LAY_EXCEL = {
  16. /**
  17. * 兼容老版本的导出函数
  18. * @param {[type]} data [description]
  19. * @param {[type]} filename [description]
  20. * @param {[type]} type [description]
  21. * @return {[type]} [description]
  22. */
  23. downloadExl: function(data, filename, type) {
  24. type = type ? type : 'xlsx';
  25. this.exportExcel({sheet1: data}, filename+'.'+type, type, null);
  26. },
  27. /**
  28. * 导出Excel并弹出下载框,具体使用方法和范围请参考文档
  29. * @param data object
  30. * @param {[type]} filename [description]
  31. * @param {[type]} type [description]
  32. * @param {[type]} opt [description]
  33. * @return {[type]} [description]
  34. */
  35. exportExcel : function(data, filename, type, opt) {
  36. type = type ? type : 'xlsx';
  37. filename = filename ? filename : '导出数据.'+type;
  38. // 创建一个 XLSX 对象
  39. var wb = XLSX.utils.book_new();
  40. // 1. 定义excel对的基本属性
  41. var Props = {
  42. Title: filename,
  43. Subject: 'Export From web browser',
  44. Author: "excel.wj2015.com",
  45. Manager: '',
  46. Company: '',
  47. Category: '',
  48. Keywords: '',
  49. Comments: '',
  50. LastAuthor: '',
  51. CreatedData: new Date(),
  52. };
  53. opt && opt.Props && (Props = $.extend(Props, opt.Props));
  54. // 默认进行压缩
  55. wb.compression = opt ? opt.compression : true
  56. if(wb.compression !== false) {
  57. wb.compression = true
  58. }
  59. wb.Props = Props;
  60. // 特殊属性实现,比如合并单元格
  61. var wbExtend = {
  62. '!merges': null
  63. ,'!margins': null
  64. ,'!cols': null
  65. ,'!rows': null
  66. ,'!protect': null
  67. ,'!autofilter': null
  68. };
  69. opt && opt.extend && (wbExtend = $.extend(wbExtend, opt.extend));
  70. // 清理空配置
  71. for (var key in wbExtend) {
  72. if (!wbExtend.hasOwnProperty(key)) {
  73. continue;
  74. }
  75. if (!wbExtend[key]) {
  76. delete wbExtend[key];
  77. }
  78. }
  79. // 判断 data 如果是 sheet 级别数据,自动加 sheet1
  80. if ($.isArray(data)) {
  81. data = {sheet1: data};
  82. }
  83. for(var sheet_name in data) {
  84. if (!data.hasOwnProperty(sheet_name)) {
  85. continue;
  86. }
  87. var content = data[sheet_name];
  88. // 2. 设置sheet名称
  89. wb.SheetNames.push(sheet_name);
  90. // 3. 分配工作表对象到 sheet
  91. var is_aoa = false;
  92. if (content.length && content[0] && $.isArray(content[0])) {
  93. is_aoa = true;
  94. }
  95. if (is_aoa) {
  96. ws = XLSX.utils.aoa_to_sheet(content);
  97. } else {
  98. var option = {};
  99. if (content.length) {
  100. option.headers = content.unshift();
  101. option.skipHeader = true;
  102. // 分离并重组样式
  103. var splitRes = this.splitContent(content);
  104. }
  105. var ws = XLSX.utils.json_to_sheet(content, option);
  106. // 特殊属性,支持单独设置某个sheet的属性
  107. if (wbExtend[sheet_name]) {
  108. $.extend(ws, wbExtend[sheet_name]);
  109. } else {
  110. $.extend(ws, wbExtend);
  111. }
  112. // 合并样式
  113. if (typeof splitRes !== 'undefined') {
  114. this.mergeCellOpt(ws, splitRes.style);
  115. }
  116. }
  117. wb.Sheets[sheet_name] = ws;
  118. };
  119. // 4. 输出工作表
  120. var wbout = XLSX.write(wb, {bookType: type, type: 'binary', cellStyles: true, compression: wb.compression});
  121. // 5. 跨浏览器支持,采用 FileSaver 三方库
  122. saveAs(new Blob([this.s2ab(wbout)], {type: "application/octet-stream"}), filename);
  123. },
  124. /**
  125. * 分离内容和样式
  126. * @param {[type]} content [description]
  127. * @return {[type]} [description]
  128. */
  129. splitContent: function(content) {
  130. var styleContent = {};
  131. // 扫描每个单元格,如果是对象则等表格转换完毕后分离出来重新赋值
  132. for (var line = 0; line < content.length; line++) {
  133. var lineData = content[line];
  134. var rowIndex = 0;
  135. for (var row in lineData) {
  136. if (!lineData.hasOwnProperty(row)) {
  137. continue;
  138. }
  139. var rowData = lineData[row];
  140. if (typeof rowData === 'object') {
  141. // typeof null == object
  142. if (rowData !== null) {
  143. styleContent[this.numToTitle(rowIndex+1)+(parseInt(line)+1)] = rowData;
  144. } else {
  145. lineData[row] = '';
  146. }
  147. } else {
  148. // JeffreyWang 2019-03-10针对 0 的hack处理
  149. if (rowData === 0) {
  150. rowData = {
  151. v: '0',
  152. s: {
  153. alignment: {
  154. horizontal: 'right'
  155. }
  156. }
  157. }
  158. }
  159. styleContent[this.numToTitle(rowIndex+1)+(parseInt(line)+1)] = rowData;
  160. }
  161. rowIndex++;
  162. }
  163. }
  164. return {
  165. content: content,
  166. style: styleContent
  167. };
  168. },
  169. /**
  170. * 合并内容和样式
  171. * @param {[type]} ws [description]
  172. * @param {[type]} style [description]
  173. * @return {[type]} [description]
  174. */
  175. mergeCellOpt: function(ws, style) {
  176. for (var row in style) {
  177. if (!style.hasOwnProperty(row)) {
  178. continue;
  179. }
  180. var rowOpt = style[row];
  181. if (ws[row]) {
  182. // 其他属性做一个初始化
  183. var otherOpt = ['t', 'w', 'f', 'r', 'h', 'c', 'z', 'l', 's'];
  184. for (var i = 0; i < otherOpt.length; i++) {
  185. ws[row][otherOpt[i]] = ws[row][otherOpt[i]];
  186. }
  187. $.extend(ws[row], rowOpt);
  188. }
  189. }
  190. },
  191. /**
  192. * 将table转换为JSON数据
  193. * @param dom
  194. */
  195. tableToJson: function(dom) {
  196. dom = $(dom)
  197. var head = []
  198. dom.find('thead > tr').each(function () {
  199. var line = []
  200. $(this).find('td,th').each(function () {
  201. line.push($(this).text())
  202. })
  203. head.push(line)
  204. })
  205. var body = [];
  206. dom.find('tbody > tr').each(function () {
  207. var line = []
  208. $(this).find('td').each(function () {
  209. line.push($(this).text())
  210. })
  211. body.push(line)
  212. })
  213. return {
  214. head: head,
  215. body: body
  216. }
  217. },
  218. // 测试代码:
  219. // for(i=1;i<100;i++){var change = layui.excel.numToTitle(i);console.log(i, change, layui.excel.titleToNum(change));}
  220. // numsToTitle备忘录提效
  221. numsTitleCache: {},
  222. // titleToTitle 备忘录提效
  223. titleNumsCache: {},
  224. /**
  225. * 将数字(从一开始)转换为 A、B、C...AA、AB
  226. * @param {[int]} num [description]
  227. * @return {[type]} [description]
  228. */
  229. numToTitle: function(num) {
  230. if (this.numsTitleCache[num]) {
  231. return this.numsTitleCache[num];
  232. }
  233. var ans = '';
  234. if (num > 26) {
  235. // 要注意小心 26 的倍数导致的无限递归问题
  236. var dec = num % 26;
  237. ans = this.numToTitle((num - dec)/26) + this.numToTitle(dec?dec:26);
  238. this.numsTitleCache[num] = ans;
  239. this.titleNumsCache[ans] = num;
  240. return ans;
  241. } else {
  242. // A 的 ascii 为 0,顺位相加
  243. ans = String.fromCharCode(64 + num);
  244. this.numsTitleCache[num] = ans;
  245. this.titleNumsCache[ans] = num;
  246. return ans;
  247. }
  248. },
  249. /**
  250. * 将A、B、AA、ABC转换为 1、2、3形式的数字
  251. * @param {[type]} title [description]
  252. * @return {number} [description]
  253. */
  254. titleToNum: function(title) {
  255. if (this.titleNumsCache[title]) {
  256. return this.titleNumsCache[title];
  257. }
  258. var len = title.length;
  259. var total = 0;
  260. for (var index in title) {
  261. if (!title.hasOwnProperty(index)) {
  262. continue;
  263. }
  264. var char = title[index];
  265. var code = char.charCodeAt() - 64;
  266. total += code * Math.pow(26, len - index - 1);
  267. }
  268. this.numsTitleCache[total] = title;
  269. this.titleNumsCache[title] = total;
  270. return total;
  271. },
  272. /**
  273. * 获取数据范围内有效范围
  274. * @param data array sheet级别的数据
  275. * @param range 范围字符串,如 A1:C12,默认从左上角到右下角
  276. */
  277. getDefaultRange: function(data, range) {
  278. // 以 rowIndex 为键,field 为值
  279. var fieldKeys = Object.keys(data[0]);
  280. var maxCol = fieldKeys.length - 1;
  281. var maxRow = data.length -1;
  282. // 默认 A1 ~ 右下角
  283. var startPos = {c: 0, r: 0};
  284. var endPos = {c: maxCol, r: maxRow};
  285. if (range && typeof range === 'string') {
  286. var rangeArr = range.split(':');
  287. if (rangeArr[0].length) {
  288. startPos = this.splitPosition(rangeArr[0]);
  289. }
  290. if (typeof rangeArr[1] !== 'undefined' && rangeArr[1] !== '') {
  291. endPos = this.splitPosition(rangeArr[1]);
  292. }
  293. } else {
  294. // pass
  295. }
  296. // position范围限制 - 考虑到特殊情况取消此限制
  297. // startPos.c = startPos.c < maxCol ? startPos.c : maxCol;
  298. // endPos.c = endPos.c < maxCol ? endPos.c : maxCol;
  299. // startPos.r = startPos.r < maxRow ? startPos.r : maxRow;
  300. // endPos.r = endPos.r < maxRow ? endPos.r : maxRow;
  301. if (startPos.c > endPos.c) {
  302. console.error('开始列不得大于结束列');
  303. }
  304. if (startPos.r > endPos.r) {
  305. console.error('开始行不得大于结束行');
  306. }
  307. return {
  308. startPos: startPos,
  309. endPos: endPos,
  310. fieldKeys: fieldKeys
  311. }
  312. },
  313. /**
  314. * 根据 startPos endPos 遍历设置单元格属性,支持 filter 回调处理
  315. * @param data array sheet级别数据
  316. * @param startPos object {c: 开始列索引, r: 开始行索引}
  317. * @param endPos object {c: 结束列索引, r: 结束行索引}
  318. * @param fieldKeys ['第一列属性Key', '第二列属性Key']
  319. * @param config object {s: {样式}, v: '值'}
  320. * @param filter callable 回调函数,入参 cell(原cell),newCell(新cell),row(当前行),config(配置), currentRow(当前行索引), currentCol(当前列索引-数字),currentColKey(当前列索引-对象)
  321. */
  322. setCellStyle: function (data, startPos, endPos, fieldKeys, config, filter) {
  323. // 遍历范围内的数据,进行样式设置,按从上到下从左到右按行遍历
  324. for (var currentRow = startPos.r; currentRow <= endPos.r; currentRow++) {
  325. for (var currentCol = startPos.c; currentCol <= endPos.c; currentCol++) {
  326. // 如果有回调则执行回调判断,否则全部更新,如果遇到超出数据范围的,自动置空
  327. var row = data[currentRow];
  328. if (!row) {
  329. row = {};
  330. for (var key = 0; key < fieldKeys.length; key++) {
  331. row[fieldKeys[key]] = '';
  332. }
  333. data[currentRow] = row;
  334. }
  335. var cell = row[fieldKeys[currentCol]];
  336. var newCell = null;
  337. if (cell === null || cell === undefined) {
  338. cell = '';
  339. }
  340. // 手工合并(相同的则以当前函数config为准)
  341. if (typeof cell === 'object') {
  342. newCell = $.extend(true, {}, cell, config);
  343. } else {
  344. newCell = $.extend(true, {}, {v: cell}, config);
  345. }
  346. if (
  347. typeof filter === 'function'
  348. ) {
  349. newCell = filter(cell, newCell, row, config, currentRow, currentCol, fieldKeys[currentCol]);
  350. } else {
  351. }
  352. // 回写
  353. data[currentRow][fieldKeys[currentCol]] = newCell;
  354. }
  355. }
  356. },
  357. /**
  358. * 设置范围内环绕的边框
  359. * @param data [sheet级别的数据]
  360. * @param range [范围字符串,如 A1:C12,默认从左上角到右下角]
  361. * @param config [border 上下左右属性配置信息(对角线的三个属性被下放到left/right/top/bottom下),如:{top: {xxx}, bottom: {}, left: {}, right: {}}]
  362. */
  363. setRoundBorder: function(data, range, config) {
  364. if (typeof data !== 'object' || !data.length || !data[0] || !Object.keys(data[0]).length) {
  365. return [];
  366. }
  367. var rangeObj = this.getDefaultRange(data, range);
  368. var startPos = rangeObj.startPos;
  369. var endPos = rangeObj.endPos;
  370. var fieldKeys = rangeObj.fieldKeys;
  371. // 顶部 border 属性取 config.top
  372. this.setCellStyle(data, startPos, {
  373. c: endPos.c,
  374. r: startPos.r
  375. }, fieldKeys, {
  376. s: {
  377. border: {
  378. top: config.top,
  379. diagonal: config.top.diagonal,
  380. diagonalUp: config.top.diagonalUp,
  381. diagonalDown: config.top.diagonalDown
  382. }
  383. }
  384. })
  385. // 右侧 border 属性取 config.right
  386. this.setCellStyle(data, {
  387. c: endPos.c,
  388. r: startPos.r
  389. }, endPos, fieldKeys, {
  390. s: {
  391. border: {
  392. right: config.right,
  393. diagonal: config.right.diagonal,
  394. diagonalUp: config.right.diagonalUp,
  395. diagonalDown: config.right.diagonalDown
  396. }
  397. }
  398. })
  399. // 底部 border 属性取 config.bottom
  400. this.setCellStyle(data, {
  401. c: startPos.c,
  402. r: endPos.r
  403. }, endPos, fieldKeys, {
  404. s: {
  405. border: {
  406. bottom: config.bottom,
  407. diagonal: config.bottom.diagonal,
  408. diagonalUp: config.bottom.diagonalUp,
  409. diagonalDown: config.bottom.diagonalDown
  410. }
  411. }
  412. })
  413. // 左侧 border 属性取 config.left
  414. this.setCellStyle(data, startPos, {
  415. c: startPos.c,
  416. r: endPos.r
  417. }, fieldKeys, {
  418. s: {
  419. border: {
  420. left: config.left,
  421. diagonal: config.left.diagonal,
  422. diagonalUp: config.left.diagonalUp,
  423. diagonalDown: config.left.diagonalDown
  424. }
  425. }
  426. })
  427. },
  428. /**
  429. * 批量设置单元格属性
  430. * @param {array} data [sheet级别的数据]
  431. * @param {string} range [范围字符串,比如 A1:C12,开始位置默认 A1,结束位置默认整个表格右下角]
  432. * @param {object} config [批量设置的单元格属性]
  433. * @param {function} filter [回调函数,传递函数生效,返回值作为新的值(可用于过滤、规则替换样式等骚操作)]
  434. * @return {array} [重新渲染后的 sheet 数据]
  435. */
  436. setExportCellStyle: function(data, range, config, filter) {
  437. if (typeof data !== 'object' || !data.length || !data[0] || !Object.keys(data[0]).length) {
  438. return [];
  439. }
  440. var rangeObj = this.getDefaultRange(data, range);
  441. var startPos = rangeObj.startPos;
  442. var endPos = rangeObj.endPos;
  443. var fieldKeys = rangeObj.fieldKeys;
  444. this.setCellStyle(data, startPos, endPos, fieldKeys, config, filter);
  445. return data;
  446. },
  447. /**
  448. * 合并单元格快速生成配置的函数 传入 [ ['开始坐标 A1', '结束坐标 D2'], ['开始坐标 B2', '结束坐标 E3'] ]
  449. * @param {[type]} origin [description]
  450. * @return {[type]} [description]
  451. */
  452. makeMergeConfig: function(origin) {
  453. var merge = [];
  454. for (var index = 0; index < origin.length; index++) {
  455. merge.push({
  456. s: this.splitPosition(origin[index][0]),
  457. e: this.splitPosition(origin[index][1])
  458. });
  459. }
  460. return merge;
  461. },
  462. /**
  463. * 自动生成列宽配置
  464. * @param {$ObjMap} data [A、B、C的宽度映射]
  465. * @param {number} defaultNum [description]
  466. * @return {$ObjMap} [description]
  467. */
  468. makeColConfig: function(data, defaultNum) {
  469. defaultNum = defaultNum > 0 ? defaultNum : 50;
  470. // 将列的 ABC 转换为 index
  471. var change = [];
  472. var startIndex = 0;
  473. for (var index in data) {
  474. if (!data.hasOwnProperty(index)) {
  475. continue;
  476. }
  477. var item = data[index];
  478. if (index.match && index.match(/[A-Z]*/)) {
  479. var currentIndex = this.titleToNum(index) - 1;
  480. // 填充未配置的单元格
  481. while (startIndex < currentIndex) {
  482. change.push({wpx: defaultNum});
  483. startIndex++;
  484. }
  485. startIndex = currentIndex+1;
  486. change.push({wpx: item > 0 ? item : defaultNum});
  487. }
  488. };
  489. return change;
  490. },
  491. /**
  492. * 自动生成列高配置
  493. * @param {[type]} data [description]
  494. * @param {[type]} defaultNum [description]
  495. * @return {[type]} [description]
  496. */
  497. makeRowConfig: function(data, defaultNum) {
  498. defaultNum = defaultNum > 0 ? defaultNum : 10;
  499. // 将列的 ABC 转换为 index
  500. var change = [];
  501. var startIndex = 0;
  502. for (var index in data) {
  503. if (!data.hasOwnProperty(index)) {
  504. continue;
  505. }
  506. var item = data[index];
  507. if (index.match && index.match(/[0-9]*/)) {
  508. var currentIndex = parseInt(index) - 1;
  509. // 填充未配置的行
  510. while (startIndex < currentIndex) {
  511. change.push({hpx: defaultNum});
  512. startIndex++;
  513. }
  514. startIndex = currentIndex+1;
  515. change.push({hpx: item > 0 ? item : defaultNum});
  516. }
  517. };
  518. return change;
  519. },
  520. /**
  521. * 将A1分离成 {c: 0, r: 0} 格式的数据
  522. * @param {string} pos [description]
  523. * @return {{r: number, c: number}} [description]
  524. */
  525. splitPosition: function(pos) {
  526. var res = pos.match('^([A-Z]+)([0-9]+)$');
  527. if (!res) {
  528. return {c: 0, r: 0};
  529. }
  530. // 转换结果相比需要的结果需要减一转换
  531. return {
  532. c: this.titleToNum(res[1]) - 1,
  533. r: parseInt(res[2]) - 1
  534. }
  535. },
  536. /**
  537. * 将二进制数据转为8位字节
  538. * @param {[type]} s [description]
  539. * @return {[type]} [description]
  540. */
  541. s2ab: function(s) {
  542. var buf = new ArrayBuffer(s.length);
  543. var view = new Uint8Array(buf);
  544. for (var i = 0; i < s.length; i++) {
  545. view[i] = s.charCodeAt(i) & 0xFF;
  546. }
  547. return buf;
  548. },
  549. /**
  550. * 将导出的数据格式,转换为可以aoa导出的格式
  551. * @return {[type]} [description]
  552. */
  553. filterDataToAoaData: function(filterData){
  554. var aoaData = [];
  555. $.each(filterData, function(index, item) {
  556. var itemData = [];
  557. for (var i in item) {
  558. if (!item.hasOwnProperty(i)) {
  559. continue;
  560. }
  561. itemData.push(item[i]);
  562. }
  563. aoaData.push(itemData);
  564. });
  565. return aoaData;
  566. },
  567. /**
  568. * 梳理导出的数据,包括字段排序和多余数据过滤,具体功能请参见文档
  569. * @param {[type]} data [需要梳理的数据]
  570. * @param {[type]} fields [支持数组和对象,用于映射关系和字段排序]
  571. * @return {[type]} [description]
  572. */
  573. filterExportData: function(data, fields) {
  574. // PS:之所以不直接引用 data 节省内存,是因为担心如果 fields 可能存在如下情况: { "id": 'test_id', 'test_id': 'new_id' },会导致处理异常
  575. var exportData = [];
  576. var true_fields = [];
  577. // filed 支持两种模式,数组则单纯排序,对象则转换映射关系,为了统一处理,将数组转换为符合要求的映射关系对象
  578. if (Array.isArray(fields)) {
  579. for (var i = 0; i< fields.length; i++) {
  580. true_fields[fields[i]] = fields[i];
  581. }
  582. } else {
  583. true_fields = fields;
  584. }
  585. for (var i = 0; i < data.length; i++) {
  586. var item = data[i];
  587. exportData[i] = {};
  588. for (var key in true_fields) {
  589. if (!true_fields.hasOwnProperty(key)) {
  590. continue;
  591. }
  592. var new_field_name = key;
  593. var old_field_name = true_fields[key];
  594. // 如果传入的是回调,则回调的值则为新值
  595. if (typeof old_field_name === 'function' && old_field_name.apply) {
  596. exportData[i][new_field_name] = old_field_name.apply(window, [item[new_field_name], item, data, i, new_field_name]);
  597. } else {
  598. if (typeof item[old_field_name] !== 'undefined') {
  599. exportData[i][new_field_name] = item[old_field_name];
  600. } else {
  601. exportData[i][new_field_name] = '';
  602. }
  603. }
  604. }
  605. }
  606. return exportData;
  607. },
  608. /**
  609. * 梳理导入的数据,参数意义可参考 filterExportData
  610. * @param {[type]} data [description]
  611. * @param {[type]} fields [description]
  612. * @return {[type]} [description]
  613. */
  614. filterImportData: function(data, fields) {
  615. var that = this;
  616. $.each(data, function(fileindex, xlsx) {
  617. $.each(xlsx, function(sheetname, content) {
  618. xlsx[sheetname] = that.filterExportData(content, fields);
  619. });
  620. });
  621. return data;
  622. },
  623. /**
  624. * 读取Excel,支持多文件多表格读取
  625. * @param {[type]} files [description]
  626. * @param {[type]} opt [description]
  627. * @param {Function} callback [description]
  628. * @return {[type]} [description]
  629. */
  630. importExcel: function(files, opt, callback) {
  631. var option = {
  632. header: 'A',
  633. range: null,
  634. fields: null
  635. };
  636. $.extend(option, opt);
  637. var that = this;
  638. if (files.length < 1) {
  639. throw {code: 999, 'message': '传入文件为空'};
  640. }
  641. // 按照二进制读取
  642. var data = {};
  643. var book = {};
  644. $.each(files, function(index, item) {
  645. var reader = new FileReader();
  646. if (!reader) {
  647. throw {code: 999, message: '不支持FileReader,请更换更新的浏览器'};
  648. }
  649. // 读取excel表格对象
  650. reader.onload = function(ev) {
  651. var wb = XLSX.read(ev.target.result, {
  652. type: 'binary'
  653. });
  654. var excelData = {};
  655. $.each(wb.Sheets, function(sheet, sheetObj) {
  656. // 全为空的去掉
  657. if (wb.Sheets.hasOwnProperty(sheet)) {
  658. var opt = {
  659. header: option.header,
  660. defval: ''
  661. };
  662. if (option.range) {
  663. opt.range = option.range;
  664. }
  665. excelData[sheet] = XLSX.utils.sheet_to_json(sheetObj, opt);
  666. // 支持梳理数据
  667. if (option.fields) {
  668. excelData[sheet] = that.filterExportData(excelData[sheet], option.fields);
  669. }
  670. }
  671. });
  672. data[index] = excelData;
  673. book[index] = wb;
  674. // 全部读取完毕才执行
  675. if (index === files.length - 1) {
  676. callback && callback.apply && callback.apply(window, [data, book]);
  677. }
  678. };
  679. reader.readAsBinaryString(item);
  680. });
  681. },
  682. /**
  683. * EXCEL日期码转换为Date对象
  684. * @param code double excel中存储的日期格式码
  685. */
  686. dateCodeToDate: function(code)
  687. {
  688. var obj = XLSX.SSF.parse_date_code(code);
  689. return (new Date(obj.y + '-' + obj.m + '-' + obj.d + ' ' + obj.H + ':' + obj.M + ':' + obj.S));
  690. },
  691. /**
  692. * 字符补全函数
  693. * @param str
  694. * @param maxLength
  695. * @param padString
  696. * @returns {*}
  697. */
  698. strPad: function(str, maxLength, padString) {
  699. str = str + ''
  700. if (typeof maxLength === 'undefined') {
  701. maxLength = 2
  702. }
  703. if (typeof padString === 'undefined') {
  704. padString = '0'
  705. }
  706. if (padString.length <= 0) {
  707. console.error('strPad error');
  708. return str;
  709. }
  710. if (str.length < maxLength) {
  711. var repeatCount = Math.floor((maxLength - str.length) / padString.length);
  712. var exceptStr = '';
  713. if (repeatCount * padString.length < maxLength - 1) {
  714. exceptStr = padString.substr(0, maxLength - 1 - repeatCount * padString.length)
  715. }
  716. return padString * repeatCount + exceptStr + str
  717. } else {
  718. return str
  719. }
  720. },
  721. /**
  722. * 简易格式转换
  723. * @param date Date 待转换时间
  724. * @param format String 日期格式 YYYY-MM-DD HH:ii:ss
  725. */
  726. dateFormat: function(date, format)
  727. {
  728. if (!(date instanceof Date)) {
  729. console.error(date+'需要是时间日期对象');
  730. }
  731. if (typeof format === 'undefined') {
  732. format = 'YYYY-MM-DD HH:ii:ss';
  733. }
  734. // 制造 format 相关参数
  735. var YYYY = date.getFullYear();
  736. var YY = (YYYY + '').substr(2, 2)
  737. var M = date.getMonth();
  738. var MM = this.strPad(M, 2, '0');
  739. var D = date.getDay();
  740. var DD = this.strPad(D, 2, '0');
  741. var H = date.getHours();
  742. var HH = this.strPad(H, 2, '0');
  743. var i = date.getMinutes();
  744. var ii = this.strPad(i, 2, '0');
  745. var s = date.getSeconds();
  746. var ss = this.strPad(s, 2, '0');
  747. var config = {
  748. 'YYYY': YYYY,
  749. 'YY': YY,
  750. 'MM': MM,
  751. 'M': M,
  752. 'DD': DD,
  753. 'D': D,
  754. 'HH': HH,
  755. 'H': H,
  756. 'ii': ii,
  757. 'i': i,
  758. 'ss': ss,
  759. 's': s
  760. };
  761. for (var key in config) {
  762. if (!config.hasOwnProperty(key)) {
  763. continue;
  764. }
  765. var reg = RegExp(key, 'g');
  766. format = format.replace(reg, config[key]);
  767. }
  768. return format;
  769. },
  770. /**
  771. * excel的日期CODE格式化
  772. * @param code
  773. * @param format
  774. * @returns {*|void|string}
  775. */
  776. dateCodeFormat: function (code, format) {
  777. return this.dateFormat(this.dateCodeToDate(code), format)
  778. }
  779. }
  780. if (typeof layui !== 'undefined') {
  781. layui.define(['jquery'], function(exports){
  782. $ = layui.jquery;
  783. exports('excel', LAY_EXCEL);
  784. });
  785. }
  786. /*---------split--------*//* Blob.js
  787. * A Blob, File, FileReader & URL implementation.
  788. * 2018-08-09
  789. *
  790. * By Eli Grey, http://eligrey.com
  791. * By Jimmy Wärting, https://github.com/jimmywarting
  792. * License: MIT
  793. * See https://github.com/eligrey/Blob.js/blob/master/LICENSE.md
  794. */
  795. ;(function(){
  796. var global = typeof window === 'object'
  797. ? window : typeof self === 'object'
  798. ? self : this
  799. var BlobBuilder = global.BlobBuilder
  800. || global.WebKitBlobBuilder
  801. || global.MSBlobBuilder
  802. || global.MozBlobBuilder;
  803. global.URL = global.URL || global.webkitURL || function(href, a) {
  804. a = document.createElement('a')
  805. a.href = href
  806. return a
  807. }
  808. var origBlob = global.Blob
  809. var createObjectURL = URL.createObjectURL
  810. var revokeObjectURL = URL.revokeObjectURL
  811. var strTag = global.Symbol && global.Symbol.toStringTag
  812. var blobSupported = false
  813. var blobSupportsArrayBufferView = false
  814. var arrayBufferSupported = !!global.ArrayBuffer
  815. var blobBuilderSupported = BlobBuilder
  816. && BlobBuilder.prototype.append
  817. && BlobBuilder.prototype.getBlob;
  818. try {
  819. // Check if Blob constructor is supported
  820. blobSupported = new Blob(['ä']).size === 2
  821. // Check if Blob constructor supports ArrayBufferViews
  822. // Fails in Safari 6, so we need to map to ArrayBuffers there.
  823. blobSupportsArrayBufferView = new Blob([new Uint8Array([1,2])]).size === 2
  824. } catch(e) {}
  825. /**
  826. * Helper function that maps ArrayBufferViews to ArrayBuffers
  827. * Used by BlobBuilder constructor and old browsers that didn't
  828. * support it in the Blob constructor.
  829. */
  830. function mapArrayBufferViews(ary) {
  831. return ary.map(function(chunk) {
  832. if (chunk.buffer instanceof ArrayBuffer) {
  833. var buf = chunk.buffer;
  834. // if this is a subarray, make a copy so we only
  835. // include the subarray region from the underlying buffer
  836. if (chunk.byteLength !== buf.byteLength) {
  837. var copy = new Uint8Array(chunk.byteLength);
  838. copy.set(new Uint8Array(buf, chunk.byteOffset, chunk.byteLength));
  839. buf = copy.buffer;
  840. }
  841. return buf;
  842. }
  843. return chunk;
  844. });
  845. }
  846. function BlobBuilderConstructor(ary, options) {
  847. options = options || {};
  848. var bb = new BlobBuilder();
  849. mapArrayBufferViews(ary).forEach(function(part) {
  850. bb.append(part);
  851. });
  852. return options.type ? bb.getBlob(options.type) : bb.getBlob();
  853. };
  854. function BlobConstructor(ary, options) {
  855. return new origBlob(mapArrayBufferViews(ary), options || {});
  856. };
  857. if (global.Blob) {
  858. BlobBuilderConstructor.prototype = Blob.prototype;
  859. BlobConstructor.prototype = Blob.prototype;
  860. }
  861. function FakeBlobBuilder() {
  862. function toUTF8Array(str) {
  863. var utf8 = [];
  864. for (var i=0; i < str.length; i++) {
  865. var charcode = str.charCodeAt(i);
  866. if (charcode < 0x80) utf8.push(charcode);
  867. else if (charcode < 0x800) {
  868. utf8.push(0xc0 | (charcode >> 6),
  869. 0x80 | (charcode & 0x3f));
  870. }
  871. else if (charcode < 0xd800 || charcode >= 0xe000) {
  872. utf8.push(0xe0 | (charcode >> 12),
  873. 0x80 | ((charcode>>6) & 0x3f),
  874. 0x80 | (charcode & 0x3f));
  875. }
  876. // surrogate pair
  877. else {
  878. i++;
  879. // UTF-16 encodes 0x10000-0x10FFFF by
  880. // subtracting 0x10000 and splitting the
  881. // 20 bits of 0x0-0xFFFFF into two halves
  882. charcode = 0x10000 + (((charcode & 0x3ff)<<10)
  883. | (str.charCodeAt(i) & 0x3ff));
  884. utf8.push(0xf0 | (charcode >>18),
  885. 0x80 | ((charcode>>12) & 0x3f),
  886. 0x80 | ((charcode>>6) & 0x3f),
  887. 0x80 | (charcode & 0x3f));
  888. }
  889. }
  890. return utf8;
  891. }
  892. function fromUtf8Array(array) {
  893. var out, i, len, c;
  894. var char2, char3;
  895. out = "";
  896. len = array.length;
  897. i = 0;
  898. while (i < len) {
  899. c = array[i++];
  900. switch (c >> 4)
  901. {
  902. case 0: case 1: case 2: case 3: case 4: case 5: case 6: case 7:
  903. // 0xxxxxxx
  904. out += String.fromCharCode(c);
  905. break;
  906. case 12: case 13:
  907. // 110x xxxx 10xx xxxx
  908. char2 = array[i++];
  909. out += String.fromCharCode(((c & 0x1F) << 6) | (char2 & 0x3F));
  910. break;
  911. case 14:
  912. // 1110 xxxx 10xx xxxx 10xx xxxx
  913. char2 = array[i++];
  914. char3 = array[i++];
  915. out += String.fromCharCode(((c & 0x0F) << 12) |
  916. ((char2 & 0x3F) << 6) |
  917. ((char3 & 0x3F) << 0));
  918. break;
  919. }
  920. }
  921. return out;
  922. }
  923. function isDataView(obj) {
  924. return obj && DataView.prototype.isPrototypeOf(obj)
  925. }
  926. function bufferClone(buf) {
  927. var view = new Array(buf.byteLength)
  928. var array = new Uint8Array(buf)
  929. var i = view.length
  930. while(i--) {
  931. view[i] = array[i]
  932. }
  933. return view
  934. }
  935. function encodeByteArray(input) {
  936. var byteToCharMap = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'
  937. var output = [];
  938. for (var i = 0; i < input.length; i += 3) {
  939. var byte1 = input[i];
  940. var haveByte2 = i + 1 < input.length;
  941. var byte2 = haveByte2 ? input[i + 1] : 0;
  942. var haveByte3 = i + 2 < input.length;
  943. var byte3 = haveByte3 ? input[i + 2] : 0;
  944. var outByte1 = byte1 >> 2;
  945. var outByte2 = ((byte1 & 0x03) << 4) | (byte2 >> 4);
  946. var outByte3 = ((byte2 & 0x0F) << 2) | (byte3 >> 6);
  947. var outByte4 = byte3 & 0x3F;
  948. if (!haveByte3) {
  949. outByte4 = 64;
  950. if (!haveByte2) {
  951. outByte3 = 64;
  952. }
  953. }
  954. output.push(
  955. byteToCharMap[outByte1], byteToCharMap[outByte2],
  956. byteToCharMap[outByte3], byteToCharMap[outByte4])
  957. }
  958. return output.join('')
  959. }
  960. var create = Object.create || function (a) {
  961. function c() {}
  962. c.prototype = a;
  963. return new c
  964. }
  965. if (arrayBufferSupported) {
  966. var viewClasses = [
  967. '[object Int8Array]',
  968. '[object Uint8Array]',
  969. '[object Uint8ClampedArray]',
  970. '[object Int16Array]',
  971. '[object Uint16Array]',
  972. '[object Int32Array]',
  973. '[object Uint32Array]',
  974. '[object Float32Array]',
  975. '[object Float64Array]'
  976. ]
  977. var isArrayBufferView = ArrayBuffer.isView || function(obj) {
  978. return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1
  979. }
  980. }
  981. /********************************************************/
  982. /* Blob constructor */
  983. /********************************************************/
  984. function Blob(chunks, opts) {
  985. chunks = chunks || []
  986. for (var i = 0, len = chunks.length; i < len; i++) {
  987. var chunk = chunks[i]
  988. if (chunk instanceof Blob) {
  989. chunks[i] = chunk._buffer
  990. } else if (typeof chunk === 'string') {
  991. chunks[i] = toUTF8Array(chunk)
  992. } else if (arrayBufferSupported && (ArrayBuffer.prototype.isPrototypeOf(chunk) || isArrayBufferView(chunk))) {
  993. chunks[i] = bufferClone(chunk)
  994. } else if (arrayBufferSupported && isDataView(chunk)) {
  995. chunks[i] = bufferClone(chunk.buffer)
  996. } else {
  997. chunks[i] = toUTF8Array(String(chunk))
  998. }
  999. }
  1000. this._buffer = [].concat.apply([], chunks)
  1001. this.size = this._buffer.length
  1002. this.type = opts ? opts.type || '' : ''
  1003. }
  1004. Blob.prototype.slice = function(start, end, type) {
  1005. var slice = this._buffer.slice(start || 0, end || this._buffer.length)
  1006. return new Blob([slice], {type: type})
  1007. }
  1008. Blob.prototype.toString = function() {
  1009. return '[object Blob]'
  1010. }
  1011. /********************************************************/
  1012. /* File constructor */
  1013. /********************************************************/
  1014. function File(chunks, name, opts) {
  1015. opts = opts || {}
  1016. var a = Blob.call(this, chunks, opts) || this
  1017. a.name = name
  1018. a.lastModifiedDate = opts.lastModified ? new Date(opts.lastModified) : new Date
  1019. a.lastModified = +a.lastModifiedDate
  1020. return a
  1021. }
  1022. File.prototype = create(Blob.prototype);
  1023. File.prototype.constructor = File;
  1024. if (Object.setPrototypeOf)
  1025. Object.setPrototypeOf(File, Blob);
  1026. else {
  1027. try {File.__proto__ = Blob} catch (e) {}
  1028. }
  1029. File.prototype.toString = function() {
  1030. return '[object File]'
  1031. }
  1032. /********************************************************/
  1033. /* FileReader constructor */
  1034. /********************************************************/
  1035. function FileReader() {
  1036. if (!(this instanceof FileReader))
  1037. throw new TypeError("Failed to construct 'FileReader': Please use the 'new' operator, this DOM object constructor cannot be called as a function.")
  1038. var delegate = document.createDocumentFragment()
  1039. this.addEventListener = delegate.addEventListener
  1040. this.dispatchEvent = function(evt) {
  1041. var local = this['on' + evt.type]
  1042. if (typeof local === 'function') local(evt)
  1043. delegate.dispatchEvent(evt)
  1044. }
  1045. this.removeEventListener = delegate.removeEventListener
  1046. }
  1047. function _read(fr, blob, kind) {
  1048. if (!(blob instanceof Blob))
  1049. throw new TypeError("Failed to execute '" + kind + "' on 'FileReader': parameter 1 is not of type 'Blob'.")
  1050. fr.result = ''
  1051. setTimeout(function(){
  1052. this.readyState = FileReader.LOADING
  1053. fr.dispatchEvent(new Event('load'))
  1054. fr.dispatchEvent(new Event('loadend'))
  1055. })
  1056. }
  1057. FileReader.EMPTY = 0
  1058. FileReader.LOADING = 1
  1059. FileReader.DONE = 2
  1060. FileReader.prototype.error = null
  1061. FileReader.prototype.onabort = null
  1062. FileReader.prototype.onerror = null
  1063. FileReader.prototype.onload = null
  1064. FileReader.prototype.onloadend = null
  1065. FileReader.prototype.onloadstart = null
  1066. FileReader.prototype.onprogress = null
  1067. FileReader.prototype.readAsDataURL = function(blob) {
  1068. _read(this, blob, 'readAsDataURL')
  1069. this.result = 'data:' + blob.type + ';base64,' + encodeByteArray(blob._buffer)
  1070. }
  1071. FileReader.prototype.readAsText = function(blob) {
  1072. _read(this, blob, 'readAsText')
  1073. this.result = fromUtf8Array(blob._buffer)
  1074. }
  1075. FileReader.prototype.readAsArrayBuffer = function(blob) {
  1076. _read(this, blob, 'readAsText')
  1077. this.result = blob._buffer.slice()
  1078. }
  1079. FileReader.prototype.abort = function() {}
  1080. /********************************************************/
  1081. /* URL */
  1082. /********************************************************/
  1083. URL.createObjectURL = function(blob) {
  1084. return blob instanceof Blob
  1085. ? 'data:' + blob.type + ';base64,' + encodeByteArray(blob._buffer)
  1086. : createObjectURL.call(URL, blob)
  1087. }
  1088. URL.revokeObjectURL = function(url) {
  1089. revokeObjectURL && revokeObjectURL.call(URL, url)
  1090. }
  1091. /********************************************************/
  1092. /* XHR */
  1093. /********************************************************/
  1094. var _send = global.XMLHttpRequest && global.XMLHttpRequest.prototype.send
  1095. if (_send) {
  1096. XMLHttpRequest.prototype.send = function(data) {
  1097. if (data instanceof Blob) {
  1098. this.setRequestHeader('Content-Type', data.type)
  1099. _send.call(this, fromUtf8Array(data._buffer))
  1100. } else {
  1101. _send.call(this, data)
  1102. }
  1103. }
  1104. }
  1105. global.FileReader = FileReader
  1106. global.File = File
  1107. global.Blob = Blob
  1108. }
  1109. if (strTag) {
  1110. File.prototype[strTag] = 'File'
  1111. Blob.prototype[strTag] = 'Blob'
  1112. FileReader.prototype[strTag] = 'FileReader'
  1113. }
  1114. function fixFileAndXHR() {
  1115. var isIE = !!global.ActiveXObject || (
  1116. '-ms-scroll-limit' in document.documentElement.style &&
  1117. '-ms-ime-align' in document.documentElement.style
  1118. )
  1119. // Monkey patched
  1120. // IE don't set Content-Type header on XHR whose body is a typed Blob
  1121. // https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/6047383
  1122. var _send = global.XMLHttpRequest && global.XMLHttpRequest.prototype.send
  1123. if (isIE && _send) {
  1124. XMLHttpRequest.prototype.send = function(data) {
  1125. if (data instanceof Blob) {
  1126. this.setRequestHeader('Content-Type', data.type)
  1127. _send.call(this, data)
  1128. } else {
  1129. _send.call(this, data)
  1130. }
  1131. }
  1132. }
  1133. try {
  1134. new File([], '')
  1135. } catch(e) {
  1136. try {
  1137. var klass = new Function('class File extends Blob {' +
  1138. 'constructor(chunks, name, opts) {' +
  1139. 'opts = opts || {};' +
  1140. 'super(chunks, opts || {});' +
  1141. 'this.name = name;' +
  1142. 'this.lastModifiedDate = opts.lastModified ? new Date(opts.lastModified) : new Date;' +
  1143. 'this.lastModified = +this.lastModifiedDate;' +
  1144. '}};' +
  1145. 'return new File([], ""), File'
  1146. )()
  1147. global.File = klass
  1148. } catch(e) {
  1149. var klass = function(b, d, c) {
  1150. var blob = new Blob(b, c)
  1151. var t = c && void 0 !== c.lastModified ? new Date(c.lastModified) : new Date
  1152. blob.name = d
  1153. blob.lastModifiedDate = t
  1154. blob.lastModified = +t
  1155. blob.toString = function() {
  1156. return '[object File]'
  1157. }
  1158. if (strTag)
  1159. blob[strTag] = 'File'
  1160. return blob
  1161. }
  1162. global.File = klass
  1163. }
  1164. }
  1165. }
  1166. if (blobSupported) {
  1167. fixFileAndXHR()
  1168. global.Blob = blobSupportsArrayBufferView ? global.Blob : BlobConstructor
  1169. } else if (blobBuilderSupported) {
  1170. fixFileAndXHR()
  1171. global.Blob = BlobBuilderConstructor;
  1172. } else {
  1173. FakeBlobBuilder()
  1174. }
  1175. })();
  1176. /*---------split--------*/(function(a,b){if("function"==typeof define&&define.amd)define([],b);else if("undefined"!=typeof exports)b();else{b(),a.FileSaver={exports:{}}.exports}})(this,function(){"use strict";function b(a,b){return"undefined"==typeof b?b={autoBom:!1}:"object"!=typeof b&&(console.warn("Depricated: Expected third argument to be a object"),b={autoBom:!b}),b.autoBom&&/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob(["\uFEFF",a],{type:a.type}):a}function c(b,c,d){var e=new XMLHttpRequest;e.open("GET",b),e.responseType="blob",e.onload=function(){a(e.response,c,d)},e.onerror=function(){console.error("could not download file")},e.send()}function d(a){var b=new XMLHttpRequest;return b.open("HEAD",a,!1),b.send(),200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent("click"))}catch(c){var b=document.createEvent("MouseEvents");b.initMouseEvent("click",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b)}}var f="object"==typeof window&&window.window===window?window:"object"==typeof self&&self.self===self?self:"object"==typeof global&&global.global===global?global:void 0,a=f.saveAs||"object"!=typeof window||window!==f?function(){}:"download"in HTMLAnchorElement.prototype?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement("a");g=g||b.name||"download",j.download=g,j.rel="noopener","string"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target="_blank")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href)},4E4),setTimeout(function(){e(j)},0))}:"msSaveOrOpenBlob"in navigator?function(f,g,h){if(g=g||f.name||"download","string"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else{var i=document.createElement("a");i.href=f,i.target="_blank",setTimeout(function(){e(i)})}}:function(a,b,d,e){if(e=e||open("","_blank"),e&&(e.document.title=e.document.body.innerText="downloading..."),"string"==typeof a)return c(a,b,d);var g="application/octet-stream"===a.type,h=/constructor/i.test(f.HTMLElement)||f.safari,i=/CriOS\/[\d]+/.test(navigator.userAgent);if((i||g&&h)&&"object"==typeof FileReader){var j=new FileReader;j.onloadend=function(){var a=j.result;a=i?a:a.replace(/^data:[^;]*;/,"data:attachment/file;"),e?e.location.href=a:location=a,e=null},j.readAsDataURL(a)}else{var k=f.URL||f.webkitURL,l=k.createObjectURL(a);e?e.location=l:location.href=l,e=null,setTimeout(function(){k.revokeObjectURL(l)},4E4)}};f.saveAs=a.saveAs=a,"undefined"!=typeof module&&(module.exports=a)});
  1177. /*---------split--------*//*
  1178. JSZip - A Javascript class for generating and reading zip files
  1179. <http://stuartk.com/jszip>
  1180. (c) 2009-2014 Stuart Knightley <stuart [at] stuartk.com>
  1181. Dual licenced under the MIT license or GPLv3. See https://raw.github.com/Stuk/jszip/master/LICENSE.markdown.
  1182. JSZip uses the library pako released under the MIT license :
  1183. https://github.com/nodeca/pako/blob/master/LICENSE
  1184. Note: since JSZip 3 removed critical functionality, this version assigns to the
  1185. `JSZipSync` variable. Another JSZip version can be loaded in parallel.
  1186. */
  1187. (function(e){
  1188. if("object"==typeof exports&&"undefined"!=typeof module&&"undefined"==typeof DO_NOT_EXPORT_JSZIP)module.exports=e();
  1189. else if("function"==typeof define&&define.amd&&"undefined"==typeof DO_NOT_EXPORT_JSZIP){JSZipSync=e();define([],e);}
  1190. else{
  1191. var f;
  1192. "undefined"!=typeof window?f=window:
  1193. "undefined"!=typeof global?f=global:
  1194. "undefined"!=typeof $ && $.global?f=$.global:
  1195. "undefined"!=typeof self&&(f=self),f.JSZipSync=e()
  1196. }
  1197. }(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);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.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){
  1198. 'use strict';
  1199. // private property
  1200. var _keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  1201. // public method for encoding
  1202. exports.encode = function(input, utf8) {
  1203. var output = "";
  1204. var chr1, chr2, chr3, enc1, enc2, enc3, enc4;
  1205. var i = 0;
  1206. while (i < input.length) {
  1207. chr1 = input.charCodeAt(i++);
  1208. chr2 = input.charCodeAt(i++);
  1209. chr3 = input.charCodeAt(i++);
  1210. enc1 = chr1 >> 2;
  1211. enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
  1212. enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
  1213. enc4 = chr3 & 63;
  1214. if (isNaN(chr2)) {
  1215. enc3 = enc4 = 64;
  1216. }
  1217. else if (isNaN(chr3)) {
  1218. enc4 = 64;
  1219. }
  1220. output = output + _keyStr.charAt(enc1) + _keyStr.charAt(enc2) + _keyStr.charAt(enc3) + _keyStr.charAt(enc4);
  1221. }
  1222. return output;
  1223. };
  1224. // public method for decoding
  1225. exports.decode = function(input, utf8) {
  1226. var output = "";
  1227. var chr1, chr2, chr3;
  1228. var enc1, enc2, enc3, enc4;
  1229. var i = 0;
  1230. input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
  1231. while (i < input.length) {
  1232. enc1 = _keyStr.indexOf(input.charAt(i++));
  1233. enc2 = _keyStr.indexOf(input.charAt(i++));
  1234. enc3 = _keyStr.indexOf(input.charAt(i++));
  1235. enc4 = _keyStr.indexOf(input.charAt(i++));
  1236. chr1 = (enc1 << 2) | (enc2 >> 4);
  1237. chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
  1238. chr3 = ((enc3 & 3) << 6) | enc4;
  1239. output = output + String.fromCharCode(chr1);
  1240. if (enc3 != 64) {
  1241. output = output + String.fromCharCode(chr2);
  1242. }
  1243. if (enc4 != 64) {
  1244. output = output + String.fromCharCode(chr3);
  1245. }
  1246. }
  1247. return output;
  1248. };
  1249. },{}],2:[function(_dereq_,module,exports){
  1250. 'use strict';
  1251. function CompressedObject() {
  1252. this.compressedSize = 0;
  1253. this.uncompressedSize = 0;
  1254. this.crc32 = 0;
  1255. this.compressionMethod = null;
  1256. this.compressedContent = null;
  1257. }
  1258. CompressedObject.prototype = {
  1259. /**
  1260. * Return the decompressed content in an unspecified format.
  1261. * The format will depend on the decompressor.
  1262. * @return {Object} the decompressed content.
  1263. */
  1264. getContent: function() {
  1265. return null; // see implementation
  1266. },
  1267. /**
  1268. * Return the compressed content in an unspecified format.
  1269. * The format will depend on the compressed conten source.
  1270. * @return {Object} the compressed content.
  1271. */
  1272. getCompressedContent: function() {
  1273. return null; // see implementation
  1274. }
  1275. };
  1276. module.exports = CompressedObject;
  1277. },{}],3:[function(_dereq_,module,exports){
  1278. 'use strict';
  1279. exports.STORE = {
  1280. magic: "\x00\x00",
  1281. compress: function(content) {
  1282. return content; // no compression
  1283. },
  1284. uncompress: function(content) {
  1285. return content; // no compression
  1286. },
  1287. compressInputType: null,
  1288. uncompressInputType: null
  1289. };
  1290. exports.DEFLATE = _dereq_('./flate');
  1291. },{"./flate":8}],4:[function(_dereq_,module,exports){
  1292. 'use strict';
  1293. var utils = _dereq_('./utils');
  1294. var table = [
  1295. 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA,
  1296. 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3,
  1297. 0x0EDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988,
  1298. 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91,
  1299. 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE,
  1300. 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7,
  1301. 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC,
  1302. 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5,
  1303. 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172,
  1304. 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B,
  1305. 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940,
  1306. 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59,
  1307. 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116,
  1308. 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F,
  1309. 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924,
  1310. 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D,
  1311. 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A,
  1312. 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433,
  1313. 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818,
  1314. 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01,
  1315. 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E,
  1316. 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457,
  1317. 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C,
  1318. 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65,
  1319. 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2,
  1320. 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB,
  1321. 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0,
  1322. 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9,
  1323. 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086,
  1324. 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F,
  1325. 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4,
  1326. 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD,
  1327. 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A,
  1328. 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683,
  1329. 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8,
  1330. 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1,
  1331. 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE,
  1332. 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7,
  1333. 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC,
  1334. 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5,
  1335. 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252,
  1336. 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B,
  1337. 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60,
  1338. 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79,
  1339. 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236,
  1340. 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F,
  1341. 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04,
  1342. 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D,
  1343. 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A,
  1344. 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713,
  1345. 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38,
  1346. 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21,
  1347. 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E,
  1348. 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777,
  1349. 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C,
  1350. 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45,
  1351. 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2,
  1352. 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB,
  1353. 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0,
  1354. 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9,
  1355. 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6,
  1356. 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF,
  1357. 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94,
  1358. 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D
  1359. ];
  1360. /**
  1361. *
  1362. * Javascript crc32
  1363. * http://www.webtoolkit.info/
  1364. *
  1365. */
  1366. module.exports = function crc32(input, crc) {
  1367. if (typeof input === "undefined" || !input.length) {
  1368. return 0;
  1369. }
  1370. var isArray = utils.getTypeOf(input) !== "string";
  1371. if (typeof(crc) == "undefined") {
  1372. crc = 0;
  1373. }
  1374. var x = 0;
  1375. var y = 0;
  1376. var b = 0;
  1377. crc = crc ^ (-1);
  1378. for (var i = 0, iTop = input.length; i < iTop; i++) {
  1379. b = isArray ? input[i] : input.charCodeAt(i);
  1380. y = (crc ^ b) & 0xFF;
  1381. x = table[y];
  1382. crc = (crc >>> 8) ^ x;
  1383. }
  1384. return crc ^ (-1);
  1385. };
  1386. // vim: set shiftwidth=4 softtabstop=4:
  1387. },{"./utils":21}],5:[function(_dereq_,module,exports){
  1388. 'use strict';
  1389. var utils = _dereq_('./utils');
  1390. function DataReader(data) {
  1391. this.data = null; // type : see implementation
  1392. this.length = 0;
  1393. this.index = 0;
  1394. }
  1395. DataReader.prototype = {
  1396. /**
  1397. * Check that the offset will not go too far.
  1398. * @param {string} offset the additional offset to check.
  1399. * @throws {Error} an error if the offset is out of bounds.
  1400. */
  1401. checkOffset: function(offset) {
  1402. this.checkIndex(this.index + offset);
  1403. },
  1404. /**
  1405. * Check that the specifed index will not be too far.
  1406. * @param {string} newIndex the index to check.
  1407. * @throws {Error} an error if the index is out of bounds.
  1408. */
  1409. checkIndex: function(newIndex) {
  1410. if (this.length < newIndex || newIndex < 0) {
  1411. throw new Error("End of data reached (data length = " + this.length + ", asked index = " + (newIndex) + "). Corrupted zip ?");
  1412. }
  1413. },
  1414. /**
  1415. * Change the index.
  1416. * @param {number} newIndex The new index.
  1417. * @throws {Error} if the new index is out of the data.
  1418. */
  1419. setIndex: function(newIndex) {
  1420. this.checkIndex(newIndex);
  1421. this.index = newIndex;
  1422. },
  1423. /**
  1424. * Skip the next n bytes.
  1425. * @param {number} n the number of bytes to skip.
  1426. * @throws {Error} if the new index is out of the data.
  1427. */
  1428. skip: function(n) {
  1429. this.setIndex(this.index + n);
  1430. },
  1431. /**
  1432. * Get the byte at the specified index.
  1433. * @param {number} i the index to use.
  1434. * @return {number} a byte.
  1435. */
  1436. byteAt: function(i) {
  1437. // see implementations
  1438. },
  1439. /**
  1440. * Get the next number with a given byte size.
  1441. * @param {number} size the number of bytes to read.
  1442. * @return {number} the corresponding number.
  1443. */
  1444. readInt: function(size) {
  1445. var result = 0,
  1446. i;
  1447. this.checkOffset(size);
  1448. for (i = this.index + size - 1; i >= this.index; i--) {
  1449. result = (result << 8) + this.byteAt(i);
  1450. }
  1451. this.index += size;
  1452. return result;
  1453. },
  1454. /**
  1455. * Get the next string with a given byte size.
  1456. * @param {number} size the number of bytes to read.
  1457. * @return {string} the corresponding string.
  1458. */
  1459. readString: function(size) {
  1460. return utils.transformTo("string", this.readData(size));
  1461. },
  1462. /**
  1463. * Get raw data without conversion, <size> bytes.
  1464. * @param {number} size the number of bytes to read.
  1465. * @return {Object} the raw data, implementation specific.
  1466. */
  1467. readData: function(size) {
  1468. // see implementations
  1469. },
  1470. /**
  1471. * Find the last occurence of a zip signature (4 bytes).
  1472. * @param {string} sig the signature to find.
  1473. * @return {number} the index of the last occurence, -1 if not found.
  1474. */
  1475. lastIndexOfSignature: function(sig) {
  1476. // see implementations
  1477. },
  1478. /**
  1479. * Get the next date.
  1480. * @return {Date} the date.
  1481. */
  1482. readDate: function() {
  1483. var dostime = this.readInt(4);
  1484. return new Date(
  1485. ((dostime >> 25) & 0x7f) + 1980, // year
  1486. ((dostime >> 21) & 0x0f) - 1, // month
  1487. (dostime >> 16) & 0x1f, // day
  1488. (dostime >> 11) & 0x1f, // hour
  1489. (dostime >> 5) & 0x3f, // minute
  1490. (dostime & 0x1f) << 1); // second
  1491. }
  1492. };
  1493. module.exports = DataReader;
  1494. },{"./utils":21}],6:[function(_dereq_,module,exports){
  1495. 'use strict';
  1496. exports.base64 = false;
  1497. exports.binary = false;
  1498. exports.dir = false;
  1499. exports.createFolders = false;
  1500. exports.date = null;
  1501. exports.compression = null;
  1502. exports.comment = null;
  1503. },{}],7:[function(_dereq_,module,exports){
  1504. 'use strict';
  1505. var utils = _dereq_('./utils');
  1506. /**
  1507. * @deprecated
  1508. * This function will be removed in a future version without replacement.
  1509. */
  1510. exports.string2binary = function(str) {
  1511. return utils.string2binary(str);
  1512. };
  1513. /**
  1514. * @deprecated
  1515. * This function will be removed in a future version without replacement.
  1516. */
  1517. exports.string2Uint8Array = function(str) {
  1518. return utils.transformTo("uint8array", str);
  1519. };
  1520. /**
  1521. * @deprecated
  1522. * This function will be removed in a future version without replacement.
  1523. */
  1524. exports.uint8Array2String = function(array) {
  1525. return utils.transformTo("string", array);
  1526. };
  1527. /**
  1528. * @deprecated
  1529. * This function will be removed in a future version without replacement.
  1530. */
  1531. exports.string2Blob = function(str) {
  1532. var buffer = utils.transformTo("arraybuffer", str);
  1533. return utils.arrayBuffer2Blob(buffer);
  1534. };
  1535. /**
  1536. * @deprecated
  1537. * This function will be removed in a future version without replacement.
  1538. */
  1539. exports.arrayBuffer2Blob = function(buffer) {
  1540. return utils.arrayBuffer2Blob(buffer);
  1541. };
  1542. /**
  1543. * @deprecated
  1544. * This function will be removed in a future version without replacement.
  1545. */
  1546. exports.transformTo = function(outputType, input) {
  1547. return utils.transformTo(outputType, input);
  1548. };
  1549. /**
  1550. * @deprecated
  1551. * This function will be removed in a future version without replacement.
  1552. */
  1553. exports.getTypeOf = function(input) {
  1554. return utils.getTypeOf(input);
  1555. };
  1556. /**
  1557. * @deprecated
  1558. * This function will be removed in a future version without replacement.
  1559. */
  1560. exports.checkSupport = function(type) {
  1561. return utils.checkSupport(type);
  1562. };
  1563. /**
  1564. * @deprecated
  1565. * This value will be removed in a future version without replacement.
  1566. */
  1567. exports.MAX_VALUE_16BITS = utils.MAX_VALUE_16BITS;
  1568. /**
  1569. * @deprecated
  1570. * This value will be removed in a future version without replacement.
  1571. */
  1572. exports.MAX_VALUE_32BITS = utils.MAX_VALUE_32BITS;
  1573. /**
  1574. * @deprecated
  1575. * This function will be removed in a future version without replacement.
  1576. */
  1577. exports.pretty = function(str) {
  1578. return utils.pretty(str);
  1579. };
  1580. /**
  1581. * @deprecated
  1582. * This function will be removed in a future version without replacement.
  1583. */
  1584. exports.findCompression = function(compressionMethod) {
  1585. return utils.findCompression(compressionMethod);
  1586. };
  1587. /**
  1588. * @deprecated
  1589. * This function will be removed in a future version without replacement.
  1590. */
  1591. exports.isRegExp = function (object) {
  1592. return utils.isRegExp(object);
  1593. };
  1594. },{"./utils":21}],8:[function(_dereq_,module,exports){
  1595. 'use strict';
  1596. var USE_TYPEDARRAY = (typeof Uint8Array !== 'undefined') && (typeof Uint16Array !== 'undefined') && (typeof Uint32Array !== 'undefined');
  1597. var pako = _dereq_("pako");
  1598. exports.uncompressInputType = USE_TYPEDARRAY ? "uint8array" : "array";
  1599. exports.compressInputType = USE_TYPEDARRAY ? "uint8array" : "array";
  1600. exports.magic = "\x08\x00";
  1601. exports.compress = function(input) {
  1602. return pako.deflateRaw(input);
  1603. };
  1604. exports.uncompress = function(input) {
  1605. return pako.inflateRaw(input);
  1606. };
  1607. },{"pako":24}],9:[function(_dereq_,module,exports){
  1608. 'use strict';
  1609. var base64 = _dereq_('./base64');
  1610. /**
  1611. Usage:
  1612. zip = new JSZip();
  1613. zip.file("hello.txt", "Hello, World!").file("tempfile", "nothing");
  1614. zip.folder("images").file("smile.gif", base64Data, {base64: true});
  1615. zip.file("Xmas.txt", "Ho ho ho !", {date : new Date("December 25, 2007 00:00:01")});
  1616. zip.remove("tempfile");
  1617. base64zip = zip.generate();
  1618. **/
  1619. /**
  1620. * Representation a of zip file in js
  1621. * @constructor
  1622. * @param {String=|ArrayBuffer=|Uint8Array=} data the data to load, if any (optional).
  1623. * @param {Object=} options the options for creating this objects (optional).
  1624. */
  1625. function JSZipSync(data, options) {
  1626. // if this constructor is used without `new`, it adds `new` before itself:
  1627. if(!(this instanceof JSZipSync)) return new JSZipSync(data, options);
  1628. // object containing the files :
  1629. // {
  1630. // "folder/" : {...},
  1631. // "folder/data.txt" : {...}
  1632. // }
  1633. this.files = {};
  1634. this.comment = null;
  1635. // Where we are in the hierarchy
  1636. this.root = "";
  1637. if (data) {
  1638. this.load(data, options);
  1639. }
  1640. this.clone = function() {
  1641. var newObj = new JSZipSync();
  1642. for (var i in this) {
  1643. if (typeof this[i] !== "function") {
  1644. newObj[i] = this[i];
  1645. }
  1646. }
  1647. return newObj;
  1648. };
  1649. }
  1650. JSZipSync.prototype = _dereq_('./object');
  1651. JSZipSync.prototype.load = _dereq_('./load');
  1652. JSZipSync.support = _dereq_('./support');
  1653. JSZipSync.defaults = _dereq_('./defaults');
  1654. /**
  1655. * @deprecated
  1656. * This namespace will be removed in a future version without replacement.
  1657. */
  1658. JSZipSync.utils = _dereq_('./deprecatedPublicUtils');
  1659. JSZipSync.base64 = {
  1660. /**
  1661. * @deprecated
  1662. * This method will be removed in a future version without replacement.
  1663. */
  1664. encode : function(input) {
  1665. return base64.encode(input);
  1666. },
  1667. /**
  1668. * @deprecated
  1669. * This method will be removed in a future version without replacement.
  1670. */
  1671. decode : function(input) {
  1672. return base64.decode(input);
  1673. }
  1674. };
  1675. JSZipSync.compressions = _dereq_('./compressions');
  1676. module.exports = JSZipSync;
  1677. },{"./base64":1,"./compressions":3,"./defaults":6,"./deprecatedPublicUtils":7,"./load":10,"./object":13,"./support":17}],10:[function(_dereq_,module,exports){
  1678. 'use strict';
  1679. var base64 = _dereq_('./base64');
  1680. var ZipEntries = _dereq_('./zipEntries');
  1681. module.exports = function(data, options) {
  1682. var files, zipEntries, i, input;
  1683. options = options || {};
  1684. if (options.base64) {
  1685. data = base64.decode(data);
  1686. }
  1687. zipEntries = new ZipEntries(data, options);
  1688. files = zipEntries.files;
  1689. for (i = 0; i < files.length; i++) {
  1690. input = files[i];
  1691. this.file(input.fileName, input.decompressed, {
  1692. binary: true,
  1693. optimizedBinaryString: true,
  1694. date: input.date,
  1695. dir: input.dir,
  1696. comment : input.fileComment.length ? input.fileComment : null,
  1697. createFolders: options.createFolders
  1698. });
  1699. }
  1700. if (zipEntries.zipComment.length) {
  1701. this.comment = zipEntries.zipComment;
  1702. }
  1703. return this;
  1704. };
  1705. },{"./base64":1,"./zipEntries":22}],11:[function(_dereq_,module,exports){
  1706. (function (Buffer){
  1707. 'use strict';
  1708. var Buffer_from = /*::(*/function(){}/*:: :any)*/;
  1709. if(typeof Buffer !== 'undefined') {
  1710. var nbfs = !Buffer.from;
  1711. if(!nbfs) try { Buffer.from("foo", "utf8"); } catch(e) { nbfs = true; }
  1712. Buffer_from = nbfs ? function(buf, enc) { return (enc) ? new Buffer(buf, enc) : new Buffer(buf); } : Buffer.from.bind(Buffer);
  1713. // $FlowIgnore
  1714. if(!Buffer.alloc) Buffer.alloc = function(n) { return new Buffer(n); };
  1715. }
  1716. module.exports = function(data, encoding){
  1717. return typeof data == 'number' ? Buffer.alloc(data) : Buffer_from(data, encoding);
  1718. };
  1719. module.exports.test = function(b){
  1720. return Buffer.isBuffer(b);
  1721. };
  1722. }).call(this,(typeof Buffer !== "undefined" ? Buffer : undefined))
  1723. },{}],12:[function(_dereq_,module,exports){
  1724. 'use strict';
  1725. var Uint8ArrayReader = _dereq_('./uint8ArrayReader');
  1726. function NodeBufferReader(data) {
  1727. this.data = data;
  1728. this.length = this.data.length;
  1729. this.index = 0;
  1730. }
  1731. NodeBufferReader.prototype = new Uint8ArrayReader();
  1732. /**
  1733. * @see DataReader.readData
  1734. */
  1735. NodeBufferReader.prototype.readData = function(size) {
  1736. this.checkOffset(size);
  1737. var result = this.data.slice(this.index, this.index + size);
  1738. this.index += size;
  1739. return result;
  1740. };
  1741. module.exports = NodeBufferReader;
  1742. },{"./uint8ArrayReader":18}],13:[function(_dereq_,module,exports){
  1743. 'use strict';
  1744. var support = _dereq_('./support');
  1745. var utils = _dereq_('./utils');
  1746. var crc32 = _dereq_('./crc32');
  1747. var signature = _dereq_('./signature');
  1748. var defaults = _dereq_('./defaults');
  1749. var base64 = _dereq_('./base64');
  1750. var compressions = _dereq_('./compressions');
  1751. var CompressedObject = _dereq_('./compressedObject');
  1752. var nodeBuffer = _dereq_('./nodeBuffer');
  1753. var utf8 = _dereq_('./utf8');
  1754. var StringWriter = _dereq_('./stringWriter');
  1755. var Uint8ArrayWriter = _dereq_('./uint8ArrayWriter');
  1756. /**
  1757. * Returns the raw data of a ZipObject, decompress the content if necessary.
  1758. * @param {ZipObject} file the file to use.
  1759. * @return {String|ArrayBuffer|Uint8Array|Buffer} the data.
  1760. */
  1761. var getRawData = function(file) {
  1762. if (file._data instanceof CompressedObject) {
  1763. file._data = file._data.getContent();
  1764. file.options.binary = true;
  1765. file.options.base64 = false;
  1766. if (utils.getTypeOf(file._data) === "uint8array") {
  1767. var copy = file._data;
  1768. // when reading an arraybuffer, the CompressedObject mechanism will keep it and subarray() a Uint8Array.
  1769. // if we request a file in the same format, we might get the same Uint8Array or its ArrayBuffer (the original zip file).
  1770. file._data = new Uint8Array(copy.length);
  1771. // with an empty Uint8Array, Opera fails with a "Offset larger than array size"
  1772. if (copy.length !== 0) {
  1773. file._data.set(copy, 0);
  1774. }
  1775. }
  1776. }
  1777. return file._data;
  1778. };
  1779. /**
  1780. * Returns the data of a ZipObject in a binary form. If the content is an unicode string, encode it.
  1781. * @param {ZipObject} file the file to use.
  1782. * @return {String|ArrayBuffer|Uint8Array|Buffer} the data.
  1783. */
  1784. var getBinaryData = function(file) {
  1785. var result = getRawData(file),
  1786. type = utils.getTypeOf(result);
  1787. if (type === "string") {
  1788. if (!file.options.binary) {
  1789. // unicode text !
  1790. // unicode string => binary string is a painful process, check if we can avoid it.
  1791. if (support.nodebuffer) {
  1792. return nodeBuffer(result, "utf-8");
  1793. }
  1794. }
  1795. return file.asBinary();
  1796. }
  1797. return result;
  1798. };
  1799. /**
  1800. * Transform this._data into a string.
  1801. * @param {function} filter a function String -> String, applied if not null on the result.
  1802. * @return {String} the string representing this._data.
  1803. */
  1804. var dataToString = function(asUTF8) {
  1805. var result = getRawData(this);
  1806. if (result === null || typeof result === "undefined") {
  1807. return "";
  1808. }
  1809. // if the data is a base64 string, we decode it before checking the encoding !
  1810. if (this.options.base64) {
  1811. result = base64.decode(result);
  1812. }
  1813. if (asUTF8 && this.options.binary) {
  1814. // JSZip.prototype.utf8decode supports arrays as input
  1815. // skip to array => string step, utf8decode will do it.
  1816. result = out.utf8decode(result);
  1817. }
  1818. else {
  1819. // no utf8 transformation, do the array => string step.
  1820. result = utils.transformTo("string", result);
  1821. }
  1822. if (!asUTF8 && !this.options.binary) {
  1823. result = utils.transformTo("string", out.utf8encode(result));
  1824. }
  1825. return result;
  1826. };
  1827. /**
  1828. * A simple object representing a file in the zip file.
  1829. * @constructor
  1830. * @param {string} name the name of the file
  1831. * @param {String|ArrayBuffer|Uint8Array|Buffer} data the data
  1832. * @param {Object} options the options of the file
  1833. */
  1834. var ZipObject = function(name, data, options) {
  1835. this.name = name;
  1836. this.dir = options.dir;
  1837. this.date = options.date;
  1838. this.comment = options.comment;
  1839. this._data = data;
  1840. this.options = options;
  1841. /*
  1842. * This object contains initial values for dir and date.
  1843. * With them, we can check if the user changed the deprecated metadata in
  1844. * `ZipObject#options` or not.
  1845. */
  1846. this._initialMetadata = {
  1847. dir : options.dir,
  1848. date : options.date
  1849. };
  1850. };
  1851. ZipObject.prototype = {
  1852. /**
  1853. * Return the content as UTF8 string.
  1854. * @return {string} the UTF8 string.
  1855. */
  1856. asText: function() {
  1857. return dataToString.call(this, true);
  1858. },
  1859. /**
  1860. * Returns the binary content.
  1861. * @return {string} the content as binary.
  1862. */
  1863. asBinary: function() {
  1864. return dataToString.call(this, false);
  1865. },
  1866. /**
  1867. * Returns the content as a nodejs Buffer.
  1868. * @return {Buffer} the content as a Buffer.
  1869. */
  1870. asNodeBuffer: function() {
  1871. var result = getBinaryData(this);
  1872. return utils.transformTo("nodebuffer", result);
  1873. },
  1874. /**
  1875. * Returns the content as an Uint8Array.
  1876. * @return {Uint8Array} the content as an Uint8Array.
  1877. */
  1878. asUint8Array: function() {
  1879. var result = getBinaryData(this);
  1880. return utils.transformTo("uint8array", result);
  1881. },
  1882. /**
  1883. * Returns the content as an ArrayBuffer.
  1884. * @return {ArrayBuffer} the content as an ArrayBufer.
  1885. */
  1886. asArrayBuffer: function() {
  1887. return this.asUint8Array().buffer;
  1888. }
  1889. };
  1890. /**
  1891. * Transform an integer into a string in hexadecimal.
  1892. * @private
  1893. * @param {number} dec the number to convert.
  1894. * @param {number} bytes the number of bytes to generate.
  1895. * @returns {string} the result.
  1896. */
  1897. var decToHex = function(dec, bytes) {
  1898. var hex = "",
  1899. i;
  1900. for (i = 0; i < bytes; i++) {
  1901. hex += String.fromCharCode(dec & 0xff);
  1902. dec = dec >>> 8;
  1903. }
  1904. return hex;
  1905. };
  1906. /**
  1907. * Merge the objects passed as parameters into a new one.
  1908. * @private
  1909. * @param {...Object} var_args All objects to merge.
  1910. * @return {Object} a new object with the data of the others.
  1911. */
  1912. var extend = function() {
  1913. var result = {}, i, attr;
  1914. for (i = 0; i < arguments.length; i++) { // arguments is not enumerable in some browsers
  1915. for (attr in arguments[i]) {
  1916. if (arguments[i].hasOwnProperty(attr) && typeof result[attr] === "undefined") {
  1917. result[attr] = arguments[i][attr];
  1918. }
  1919. }
  1920. }
  1921. return result;
  1922. };
  1923. /**
  1924. * Transforms the (incomplete) options from the user into the complete
  1925. * set of options to create a file.
  1926. * @private
  1927. * @param {Object} o the options from the user.
  1928. * @return {Object} the complete set of options.
  1929. */
  1930. var prepareFileAttrs = function(o) {
  1931. o = o || {};
  1932. if (o.base64 === true && (o.binary === null || o.binary === undefined)) {
  1933. o.binary = true;
  1934. }
  1935. o = extend(o, defaults);
  1936. o.date = o.date || new Date();
  1937. if (o.compression !== null) o.compression = o.compression.toUpperCase();
  1938. return o;
  1939. };
  1940. /**
  1941. * Add a file in the current folder.
  1942. * @private
  1943. * @param {string} name the name of the file
  1944. * @param {String|ArrayBuffer|Uint8Array|Buffer} data the data of the file
  1945. * @param {Object} o the options of the file
  1946. * @return {Object} the new file.
  1947. */
  1948. var fileAdd = function(name, data, o) {
  1949. // be sure sub folders exist
  1950. var dataType = utils.getTypeOf(data),
  1951. parent;
  1952. o = prepareFileAttrs(o);
  1953. if (o.createFolders && (parent = parentFolder(name))) {
  1954. folderAdd.call(this, parent, true);
  1955. }
  1956. if (o.dir || data === null || typeof data === "undefined") {
  1957. o.base64 = false;
  1958. o.binary = false;
  1959. data = null;
  1960. }
  1961. else if (dataType === "string") {
  1962. if (o.binary && !o.base64) {
  1963. // optimizedBinaryString == true means that the file has already been filtered with a 0xFF mask
  1964. if (o.optimizedBinaryString !== true) {
  1965. // this is a string, not in a base64 format.
  1966. // Be sure that this is a correct "binary string"
  1967. data = utils.string2binary(data);
  1968. }
  1969. }
  1970. }
  1971. else { // arraybuffer, uint8array, ...
  1972. o.base64 = false;
  1973. o.binary = true;
  1974. if (!dataType && !(data instanceof CompressedObject)) {
  1975. throw new Error("The data of '" + name + "' is in an unsupported format !");
  1976. }
  1977. // special case : it's way easier to work with Uint8Array than with ArrayBuffer
  1978. if (dataType === "arraybuffer") {
  1979. data = utils.transformTo("uint8array", data);
  1980. }
  1981. }
  1982. var object = new ZipObject(name, data, o);
  1983. this.files[name] = object;
  1984. return object;
  1985. };
  1986. /**
  1987. * Find the parent folder of the path.
  1988. * @private
  1989. * @param {string} path the path to use
  1990. * @return {string} the parent folder, or ""
  1991. */
  1992. var parentFolder = function (path) {
  1993. if (path.slice(-1) == '/') {
  1994. path = path.substring(0, path.length - 1);
  1995. }
  1996. var lastSlash = path.lastIndexOf('/');
  1997. return (lastSlash > 0) ? path.substring(0, lastSlash) : "";
  1998. };
  1999. /**
  2000. * Add a (sub) folder in the current folder.
  2001. * @private
  2002. * @param {string} name the folder's name
  2003. * @param {boolean=} [createFolders] If true, automatically create sub
  2004. * folders. Defaults to false.
  2005. * @return {Object} the new folder.
  2006. */
  2007. var folderAdd = function(name, createFolders) {
  2008. // Check the name ends with a /
  2009. if (name.slice(-1) != "/") {
  2010. name += "/"; // IE doesn't like substr(-1)
  2011. }
  2012. createFolders = (typeof createFolders !== 'undefined') ? createFolders : false;
  2013. // Does this folder already exist?
  2014. if (!this.files[name]) {
  2015. fileAdd.call(this, name, null, {
  2016. dir: true,
  2017. createFolders: createFolders
  2018. });
  2019. }
  2020. return this.files[name];
  2021. };
  2022. /**
  2023. * Generate a JSZip.CompressedObject for a given zipOject.
  2024. * @param {ZipObject} file the object to read.
  2025. * @param {JSZip.compression} compression the compression to use.
  2026. * @return {JSZip.CompressedObject} the compressed result.
  2027. */
  2028. var generateCompressedObjectFrom = function(file, compression) {
  2029. var result = new CompressedObject(),
  2030. content;
  2031. // the data has not been decompressed, we might reuse things !
  2032. if (file._data instanceof CompressedObject) {
  2033. result.uncompressedSize = file._data.uncompressedSize;
  2034. result.crc32 = file._data.crc32;
  2035. if (result.uncompressedSize === 0 || file.dir) {
  2036. compression = compressions['STORE'];
  2037. result.compressedContent = "";
  2038. result.crc32 = 0;
  2039. }
  2040. else if (file._data.compressionMethod === compression.magic) {
  2041. result.compressedContent = file._data.getCompressedContent();
  2042. }
  2043. else {
  2044. content = file._data.getContent();
  2045. // need to decompress / recompress
  2046. result.compressedContent = compression.compress(utils.transformTo(compression.compressInputType, content));
  2047. }
  2048. }
  2049. else {
  2050. // have uncompressed data
  2051. content = getBinaryData(file);
  2052. if (!content || content.length === 0 || file.dir) {
  2053. compression = compressions['STORE'];
  2054. content = "";
  2055. }
  2056. result.uncompressedSize = content.length;
  2057. result.crc32 = crc32(content);
  2058. result.compressedContent = compression.compress(utils.transformTo(compression.compressInputType, content));
  2059. }
  2060. result.compressedSize = result.compressedContent.length;
  2061. result.compressionMethod = compression.magic;
  2062. return result;
  2063. };
  2064. /**
  2065. * Generate the various parts used in the construction of the final zip file.
  2066. * @param {string} name the file name.
  2067. * @param {ZipObject} file the file content.
  2068. * @param {JSZip.CompressedObject} compressedObject the compressed object.
  2069. * @param {number} offset the current offset from the start of the zip file.
  2070. * @return {object} the zip parts.
  2071. */
  2072. var generateZipParts = function(name, file, compressedObject, offset) {
  2073. var data = compressedObject.compressedContent,
  2074. utfEncodedFileName = utils.transformTo("string", utf8.utf8encode(file.name)),
  2075. comment = file.comment || "",
  2076. utfEncodedComment = utils.transformTo("string", utf8.utf8encode(comment)),
  2077. useUTF8ForFileName = utfEncodedFileName.length !== file.name.length,
  2078. useUTF8ForComment = utfEncodedComment.length !== comment.length,
  2079. o = file.options,
  2080. dosTime,
  2081. dosDate,
  2082. extraFields = "",
  2083. unicodePathExtraField = "",
  2084. unicodeCommentExtraField = "",
  2085. dir, date;
  2086. // handle the deprecated options.dir
  2087. if (file._initialMetadata.dir !== file.dir) {
  2088. dir = file.dir;
  2089. } else {
  2090. dir = o.dir;
  2091. }
  2092. // handle the deprecated options.date
  2093. if(file._initialMetadata.date !== file.date) {
  2094. date = file.date;
  2095. } else {
  2096. date = o.date;
  2097. }
  2098. dosTime = date.getHours();
  2099. dosTime = dosTime << 6;
  2100. dosTime = dosTime | date.getMinutes();
  2101. dosTime = dosTime << 5;
  2102. dosTime = dosTime | date.getSeconds() / 2;
  2103. dosDate = date.getFullYear() - 1980;
  2104. dosDate = dosDate << 4;
  2105. dosDate = dosDate | (date.getMonth() + 1);
  2106. dosDate = dosDate << 5;
  2107. dosDate = dosDate | date.getDate();
  2108. if (useUTF8ForFileName) {
  2109. // set the unicode path extra field. unzip needs at least one extra
  2110. // field to correctly handle unicode path, so using the path is as good
  2111. // as any other information. This could improve the situation with
  2112. // other archive managers too.
  2113. // This field is usually used without the utf8 flag, with a non
  2114. // unicode path in the header (winrar, winzip). This helps (a bit)
  2115. // with the messy Windows' default compressed folders feature but
  2116. // breaks on p7zip which doesn't seek the unicode path extra field.
  2117. // So for now, UTF-8 everywhere !
  2118. unicodePathExtraField =
  2119. // Version
  2120. decToHex(1, 1) +
  2121. // NameCRC32
  2122. decToHex(crc32(utfEncodedFileName), 4) +
  2123. // UnicodeName
  2124. utfEncodedFileName;
  2125. extraFields +=
  2126. // Info-ZIP Unicode Path Extra Field
  2127. "\x75\x70" +
  2128. // size
  2129. decToHex(unicodePathExtraField.length, 2) +
  2130. // content
  2131. unicodePathExtraField;
  2132. }
  2133. if(useUTF8ForComment) {
  2134. unicodeCommentExtraField =
  2135. // Version
  2136. decToHex(1, 1) +
  2137. // CommentCRC32
  2138. decToHex(this.crc32(utfEncodedComment), 4) +
  2139. // UnicodeName
  2140. utfEncodedComment;
  2141. extraFields +=
  2142. // Info-ZIP Unicode Path Extra Field
  2143. "\x75\x63" +
  2144. // size
  2145. decToHex(unicodeCommentExtraField.length, 2) +
  2146. // content
  2147. unicodeCommentExtraField;
  2148. }
  2149. var header = "";
  2150. // version needed to extract
  2151. header += "\x0A\x00";
  2152. // general purpose bit flag
  2153. // set bit 11 if utf8
  2154. header += (useUTF8ForFileName || useUTF8ForComment) ? "\x00\x08" : "\x00\x00";
  2155. // compression method
  2156. header += compressedObject.compressionMethod;
  2157. // last mod file time
  2158. header += decToHex(dosTime, 2);
  2159. // last mod file date
  2160. header += decToHex(dosDate, 2);
  2161. // crc-32
  2162. header += decToHex(compressedObject.crc32, 4);
  2163. // compressed size
  2164. header += decToHex(compressedObject.compressedSize, 4);
  2165. // uncompressed size
  2166. header += decToHex(compressedObject.uncompressedSize, 4);
  2167. // file name length
  2168. header += decToHex(utfEncodedFileName.length, 2);
  2169. // extra field length
  2170. header += decToHex(extraFields.length, 2);
  2171. var fileRecord = signature.LOCAL_FILE_HEADER + header + utfEncodedFileName + extraFields;
  2172. var dirRecord = signature.CENTRAL_FILE_HEADER +
  2173. // version made by (00: DOS)
  2174. "\x14\x00" +
  2175. // file header (common to file and central directory)
  2176. header +
  2177. // file comment length
  2178. decToHex(utfEncodedComment.length, 2) +
  2179. // disk number start
  2180. "\x00\x00" +
  2181. // internal file attributes TODO
  2182. "\x00\x00" +
  2183. // external file attributes
  2184. (dir === true ? "\x10\x00\x00\x00" : "\x00\x00\x00\x00") +
  2185. // relative offset of local header
  2186. decToHex(offset, 4) +
  2187. // file name
  2188. utfEncodedFileName +
  2189. // extra field
  2190. extraFields +
  2191. // file comment
  2192. utfEncodedComment;
  2193. return {
  2194. fileRecord: fileRecord,
  2195. dirRecord: dirRecord,
  2196. compressedObject: compressedObject
  2197. };
  2198. };
  2199. // return the actual prototype of JSZip
  2200. var out = {
  2201. /**
  2202. * Read an existing zip and merge the data in the current JSZip object.
  2203. * The implementation is in jszip-load.js, don't forget to include it.
  2204. * @param {String|ArrayBuffer|Uint8Array|Buffer} stream The stream to load
  2205. * @param {Object} options Options for loading the stream.
  2206. * options.base64 : is the stream in base64 ? default : false
  2207. * @return {JSZip} the current JSZip object
  2208. */
  2209. load: function(stream, options) {
  2210. throw new Error("Load method is not defined. Is the file jszip-load.js included ?");
  2211. },
  2212. /**
  2213. * Filter nested files/folders with the specified function.
  2214. * @param {Function} search the predicate to use :
  2215. * function (relativePath, file) {...}
  2216. * It takes 2 arguments : the relative path and the file.
  2217. * @return {Array} An array of matching elements.
  2218. */
  2219. filter: function(search) {
  2220. var result = [],
  2221. filename, relativePath, file, fileClone;
  2222. for (filename in this.files) {
  2223. if (!this.files.hasOwnProperty(filename)) {
  2224. continue;
  2225. }
  2226. file = this.files[filename];
  2227. // return a new object, don't let the user mess with our internal objects :)
  2228. fileClone = new ZipObject(file.name, file._data, extend(file.options));
  2229. relativePath = filename.slice(this.root.length, filename.length);
  2230. if (filename.slice(0, this.root.length) === this.root && // the file is in the current root
  2231. search(relativePath, fileClone)) { // and the file matches the function
  2232. result.push(fileClone);
  2233. }
  2234. }
  2235. return result;
  2236. },
  2237. /**
  2238. * Add a file to the zip file, or search a file.
  2239. * @param {string|RegExp} name The name of the file to add (if data is defined),
  2240. * the name of the file to find (if no data) or a regex to match files.
  2241. * @param {String|ArrayBuffer|Uint8Array|Buffer} data The file data, either raw or base64 encoded
  2242. * @param {Object} o File options
  2243. * @return {JSZip|Object|Array} this JSZip object (when adding a file),
  2244. * a file (when searching by string) or an array of files (when searching by regex).
  2245. */
  2246. file: function(name, data, o) {
  2247. if (arguments.length === 1) {
  2248. if (utils.isRegExp(name)) {
  2249. var regexp = name;
  2250. return this.filter(function(relativePath, file) {
  2251. return !file.dir && regexp.test(relativePath);
  2252. });
  2253. }
  2254. else { // text
  2255. return this.filter(function(relativePath, file) {
  2256. return !file.dir && relativePath === name;
  2257. })[0] || null;
  2258. }
  2259. }
  2260. else { // more than one argument : we have data !
  2261. name = this.root + name;
  2262. fileAdd.call(this, name, data, o);
  2263. }
  2264. return this;
  2265. },
  2266. /**
  2267. * Add a directory to the zip file, or search.
  2268. * @param {String|RegExp} arg The name of the directory to add, or a regex to search folders.
  2269. * @return {JSZip} an object with the new directory as the root, or an array containing matching folders.
  2270. */
  2271. folder: function(arg) {
  2272. if (!arg) {
  2273. return this;
  2274. }
  2275. if (utils.isRegExp(arg)) {
  2276. return this.filter(function(relativePath, file) {
  2277. return file.dir && arg.test(relativePath);
  2278. });
  2279. }
  2280. // else, name is a new folder
  2281. var name = this.root + arg;
  2282. var newFolder = folderAdd.call(this, name);
  2283. // Allow chaining by returning a new object with this folder as the root
  2284. var ret = this.clone();
  2285. ret.root = newFolder.name;
  2286. return ret;
  2287. },
  2288. /**
  2289. * Delete a file, or a directory and all sub-files, from the zip
  2290. * @param {string} name the name of the file to delete
  2291. * @return {JSZip} this JSZip object
  2292. */
  2293. remove: function(name) {
  2294. name = this.root + name;
  2295. var file = this.files[name];
  2296. if (!file) {
  2297. // Look for any folders
  2298. if (name.slice(-1) != "/") {
  2299. name += "/";
  2300. }
  2301. file = this.files[name];
  2302. }
  2303. if (file && !file.dir) {
  2304. // file
  2305. delete this.files[name];
  2306. } else {
  2307. // maybe a folder, delete recursively
  2308. var kids = this.filter(function(relativePath, file) {
  2309. return file.name.slice(0, name.length) === name;
  2310. });
  2311. for (var i = 0; i < kids.length; i++) {
  2312. delete this.files[kids[i].name];
  2313. }
  2314. }
  2315. return this;
  2316. },
  2317. /**
  2318. * Generate the complete zip file
  2319. * @param {Object} options the options to generate the zip file :
  2320. * - base64, (deprecated, use type instead) true to generate base64.
  2321. * - compression, "STORE" by default.
  2322. * - type, "base64" by default. Values are : string, base64, uint8array, arraybuffer, blob.
  2323. * @return {String|Uint8Array|ArrayBuffer|Buffer|Blob} the zip file
  2324. */
  2325. generate: function(options) {
  2326. options = extend(options || {}, {
  2327. base64: true,
  2328. compression: "STORE",
  2329. type: "base64",
  2330. comment: null
  2331. });
  2332. utils.checkSupport(options.type);
  2333. var zipData = [],
  2334. localDirLength = 0,
  2335. centralDirLength = 0,
  2336. writer, i,
  2337. utfEncodedComment = utils.transformTo("string", this.utf8encode(options.comment || this.comment || ""));
  2338. // first, generate all the zip parts.
  2339. for (var name in this.files) {
  2340. if (!this.files.hasOwnProperty(name)) {
  2341. continue;
  2342. }
  2343. var file = this.files[name];
  2344. var compressionName = file.options.compression || options.compression.toUpperCase();
  2345. var compression = compressions[compressionName];
  2346. if (!compression) {
  2347. throw new Error(compressionName + " is not a valid compression method !");
  2348. }
  2349. var compressedObject = generateCompressedObjectFrom.call(this, file, compression);
  2350. var zipPart = generateZipParts.call(this, name, file, compressedObject, localDirLength);
  2351. localDirLength += zipPart.fileRecord.length + compressedObject.compressedSize;
  2352. centralDirLength += zipPart.dirRecord.length;
  2353. zipData.push(zipPart);
  2354. }
  2355. var dirEnd = "";
  2356. // end of central dir signature
  2357. dirEnd = signature.CENTRAL_DIRECTORY_END +
  2358. // number of this disk
  2359. "\x00\x00" +
  2360. // number of the disk with the start of the central directory
  2361. "\x00\x00" +
  2362. // total number of entries in the central directory on this disk
  2363. decToHex(zipData.length, 2) +
  2364. // total number of entries in the central directory
  2365. decToHex(zipData.length, 2) +
  2366. // size of the central directory 4 bytes
  2367. decToHex(centralDirLength, 4) +
  2368. // offset of start of central directory with respect to the starting disk number
  2369. decToHex(localDirLength, 4) +
  2370. // .ZIP file comment length
  2371. decToHex(utfEncodedComment.length, 2) +
  2372. // .ZIP file comment
  2373. utfEncodedComment;
  2374. // we have all the parts (and the total length)
  2375. // time to create a writer !
  2376. var typeName = options.type.toLowerCase();
  2377. if(typeName==="uint8array"||typeName==="arraybuffer"||typeName==="blob"||typeName==="nodebuffer") {
  2378. writer = new Uint8ArrayWriter(localDirLength + centralDirLength + dirEnd.length);
  2379. }else{
  2380. writer = new StringWriter(localDirLength + centralDirLength + dirEnd.length);
  2381. }
  2382. for (i = 0; i < zipData.length; i++) {
  2383. writer.append(zipData[i].fileRecord);
  2384. writer.append(zipData[i].compressedObject.compressedContent);
  2385. }
  2386. for (i = 0; i < zipData.length; i++) {
  2387. writer.append(zipData[i].dirRecord);
  2388. }
  2389. writer.append(dirEnd);
  2390. var zip = writer.finalize();
  2391. switch(options.type.toLowerCase()) {
  2392. // case "zip is an Uint8Array"
  2393. case "uint8array" :
  2394. case "arraybuffer" :
  2395. case "nodebuffer" :
  2396. return utils.transformTo(options.type.toLowerCase(), zip);
  2397. case "blob" :
  2398. return utils.arrayBuffer2Blob(utils.transformTo("arraybuffer", zip));
  2399. // case "zip is a string"
  2400. case "base64" :
  2401. return (options.base64) ? base64.encode(zip) : zip;
  2402. default : // case "string" :
  2403. return zip;
  2404. }
  2405. },
  2406. /**
  2407. * @deprecated
  2408. * This method will be removed in a future version without replacement.
  2409. */
  2410. crc32: function (input, crc) {
  2411. return crc32(input, crc);
  2412. },
  2413. /**
  2414. * @deprecated
  2415. * This method will be removed in a future version without replacement.
  2416. */
  2417. utf8encode: function (string) {
  2418. return utils.transformTo("string", utf8.utf8encode(string));
  2419. },
  2420. /**
  2421. * @deprecated
  2422. * This method will be removed in a future version without replacement.
  2423. */
  2424. utf8decode: function (input) {
  2425. return utf8.utf8decode(input);
  2426. }
  2427. };
  2428. module.exports = out;
  2429. },{"./base64":1,"./compressedObject":2,"./compressions":3,"./crc32":4,"./defaults":6,"./nodeBuffer":11,"./signature":14,"./stringWriter":16,"./support":17,"./uint8ArrayWriter":19,"./utf8":20,"./utils":21}],14:[function(_dereq_,module,exports){
  2430. 'use strict';
  2431. exports.LOCAL_FILE_HEADER = "PK\x03\x04";
  2432. exports.CENTRAL_FILE_HEADER = "PK\x01\x02";
  2433. exports.CENTRAL_DIRECTORY_END = "PK\x05\x06";
  2434. exports.ZIP64_CENTRAL_DIRECTORY_LOCATOR = "PK\x06\x07";
  2435. exports.ZIP64_CENTRAL_DIRECTORY_END = "PK\x06\x06";
  2436. exports.DATA_DESCRIPTOR = "PK\x07\x08";
  2437. },{}],15:[function(_dereq_,module,exports){
  2438. 'use strict';
  2439. var DataReader = _dereq_('./dataReader');
  2440. var utils = _dereq_('./utils');
  2441. function StringReader(data, optimizedBinaryString) {
  2442. this.data = data;
  2443. if (!optimizedBinaryString) {
  2444. this.data = utils.string2binary(this.data);
  2445. }
  2446. this.length = this.data.length;
  2447. this.index = 0;
  2448. }
  2449. StringReader.prototype = new DataReader();
  2450. /**
  2451. * @see DataReader.byteAt
  2452. */
  2453. StringReader.prototype.byteAt = function(i) {
  2454. return this.data.charCodeAt(i);
  2455. };
  2456. /**
  2457. * @see DataReader.lastIndexOfSignature
  2458. */
  2459. StringReader.prototype.lastIndexOfSignature = function(sig) {
  2460. return this.data.lastIndexOf(sig);
  2461. };
  2462. /**
  2463. * @see DataReader.readData
  2464. */
  2465. StringReader.prototype.readData = function(size) {
  2466. this.checkOffset(size);
  2467. // this will work because the constructor applied the "& 0xff" mask.
  2468. var result = this.data.slice(this.index, this.index + size);
  2469. this.index += size;
  2470. return result;
  2471. };
  2472. module.exports = StringReader;
  2473. },{"./dataReader":5,"./utils":21}],16:[function(_dereq_,module,exports){
  2474. 'use strict';
  2475. var utils = _dereq_('./utils');
  2476. /**
  2477. * An object to write any content to a string.
  2478. * @constructor
  2479. */
  2480. var StringWriter = function() {
  2481. this.data = [];
  2482. };
  2483. StringWriter.prototype = {
  2484. /**
  2485. * Append any content to the current string.
  2486. * @param {Object} input the content to add.
  2487. */
  2488. append: function(input) {
  2489. input = utils.transformTo("string", input);
  2490. this.data.push(input);
  2491. },
  2492. /**
  2493. * Finalize the construction an return the result.
  2494. * @return {string} the generated string.
  2495. */
  2496. finalize: function() {
  2497. return this.data.join("");
  2498. }
  2499. };
  2500. module.exports = StringWriter;
  2501. },{"./utils":21}],17:[function(_dereq_,module,exports){
  2502. (function (Buffer){
  2503. 'use strict';
  2504. exports.base64 = true;
  2505. exports.array = true;
  2506. exports.string = true;
  2507. exports.arraybuffer = typeof ArrayBuffer !== "undefined" && typeof Uint8Array !== "undefined";
  2508. // contains true if JSZip can read/generate nodejs Buffer, false otherwise.
  2509. // Browserify will provide a Buffer implementation for browsers, which is
  2510. // an augmented Uint8Array (i.e., can be used as either Buffer or U8).
  2511. exports.nodebuffer = typeof Buffer !== "undefined";
  2512. // contains true if JSZip can read/generate Uint8Array, false otherwise.
  2513. exports.uint8array = typeof Uint8Array !== "undefined";
  2514. if (typeof ArrayBuffer === "undefined") {
  2515. exports.blob = false;
  2516. }
  2517. else {
  2518. var buffer = new ArrayBuffer(0);
  2519. try {
  2520. exports.blob = new Blob([buffer], {
  2521. type: "application/zip"
  2522. }).size === 0;
  2523. }
  2524. catch (e) {
  2525. try {
  2526. var Builder = window.BlobBuilder || window.WebKitBlobBuilder || window.MozBlobBuilder || window.MSBlobBuilder;
  2527. var builder = new Builder();
  2528. builder.append(buffer);
  2529. exports.blob = builder.getBlob('application/zip').size === 0;
  2530. }
  2531. catch (e) {
  2532. exports.blob = false;
  2533. }
  2534. }
  2535. }
  2536. }).call(this,(typeof Buffer !== "undefined" ? Buffer : undefined))
  2537. },{}],18:[function(_dereq_,module,exports){
  2538. 'use strict';
  2539. var DataReader = _dereq_('./dataReader');
  2540. function Uint8ArrayReader(data) {
  2541. if (data) {
  2542. this.data = data;
  2543. this.length = this.data.length;
  2544. this.index = 0;
  2545. }
  2546. }
  2547. Uint8ArrayReader.prototype = new DataReader();
  2548. /**
  2549. * @see DataReader.byteAt
  2550. */
  2551. Uint8ArrayReader.prototype.byteAt = function(i) {
  2552. return this.data[i];
  2553. };
  2554. /**
  2555. * @see DataReader.lastIndexOfSignature
  2556. */
  2557. Uint8ArrayReader.prototype.lastIndexOfSignature = function(sig) {
  2558. var sig0 = sig.charCodeAt(0),
  2559. sig1 = sig.charCodeAt(1),
  2560. sig2 = sig.charCodeAt(2),
  2561. sig3 = sig.charCodeAt(3);
  2562. for (var i = this.length - 4; i >= 0; --i) {
  2563. if (this.data[i] === sig0 && this.data[i + 1] === sig1 && this.data[i + 2] === sig2 && this.data[i + 3] === sig3) {
  2564. return i;
  2565. }
  2566. }
  2567. return -1;
  2568. };
  2569. /**
  2570. * @see DataReader.readData
  2571. */
  2572. Uint8ArrayReader.prototype.readData = function(size) {
  2573. this.checkOffset(size);
  2574. if(size === 0) {
  2575. // in IE10, when using subarray(idx, idx), we get the array [0x00] instead of [].
  2576. return new Uint8Array(0);
  2577. }
  2578. var result = this.data.subarray(this.index, this.index + size);
  2579. this.index += size;
  2580. return result;
  2581. };
  2582. module.exports = Uint8ArrayReader;
  2583. },{"./dataReader":5}],19:[function(_dereq_,module,exports){
  2584. 'use strict';
  2585. var utils = _dereq_('./utils');
  2586. /**
  2587. * An object to write any content to an Uint8Array.
  2588. * @constructor
  2589. * @param {number} length The length of the array.
  2590. */
  2591. var Uint8ArrayWriter = function(length) {
  2592. this.data = new Uint8Array(length);
  2593. this.index = 0;
  2594. };
  2595. Uint8ArrayWriter.prototype = {
  2596. /**
  2597. * Append any content to the current array.
  2598. * @param {Object} input the content to add.
  2599. */
  2600. append: function(input) {
  2601. if (input.length !== 0) {
  2602. // with an empty Uint8Array, Opera fails with a "Offset larger than array size"
  2603. input = utils.transformTo("uint8array", input);
  2604. this.data.set(input, this.index);
  2605. this.index += input.length;
  2606. }
  2607. },
  2608. /**
  2609. * Finalize the construction an return the result.
  2610. * @return {Uint8Array} the generated array.
  2611. */
  2612. finalize: function() {
  2613. return this.data;
  2614. }
  2615. };
  2616. module.exports = Uint8ArrayWriter;
  2617. },{"./utils":21}],20:[function(_dereq_,module,exports){
  2618. 'use strict';
  2619. var utils = _dereq_('./utils');
  2620. var support = _dereq_('./support');
  2621. var nodeBuffer = _dereq_('./nodeBuffer');
  2622. /**
  2623. * The following functions come from pako, from pako/lib/utils/strings
  2624. * released under the MIT license, see pako https://github.com/nodeca/pako/
  2625. */
  2626. // Table with utf8 lengths (calculated by first byte of sequence)
  2627. // Note, that 5 & 6-byte values and some 4-byte values can not be represented in JS,
  2628. // because max possible codepoint is 0x10ffff
  2629. var _utf8len = new Array(256);
  2630. for (var i=0; i<256; i++) {
  2631. _utf8len[i] = (i >= 252 ? 6 : i >= 248 ? 5 : i >= 240 ? 4 : i >= 224 ? 3 : i >= 192 ? 2 : 1);
  2632. }
  2633. _utf8len[254]=_utf8len[254]=1; // Invalid sequence start
  2634. // convert string to array (typed, when possible)
  2635. var string2buf = function (str) {
  2636. var buf, c, c2, m_pos, i, str_len = str.length, buf_len = 0;
  2637. // count binary size
  2638. for (m_pos = 0; m_pos < str_len; m_pos++) {
  2639. c = str.charCodeAt(m_pos);
  2640. if (((c & 0xfc00) === 0xd800) && (m_pos+1 < str_len)) {
  2641. c2 = str.charCodeAt(m_pos+1);
  2642. if ((c2 & 0xfc00) === 0xdc00) {
  2643. c = 0x10000 + ((c - 0xd800) << 10) + (c2 - 0xdc00);
  2644. m_pos++;
  2645. }
  2646. }
  2647. buf_len += (c < 0x80) ? 1 : ((c < 0x800) ? 2 : ((c < 0x10000) ? 3 : 4));
  2648. }
  2649. // allocate buffer
  2650. if (support.uint8array) {
  2651. buf = new Uint8Array(buf_len);
  2652. } else {
  2653. buf = new Array(buf_len);
  2654. }
  2655. // convert
  2656. for (i=0, m_pos = 0; i < buf_len; m_pos++) {
  2657. c = str.charCodeAt(m_pos);
  2658. if ((c & 0xfc00) === 0xd800 && (m_pos+1 < str_len)) {
  2659. c2 = str.charCodeAt(m_pos+1);
  2660. if ((c2 & 0xfc00) === 0xdc00) {
  2661. c = 0x10000 + ((c - 0xd800) << 10) + (c2 - 0xdc00);
  2662. m_pos++;
  2663. }
  2664. }
  2665. if (c < 0x80) {
  2666. /* one byte */
  2667. buf[i++] = c;
  2668. } else if (c < 0x800) {
  2669. /* two bytes */
  2670. buf[i++] = 0xC0 | (c >>> 6);
  2671. buf[i++] = 0x80 | (c & 0x3f);
  2672. } else if (c < 0x10000) {
  2673. /* three bytes */
  2674. buf[i++] = 0xE0 | (c >>> 12);
  2675. buf[i++] = 0x80 | ((c >>> 6) & 0x3f);
  2676. buf[i++] = 0x80 | (c & 0x3f);
  2677. } else {
  2678. /* four bytes */
  2679. buf[i++] = 0xf0 | (c >>> 18);
  2680. buf[i++] = 0x80 | ((c >>> 12) & 0x3f);
  2681. buf[i++] = 0x80 | ((c >>> 6) & 0x3f);
  2682. buf[i++] = 0x80 | (c & 0x3f);
  2683. }
  2684. }
  2685. return buf;
  2686. };
  2687. // Calculate max possible position in utf8 buffer,
  2688. // that will not break sequence. If that's not possible
  2689. // - (very small limits) return max size as is.
  2690. //
  2691. // buf[] - utf8 bytes array
  2692. // max - length limit (mandatory);
  2693. var utf8border = function(buf, max) {
  2694. var pos;
  2695. max = max || buf.length;
  2696. if (max > buf.length) { max = buf.length; }
  2697. // go back from last position, until start of sequence found
  2698. pos = max-1;
  2699. while (pos >= 0 && (buf[pos] & 0xC0) === 0x80) { pos--; }
  2700. // Fuckup - very small and broken sequence,
  2701. // return max, because we should return something anyway.
  2702. if (pos < 0) { return max; }
  2703. // If we came to start of buffer - that means vuffer is too small,
  2704. // return max too.
  2705. if (pos === 0) { return max; }
  2706. return (pos + _utf8len[buf[pos]] > max) ? pos : max;
  2707. };
  2708. // convert array to string
  2709. var buf2string = function (buf) {
  2710. var str, i, out, c, c_len;
  2711. var len = buf.length;
  2712. // Reserve max possible length (2 words per char)
  2713. // NB: by unknown reasons, Array is significantly faster for
  2714. // String.fromCharCode.apply than Uint16Array.
  2715. var utf16buf = new Array(len*2);
  2716. for (out=0, i=0; i<len;) {
  2717. c = buf[i++];
  2718. // quick process ascii
  2719. if (c < 0x80) { utf16buf[out++] = c; continue; }
  2720. c_len = _utf8len[c];
  2721. // skip 5 & 6 byte codes
  2722. if (c_len > 4) { utf16buf[out++] = 0xfffd; i += c_len-1; continue; }
  2723. // apply mask on first byte
  2724. c &= c_len === 2 ? 0x1f : c_len === 3 ? 0x0f : 0x07;
  2725. // join the rest
  2726. while (c_len > 1 && i < len) {
  2727. c = (c << 6) | (buf[i++] & 0x3f);
  2728. c_len--;
  2729. }
  2730. // terminated by end of string?
  2731. if (c_len > 1) { utf16buf[out++] = 0xfffd; continue; }
  2732. if (c < 0x10000) {
  2733. utf16buf[out++] = c;
  2734. } else {
  2735. c -= 0x10000;
  2736. utf16buf[out++] = 0xd800 | ((c >> 10) & 0x3ff);
  2737. utf16buf[out++] = 0xdc00 | (c & 0x3ff);
  2738. }
  2739. }
  2740. // shrinkBuf(utf16buf, out)
  2741. if (utf16buf.length !== out) {
  2742. if(utf16buf.subarray) {
  2743. utf16buf = utf16buf.subarray(0, out);
  2744. } else {
  2745. utf16buf.length = out;
  2746. }
  2747. }
  2748. // return String.fromCharCode.apply(null, utf16buf);
  2749. return utils.applyFromCharCode(utf16buf);
  2750. };
  2751. // That's all for the pako functions.
  2752. /**
  2753. * Transform a javascript string into an array (typed if possible) of bytes,
  2754. * UTF-8 encoded.
  2755. * @param {String} str the string to encode
  2756. * @return {Array|Uint8Array|Buffer} the UTF-8 encoded string.
  2757. */
  2758. exports.utf8encode = function utf8encode(str) {
  2759. if (support.nodebuffer) {
  2760. return nodeBuffer(str, "utf-8");
  2761. }
  2762. return string2buf(str);
  2763. };
  2764. /**
  2765. * Transform a bytes array (or a representation) representing an UTF-8 encoded
  2766. * string into a javascript string.
  2767. * @param {Array|Uint8Array|Buffer} buf the data de decode
  2768. * @return {String} the decoded string.
  2769. */
  2770. exports.utf8decode = function utf8decode(buf) {
  2771. if (support.nodebuffer) {
  2772. return utils.transformTo("nodebuffer", buf).toString("utf-8");
  2773. }
  2774. buf = utils.transformTo(support.uint8array ? "uint8array" : "array", buf);
  2775. // return buf2string(buf);
  2776. // Chrome prefers to work with "small" chunks of data
  2777. // for the method buf2string.
  2778. // Firefox and Chrome has their own shortcut, IE doesn't seem to really care.
  2779. var result = [], k = 0, len = buf.length, chunk = 65536;
  2780. while (k < len) {
  2781. var nextBoundary = utf8border(buf, Math.min(k + chunk, len));
  2782. if (support.uint8array) {
  2783. result.push(buf2string(buf.subarray(k, nextBoundary)));
  2784. } else {
  2785. result.push(buf2string(buf.slice(k, nextBoundary)));
  2786. }
  2787. k = nextBoundary;
  2788. }
  2789. return result.join("");
  2790. };
  2791. // vim: set shiftwidth=4 softtabstop=4:
  2792. },{"./nodeBuffer":11,"./support":17,"./utils":21}],21:[function(_dereq_,module,exports){
  2793. 'use strict';
  2794. var support = _dereq_('./support');
  2795. var compressions = _dereq_('./compressions');
  2796. var nodeBuffer = _dereq_('./nodeBuffer');
  2797. /**
  2798. * Convert a string to a "binary string" : a string containing only char codes between 0 and 255.
  2799. * @param {string} str the string to transform.
  2800. * @return {String} the binary string.
  2801. */
  2802. exports.string2binary = function(str) {
  2803. var result = "";
  2804. for (var i = 0; i < str.length; i++) {
  2805. result += String.fromCharCode(str.charCodeAt(i) & 0xff);
  2806. }
  2807. return result;
  2808. };
  2809. exports.arrayBuffer2Blob = function(buffer) {
  2810. exports.checkSupport("blob");
  2811. try {
  2812. // Blob constructor
  2813. return new Blob([buffer], {
  2814. type: "application/zip"
  2815. });
  2816. }
  2817. catch (e) {
  2818. try {
  2819. // deprecated, browser only, old way
  2820. var Builder = window.BlobBuilder || window.WebKitBlobBuilder || window.MozBlobBuilder || window.MSBlobBuilder;
  2821. var builder = new Builder();
  2822. builder.append(buffer);
  2823. return builder.getBlob('application/zip');
  2824. }
  2825. catch (e) {
  2826. // well, fuck ?!
  2827. throw new Error("Bug : can't construct the Blob.");
  2828. }
  2829. }
  2830. };
  2831. /**
  2832. * The identity function.
  2833. * @param {Object} input the input.
  2834. * @return {Object} the same input.
  2835. */
  2836. function identity(input) {
  2837. return input;
  2838. }
  2839. /**
  2840. * Fill in an array with a string.
  2841. * @param {String} str the string to use.
  2842. * @param {Array|ArrayBuffer|Uint8Array|Buffer} array the array to fill in (will be mutated).
  2843. * @return {Array|ArrayBuffer|Uint8Array|Buffer} the updated array.
  2844. */
  2845. function stringToArrayLike(str, array) {
  2846. for (var i = 0; i < str.length; ++i) {
  2847. array[i] = str.charCodeAt(i) & 0xFF;
  2848. }
  2849. return array;
  2850. }
  2851. /**
  2852. * Transform an array-like object to a string.
  2853. * @param {Array|ArrayBuffer|Uint8Array|Buffer} array the array to transform.
  2854. * @return {String} the result.
  2855. */
  2856. function arrayLikeToString(array) {
  2857. // Performances notes :
  2858. // --------------------
  2859. // String.fromCharCode.apply(null, array) is the fastest, see
  2860. // see http://jsperf.com/converting-a-uint8array-to-a-string/2
  2861. // but the stack is limited (and we can get huge arrays !).
  2862. //
  2863. // result += String.fromCharCode(array[i]); generate too many strings !
  2864. //
  2865. // This code is inspired by http://jsperf.com/arraybuffer-to-string-apply-performance/2
  2866. var chunk = 65536;
  2867. var result = [],
  2868. len = array.length,
  2869. type = exports.getTypeOf(array),
  2870. k = 0,
  2871. canUseApply = true;
  2872. try {
  2873. switch(type) {
  2874. case "uint8array":
  2875. String.fromCharCode.apply(null, new Uint8Array(0));
  2876. break;
  2877. case "nodebuffer":
  2878. String.fromCharCode.apply(null, nodeBuffer(0));
  2879. break;
  2880. }
  2881. } catch(e) {
  2882. canUseApply = false;
  2883. }
  2884. // no apply : slow and painful algorithm
  2885. // default browser on android 4.*
  2886. if (!canUseApply) {
  2887. var resultStr = "";
  2888. for(var i = 0; i < array.length;i++) {
  2889. resultStr += String.fromCharCode(array[i]);
  2890. }
  2891. return resultStr;
  2892. }
  2893. while (k < len && chunk > 1) {
  2894. try {
  2895. if (type === "array" || type === "nodebuffer") {
  2896. result.push(String.fromCharCode.apply(null, array.slice(k, Math.min(k + chunk, len))));
  2897. }
  2898. else {
  2899. result.push(String.fromCharCode.apply(null, array.subarray(k, Math.min(k + chunk, len))));
  2900. }
  2901. k += chunk;
  2902. }
  2903. catch (e) {
  2904. chunk = Math.floor(chunk / 2);
  2905. }
  2906. }
  2907. return result.join("");
  2908. }
  2909. exports.applyFromCharCode = arrayLikeToString;
  2910. /**
  2911. * Copy the data from an array-like to an other array-like.
  2912. * @param {Array|ArrayBuffer|Uint8Array|Buffer} arrayFrom the origin array.
  2913. * @param {Array|ArrayBuffer|Uint8Array|Buffer} arrayTo the destination array which will be mutated.
  2914. * @return {Array|ArrayBuffer|Uint8Array|Buffer} the updated destination array.
  2915. */
  2916. function arrayLikeToArrayLike(arrayFrom, arrayTo) {
  2917. for (var i = 0; i < arrayFrom.length; i++) {
  2918. arrayTo[i] = arrayFrom[i];
  2919. }
  2920. return arrayTo;
  2921. }
  2922. // a matrix containing functions to transform everything into everything.
  2923. var transform = {};
  2924. // string to ?
  2925. transform["string"] = {
  2926. "string": identity,
  2927. "array": function(input) {
  2928. return stringToArrayLike(input, new Array(input.length));
  2929. },
  2930. "arraybuffer": function(input) {
  2931. return transform["string"]["uint8array"](input).buffer;
  2932. },
  2933. "uint8array": function(input) {
  2934. return stringToArrayLike(input, new Uint8Array(input.length));
  2935. },
  2936. "nodebuffer": function(input) {
  2937. return stringToArrayLike(input, nodeBuffer(input.length));
  2938. }
  2939. };
  2940. // array to ?
  2941. transform["array"] = {
  2942. "string": arrayLikeToString,
  2943. "array": identity,
  2944. "arraybuffer": function(input) {
  2945. return (new Uint8Array(input)).buffer;
  2946. },
  2947. "uint8array": function(input) {
  2948. return new Uint8Array(input);
  2949. },
  2950. "nodebuffer": function(input) {
  2951. return nodeBuffer(input);
  2952. }
  2953. };
  2954. // arraybuffer to ?
  2955. transform["arraybuffer"] = {
  2956. "string": function(input) {
  2957. return arrayLikeToString(new Uint8Array(input));
  2958. },
  2959. "array": function(input) {
  2960. return arrayLikeToArrayLike(new Uint8Array(input), new Array(input.byteLength));
  2961. },
  2962. "arraybuffer": identity,
  2963. "uint8array": function(input) {
  2964. return new Uint8Array(input);
  2965. },
  2966. "nodebuffer": function(input) {
  2967. return nodeBuffer(new Uint8Array(input));
  2968. }
  2969. };
  2970. // uint8array to ?
  2971. transform["uint8array"] = {
  2972. "string": arrayLikeToString,
  2973. "array": function(input) {
  2974. return arrayLikeToArrayLike(input, new Array(input.length));
  2975. },
  2976. "arraybuffer": function(input) {
  2977. return input.buffer;
  2978. },
  2979. "uint8array": identity,
  2980. "nodebuffer": function(input) {
  2981. return nodeBuffer(input);
  2982. }
  2983. };
  2984. // nodebuffer to ?
  2985. transform["nodebuffer"] = {
  2986. "string": arrayLikeToString,
  2987. "array": function(input) {
  2988. return arrayLikeToArrayLike(input, new Array(input.length));
  2989. },
  2990. "arraybuffer": function(input) {
  2991. return transform["nodebuffer"]["uint8array"](input).buffer;
  2992. },
  2993. "uint8array": function(input) {
  2994. return arrayLikeToArrayLike(input, new Uint8Array(input.length));
  2995. },
  2996. "nodebuffer": identity
  2997. };
  2998. /**
  2999. * Transform an input into any type.
  3000. * The supported output type are : string, array, uint8array, arraybuffer, nodebuffer.
  3001. * If no output type is specified, the unmodified input will be returned.
  3002. * @param {String} outputType the output type.
  3003. * @param {String|Array|ArrayBuffer|Uint8Array|Buffer} input the input to convert.
  3004. * @throws {Error} an error if the browser doesn't support the requested output type.
  3005. */
  3006. exports.transformTo = function(outputType, input) {
  3007. if (!input) {
  3008. // undefined, null, etc
  3009. // an empty string won't harm.
  3010. input = "";
  3011. }
  3012. if (!outputType) {
  3013. return input;
  3014. }
  3015. exports.checkSupport(outputType);
  3016. var inputType = exports.getTypeOf(input);
  3017. var result = transform[inputType][outputType](input);
  3018. return result;
  3019. };
  3020. /**
  3021. * Return the type of the input.
  3022. * The type will be in a format valid for JSZip.utils.transformTo : string, array, uint8array, arraybuffer.
  3023. * @param {Object} input the input to identify.
  3024. * @return {String} the (lowercase) type of the input.
  3025. */
  3026. exports.getTypeOf = function(input) {
  3027. if (typeof input === "string") {
  3028. return "string";
  3029. }
  3030. if (Object.prototype.toString.call(input) === "[object Array]") {
  3031. return "array";
  3032. }
  3033. if (support.nodebuffer && nodeBuffer.test(input)) {
  3034. return "nodebuffer";
  3035. }
  3036. if (support.uint8array && input instanceof Uint8Array) {
  3037. return "uint8array";
  3038. }
  3039. if (support.arraybuffer && input instanceof ArrayBuffer) {
  3040. return "arraybuffer";
  3041. }
  3042. };
  3043. /**
  3044. * Throw an exception if the type is not supported.
  3045. * @param {String} type the type to check.
  3046. * @throws {Error} an error if the browser doesn't support the requested type.
  3047. */
  3048. exports.checkSupport = function(type) {
  3049. var supported = support[type.toLowerCase()];
  3050. if (!supported) {
  3051. throw new Error(type + " is not supported by this browser");
  3052. }
  3053. };
  3054. exports.MAX_VALUE_16BITS = 65535;
  3055. exports.MAX_VALUE_32BITS = -1; // well, "\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF" is parsed as -1
  3056. /**
  3057. * Prettify a string read as binary.
  3058. * @param {string} str the string to prettify.
  3059. * @return {string} a pretty string.
  3060. */
  3061. exports.pretty = function(str) {
  3062. var res = '',
  3063. code, i;
  3064. for (i = 0; i < (str || "").length; i++) {
  3065. code = str.charCodeAt(i);
  3066. res += '\\x' + (code < 16 ? "0" : "") + code.toString(16).toUpperCase();
  3067. }
  3068. return res;
  3069. };
  3070. /**
  3071. * Find a compression registered in JSZip.
  3072. * @param {string} compressionMethod the method magic to find.
  3073. * @return {Object|null} the JSZip compression object, null if none found.
  3074. */
  3075. exports.findCompression = function(compressionMethod) {
  3076. for (var method in compressions) {
  3077. if (!compressions.hasOwnProperty(method)) {
  3078. continue;
  3079. }
  3080. if (compressions[method].magic === compressionMethod) {
  3081. return compressions[method];
  3082. }
  3083. }
  3084. return null;
  3085. };
  3086. /**
  3087. * Cross-window, cross-Node-context regular expression detection
  3088. * @param {Object} object Anything
  3089. * @return {Boolean} true if the object is a regular expression,
  3090. * false otherwise
  3091. */
  3092. exports.isRegExp = function (object) {
  3093. return Object.prototype.toString.call(object) === "[object RegExp]";
  3094. };
  3095. },{"./compressions":3,"./nodeBuffer":11,"./support":17}],22:[function(_dereq_,module,exports){
  3096. 'use strict';
  3097. var StringReader = _dereq_('./stringReader');
  3098. var NodeBufferReader = _dereq_('./nodeBufferReader');
  3099. var Uint8ArrayReader = _dereq_('./uint8ArrayReader');
  3100. var utils = _dereq_('./utils');
  3101. var sig = _dereq_('./signature');
  3102. var ZipEntry = _dereq_('./zipEntry');
  3103. var support = _dereq_('./support');
  3104. var jszipProto = _dereq_('./object');
  3105. // class ZipEntries {{{
  3106. /**
  3107. * All the entries in the zip file.
  3108. * @constructor
  3109. * @param {String|ArrayBuffer|Uint8Array} data the binary stream to load.
  3110. * @param {Object} loadOptions Options for loading the stream.
  3111. */
  3112. function ZipEntries(data, loadOptions) {
  3113. this.files = [];
  3114. this.loadOptions = loadOptions;
  3115. if (data) {
  3116. this.load(data);
  3117. }
  3118. }
  3119. ZipEntries.prototype = {
  3120. /**
  3121. * Check that the reader is on the speficied signature.
  3122. * @param {string} expectedSignature the expected signature.
  3123. * @throws {Error} if it is an other signature.
  3124. */
  3125. checkSignature: function(expectedSignature) {
  3126. var signature = this.reader.readString(4);
  3127. if (signature !== expectedSignature) {
  3128. throw new Error("Corrupted zip or bug : unexpected signature " + "(" + utils.pretty(signature) + ", expected " + utils.pretty(expectedSignature) + ")");
  3129. }
  3130. },
  3131. /**
  3132. * Read the end of the central directory.
  3133. */
  3134. readBlockEndOfCentral: function() {
  3135. this.diskNumber = this.reader.readInt(2);
  3136. this.diskWithCentralDirStart = this.reader.readInt(2);
  3137. this.centralDirRecordsOnThisDisk = this.reader.readInt(2);
  3138. this.centralDirRecords = this.reader.readInt(2);
  3139. this.centralDirSize = this.reader.readInt(4);
  3140. this.centralDirOffset = this.reader.readInt(4);
  3141. this.zipCommentLength = this.reader.readInt(2);
  3142. // warning : the encoding depends of the system locale
  3143. // On a linux machine with LANG=en_US.utf8, this field is utf8 encoded.
  3144. // On a windows machine, this field is encoded with the localized windows code page.
  3145. this.zipComment = this.reader.readString(this.zipCommentLength);
  3146. // To get consistent behavior with the generation part, we will assume that
  3147. // this is utf8 encoded.
  3148. this.zipComment = jszipProto.utf8decode(this.zipComment);
  3149. },
  3150. /**
  3151. * Read the end of the Zip 64 central directory.
  3152. * Not merged with the method readEndOfCentral :
  3153. * The end of central can coexist with its Zip64 brother,
  3154. * I don't want to read the wrong number of bytes !
  3155. */
  3156. readBlockZip64EndOfCentral: function() {
  3157. this.zip64EndOfCentralSize = this.reader.readInt(8);
  3158. this.versionMadeBy = this.reader.readString(2);
  3159. this.versionNeeded = this.reader.readInt(2);
  3160. this.diskNumber = this.reader.readInt(4);
  3161. this.diskWithCentralDirStart = this.reader.readInt(4);
  3162. this.centralDirRecordsOnThisDisk = this.reader.readInt(8);
  3163. this.centralDirRecords = this.reader.readInt(8);
  3164. this.centralDirSize = this.reader.readInt(8);
  3165. this.centralDirOffset = this.reader.readInt(8);
  3166. this.zip64ExtensibleData = {};
  3167. var extraDataSize = this.zip64EndOfCentralSize - 44,
  3168. index = 0,
  3169. extraFieldId,
  3170. extraFieldLength,
  3171. extraFieldValue;
  3172. while (index < extraDataSize) {
  3173. extraFieldId = this.reader.readInt(2);
  3174. extraFieldLength = this.reader.readInt(4);
  3175. extraFieldValue = this.reader.readString(extraFieldLength);
  3176. this.zip64ExtensibleData[extraFieldId] = {
  3177. id: extraFieldId,
  3178. length: extraFieldLength,
  3179. value: extraFieldValue
  3180. };
  3181. }
  3182. },
  3183. /**
  3184. * Read the end of the Zip 64 central directory locator.
  3185. */
  3186. readBlockZip64EndOfCentralLocator: function() {
  3187. this.diskWithZip64CentralDirStart = this.reader.readInt(4);
  3188. this.relativeOffsetEndOfZip64CentralDir = this.reader.readInt(8);
  3189. this.disksCount = this.reader.readInt(4);
  3190. if (this.disksCount > 1) {
  3191. throw new Error("Multi-volumes zip are not supported");
  3192. }
  3193. },
  3194. /**
  3195. * Read the local files, based on the offset read in the central part.
  3196. */
  3197. readLocalFiles: function() {
  3198. var i, file;
  3199. for (i = 0; i < this.files.length; i++) {
  3200. file = this.files[i];
  3201. this.reader.setIndex(file.localHeaderOffset);
  3202. this.checkSignature(sig.LOCAL_FILE_HEADER);
  3203. file.readLocalPart(this.reader);
  3204. file.handleUTF8();
  3205. }
  3206. },
  3207. /**
  3208. * Read the central directory.
  3209. */
  3210. readCentralDir: function() {
  3211. var file;
  3212. this.reader.setIndex(this.centralDirOffset);
  3213. while (this.reader.readString(4) === sig.CENTRAL_FILE_HEADER) {
  3214. file = new ZipEntry({
  3215. zip64: this.zip64
  3216. }, this.loadOptions);
  3217. file.readCentralPart(this.reader);
  3218. this.files.push(file);
  3219. }
  3220. },
  3221. /**
  3222. * Read the end of central directory.
  3223. */
  3224. readEndOfCentral: function() {
  3225. var offset = this.reader.lastIndexOfSignature(sig.CENTRAL_DIRECTORY_END);
  3226. if (offset === -1) {
  3227. throw new Error("Corrupted zip : can't find end of central directory");
  3228. }
  3229. this.reader.setIndex(offset);
  3230. this.checkSignature(sig.CENTRAL_DIRECTORY_END);
  3231. this.readBlockEndOfCentral();
  3232. /* extract from the zip spec :
  3233. 4) If one of the fields in the end of central directory
  3234. record is too small to hold required data, the field
  3235. should be set to -1 (0xFFFF or 0xFFFFFFFF) and the
  3236. ZIP64 format record should be created.
  3237. 5) The end of central directory record and the
  3238. Zip64 end of central directory locator record must
  3239. reside on the same disk when splitting or spanning
  3240. an archive.
  3241. */
  3242. if (this.diskNumber === utils.MAX_VALUE_16BITS || this.diskWithCentralDirStart === utils.MAX_VALUE_16BITS || this.centralDirRecordsOnThisDisk === utils.MAX_VALUE_16BITS || this.centralDirRecords === utils.MAX_VALUE_16BITS || this.centralDirSize === utils.MAX_VALUE_32BITS || this.centralDirOffset === utils.MAX_VALUE_32BITS) {
  3243. this.zip64 = true;
  3244. /*
  3245. Warning : the zip64 extension is supported, but ONLY if the 64bits integer read from
  3246. the zip file can fit into a 32bits integer. This cannot be solved : Javascript represents
  3247. all numbers as 64-bit double precision IEEE 754 floating point numbers.
  3248. So, we have 53bits for integers and bitwise operations treat everything as 32bits.
  3249. see https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Operators/Bitwise_Operators
  3250. and http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-262.pdf section 8.5
  3251. */
  3252. // should look for a zip64 EOCD locator
  3253. offset = this.reader.lastIndexOfSignature(sig.ZIP64_CENTRAL_DIRECTORY_LOCATOR);
  3254. if (offset === -1) {
  3255. throw new Error("Corrupted zip : can't find the ZIP64 end of central directory locator");
  3256. }
  3257. this.reader.setIndex(offset);
  3258. this.checkSignature(sig.ZIP64_CENTRAL_DIRECTORY_LOCATOR);
  3259. this.readBlockZip64EndOfCentralLocator();
  3260. // now the zip64 EOCD record
  3261. this.reader.setIndex(this.relativeOffsetEndOfZip64CentralDir);
  3262. this.checkSignature(sig.ZIP64_CENTRAL_DIRECTORY_END);
  3263. this.readBlockZip64EndOfCentral();
  3264. }
  3265. },
  3266. prepareReader: function(data) {
  3267. var type = utils.getTypeOf(data);
  3268. if (type === "string" && !support.uint8array) {
  3269. this.reader = new StringReader(data, this.loadOptions.optimizedBinaryString);
  3270. }
  3271. else if (type === "nodebuffer") {
  3272. this.reader = new NodeBufferReader(data);
  3273. }
  3274. else {
  3275. this.reader = new Uint8ArrayReader(utils.transformTo("uint8array", data));
  3276. }
  3277. },
  3278. /**
  3279. * Read a zip file and create ZipEntries.
  3280. * @param {String|ArrayBuffer|Uint8Array|Buffer} data the binary string representing a zip file.
  3281. */
  3282. load: function(data) {
  3283. this.prepareReader(data);
  3284. this.readEndOfCentral();
  3285. this.readCentralDir();
  3286. this.readLocalFiles();
  3287. }
  3288. };
  3289. // }}} end of ZipEntries
  3290. module.exports = ZipEntries;
  3291. },{"./nodeBufferReader":12,"./object":13,"./signature":14,"./stringReader":15,"./support":17,"./uint8ArrayReader":18,"./utils":21,"./zipEntry":23}],23:[function(_dereq_,module,exports){
  3292. 'use strict';
  3293. var StringReader = _dereq_('./stringReader');
  3294. var utils = _dereq_('./utils');
  3295. var CompressedObject = _dereq_('./compressedObject');
  3296. var jszipProto = _dereq_('./object');
  3297. // class ZipEntry {{{
  3298. /**
  3299. * An entry in the zip file.
  3300. * @constructor
  3301. * @param {Object} options Options of the current file.
  3302. * @param {Object} loadOptions Options for loading the stream.
  3303. */
  3304. function ZipEntry(options, loadOptions) {
  3305. this.options = options;
  3306. this.loadOptions = loadOptions;
  3307. }
  3308. ZipEntry.prototype = {
  3309. /**
  3310. * say if the file is encrypted.
  3311. * @return {boolean} true if the file is encrypted, false otherwise.
  3312. */
  3313. isEncrypted: function() {
  3314. // bit 1 is set
  3315. return (this.bitFlag & 0x0001) === 0x0001;
  3316. },
  3317. /**
  3318. * say if the file has utf-8 filename/comment.
  3319. * @return {boolean} true if the filename/comment is in utf-8, false otherwise.
  3320. */
  3321. useUTF8: function() {
  3322. // bit 11 is set
  3323. return (this.bitFlag & 0x0800) === 0x0800;
  3324. },
  3325. /**
  3326. * Prepare the function used to generate the compressed content from this ZipFile.
  3327. * @param {DataReader} reader the reader to use.
  3328. * @param {number} from the offset from where we should read the data.
  3329. * @param {number} length the length of the data to read.
  3330. * @return {Function} the callback to get the compressed content (the type depends of the DataReader class).
  3331. */
  3332. prepareCompressedContent: function(reader, from, length) {
  3333. return function() {
  3334. var previousIndex = reader.index;
  3335. reader.setIndex(from);
  3336. var compressedFileData = reader.readData(length);
  3337. reader.setIndex(previousIndex);
  3338. return compressedFileData;
  3339. };
  3340. },
  3341. /**
  3342. * Prepare the function used to generate the uncompressed content from this ZipFile.
  3343. * @param {DataReader} reader the reader to use.
  3344. * @param {number} from the offset from where we should read the data.
  3345. * @param {number} length the length of the data to read.
  3346. * @param {JSZip.compression} compression the compression used on this file.
  3347. * @param {number} uncompressedSize the uncompressed size to expect.
  3348. * @return {Function} the callback to get the uncompressed content (the type depends of the DataReader class).
  3349. */
  3350. prepareContent: function(reader, from, length, compression, uncompressedSize) {
  3351. return function() {
  3352. var compressedFileData = utils.transformTo(compression.uncompressInputType, this.getCompressedContent());
  3353. var uncompressedFileData = compression.uncompress(compressedFileData);
  3354. if (uncompressedFileData.length !== uncompressedSize) {
  3355. throw new Error("Bug : uncompressed data size mismatch");
  3356. }
  3357. return uncompressedFileData;
  3358. };
  3359. },
  3360. /**
  3361. * Read the local part of a zip file and add the info in this object.
  3362. * @param {DataReader} reader the reader to use.
  3363. */
  3364. readLocalPart: function(reader) {
  3365. var compression, localExtraFieldsLength;
  3366. // we already know everything from the central dir !
  3367. // If the central dir data are false, we are doomed.
  3368. // On the bright side, the local part is scary : zip64, data descriptors, both, etc.
  3369. // The less data we get here, the more reliable this should be.
  3370. // Let's skip the whole header and dash to the data !
  3371. reader.skip(22);
  3372. // in some zip created on windows, the filename stored in the central dir contains \ instead of /.
  3373. // Strangely, the filename here is OK.
  3374. // I would love to treat these zip files as corrupted (see http://www.info-zip.org/FAQ.html#backslashes
  3375. // or APPNOTE#4.4.17.1, "All slashes MUST be forward slashes '/'") but there are a lot of bad zip generators...
  3376. // Search "unzip mismatching "local" filename continuing with "central" filename version" on
  3377. // the internet.
  3378. //
  3379. // I think I see the logic here : the central directory is used to display
  3380. // content and the local directory is used to extract the files. Mixing / and \
  3381. // may be used to display \ to windows users and use / when extracting the files.
  3382. // Unfortunately, this lead also to some issues : http://seclists.org/fulldisclosure/2009/Sep/394
  3383. this.fileNameLength = reader.readInt(2);
  3384. localExtraFieldsLength = reader.readInt(2); // can't be sure this will be the same as the central dir
  3385. this.fileName = reader.readString(this.fileNameLength);
  3386. reader.skip(localExtraFieldsLength);
  3387. if (this.compressedSize == -1 || this.uncompressedSize == -1) {
  3388. throw new Error("Bug or corrupted zip : didn't get enough informations from the central directory " + "(compressedSize == -1 || uncompressedSize == -1)");
  3389. }
  3390. compression = utils.findCompression(this.compressionMethod);
  3391. if (compression === null) { // no compression found
  3392. throw new Error("Corrupted zip : compression " + utils.pretty(this.compressionMethod) + " unknown (inner file : " + this.fileName + ")");
  3393. }
  3394. this.decompressed = new CompressedObject();
  3395. this.decompressed.compressedSize = this.compressedSize;
  3396. this.decompressed.uncompressedSize = this.uncompressedSize;
  3397. this.decompressed.crc32 = this.crc32;
  3398. this.decompressed.compressionMethod = this.compressionMethod;
  3399. this.decompressed.getCompressedContent = this.prepareCompressedContent(reader, reader.index, this.compressedSize, compression);
  3400. this.decompressed.getContent = this.prepareContent(reader, reader.index, this.compressedSize, compression, this.uncompressedSize);
  3401. // we need to compute the crc32...
  3402. if (this.loadOptions.checkCRC32) {
  3403. this.decompressed = utils.transformTo("string", this.decompressed.getContent());
  3404. if (jszipProto.crc32(this.decompressed) !== this.crc32) {
  3405. throw new Error("Corrupted zip : CRC32 mismatch");
  3406. }
  3407. }
  3408. },
  3409. /**
  3410. * Read the central part of a zip file and add the info in this object.
  3411. * @param {DataReader} reader the reader to use.
  3412. */
  3413. readCentralPart: function(reader) {
  3414. this.versionMadeBy = reader.readString(2);
  3415. this.versionNeeded = reader.readInt(2);
  3416. this.bitFlag = reader.readInt(2);
  3417. this.compressionMethod = reader.readString(2);
  3418. this.date = reader.readDate();
  3419. this.crc32 = reader.readInt(4);
  3420. this.compressedSize = reader.readInt(4);
  3421. this.uncompressedSize = reader.readInt(4);
  3422. this.fileNameLength = reader.readInt(2);
  3423. this.extraFieldsLength = reader.readInt(2);
  3424. this.fileCommentLength = reader.readInt(2);
  3425. this.diskNumberStart = reader.readInt(2);
  3426. this.internalFileAttributes = reader.readInt(2);
  3427. this.externalFileAttributes = reader.readInt(4);
  3428. this.localHeaderOffset = reader.readInt(4);
  3429. if (this.isEncrypted()) {
  3430. throw new Error("Encrypted zip are not supported");
  3431. }
  3432. this.fileName = reader.readString(this.fileNameLength);
  3433. this.readExtraFields(reader);
  3434. this.parseZIP64ExtraField(reader);
  3435. this.fileComment = reader.readString(this.fileCommentLength);
  3436. // warning, this is true only for zip with madeBy == DOS (plateform dependent feature)
  3437. this.dir = this.externalFileAttributes & 0x00000010 ? true : false;
  3438. },
  3439. /**
  3440. * Parse the ZIP64 extra field and merge the info in the current ZipEntry.
  3441. * @param {DataReader} reader the reader to use.
  3442. */
  3443. parseZIP64ExtraField: function(reader) {
  3444. if (!this.extraFields[0x0001]) {
  3445. return;
  3446. }
  3447. // should be something, preparing the extra reader
  3448. var extraReader = new StringReader(this.extraFields[0x0001].value);
  3449. // I really hope that these 64bits integer can fit in 32 bits integer, because js
  3450. // won't let us have more.
  3451. if (this.uncompressedSize === utils.MAX_VALUE_32BITS) {
  3452. this.uncompressedSize = extraReader.readInt(8);
  3453. }
  3454. if (this.compressedSize === utils.MAX_VALUE_32BITS) {
  3455. this.compressedSize = extraReader.readInt(8);
  3456. }
  3457. if (this.localHeaderOffset === utils.MAX_VALUE_32BITS) {
  3458. this.localHeaderOffset = extraReader.readInt(8);
  3459. }
  3460. if (this.diskNumberStart === utils.MAX_VALUE_32BITS) {
  3461. this.diskNumberStart = extraReader.readInt(4);
  3462. }
  3463. },
  3464. /**
  3465. * Read the central part of a zip file and add the info in this object.
  3466. * @param {DataReader} reader the reader to use.
  3467. */
  3468. readExtraFields: function(reader) {
  3469. var start = reader.index,
  3470. extraFieldId,
  3471. extraFieldLength,
  3472. extraFieldValue;
  3473. this.extraFields = this.extraFields || {};
  3474. while (reader.index < start + this.extraFieldsLength) {
  3475. extraFieldId = reader.readInt(2);
  3476. extraFieldLength = reader.readInt(2);
  3477. extraFieldValue = reader.readString(extraFieldLength);
  3478. this.extraFields[extraFieldId] = {
  3479. id: extraFieldId,
  3480. length: extraFieldLength,
  3481. value: extraFieldValue
  3482. };
  3483. }
  3484. },
  3485. /**
  3486. * Apply an UTF8 transformation if needed.
  3487. */
  3488. handleUTF8: function() {
  3489. if (this.useUTF8()) {
  3490. this.fileName = jszipProto.utf8decode(this.fileName);
  3491. this.fileComment = jszipProto.utf8decode(this.fileComment);
  3492. } else {
  3493. var upath = this.findExtraFieldUnicodePath();
  3494. if (upath !== null) {
  3495. this.fileName = upath;
  3496. }
  3497. var ucomment = this.findExtraFieldUnicodeComment();
  3498. if (ucomment !== null) {
  3499. this.fileComment = ucomment;
  3500. }
  3501. }
  3502. },
  3503. /**
  3504. * Find the unicode path declared in the extra field, if any.
  3505. * @return {String} the unicode path, null otherwise.
  3506. */
  3507. findExtraFieldUnicodePath: function() {
  3508. var upathField = this.extraFields[0x7075];
  3509. if (upathField) {
  3510. var extraReader = new StringReader(upathField.value);
  3511. // wrong version
  3512. if (extraReader.readInt(1) !== 1) {
  3513. return null;
  3514. }
  3515. // the crc of the filename changed, this field is out of date.
  3516. if (jszipProto.crc32(this.fileName) !== extraReader.readInt(4)) {
  3517. return null;
  3518. }
  3519. return jszipProto.utf8decode(extraReader.readString(upathField.length - 5));
  3520. }
  3521. return null;
  3522. },
  3523. /**
  3524. * Find the unicode comment declared in the extra field, if any.
  3525. * @return {String} the unicode comment, null otherwise.
  3526. */
  3527. findExtraFieldUnicodeComment: function() {
  3528. var ucommentField = this.extraFields[0x6375];
  3529. if (ucommentField) {
  3530. var extraReader = new StringReader(ucommentField.value);
  3531. // wrong version
  3532. if (extraReader.readInt(1) !== 1) {
  3533. return null;
  3534. }
  3535. // the crc of the comment changed, this field is out of date.
  3536. if (jszipProto.crc32(this.fileComment) !== extraReader.readInt(4)) {
  3537. return null;
  3538. }
  3539. return jszipProto.utf8decode(extraReader.readString(ucommentField.length - 5));
  3540. }
  3541. return null;
  3542. }
  3543. };
  3544. module.exports = ZipEntry;
  3545. },{"./compressedObject":2,"./object":13,"./stringReader":15,"./utils":21}],24:[function(_dereq_,module,exports){
  3546. // Top level file is just a mixin of submodules & constants
  3547. 'use strict';
  3548. var assign = _dereq_('./lib/utils/common').assign;
  3549. var deflate = _dereq_('./lib/deflate');
  3550. var inflate = _dereq_('./lib/inflate');
  3551. var constants = _dereq_('./lib/zlib/constants');
  3552. var pako = {};
  3553. assign(pako, deflate, inflate, constants);
  3554. module.exports = pako;
  3555. },{"./lib/deflate":25,"./lib/inflate":26,"./lib/utils/common":27,"./lib/zlib/constants":30}],25:[function(_dereq_,module,exports){
  3556. 'use strict';
  3557. var zlib_deflate = _dereq_('./zlib/deflate.js');
  3558. var utils = _dereq_('./utils/common');
  3559. var strings = _dereq_('./utils/strings');
  3560. var msg = _dereq_('./zlib/messages');
  3561. var zstream = _dereq_('./zlib/zstream');
  3562. /* Public constants ==========================================================*/
  3563. /* ===========================================================================*/
  3564. var Z_NO_FLUSH = 0;
  3565. var Z_FINISH = 4;
  3566. var Z_OK = 0;
  3567. var Z_STREAM_END = 1;
  3568. var Z_DEFAULT_COMPRESSION = -1;
  3569. var Z_DEFAULT_STRATEGY = 0;
  3570. var Z_DEFLATED = 8;
  3571. /* ===========================================================================*/
  3572. /**
  3573. * class Deflate
  3574. *
  3575. * Generic JS-style wrapper for zlib calls. If you don't need
  3576. * streaming behaviour - use more simple functions: [[deflate]],
  3577. * [[deflateRaw]] and [[gzip]].
  3578. **/
  3579. /* internal
  3580. * Deflate.chunks -> Array
  3581. *
  3582. * Chunks of output data, if [[Deflate#onData]] not overriden.
  3583. **/
  3584. /**
  3585. * Deflate.result -> Uint8Array|Array
  3586. *
  3587. * Compressed result, generated by default [[Deflate#onData]]
  3588. * and [[Deflate#onEnd]] handlers. Filled after you push last chunk
  3589. * (call [[Deflate#push]] with `Z_FINISH` / `true` param).
  3590. **/
  3591. /**
  3592. * Deflate.err -> Number
  3593. *
  3594. * error code after deflate finished. 0 (Z_OK) on success.
  3595. * You will not need it in real life, because deflate errors
  3596. * are possible only on wrong options or bad `onData` / `onEnd`
  3597. * custom handlers.
  3598. **/
  3599. /**
  3600. * Deflate.msg -> String
  3601. *
  3602. * error message, if [[Deflate.err]] != 0
  3603. **/
  3604. /**
  3605. * new Deflate(options)
  3606. * - options (Object): zlib deflate options.
  3607. *
  3608. * Creates new deflator instance with specified params. Throws exception
  3609. * on bad params. Supported options:
  3610. *
  3611. * - `level`
  3612. * - `windowBits`
  3613. * - `memLevel`
  3614. * - `strategy`
  3615. *
  3616. * [http://zlib.net/manual.html#Advanced](http://zlib.net/manual.html#Advanced)
  3617. * for more information on these.
  3618. *
  3619. * Additional options, for internal needs:
  3620. *
  3621. * - `chunkSize` - size of generated data chunks (16K by default)
  3622. * - `raw` (Boolean) - do raw deflate
  3623. * - `gzip` (Boolean) - create gzip wrapper
  3624. * - `to` (String) - if equal to 'string', then result will be "binary string"
  3625. * (each char code [0..255])
  3626. * - `header` (Object) - custom header for gzip
  3627. * - `text` (Boolean) - true if compressed data believed to be text
  3628. * - `time` (Number) - modification time, unix timestamp
  3629. * - `os` (Number) - operation system code
  3630. * - `extra` (Array) - array of bytes with extra data (max 65536)
  3631. * - `name` (String) - file name (binary string)
  3632. * - `comment` (String) - comment (binary string)
  3633. * - `hcrc` (Boolean) - true if header crc should be added
  3634. *
  3635. * ##### Example:
  3636. *
  3637. * ```javascript
  3638. * var pako = require('pako')
  3639. * , chunk1 = Uint8Array([1,2,3,4,5,6,7,8,9])
  3640. * , chunk2 = Uint8Array([10,11,12,13,14,15,16,17,18,19]);
  3641. *
  3642. * var deflate = new pako.Deflate({ level: 3});
  3643. *
  3644. * deflate.push(chunk1, false);
  3645. * deflate.push(chunk2, true); // true -> last chunk
  3646. *
  3647. * if (deflate.err) { throw new error(deflate.err); }
  3648. *
  3649. * console.log(deflate.result);
  3650. * ```
  3651. **/
  3652. var Deflate = function(options) {
  3653. this.options = utils.assign({
  3654. level: Z_DEFAULT_COMPRESSION,
  3655. method: Z_DEFLATED,
  3656. chunkSize: 16384,
  3657. windowBits: 15,
  3658. memLevel: 8,
  3659. strategy: Z_DEFAULT_STRATEGY,
  3660. to: ''
  3661. }, options || {});
  3662. var opt = this.options;
  3663. if (opt.raw && (opt.windowBits > 0)) {
  3664. opt.windowBits = -opt.windowBits;
  3665. }
  3666. else if (opt.gzip && (opt.windowBits > 0) && (opt.windowBits < 16)) {
  3667. opt.windowBits += 16;
  3668. }
  3669. this.err = 0; // error code, if happens (0 = Z_OK)
  3670. this.msg = ''; // error message
  3671. this.ended = false; // used to avoid multiple onEnd() calls
  3672. this.chunks = []; // chunks of compressed data
  3673. this.strm = new zstream();
  3674. this.strm.avail_out = 0;
  3675. var status = zlib_deflate.deflateInit2(
  3676. this.strm,
  3677. opt.level,
  3678. opt.method,
  3679. opt.windowBits,
  3680. opt.memLevel,
  3681. opt.strategy
  3682. );
  3683. if (status !== Z_OK) {
  3684. throw new Error(msg[status]);
  3685. }
  3686. if (opt.header) {
  3687. zlib_deflate.deflateSetHeader(this.strm, opt.header);
  3688. }
  3689. };
  3690. /**
  3691. * Deflate#push(data[, mode]) -> Boolean
  3692. * - data (Uint8Array|Array|String): input data. Strings will be converted to
  3693. * utf8 byte sequence.
  3694. * - mode (Number|Boolean): 0..6 for corresponding Z_NO_FLUSH..Z_TREE modes.
  3695. * See constants. Skipped or `false` means Z_NO_FLUSH, `true` meansh Z_FINISH.
  3696. *
  3697. * Sends input data to deflate pipe, generating [[Deflate#onData]] calls with
  3698. * new compressed chunks. Returns `true` on success. The last data block must have
  3699. * mode Z_FINISH (or `true`). That flush internal pending buffers and call
  3700. * [[Deflate#onEnd]].
  3701. *
  3702. * On fail call [[Deflate#onEnd]] with error code and return false.
  3703. *
  3704. * We strongly recommend to use `Uint8Array` on input for best speed (output
  3705. * array format is detected automatically). Also, don't skip last param and always
  3706. * use the same type in your code (boolean or number). That will improve JS speed.
  3707. *
  3708. * For regular `Array`-s make sure all elements are [0..255].
  3709. *
  3710. * ##### Example
  3711. *
  3712. * ```javascript
  3713. * push(chunk, false); // push one of data chunks
  3714. * ...
  3715. * push(chunk, true); // push last chunk
  3716. * ```
  3717. **/
  3718. Deflate.prototype.push = function(data, mode) {
  3719. var strm = this.strm;
  3720. var chunkSize = this.options.chunkSize;
  3721. var status, _mode;
  3722. if (this.ended) { return false; }
  3723. _mode = (mode === ~~mode) ? mode : ((mode === true) ? Z_FINISH : Z_NO_FLUSH);
  3724. // Convert data if needed
  3725. if (typeof data === 'string') {
  3726. // If we need to compress text, change encoding to utf8.
  3727. strm.input = strings.string2buf(data);
  3728. } else {
  3729. strm.input = data;
  3730. }
  3731. strm.next_in = 0;
  3732. strm.avail_in = strm.input.length;
  3733. do {
  3734. if (strm.avail_out === 0) {
  3735. strm.output = new utils.Buf8(chunkSize);
  3736. strm.next_out = 0;
  3737. strm.avail_out = chunkSize;
  3738. }
  3739. status = zlib_deflate.deflate(strm, _mode); /* no bad return value */
  3740. if (status !== Z_STREAM_END && status !== Z_OK) {
  3741. this.onEnd(status);
  3742. this.ended = true;
  3743. return false;
  3744. }
  3745. if (strm.avail_out === 0 || (strm.avail_in === 0 && _mode === Z_FINISH)) {
  3746. if (this.options.to === 'string') {
  3747. this.onData(strings.buf2binstring(utils.shrinkBuf(strm.output, strm.next_out)));
  3748. } else {
  3749. this.onData(utils.shrinkBuf(strm.output, strm.next_out));
  3750. }
  3751. }
  3752. } while ((strm.avail_in > 0 || strm.avail_out === 0) && status !== Z_STREAM_END);
  3753. // Finalize on the last chunk.
  3754. if (_mode === Z_FINISH) {
  3755. status = zlib_deflate.deflateEnd(this.strm);
  3756. this.onEnd(status);
  3757. this.ended = true;
  3758. return status === Z_OK;
  3759. }
  3760. return true;
  3761. };
  3762. /**
  3763. * Deflate#onData(chunk) -> Void
  3764. * - chunk (Uint8Array|Array|String): ouput data. Type of array depends
  3765. * on js engine support. When string output requested, each chunk
  3766. * will be string.
  3767. *
  3768. * By default, stores data blocks in `chunks[]` property and glue
  3769. * those in `onEnd`. Override this handler, if you need another behaviour.
  3770. **/
  3771. Deflate.prototype.onData = function(chunk) {
  3772. this.chunks.push(chunk);
  3773. };
  3774. /**
  3775. * Deflate#onEnd(status) -> Void
  3776. * - status (Number): deflate status. 0 (Z_OK) on success,
  3777. * other if not.
  3778. *
  3779. * Called once after you tell deflate that input stream complete
  3780. * or error happenned. By default - join collected chunks,
  3781. * free memory and fill `results` / `err` properties.
  3782. **/
  3783. Deflate.prototype.onEnd = function(status) {
  3784. // On success - join
  3785. if (status === Z_OK) {
  3786. if (this.options.to === 'string') {
  3787. this.result = this.chunks.join('');
  3788. } else {
  3789. this.result = utils.flattenChunks(this.chunks);
  3790. }
  3791. }
  3792. this.chunks = [];
  3793. this.err = status;
  3794. this.msg = this.strm.msg;
  3795. };
  3796. /**
  3797. * deflate(data[, options]) -> Uint8Array|Array|String
  3798. * - data (Uint8Array|Array|String): input data to compress.
  3799. * - options (Object): zlib deflate options.
  3800. *
  3801. * Compress `data` with deflate alrorythm and `options`.
  3802. *
  3803. * Supported options are:
  3804. *
  3805. * - level
  3806. * - windowBits
  3807. * - memLevel
  3808. * - strategy
  3809. *
  3810. * [http://zlib.net/manual.html#Advanced](http://zlib.net/manual.html#Advanced)
  3811. * for more information on these.
  3812. *
  3813. * Sugar (options):
  3814. *
  3815. * - `raw` (Boolean) - say that we work with raw stream, if you don't wish to specify
  3816. * negative windowBits implicitly.
  3817. * - `to` (String) - if equal to 'string', then result will be "binary string"
  3818. * (each char code [0..255])
  3819. *
  3820. * ##### Example:
  3821. *
  3822. * ```javascript
  3823. * var pako = require('pako')
  3824. * , data = Uint8Array([1,2,3,4,5,6,7,8,9]);
  3825. *
  3826. * console.log(pako.deflate(data));
  3827. * ```
  3828. **/
  3829. function deflate(input, options) {
  3830. var deflator = new Deflate(options);
  3831. deflator.push(input, true);
  3832. // That will never happens, if you don't cheat with options :)
  3833. if (deflator.err) { throw deflator.msg; }
  3834. return deflator.result;
  3835. }
  3836. /**
  3837. * deflateRaw(data[, options]) -> Uint8Array|Array|String
  3838. * - data (Uint8Array|Array|String): input data to compress.
  3839. * - options (Object): zlib deflate options.
  3840. *
  3841. * The same as [[deflate]], but creates raw data, without wrapper
  3842. * (header and adler32 crc).
  3843. **/
  3844. function deflateRaw(input, options) {
  3845. options = options || {};
  3846. options.raw = true;
  3847. return deflate(input, options);
  3848. }
  3849. /**
  3850. * gzip(data[, options]) -> Uint8Array|Array|String
  3851. * - data (Uint8Array|Array|String): input data to compress.
  3852. * - options (Object): zlib deflate options.
  3853. *
  3854. * The same as [[deflate]], but create gzip wrapper instead of
  3855. * deflate one.
  3856. **/
  3857. function gzip(input, options) {
  3858. options = options || {};
  3859. options.gzip = true;
  3860. return deflate(input, options);
  3861. }
  3862. exports.Deflate = Deflate;
  3863. exports.deflate = deflate;
  3864. exports.deflateRaw = deflateRaw;
  3865. exports.gzip = gzip;
  3866. },{"./utils/common":27,"./utils/strings":28,"./zlib/deflate.js":32,"./zlib/messages":37,"./zlib/zstream":39}],26:[function(_dereq_,module,exports){
  3867. 'use strict';
  3868. var zlib_inflate = _dereq_('./zlib/inflate.js');
  3869. var utils = _dereq_('./utils/common');
  3870. var strings = _dereq_('./utils/strings');
  3871. var c = _dereq_('./zlib/constants');
  3872. var msg = _dereq_('./zlib/messages');
  3873. var zstream = _dereq_('./zlib/zstream');
  3874. var gzheader = _dereq_('./zlib/gzheader');
  3875. /**
  3876. * class Inflate
  3877. *
  3878. * Generic JS-style wrapper for zlib calls. If you don't need
  3879. * streaming behaviour - use more simple functions: [[inflate]]
  3880. * and [[inflateRaw]].
  3881. **/
  3882. /* internal
  3883. * inflate.chunks -> Array
  3884. *
  3885. * Chunks of output data, if [[Inflate#onData]] not overriden.
  3886. **/
  3887. /**
  3888. * Inflate.result -> Uint8Array|Array|String
  3889. *
  3890. * Uncompressed result, generated by default [[Inflate#onData]]
  3891. * and [[Inflate#onEnd]] handlers. Filled after you push last chunk
  3892. * (call [[Inflate#push]] with `Z_FINISH` / `true` param).
  3893. **/
  3894. /**
  3895. * Inflate.err -> Number
  3896. *
  3897. * error code after inflate finished. 0 (Z_OK) on success.
  3898. * Should be checked if broken data possible.
  3899. **/
  3900. /**
  3901. * Inflate.msg -> String
  3902. *
  3903. * error message, if [[Inflate.err]] != 0
  3904. **/
  3905. /**
  3906. * new Inflate(options)
  3907. * - options (Object): zlib inflate options.
  3908. *
  3909. * Creates new inflator instance with specified params. Throws exception
  3910. * on bad params. Supported options:
  3911. *
  3912. * - `windowBits`
  3913. *
  3914. * [http://zlib.net/manual.html#Advanced](http://zlib.net/manual.html#Advanced)
  3915. * for more information on these.
  3916. *
  3917. * Additional options, for internal needs:
  3918. *
  3919. * - `chunkSize` - size of generated data chunks (16K by default)
  3920. * - `raw` (Boolean) - do raw inflate
  3921. * - `to` (String) - if equal to 'string', then result will be converted
  3922. * from utf8 to utf16 (javascript) string. When string output requested,
  3923. * chunk length can differ from `chunkSize`, depending on content.
  3924. *
  3925. * By default, when no options set, autodetect deflate/gzip data format via
  3926. * wrapper header.
  3927. *
  3928. * ##### Example:
  3929. *
  3930. * ```javascript
  3931. * var pako = require('pako')
  3932. * , chunk1 = Uint8Array([1,2,3,4,5,6,7,8,9])
  3933. * , chunk2 = Uint8Array([10,11,12,13,14,15,16,17,18,19]);
  3934. *
  3935. * var inflate = new pako.Inflate({ level: 3});
  3936. *
  3937. * inflate.push(chunk1, false);
  3938. * inflate.push(chunk2, true); // true -> last chunk
  3939. *
  3940. * if (inflate.err) { throw new error(inflate.err); }
  3941. *
  3942. * console.log(inflate.result);
  3943. * ```
  3944. **/
  3945. var Inflate = function(options) {
  3946. this.options = utils.assign({
  3947. chunkSize: 16384,
  3948. windowBits: 0,
  3949. to: ''
  3950. }, options || {});
  3951. var opt = this.options;
  3952. // Force window size for `raw` data, if not set directly,
  3953. // because we have no header for autodetect.
  3954. if (opt.raw && (opt.windowBits >= 0) && (opt.windowBits < 16)) {
  3955. opt.windowBits = -opt.windowBits;
  3956. if (opt.windowBits === 0) { opt.windowBits = -15; }
  3957. }
  3958. // If `windowBits` not defined (and mode not raw) - set autodetect flag for gzip/deflate
  3959. if ((opt.windowBits >= 0) && (opt.windowBits < 16) &&
  3960. !(options && options.windowBits)) {
  3961. opt.windowBits += 32;
  3962. }
  3963. // Gzip header has no info about windows size, we can do autodetect only
  3964. // for deflate. So, if window size not set, force it to max when gzip possible
  3965. if ((opt.windowBits > 15) && (opt.windowBits < 48)) {
  3966. // bit 3 (16) -> gzipped data
  3967. // bit 4 (32) -> autodetect gzip/deflate
  3968. if ((opt.windowBits & 15) === 0) {
  3969. opt.windowBits |= 15;
  3970. }
  3971. }
  3972. this.err = 0; // error code, if happens (0 = Z_OK)
  3973. this.msg = ''; // error message
  3974. this.ended = false; // used to avoid multiple onEnd() calls
  3975. this.chunks = []; // chunks of compressed data
  3976. this.strm = new zstream();
  3977. this.strm.avail_out = 0;
  3978. var status = zlib_inflate.inflateInit2(
  3979. this.strm,
  3980. opt.windowBits
  3981. );
  3982. if (status !== c.Z_OK) {
  3983. throw new Error(msg[status]);
  3984. }
  3985. this.header = new gzheader();
  3986. zlib_inflate.inflateGetHeader(this.strm, this.header);
  3987. };
  3988. /**
  3989. * Inflate#push(data[, mode]) -> Boolean
  3990. * - data (Uint8Array|Array|String): input data
  3991. * - mode (Number|Boolean): 0..6 for corresponding Z_NO_FLUSH..Z_TREE modes.
  3992. * See constants. Skipped or `false` means Z_NO_FLUSH, `true` meansh Z_FINISH.
  3993. *
  3994. * Sends input data to inflate pipe, generating [[Inflate#onData]] calls with
  3995. * new output chunks. Returns `true` on success. The last data block must have
  3996. * mode Z_FINISH (or `true`). That flush internal pending buffers and call
  3997. * [[Inflate#onEnd]].
  3998. *
  3999. * On fail call [[Inflate#onEnd]] with error code and return false.
  4000. *
  4001. * We strongly recommend to use `Uint8Array` on input for best speed (output
  4002. * format is detected automatically). Also, don't skip last param and always
  4003. * use the same type in your code (boolean or number). That will improve JS speed.
  4004. *
  4005. * For regular `Array`-s make sure all elements are [0..255].
  4006. *
  4007. * ##### Example
  4008. *
  4009. * ```javascript
  4010. * push(chunk, false); // push one of data chunks
  4011. * ...
  4012. * push(chunk, true); // push last chunk
  4013. * ```
  4014. **/
  4015. Inflate.prototype.push = function(data, mode) {
  4016. var strm = this.strm;
  4017. var chunkSize = this.options.chunkSize;
  4018. var status, _mode;
  4019. var next_out_utf8, tail, utf8str;
  4020. if (this.ended) { return false; }
  4021. _mode = (mode === ~~mode) ? mode : ((mode === true) ? c.Z_FINISH : c.Z_NO_FLUSH);
  4022. // Convert data if needed
  4023. if (typeof data === 'string') {
  4024. // Only binary strings can be decompressed on practice
  4025. strm.input = strings.binstring2buf(data);
  4026. } else {
  4027. strm.input = data;
  4028. }
  4029. strm.next_in = 0;
  4030. strm.avail_in = strm.input.length;
  4031. do {
  4032. if (strm.avail_out === 0) {
  4033. strm.output = new utils.Buf8(chunkSize);
  4034. strm.next_out = 0;
  4035. strm.avail_out = chunkSize;
  4036. }
  4037. status = zlib_inflate.inflate(strm, c.Z_NO_FLUSH); /* no bad return value */
  4038. if (status !== c.Z_STREAM_END && status !== c.Z_OK) {
  4039. this.onEnd(status);
  4040. this.ended = true;
  4041. return false;
  4042. }
  4043. if (strm.next_out) {
  4044. if (strm.avail_out === 0 || status === c.Z_STREAM_END || (strm.avail_in === 0 && _mode === c.Z_FINISH)) {
  4045. if (this.options.to === 'string') {
  4046. next_out_utf8 = strings.utf8border(strm.output, strm.next_out);
  4047. tail = strm.next_out - next_out_utf8;
  4048. utf8str = strings.buf2string(strm.output, next_out_utf8);
  4049. // move tail
  4050. strm.next_out = tail;
  4051. strm.avail_out = chunkSize - tail;
  4052. if (tail) { utils.arraySet(strm.output, strm.output, next_out_utf8, tail, 0); }
  4053. this.onData(utf8str);
  4054. } else {
  4055. this.onData(utils.shrinkBuf(strm.output, strm.next_out));
  4056. }
  4057. }
  4058. }
  4059. } while ((strm.avail_in > 0) && status !== c.Z_STREAM_END);
  4060. if (status === c.Z_STREAM_END) {
  4061. _mode = c.Z_FINISH;
  4062. }
  4063. // Finalize on the last chunk.
  4064. if (_mode === c.Z_FINISH) {
  4065. status = zlib_inflate.inflateEnd(this.strm);
  4066. this.onEnd(status);
  4067. this.ended = true;
  4068. return status === c.Z_OK;
  4069. }
  4070. return true;
  4071. };
  4072. /**
  4073. * Inflate#onData(chunk) -> Void
  4074. * - chunk (Uint8Array|Array|String): ouput data. Type of array depends
  4075. * on js engine support. When string output requested, each chunk
  4076. * will be string.
  4077. *
  4078. * By default, stores data blocks in `chunks[]` property and glue
  4079. * those in `onEnd`. Override this handler, if you need another behaviour.
  4080. **/
  4081. Inflate.prototype.onData = function(chunk) {
  4082. this.chunks.push(chunk);
  4083. };
  4084. /**
  4085. * Inflate#onEnd(status) -> Void
  4086. * - status (Number): inflate status. 0 (Z_OK) on success,
  4087. * other if not.
  4088. *
  4089. * Called once after you tell inflate that input stream complete
  4090. * or error happenned. By default - join collected chunks,
  4091. * free memory and fill `results` / `err` properties.
  4092. **/
  4093. Inflate.prototype.onEnd = function(status) {
  4094. // On success - join
  4095. if (status === c.Z_OK) {
  4096. if (this.options.to === 'string') {
  4097. // Glue & convert here, until we teach pako to send
  4098. // utf8 alligned strings to onData
  4099. this.result = this.chunks.join('');
  4100. } else {
  4101. this.result = utils.flattenChunks(this.chunks);
  4102. }
  4103. }
  4104. this.chunks = [];
  4105. this.err = status;
  4106. this.msg = this.strm.msg;
  4107. };
  4108. /**
  4109. * inflate(data[, options]) -> Uint8Array|Array|String
  4110. * - data (Uint8Array|Array|String): input data to decompress.
  4111. * - options (Object): zlib inflate options.
  4112. *
  4113. * Decompress `data` with inflate/ungzip and `options`. Autodetect
  4114. * format via wrapper header by default. That's why we don't provide
  4115. * separate `ungzip` method.
  4116. *
  4117. * Supported options are:
  4118. *
  4119. * - windowBits
  4120. *
  4121. * [http://zlib.net/manual.html#Advanced](http://zlib.net/manual.html#Advanced)
  4122. * for more information.
  4123. *
  4124. * Sugar (options):
  4125. *
  4126. * - `raw` (Boolean) - say that we work with raw stream, if you don't wish to specify
  4127. * negative windowBits implicitly.
  4128. * - `to` (String) - if equal to 'string', then result will be converted
  4129. * from utf8 to utf16 (javascript) string. When string output requested,
  4130. * chunk length can differ from `chunkSize`, depending on content.
  4131. *
  4132. *
  4133. * ##### Example:
  4134. *
  4135. * ```javascript
  4136. * var pako = require('pako')
  4137. * , input = pako.deflate([1,2,3,4,5,6,7,8,9])
  4138. * , output;
  4139. *
  4140. * try {
  4141. * output = pako.inflate(input);
  4142. * } catch (err)
  4143. * console.log(err);
  4144. * }
  4145. * ```
  4146. **/
  4147. function inflate(input, options) {
  4148. var inflator = new Inflate(options);
  4149. inflator.push(input, true);
  4150. // That will never happens, if you don't cheat with options :)
  4151. if (inflator.err) { throw inflator.msg; }
  4152. return inflator.result;
  4153. }
  4154. /**
  4155. * inflateRaw(data[, options]) -> Uint8Array|Array|String
  4156. * - data (Uint8Array|Array|String): input data to decompress.
  4157. * - options (Object): zlib inflate options.
  4158. *
  4159. * The same as [[inflate]], but creates raw data, without wrapper
  4160. * (header and adler32 crc).
  4161. **/
  4162. function inflateRaw(input, options) {
  4163. options = options || {};
  4164. options.raw = true;
  4165. return inflate(input, options);
  4166. }
  4167. /**
  4168. * ungzip(data[, options]) -> Uint8Array|Array|String
  4169. * - data (Uint8Array|Array|String): input data to decompress.
  4170. * - options (Object): zlib inflate options.
  4171. *
  4172. * Just shortcut to [[inflate]], because it autodetects format
  4173. * by header.content. Done for convenience.
  4174. **/
  4175. exports.Inflate = Inflate;
  4176. exports.inflate = inflate;
  4177. exports.inflateRaw = inflateRaw;
  4178. exports.ungzip = inflate;
  4179. },{"./utils/common":27,"./utils/strings":28,"./zlib/constants":30,"./zlib/gzheader":33,"./zlib/inflate.js":35,"./zlib/messages":37,"./zlib/zstream":39}],27:[function(_dereq_,module,exports){
  4180. 'use strict';
  4181. var TYPED_OK = (typeof Uint8Array !== 'undefined') &&
  4182. (typeof Uint16Array !== 'undefined') &&
  4183. (typeof Int32Array !== 'undefined');
  4184. exports.assign = function (obj /*from1, from2, from3, ...*/) {
  4185. var sources = Array.prototype.slice.call(arguments, 1);
  4186. while (sources.length) {
  4187. var source = sources.shift();
  4188. if (!source) { continue; }
  4189. if (typeof(source) !== 'object') {
  4190. throw new TypeError(source + 'must be non-object');
  4191. }
  4192. for (var p in source) {
  4193. if (source.hasOwnProperty(p)) {
  4194. obj[p] = source[p];
  4195. }
  4196. }
  4197. }
  4198. return obj;
  4199. };
  4200. // reduce buffer size, avoiding mem copy
  4201. exports.shrinkBuf = function (buf, size) {
  4202. if (buf.length === size) { return buf; }
  4203. if (buf.subarray) { return buf.subarray(0, size); }
  4204. buf.length = size;
  4205. return buf;
  4206. };
  4207. var fnTyped = {
  4208. arraySet: function (dest, src, src_offs, len, dest_offs) {
  4209. if (src.subarray && dest.subarray) {
  4210. dest.set(src.subarray(src_offs, src_offs+len), dest_offs);
  4211. return;
  4212. }
  4213. // Fallback to ordinary array
  4214. for(var i=0; i<len; i++) {
  4215. dest[dest_offs + i] = src[src_offs + i];
  4216. }
  4217. },
  4218. // Join array of chunks to single array.
  4219. flattenChunks: function(chunks) {
  4220. var i, l, len, pos, chunk, result;
  4221. // calculate data length
  4222. len = 0;
  4223. for (i=0, l=chunks.length; i<l; i++) {
  4224. len += chunks[i].length;
  4225. }
  4226. // join chunks
  4227. result = new Uint8Array(len);
  4228. pos = 0;
  4229. for (i=0, l=chunks.length; i<l; i++) {
  4230. chunk = chunks[i];
  4231. result.set(chunk, pos);
  4232. pos += chunk.length;
  4233. }
  4234. return result;
  4235. }
  4236. };
  4237. var fnUntyped = {
  4238. arraySet: function (dest, src, src_offs, len, dest_offs) {
  4239. for(var i=0; i<len; i++) {
  4240. dest[dest_offs + i] = src[src_offs + i];
  4241. }
  4242. },
  4243. // Join array of chunks to single array.
  4244. flattenChunks: function(chunks) {
  4245. return [].concat.apply([], chunks);
  4246. }
  4247. };
  4248. // Enable/Disable typed arrays use, for testing
  4249. //
  4250. exports.setTyped = function (on) {
  4251. if (on) {
  4252. exports.Buf8 = Uint8Array;
  4253. exports.Buf16 = Uint16Array;
  4254. exports.Buf32 = Int32Array;
  4255. exports.assign(exports, fnTyped);
  4256. } else {
  4257. exports.Buf8 = Array;
  4258. exports.Buf16 = Array;
  4259. exports.Buf32 = Array;
  4260. exports.assign(exports, fnUntyped);
  4261. }
  4262. };
  4263. exports.setTyped(TYPED_OK);
  4264. },{}],28:[function(_dereq_,module,exports){
  4265. // String encode/decode helpers
  4266. 'use strict';
  4267. var utils = _dereq_('./common');
  4268. // Quick check if we can use fast array to bin string conversion
  4269. //
  4270. // - apply(Array) can fail on Android 2.2
  4271. // - apply(Uint8Array) can fail on iOS 5.1 Safary
  4272. //
  4273. var STR_APPLY_OK = true;
  4274. var STR_APPLY_UIA_OK = true;
  4275. try { String.fromCharCode.apply(null, [0]); } catch(__) { STR_APPLY_OK = false; }
  4276. try { String.fromCharCode.apply(null, new Uint8Array(1)); } catch(__) { STR_APPLY_UIA_OK = false; }
  4277. // Table with utf8 lengths (calculated by first byte of sequence)
  4278. // Note, that 5 & 6-byte values and some 4-byte values can not be represented in JS,
  4279. // because max possible codepoint is 0x10ffff
  4280. var _utf8len = new utils.Buf8(256);
  4281. for (var i=0; i<256; i++) {
  4282. _utf8len[i] = (i >= 252 ? 6 : i >= 248 ? 5 : i >= 240 ? 4 : i >= 224 ? 3 : i >= 192 ? 2 : 1);
  4283. }
  4284. _utf8len[254]=_utf8len[254]=1; // Invalid sequence start
  4285. // convert string to array (typed, when possible)
  4286. exports.string2buf = function (str) {
  4287. var buf, c, c2, m_pos, i, str_len = str.length, buf_len = 0;
  4288. // count binary size
  4289. for (m_pos = 0; m_pos < str_len; m_pos++) {
  4290. c = str.charCodeAt(m_pos);
  4291. if ((c & 0xfc00) === 0xd800 && (m_pos+1 < str_len)) {
  4292. c2 = str.charCodeAt(m_pos+1);
  4293. if ((c2 & 0xfc00) === 0xdc00) {
  4294. c = 0x10000 + ((c - 0xd800) << 10) + (c2 - 0xdc00);
  4295. m_pos++;
  4296. }
  4297. }
  4298. buf_len += c < 0x80 ? 1 : c < 0x800 ? 2 : c < 0x10000 ? 3 : 4;
  4299. }
  4300. // allocate buffer
  4301. buf = new utils.Buf8(buf_len);
  4302. // convert
  4303. for (i=0, m_pos = 0; i < buf_len; m_pos++) {
  4304. c = str.charCodeAt(m_pos);
  4305. if ((c & 0xfc00) === 0xd800 && (m_pos+1 < str_len)) {
  4306. c2 = str.charCodeAt(m_pos+1);
  4307. if ((c2 & 0xfc00) === 0xdc00) {
  4308. c = 0x10000 + ((c - 0xd800) << 10) + (c2 - 0xdc00);
  4309. m_pos++;
  4310. }
  4311. }
  4312. if (c < 0x80) {
  4313. /* one byte */
  4314. buf[i++] = c;
  4315. } else if (c < 0x800) {
  4316. /* two bytes */
  4317. buf[i++] = 0xC0 | (c >>> 6);
  4318. buf[i++] = 0x80 | (c & 0x3f);
  4319. } else if (c < 0x10000) {
  4320. /* three bytes */
  4321. buf[i++] = 0xE0 | (c >>> 12);
  4322. buf[i++] = 0x80 | (c >>> 6 & 0x3f);
  4323. buf[i++] = 0x80 | (c & 0x3f);
  4324. } else {
  4325. /* four bytes */
  4326. buf[i++] = 0xf0 | (c >>> 18);
  4327. buf[i++] = 0x80 | (c >>> 12 & 0x3f);
  4328. buf[i++] = 0x80 | (c >>> 6 & 0x3f);
  4329. buf[i++] = 0x80 | (c & 0x3f);
  4330. }
  4331. }
  4332. return buf;
  4333. };
  4334. // Helper (used in 2 places)
  4335. function buf2binstring(buf, len) {
  4336. // use fallback for big arrays to avoid stack overflow
  4337. if (len < 65537) {
  4338. if ((buf.subarray && STR_APPLY_UIA_OK) || (!buf.subarray && STR_APPLY_OK)) {
  4339. return String.fromCharCode.apply(null, utils.shrinkBuf(buf, len));
  4340. }
  4341. }
  4342. var result = '';
  4343. for(var i=0; i < len; i++) {
  4344. result += String.fromCharCode(buf[i]);
  4345. }
  4346. return result;
  4347. }
  4348. // Convert byte array to binary string
  4349. exports.buf2binstring = function(buf) {
  4350. return buf2binstring(buf, buf.length);
  4351. };
  4352. // Convert binary string (typed, when possible)
  4353. exports.binstring2buf = function(str) {
  4354. var buf = new utils.Buf8(str.length);
  4355. for(var i=0, len=buf.length; i < len; i++) {
  4356. buf[i] = str.charCodeAt(i);
  4357. }
  4358. return buf;
  4359. };
  4360. // convert array to string
  4361. exports.buf2string = function (buf, max) {
  4362. var i, out, c, c_len;
  4363. var len = max || buf.length;
  4364. // Reserve max possible length (2 words per char)
  4365. // NB: by unknown reasons, Array is significantly faster for
  4366. // String.fromCharCode.apply than Uint16Array.
  4367. var utf16buf = new Array(len*2);
  4368. for (out=0, i=0; i<len;) {
  4369. c = buf[i++];
  4370. // quick process ascii
  4371. if (c < 0x80) { utf16buf[out++] = c; continue; }
  4372. c_len = _utf8len[c];
  4373. // skip 5 & 6 byte codes
  4374. if (c_len > 4) { utf16buf[out++] = 0xfffd; i += c_len-1; continue; }
  4375. // apply mask on first byte
  4376. c &= c_len === 2 ? 0x1f : c_len === 3 ? 0x0f : 0x07;
  4377. // join the rest
  4378. while (c_len > 1 && i < len) {
  4379. c = (c << 6) | (buf[i++] & 0x3f);
  4380. c_len--;
  4381. }
  4382. // terminated by end of string?
  4383. if (c_len > 1) { utf16buf[out++] = 0xfffd; continue; }
  4384. if (c < 0x10000) {
  4385. utf16buf[out++] = c;
  4386. } else {
  4387. c -= 0x10000;
  4388. utf16buf[out++] = 0xd800 | ((c >> 10) & 0x3ff);
  4389. utf16buf[out++] = 0xdc00 | (c & 0x3ff);
  4390. }
  4391. }
  4392. return buf2binstring(utf16buf, out);
  4393. };
  4394. // Calculate max possible position in utf8 buffer,
  4395. // that will not break sequence. If that's not possible
  4396. // - (very small limits) return max size as is.
  4397. //
  4398. // buf[] - utf8 bytes array
  4399. // max - length limit (mandatory);
  4400. exports.utf8border = function(buf, max) {
  4401. var pos;
  4402. max = max || buf.length;
  4403. if (max > buf.length) { max = buf.length; }
  4404. // go back from last position, until start of sequence found
  4405. pos = max-1;
  4406. while (pos >= 0 && (buf[pos] & 0xC0) === 0x80) { pos--; }
  4407. // Fuckup - very small and broken sequence,
  4408. // return max, because we should return something anyway.
  4409. if (pos < 0) { return max; }
  4410. // If we came to start of buffer - that means vuffer is too small,
  4411. // return max too.
  4412. if (pos === 0) { return max; }
  4413. return (pos + _utf8len[buf[pos]] > max) ? pos : max;
  4414. };
  4415. },{"./common":27}],29:[function(_dereq_,module,exports){
  4416. 'use strict';
  4417. // Note: adler32 takes 12% for level 0 and 2% for level 6.
  4418. // It doesn't worth to make additional optimizationa as in original.
  4419. // Small size is preferable.
  4420. function adler32(adler, buf, len, pos) {
  4421. var s1 = (adler & 0xffff) |0
  4422. , s2 = ((adler >>> 16) & 0xffff) |0
  4423. , n = 0;
  4424. while (len !== 0) {
  4425. // Set limit ~ twice less than 5552, to keep
  4426. // s2 in 31-bits, because we force signed ints.
  4427. // in other case %= will fail.
  4428. n = len > 2000 ? 2000 : len;
  4429. len -= n;
  4430. do {
  4431. s1 = (s1 + buf[pos++]) |0;
  4432. s2 = (s2 + s1) |0;
  4433. } while (--n);
  4434. s1 %= 65521;
  4435. s2 %= 65521;
  4436. }
  4437. return (s1 | (s2 << 16)) |0;
  4438. }
  4439. module.exports = adler32;
  4440. },{}],30:[function(_dereq_,module,exports){
  4441. module.exports = {
  4442. /* Allowed flush values; see deflate() and inflate() below for details */
  4443. Z_NO_FLUSH: 0,
  4444. Z_PARTIAL_FLUSH: 1,
  4445. Z_SYNC_FLUSH: 2,
  4446. Z_FULL_FLUSH: 3,
  4447. Z_FINISH: 4,
  4448. Z_BLOCK: 5,
  4449. Z_TREES: 6,
  4450. /* Return codes for the compression/decompression functions. Negative values
  4451. * are errors, positive values are used for special but normal events.
  4452. */
  4453. Z_OK: 0,
  4454. Z_STREAM_END: 1,
  4455. Z_NEED_DICT: 2,
  4456. Z_ERRNO: -1,
  4457. Z_STREAM_ERROR: -2,
  4458. Z_DATA_ERROR: -3,
  4459. //Z_MEM_ERROR: -4,
  4460. Z_BUF_ERROR: -5,
  4461. //Z_VERSION_ERROR: -6,
  4462. /* compression levels */
  4463. Z_NO_COMPRESSION: 0,
  4464. Z_BEST_SPEED: 1,
  4465. Z_BEST_COMPRESSION: 9,
  4466. Z_DEFAULT_COMPRESSION: -1,
  4467. Z_FILTERED: 1,
  4468. Z_HUFFMAN_ONLY: 2,
  4469. Z_RLE: 3,
  4470. Z_FIXED: 4,
  4471. Z_DEFAULT_STRATEGY: 0,
  4472. /* Possible values of the data_type field (though see inflate()) */
  4473. Z_BINARY: 0,
  4474. Z_TEXT: 1,
  4475. //Z_ASCII: 1, // = Z_TEXT (deprecated)
  4476. Z_UNKNOWN: 2,
  4477. /* The deflate compression method */
  4478. Z_DEFLATED: 8
  4479. //Z_NULL: null // Use -1 or null inline, depending on var type
  4480. };
  4481. },{}],31:[function(_dereq_,module,exports){
  4482. 'use strict';
  4483. // Note: we can't get significant speed boost here.
  4484. // So write code to minimize size - no pregenerated tables
  4485. // and array tools dependencies.
  4486. // Use ordinary array, since untyped makes no boost here
  4487. function makeTable() {
  4488. var c, table = [];
  4489. for(var n =0; n < 256; n++){
  4490. c = n;
  4491. for(var k =0; k < 8; k++){
  4492. c = ((c&1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1));
  4493. }
  4494. table[n] = c;
  4495. }
  4496. return table;
  4497. }
  4498. // Create table on load. Just 255 signed longs. Not a problem.
  4499. var crcTable = makeTable();
  4500. function crc32(crc, buf, len, pos) {
  4501. var t = crcTable
  4502. , end = pos + len;
  4503. crc = crc ^ (-1);
  4504. for (var i = pos; i < end; i++ ) {
  4505. crc = (crc >>> 8) ^ t[(crc ^ buf[i]) & 0xFF];
  4506. }
  4507. return (crc ^ (-1)); // >>> 0;
  4508. }
  4509. module.exports = crc32;
  4510. },{}],32:[function(_dereq_,module,exports){
  4511. 'use strict';
  4512. var utils = _dereq_('../utils/common');
  4513. var trees = _dereq_('./trees');
  4514. var adler32 = _dereq_('./adler32');
  4515. var crc32 = _dereq_('./crc32');
  4516. var msg = _dereq_('./messages');
  4517. /* Public constants ==========================================================*/
  4518. /* ===========================================================================*/
  4519. /* Allowed flush values; see deflate() and inflate() below for details */
  4520. var Z_NO_FLUSH = 0;
  4521. var Z_PARTIAL_FLUSH = 1;
  4522. //var Z_SYNC_FLUSH = 2;
  4523. var Z_FULL_FLUSH = 3;
  4524. var Z_FINISH = 4;
  4525. var Z_BLOCK = 5;
  4526. //var Z_TREES = 6;
  4527. /* Return codes for the compression/decompression functions. Negative values
  4528. * are errors, positive values are used for special but normal events.
  4529. */
  4530. var Z_OK = 0;
  4531. var Z_STREAM_END = 1;
  4532. //var Z_NEED_DICT = 2;
  4533. //var Z_ERRNO = -1;
  4534. var Z_STREAM_ERROR = -2;
  4535. var Z_DATA_ERROR = -3;
  4536. //var Z_MEM_ERROR = -4;
  4537. var Z_BUF_ERROR = -5;
  4538. //var Z_VERSION_ERROR = -6;
  4539. /* compression levels */
  4540. //var Z_NO_COMPRESSION = 0;
  4541. //var Z_BEST_SPEED = 1;
  4542. //var Z_BEST_COMPRESSION = 9;
  4543. var Z_DEFAULT_COMPRESSION = -1;
  4544. var Z_FILTERED = 1;
  4545. var Z_HUFFMAN_ONLY = 2;
  4546. var Z_RLE = 3;
  4547. var Z_FIXED = 4;
  4548. var Z_DEFAULT_STRATEGY = 0;
  4549. /* Possible values of the data_type field (though see inflate()) */
  4550. //var Z_BINARY = 0;
  4551. //var Z_TEXT = 1;
  4552. //var Z_ASCII = 1; // = Z_TEXT
  4553. var Z_UNKNOWN = 2;
  4554. /* The deflate compression method */
  4555. var Z_DEFLATED = 8;
  4556. /*============================================================================*/
  4557. var MAX_MEM_LEVEL = 9;
  4558. /* Maximum value for memLevel in deflateInit2 */
  4559. var MAX_WBITS = 15;
  4560. /* 32K LZ77 window */
  4561. var DEF_MEM_LEVEL = 8;
  4562. var LENGTH_CODES = 29;
  4563. /* number of length codes, not counting the special END_BLOCK code */
  4564. var LITERALS = 256;
  4565. /* number of literal bytes 0..255 */
  4566. var L_CODES = LITERALS + 1 + LENGTH_CODES;
  4567. /* number of Literal or Length codes, including the END_BLOCK code */
  4568. var D_CODES = 30;
  4569. /* number of distance codes */
  4570. var BL_CODES = 19;
  4571. /* number of codes used to transfer the bit lengths */
  4572. var HEAP_SIZE = 2*L_CODES + 1;
  4573. /* maximum heap size */
  4574. var MAX_BITS = 15;
  4575. /* All codes must not exceed MAX_BITS bits */
  4576. var MIN_MATCH = 3;
  4577. var MAX_MATCH = 258;
  4578. var MIN_LOOKAHEAD = (MAX_MATCH + MIN_MATCH + 1);
  4579. var PRESET_DICT = 0x20;
  4580. var INIT_STATE = 42;
  4581. var EXTRA_STATE = 69;
  4582. var NAME_STATE = 73;
  4583. var COMMENT_STATE = 91;
  4584. var HCRC_STATE = 103;
  4585. var BUSY_STATE = 113;
  4586. var FINISH_STATE = 666;
  4587. var BS_NEED_MORE = 1; /* block not completed, need more input or more output */
  4588. var BS_BLOCK_DONE = 2; /* block flush performed */
  4589. var BS_FINISH_STARTED = 3; /* finish started, need only more output at next deflate */
  4590. var BS_FINISH_DONE = 4; /* finish done, accept no more input or output */
  4591. var OS_CODE = 0x03; // Unix :) . Don't detect, use this default.
  4592. function err(strm, errorCode) {
  4593. strm.msg = msg[errorCode];
  4594. return errorCode;
  4595. }
  4596. function rank(f) {
  4597. return ((f) << 1) - ((f) > 4 ? 9 : 0);
  4598. }
  4599. function zero(buf) { var len = buf.length; while (--len >= 0) { buf[len] = 0; } }
  4600. /* =========================================================================
  4601. * Flush as much pending output as possible. All deflate() output goes
  4602. * through this function so some applications may wish to modify it
  4603. * to avoid allocating a large strm->output buffer and copying into it.
  4604. * (See also read_buf()).
  4605. */
  4606. function flush_pending(strm) {
  4607. var s = strm.state;
  4608. //_tr_flush_bits(s);
  4609. var len = s.pending;
  4610. if (len > strm.avail_out) {
  4611. len = strm.avail_out;
  4612. }
  4613. if (len === 0) { return; }
  4614. utils.arraySet(strm.output, s.pending_buf, s.pending_out, len, strm.next_out);
  4615. strm.next_out += len;
  4616. s.pending_out += len;
  4617. strm.total_out += len;
  4618. strm.avail_out -= len;
  4619. s.pending -= len;
  4620. if (s.pending === 0) {
  4621. s.pending_out = 0;
  4622. }
  4623. }
  4624. function flush_block_only (s, last) {
  4625. trees._tr_flush_block(s, (s.block_start >= 0 ? s.block_start : -1), s.strstart - s.block_start, last);
  4626. s.block_start = s.strstart;
  4627. flush_pending(s.strm);
  4628. }
  4629. function put_byte(s, b) {
  4630. s.pending_buf[s.pending++] = b;
  4631. }
  4632. /* =========================================================================
  4633. * Put a short in the pending buffer. The 16-bit value is put in MSB order.
  4634. * IN assertion: the stream state is correct and there is enough room in
  4635. * pending_buf.
  4636. */
  4637. function putShortMSB(s, b) {
  4638. // put_byte(s, (Byte)(b >> 8));
  4639. // put_byte(s, (Byte)(b & 0xff));
  4640. s.pending_buf[s.pending++] = (b >>> 8) & 0xff;
  4641. s.pending_buf[s.pending++] = b & 0xff;
  4642. }
  4643. /* ===========================================================================
  4644. * Read a new buffer from the current input stream, update the adler32
  4645. * and total number of bytes read. All deflate() input goes through
  4646. * this function so some applications may wish to modify it to avoid
  4647. * allocating a large strm->input buffer and copying from it.
  4648. * (See also flush_pending()).
  4649. */
  4650. function read_buf(strm, buf, start, size) {
  4651. var len = strm.avail_in;
  4652. if (len > size) { len = size; }
  4653. if (len === 0) { return 0; }
  4654. strm.avail_in -= len;
  4655. utils.arraySet(buf, strm.input, strm.next_in, len, start);
  4656. if (strm.state.wrap === 1) {
  4657. strm.adler = adler32(strm.adler, buf, len, start);
  4658. }
  4659. else if (strm.state.wrap === 2) {
  4660. strm.adler = crc32(strm.adler, buf, len, start);
  4661. }
  4662. strm.next_in += len;
  4663. strm.total_in += len;
  4664. return len;
  4665. }
  4666. /* ===========================================================================
  4667. * Set match_start to the longest match starting at the given string and
  4668. * return its length. Matches shorter or equal to prev_length are discarded,
  4669. * in which case the result is equal to prev_length and match_start is
  4670. * garbage.
  4671. * IN assertions: cur_match is the head of the hash chain for the current
  4672. * string (strstart) and its distance is <= MAX_DIST, and prev_length >= 1
  4673. * OUT assertion: the match length is not greater than s->lookahead.
  4674. */
  4675. function longest_match(s, cur_match) {
  4676. var chain_length = s.max_chain_length; /* max hash chain length */
  4677. var scan = s.strstart; /* current string */
  4678. var match; /* matched string */
  4679. var len; /* length of current match */
  4680. var best_len = s.prev_length; /* best match length so far */
  4681. var nice_match = s.nice_match; /* stop if match long enough */
  4682. var limit = (s.strstart > (s.w_size - MIN_LOOKAHEAD)) ?
  4683. s.strstart - (s.w_size - MIN_LOOKAHEAD) : 0/*NIL*/;
  4684. var _win = s.window; // shortcut
  4685. var wmask = s.w_mask;
  4686. var prev = s.prev;
  4687. /* Stop when cur_match becomes <= limit. To simplify the code,
  4688. * we prevent matches with the string of window index 0.
  4689. */
  4690. var strend = s.strstart + MAX_MATCH;
  4691. var scan_end1 = _win[scan + best_len - 1];
  4692. var scan_end = _win[scan + best_len];
  4693. /* The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of 16.
  4694. * It is easy to get rid of this optimization if necessary.
  4695. */
  4696. // Assert(s->hash_bits >= 8 && MAX_MATCH == 258, "Code too clever");
  4697. /* Do not waste too much time if we already have a good match: */
  4698. if (s.prev_length >= s.good_match) {
  4699. chain_length >>= 2;
  4700. }
  4701. /* Do not look for matches beyond the end of the input. This is necessary
  4702. * to make deflate deterministic.
  4703. */
  4704. if (nice_match > s.lookahead) { nice_match = s.lookahead; }
  4705. // Assert((ulg)s->strstart <= s->window_size-MIN_LOOKAHEAD, "need lookahead");
  4706. do {
  4707. // Assert(cur_match < s->strstart, "no future");
  4708. match = cur_match;
  4709. /* Skip to next match if the match length cannot increase
  4710. * or if the match length is less than 2. Note that the checks below
  4711. * for insufficient lookahead only occur occasionally for performance
  4712. * reasons. Therefore uninitialized memory will be accessed, and
  4713. * conditional jumps will be made that depend on those values.
  4714. * However the length of the match is limited to the lookahead, so
  4715. * the output of deflate is not affected by the uninitialized values.
  4716. */
  4717. if (_win[match + best_len] !== scan_end ||
  4718. _win[match + best_len - 1] !== scan_end1 ||
  4719. _win[match] !== _win[scan] ||
  4720. _win[++match] !== _win[scan + 1]) {
  4721. continue;
  4722. }
  4723. /* The check at best_len-1 can be removed because it will be made
  4724. * again later. (This heuristic is not always a win.)
  4725. * It is not necessary to compare scan[2] and match[2] since they
  4726. * are always equal when the other bytes match, given that
  4727. * the hash keys are equal and that HASH_BITS >= 8.
  4728. */
  4729. scan += 2;
  4730. match++;
  4731. // Assert(*scan == *match, "match[2]?");
  4732. /* We check for insufficient lookahead only every 8th comparison;
  4733. * the 256th check will be made at strstart+258.
  4734. */
  4735. do {
  4736. /*jshint noempty:false*/
  4737. } while (_win[++scan] === _win[++match] && _win[++scan] === _win[++match] &&
  4738. _win[++scan] === _win[++match] && _win[++scan] === _win[++match] &&
  4739. _win[++scan] === _win[++match] && _win[++scan] === _win[++match] &&
  4740. _win[++scan] === _win[++match] && _win[++scan] === _win[++match] &&
  4741. scan < strend);
  4742. // Assert(scan <= s->window+(unsigned)(s->window_size-1), "wild scan");
  4743. len = MAX_MATCH - (strend - scan);
  4744. scan = strend - MAX_MATCH;
  4745. if (len > best_len) {
  4746. s.match_start = cur_match;
  4747. best_len = len;
  4748. if (len >= nice_match) {
  4749. break;
  4750. }
  4751. scan_end1 = _win[scan + best_len - 1];
  4752. scan_end = _win[scan + best_len];
  4753. }
  4754. } while ((cur_match = prev[cur_match & wmask]) > limit && --chain_length !== 0);
  4755. if (best_len <= s.lookahead) {
  4756. return best_len;
  4757. }
  4758. return s.lookahead;
  4759. }
  4760. /* ===========================================================================
  4761. * Fill the window when the lookahead becomes insufficient.
  4762. * Updates strstart and lookahead.
  4763. *
  4764. * IN assertion: lookahead < MIN_LOOKAHEAD
  4765. * OUT assertions: strstart <= window_size-MIN_LOOKAHEAD
  4766. * At least one byte has been read, or avail_in == 0; reads are
  4767. * performed for at least two bytes (required for the zip translate_eol
  4768. * option -- not supported here).
  4769. */
  4770. function fill_window(s) {
  4771. var _w_size = s.w_size;
  4772. var p, n, m, more, str;
  4773. //Assert(s->lookahead < MIN_LOOKAHEAD, "already enough lookahead");
  4774. do {
  4775. more = s.window_size - s.lookahead - s.strstart;
  4776. // JS ints have 32 bit, block below not needed
  4777. /* Deal with !@#$% 64K limit: */
  4778. //if (sizeof(int) <= 2) {
  4779. // if (more == 0 && s->strstart == 0 && s->lookahead == 0) {
  4780. // more = wsize;
  4781. //
  4782. // } else if (more == (unsigned)(-1)) {
  4783. // /* Very unlikely, but possible on 16 bit machine if
  4784. // * strstart == 0 && lookahead == 1 (input done a byte at time)
  4785. // */
  4786. // more--;
  4787. // }
  4788. //}
  4789. /* If the window is almost full and there is insufficient lookahead,
  4790. * move the upper half to the lower one to make room in the upper half.
  4791. */
  4792. if (s.strstart >= _w_size + (_w_size - MIN_LOOKAHEAD)) {
  4793. utils.arraySet(s.window, s.window, _w_size, _w_size, 0);
  4794. s.match_start -= _w_size;
  4795. s.strstart -= _w_size;
  4796. /* we now have strstart >= MAX_DIST */
  4797. s.block_start -= _w_size;
  4798. /* Slide the hash table (could be avoided with 32 bit values
  4799. at the expense of memory usage). We slide even when level == 0
  4800. to keep the hash table consistent if we switch back to level > 0
  4801. later. (Using level 0 permanently is not an optimal usage of
  4802. zlib, so we don't care about this pathological case.)
  4803. */
  4804. n = s.hash_size;
  4805. p = n;
  4806. do {
  4807. m = s.head[--p];
  4808. s.head[p] = (m >= _w_size ? m - _w_size : 0);
  4809. } while (--n);
  4810. n = _w_size;
  4811. p = n;
  4812. do {
  4813. m = s.prev[--p];
  4814. s.prev[p] = (m >= _w_size ? m - _w_size : 0);
  4815. /* If n is not on any hash chain, prev[n] is garbage but
  4816. * its value will never be used.
  4817. */
  4818. } while (--n);
  4819. more += _w_size;
  4820. }
  4821. if (s.strm.avail_in === 0) {
  4822. break;
  4823. }
  4824. /* If there was no sliding:
  4825. * strstart <= WSIZE+MAX_DIST-1 && lookahead <= MIN_LOOKAHEAD - 1 &&
  4826. * more == window_size - lookahead - strstart
  4827. * => more >= window_size - (MIN_LOOKAHEAD-1 + WSIZE + MAX_DIST-1)
  4828. * => more >= window_size - 2*WSIZE + 2
  4829. * In the BIG_MEM or MMAP case (not yet supported),
  4830. * window_size == input_size + MIN_LOOKAHEAD &&
  4831. * strstart + s->lookahead <= input_size => more >= MIN_LOOKAHEAD.
  4832. * Otherwise, window_size == 2*WSIZE so more >= 2.
  4833. * If there was sliding, more >= WSIZE. So in all cases, more >= 2.
  4834. */
  4835. //Assert(more >= 2, "more < 2");
  4836. n = read_buf(s.strm, s.window, s.strstart + s.lookahead, more);
  4837. s.lookahead += n;
  4838. /* Initialize the hash value now that we have some input: */
  4839. if (s.lookahead + s.insert >= MIN_MATCH) {
  4840. str = s.strstart - s.insert;
  4841. s.ins_h = s.window[str];
  4842. /* UPDATE_HASH(s, s->ins_h, s->window[str + 1]); */
  4843. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[str + 1]) & s.hash_mask;
  4844. //#if MIN_MATCH != 3
  4845. // Call update_hash() MIN_MATCH-3 more times
  4846. //#endif
  4847. while (s.insert) {
  4848. /* UPDATE_HASH(s, s->ins_h, s->window[str + MIN_MATCH-1]); */
  4849. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[str + MIN_MATCH-1]) & s.hash_mask;
  4850. s.prev[str & s.w_mask] = s.head[s.ins_h];
  4851. s.head[s.ins_h] = str;
  4852. str++;
  4853. s.insert--;
  4854. if (s.lookahead + s.insert < MIN_MATCH) {
  4855. break;
  4856. }
  4857. }
  4858. }
  4859. /* If the whole input has less than MIN_MATCH bytes, ins_h is garbage,
  4860. * but this is not important since only literal bytes will be emitted.
  4861. */
  4862. } while (s.lookahead < MIN_LOOKAHEAD && s.strm.avail_in !== 0);
  4863. /* If the WIN_INIT bytes after the end of the current data have never been
  4864. * written, then zero those bytes in order to avoid memory check reports of
  4865. * the use of uninitialized (or uninitialised as Julian writes) bytes by
  4866. * the longest match routines. Update the high water mark for the next
  4867. * time through here. WIN_INIT is set to MAX_MATCH since the longest match
  4868. * routines allow scanning to strstart + MAX_MATCH, ignoring lookahead.
  4869. */
  4870. // if (s.high_water < s.window_size) {
  4871. // var curr = s.strstart + s.lookahead;
  4872. // var init = 0;
  4873. //
  4874. // if (s.high_water < curr) {
  4875. // /* Previous high water mark below current data -- zero WIN_INIT
  4876. // * bytes or up to end of window, whichever is less.
  4877. // */
  4878. // init = s.window_size - curr;
  4879. // if (init > WIN_INIT)
  4880. // init = WIN_INIT;
  4881. // zmemzero(s->window + curr, (unsigned)init);
  4882. // s->high_water = curr + init;
  4883. // }
  4884. // else if (s->high_water < (ulg)curr + WIN_INIT) {
  4885. // /* High water mark at or above current data, but below current data
  4886. // * plus WIN_INIT -- zero out to current data plus WIN_INIT, or up
  4887. // * to end of window, whichever is less.
  4888. // */
  4889. // init = (ulg)curr + WIN_INIT - s->high_water;
  4890. // if (init > s->window_size - s->high_water)
  4891. // init = s->window_size - s->high_water;
  4892. // zmemzero(s->window + s->high_water, (unsigned)init);
  4893. // s->high_water += init;
  4894. // }
  4895. // }
  4896. //
  4897. // Assert((ulg)s->strstart <= s->window_size - MIN_LOOKAHEAD,
  4898. // "not enough room for search");
  4899. }
  4900. /* ===========================================================================
  4901. * Copy without compression as much as possible from the input stream, return
  4902. * the current block state.
  4903. * This function does not insert new strings in the dictionary since
  4904. * uncompressible data is probably not useful. This function is used
  4905. * only for the level=0 compression option.
  4906. * NOTE: this function should be optimized to avoid extra copying from
  4907. * window to pending_buf.
  4908. */
  4909. function deflate_stored(s, flush) {
  4910. /* Stored blocks are limited to 0xffff bytes, pending_buf is limited
  4911. * to pending_buf_size, and each stored block has a 5 byte header:
  4912. */
  4913. var max_block_size = 0xffff;
  4914. if (max_block_size > s.pending_buf_size - 5) {
  4915. max_block_size = s.pending_buf_size - 5;
  4916. }
  4917. /* Copy as much as possible from input to output: */
  4918. for (;;) {
  4919. /* Fill the window as much as possible: */
  4920. if (s.lookahead <= 1) {
  4921. //Assert(s->strstart < s->w_size+MAX_DIST(s) ||
  4922. // s->block_start >= (long)s->w_size, "slide too late");
  4923. // if (!(s.strstart < s.w_size + (s.w_size - MIN_LOOKAHEAD) ||
  4924. // s.block_start >= s.w_size)) {
  4925. // throw new error("slide too late");
  4926. // }
  4927. fill_window(s);
  4928. if (s.lookahead === 0 && flush === Z_NO_FLUSH) {
  4929. return BS_NEED_MORE;
  4930. }
  4931. if (s.lookahead === 0) {
  4932. break;
  4933. }
  4934. /* flush the current block */
  4935. }
  4936. //Assert(s->block_start >= 0L, "block gone");
  4937. // if (s.block_start < 0) throw new error("block gone");
  4938. s.strstart += s.lookahead;
  4939. s.lookahead = 0;
  4940. /* Emit a stored block if pending_buf will be full: */
  4941. var max_start = s.block_start + max_block_size;
  4942. if (s.strstart === 0 || s.strstart >= max_start) {
  4943. /* strstart == 0 is possible when wraparound on 16-bit machine */
  4944. s.lookahead = s.strstart - max_start;
  4945. s.strstart = max_start;
  4946. /*** FLUSH_BLOCK(s, 0); ***/
  4947. flush_block_only(s, false);
  4948. if (s.strm.avail_out === 0) {
  4949. return BS_NEED_MORE;
  4950. }
  4951. /***/
  4952. }
  4953. /* Flush if we may have to slide, otherwise block_start may become
  4954. * negative and the data will be gone:
  4955. */
  4956. if (s.strstart - s.block_start >= (s.w_size - MIN_LOOKAHEAD)) {
  4957. /*** FLUSH_BLOCK(s, 0); ***/
  4958. flush_block_only(s, false);
  4959. if (s.strm.avail_out === 0) {
  4960. return BS_NEED_MORE;
  4961. }
  4962. /***/
  4963. }
  4964. }
  4965. s.insert = 0;
  4966. if (flush === Z_FINISH) {
  4967. /*** FLUSH_BLOCK(s, 1); ***/
  4968. flush_block_only(s, true);
  4969. if (s.strm.avail_out === 0) {
  4970. return BS_FINISH_STARTED;
  4971. }
  4972. /***/
  4973. return BS_FINISH_DONE;
  4974. }
  4975. if (s.strstart > s.block_start) {
  4976. /*** FLUSH_BLOCK(s, 0); ***/
  4977. flush_block_only(s, false);
  4978. if (s.strm.avail_out === 0) {
  4979. return BS_NEED_MORE;
  4980. }
  4981. /***/
  4982. }
  4983. return BS_NEED_MORE;
  4984. }
  4985. /* ===========================================================================
  4986. * Compress as much as possible from the input stream, return the current
  4987. * block state.
  4988. * This function does not perform lazy evaluation of matches and inserts
  4989. * new strings in the dictionary only for unmatched strings or for short
  4990. * matches. It is used only for the fast compression options.
  4991. */
  4992. function deflate_fast(s, flush) {
  4993. var hash_head; /* head of the hash chain */
  4994. var bflush; /* set if current block must be flushed */
  4995. for (;;) {
  4996. /* Make sure that we always have enough lookahead, except
  4997. * at the end of the input file. We need MAX_MATCH bytes
  4998. * for the next match, plus MIN_MATCH bytes to insert the
  4999. * string following the next match.
  5000. */
  5001. if (s.lookahead < MIN_LOOKAHEAD) {
  5002. fill_window(s);
  5003. if (s.lookahead < MIN_LOOKAHEAD && flush === Z_NO_FLUSH) {
  5004. return BS_NEED_MORE;
  5005. }
  5006. if (s.lookahead === 0) {
  5007. break; /* flush the current block */
  5008. }
  5009. }
  5010. /* Insert the string window[strstart .. strstart+2] in the
  5011. * dictionary, and set hash_head to the head of the hash chain:
  5012. */
  5013. hash_head = 0/*NIL*/;
  5014. if (s.lookahead >= MIN_MATCH) {
  5015. /*** INSERT_STRING(s, s.strstart, hash_head); ***/
  5016. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH - 1]) & s.hash_mask;
  5017. hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h];
  5018. s.head[s.ins_h] = s.strstart;
  5019. /***/
  5020. }
  5021. /* Find the longest match, discarding those <= prev_length.
  5022. * At this point we have always match_length < MIN_MATCH
  5023. */
  5024. if (hash_head !== 0/*NIL*/ && ((s.strstart - hash_head) <= (s.w_size - MIN_LOOKAHEAD))) {
  5025. /* To simplify the code, we prevent matches with the string
  5026. * of window index 0 (in particular we have to avoid a match
  5027. * of the string with itself at the start of the input file).
  5028. */
  5029. s.match_length = longest_match(s, hash_head);
  5030. /* longest_match() sets match_start */
  5031. }
  5032. if (s.match_length >= MIN_MATCH) {
  5033. // check_match(s, s.strstart, s.match_start, s.match_length); // for debug only
  5034. /*** _tr_tally_dist(s, s.strstart - s.match_start,
  5035. s.match_length - MIN_MATCH, bflush); ***/
  5036. bflush = trees._tr_tally(s, s.strstart - s.match_start, s.match_length - MIN_MATCH);
  5037. s.lookahead -= s.match_length;
  5038. /* Insert new strings in the hash table only if the match length
  5039. * is not too large. This saves time but degrades compression.
  5040. */
  5041. if (s.match_length <= s.max_lazy_match/*max_insert_length*/ && s.lookahead >= MIN_MATCH) {
  5042. s.match_length--; /* string at strstart already in table */
  5043. do {
  5044. s.strstart++;
  5045. /*** INSERT_STRING(s, s.strstart, hash_head); ***/
  5046. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH - 1]) & s.hash_mask;
  5047. hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h];
  5048. s.head[s.ins_h] = s.strstart;
  5049. /***/
  5050. /* strstart never exceeds WSIZE-MAX_MATCH, so there are
  5051. * always MIN_MATCH bytes ahead.
  5052. */
  5053. } while (--s.match_length !== 0);
  5054. s.strstart++;
  5055. } else
  5056. {
  5057. s.strstart += s.match_length;
  5058. s.match_length = 0;
  5059. s.ins_h = s.window[s.strstart];
  5060. /* UPDATE_HASH(s, s.ins_h, s.window[s.strstart+1]); */
  5061. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + 1]) & s.hash_mask;
  5062. //#if MIN_MATCH != 3
  5063. // Call UPDATE_HASH() MIN_MATCH-3 more times
  5064. //#endif
  5065. /* If lookahead < MIN_MATCH, ins_h is garbage, but it does not
  5066. * matter since it will be recomputed at next deflate call.
  5067. */
  5068. }
  5069. } else {
  5070. /* No match, output a literal byte */
  5071. //Tracevv((stderr,"%c", s.window[s.strstart]));
  5072. /*** _tr_tally_lit(s, s.window[s.strstart], bflush); ***/
  5073. bflush = trees._tr_tally(s, 0, s.window[s.strstart]);
  5074. s.lookahead--;
  5075. s.strstart++;
  5076. }
  5077. if (bflush) {
  5078. /*** FLUSH_BLOCK(s, 0); ***/
  5079. flush_block_only(s, false);
  5080. if (s.strm.avail_out === 0) {
  5081. return BS_NEED_MORE;
  5082. }
  5083. /***/
  5084. }
  5085. }
  5086. s.insert = ((s.strstart < (MIN_MATCH-1)) ? s.strstart : MIN_MATCH-1);
  5087. if (flush === Z_FINISH) {
  5088. /*** FLUSH_BLOCK(s, 1); ***/
  5089. flush_block_only(s, true);
  5090. if (s.strm.avail_out === 0) {
  5091. return BS_FINISH_STARTED;
  5092. }
  5093. /***/
  5094. return BS_FINISH_DONE;
  5095. }
  5096. if (s.last_lit) {
  5097. /*** FLUSH_BLOCK(s, 0); ***/
  5098. flush_block_only(s, false);
  5099. if (s.strm.avail_out === 0) {
  5100. return BS_NEED_MORE;
  5101. }
  5102. /***/
  5103. }
  5104. return BS_BLOCK_DONE;
  5105. }
  5106. /* ===========================================================================
  5107. * Same as above, but achieves better compression. We use a lazy
  5108. * evaluation for matches: a match is finally adopted only if there is
  5109. * no better match at the next window position.
  5110. */
  5111. function deflate_slow(s, flush) {
  5112. var hash_head; /* head of hash chain */
  5113. var bflush; /* set if current block must be flushed */
  5114. var max_insert;
  5115. /* Process the input block. */
  5116. for (;;) {
  5117. /* Make sure that we always have enough lookahead, except
  5118. * at the end of the input file. We need MAX_MATCH bytes
  5119. * for the next match, plus MIN_MATCH bytes to insert the
  5120. * string following the next match.
  5121. */
  5122. if (s.lookahead < MIN_LOOKAHEAD) {
  5123. fill_window(s);
  5124. if (s.lookahead < MIN_LOOKAHEAD && flush === Z_NO_FLUSH) {
  5125. return BS_NEED_MORE;
  5126. }
  5127. if (s.lookahead === 0) { break; } /* flush the current block */
  5128. }
  5129. /* Insert the string window[strstart .. strstart+2] in the
  5130. * dictionary, and set hash_head to the head of the hash chain:
  5131. */
  5132. hash_head = 0/*NIL*/;
  5133. if (s.lookahead >= MIN_MATCH) {
  5134. /*** INSERT_STRING(s, s.strstart, hash_head); ***/
  5135. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH - 1]) & s.hash_mask;
  5136. hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h];
  5137. s.head[s.ins_h] = s.strstart;
  5138. /***/
  5139. }
  5140. /* Find the longest match, discarding those <= prev_length.
  5141. */
  5142. s.prev_length = s.match_length;
  5143. s.prev_match = s.match_start;
  5144. s.match_length = MIN_MATCH-1;
  5145. if (hash_head !== 0/*NIL*/ && s.prev_length < s.max_lazy_match &&
  5146. s.strstart - hash_head <= (s.w_size-MIN_LOOKAHEAD)/*MAX_DIST(s)*/) {
  5147. /* To simplify the code, we prevent matches with the string
  5148. * of window index 0 (in particular we have to avoid a match
  5149. * of the string with itself at the start of the input file).
  5150. */
  5151. s.match_length = longest_match(s, hash_head);
  5152. /* longest_match() sets match_start */
  5153. if (s.match_length <= 5 &&
  5154. (s.strategy === Z_FILTERED || (s.match_length === MIN_MATCH && s.strstart - s.match_start > 4096/*TOO_FAR*/))) {
  5155. /* If prev_match is also MIN_MATCH, match_start is garbage
  5156. * but we will ignore the current match anyway.
  5157. */
  5158. s.match_length = MIN_MATCH-1;
  5159. }
  5160. }
  5161. /* If there was a match at the previous step and the current
  5162. * match is not better, output the previous match:
  5163. */
  5164. if (s.prev_length >= MIN_MATCH && s.match_length <= s.prev_length) {
  5165. max_insert = s.strstart + s.lookahead - MIN_MATCH;
  5166. /* Do not insert strings in hash table beyond this. */
  5167. //check_match(s, s.strstart-1, s.prev_match, s.prev_length);
  5168. /***_tr_tally_dist(s, s.strstart - 1 - s.prev_match,
  5169. s.prev_length - MIN_MATCH, bflush);***/
  5170. bflush = trees._tr_tally(s, s.strstart - 1- s.prev_match, s.prev_length - MIN_MATCH);
  5171. /* Insert in hash table all strings up to the end of the match.
  5172. * strstart-1 and strstart are already inserted. If there is not
  5173. * enough lookahead, the last two strings are not inserted in
  5174. * the hash table.
  5175. */
  5176. s.lookahead -= s.prev_length-1;
  5177. s.prev_length -= 2;
  5178. do {
  5179. if (++s.strstart <= max_insert) {
  5180. /*** INSERT_STRING(s, s.strstart, hash_head); ***/
  5181. s.ins_h = ((s.ins_h << s.hash_shift) ^ s.window[s.strstart + MIN_MATCH - 1]) & s.hash_mask;
  5182. hash_head = s.prev[s.strstart & s.w_mask] = s.head[s.ins_h];
  5183. s.head[s.ins_h] = s.strstart;
  5184. /***/
  5185. }
  5186. } while (--s.prev_length !== 0);
  5187. s.match_available = 0;
  5188. s.match_length = MIN_MATCH-1;
  5189. s.strstart++;
  5190. if (bflush) {
  5191. /*** FLUSH_BLOCK(s, 0); ***/
  5192. flush_block_only(s, false);
  5193. if (s.strm.avail_out === 0) {
  5194. return BS_NEED_MORE;
  5195. }
  5196. /***/
  5197. }
  5198. } else if (s.match_available) {
  5199. /* If there was no match at the previous position, output a
  5200. * single literal. If there was a match but the current match
  5201. * is longer, truncate the previous match to a single literal.
  5202. */
  5203. //Tracevv((stderr,"%c", s->window[s->strstart-1]));
  5204. /*** _tr_tally_lit(s, s.window[s.strstart-1], bflush); ***/
  5205. bflush = trees._tr_tally(s, 0, s.window[s.strstart-1]);
  5206. if (bflush) {
  5207. /*** FLUSH_BLOCK_ONLY(s, 0) ***/
  5208. flush_block_only(s, false);
  5209. /***/
  5210. }
  5211. s.strstart++;
  5212. s.lookahead--;
  5213. if (s.strm.avail_out === 0) {
  5214. return BS_NEED_MORE;
  5215. }
  5216. } else {
  5217. /* There is no previous match to compare with, wait for
  5218. * the next step to decide.
  5219. */
  5220. s.match_available = 1;
  5221. s.strstart++;
  5222. s.lookahead--;
  5223. }
  5224. }
  5225. //Assert (flush != Z_NO_FLUSH, "no flush?");
  5226. if (s.match_available) {
  5227. //Tracevv((stderr,"%c", s->window[s->strstart-1]));
  5228. /*** _tr_tally_lit(s, s.window[s.strstart-1], bflush); ***/
  5229. bflush = trees._tr_tally(s, 0, s.window[s.strstart-1]);
  5230. s.match_available = 0;
  5231. }
  5232. s.insert = s.strstart < MIN_MATCH-1 ? s.strstart : MIN_MATCH-1;
  5233. if (flush === Z_FINISH) {
  5234. /*** FLUSH_BLOCK(s, 1); ***/
  5235. flush_block_only(s, true);
  5236. if (s.strm.avail_out === 0) {
  5237. return BS_FINISH_STARTED;
  5238. }
  5239. /***/
  5240. return BS_FINISH_DONE;
  5241. }
  5242. if (s.last_lit) {
  5243. /*** FLUSH_BLOCK(s, 0); ***/
  5244. flush_block_only(s, false);
  5245. if (s.strm.avail_out === 0) {
  5246. return BS_NEED_MORE;
  5247. }
  5248. /***/
  5249. }
  5250. return BS_BLOCK_DONE;
  5251. }
  5252. /* ===========================================================================
  5253. * For Z_RLE, simply look for runs of bytes, generate matches only of distance
  5254. * one. Do not maintain a hash table. (It will be regenerated if this run of
  5255. * deflate switches away from Z_RLE.)
  5256. */
  5257. function deflate_rle(s, flush) {
  5258. var bflush; /* set if current block must be flushed */
  5259. var prev; /* byte at distance one to match */
  5260. var scan, strend; /* scan goes up to strend for length of run */
  5261. var _win = s.window;
  5262. for (;;) {
  5263. /* Make sure that we always have enough lookahead, except
  5264. * at the end of the input file. We need MAX_MATCH bytes
  5265. * for the longest run, plus one for the unrolled loop.
  5266. */
  5267. if (s.lookahead <= MAX_MATCH) {
  5268. fill_window(s);
  5269. if (s.lookahead <= MAX_MATCH && flush === Z_NO_FLUSH) {
  5270. return BS_NEED_MORE;
  5271. }
  5272. if (s.lookahead === 0) { break; } /* flush the current block */
  5273. }
  5274. /* See how many times the previous byte repeats */
  5275. s.match_length = 0;
  5276. if (s.lookahead >= MIN_MATCH && s.strstart > 0) {
  5277. scan = s.strstart - 1;
  5278. prev = _win[scan];
  5279. if (prev === _win[++scan] && prev === _win[++scan] && prev === _win[++scan]) {
  5280. strend = s.strstart + MAX_MATCH;
  5281. do {
  5282. /*jshint noempty:false*/
  5283. } while (prev === _win[++scan] && prev === _win[++scan] &&
  5284. prev === _win[++scan] && prev === _win[++scan] &&
  5285. prev === _win[++scan] && prev === _win[++scan] &&
  5286. prev === _win[++scan] && prev === _win[++scan] &&
  5287. scan < strend);
  5288. s.match_length = MAX_MATCH - (strend - scan);
  5289. if (s.match_length > s.lookahead) {
  5290. s.match_length = s.lookahead;
  5291. }
  5292. }
  5293. //Assert(scan <= s->window+(uInt)(s->window_size-1), "wild scan");
  5294. }
  5295. /* Emit match if have run of MIN_MATCH or longer, else emit literal */
  5296. if (s.match_length >= MIN_MATCH) {
  5297. //check_match(s, s.strstart, s.strstart - 1, s.match_length);
  5298. /*** _tr_tally_dist(s, 1, s.match_length - MIN_MATCH, bflush); ***/
  5299. bflush = trees._tr_tally(s, 1, s.match_length - MIN_MATCH);
  5300. s.lookahead -= s.match_length;
  5301. s.strstart += s.match_length;
  5302. s.match_length = 0;
  5303. } else {
  5304. /* No match, output a literal byte */
  5305. //Tracevv((stderr,"%c", s->window[s->strstart]));
  5306. /*** _tr_tally_lit(s, s.window[s.strstart], bflush); ***/
  5307. bflush = trees._tr_tally(s, 0, s.window[s.strstart]);
  5308. s.lookahead--;
  5309. s.strstart++;
  5310. }
  5311. if (bflush) {
  5312. /*** FLUSH_BLOCK(s, 0); ***/
  5313. flush_block_only(s, false);
  5314. if (s.strm.avail_out === 0) {
  5315. return BS_NEED_MORE;
  5316. }
  5317. /***/
  5318. }
  5319. }
  5320. s.insert = 0;
  5321. if (flush === Z_FINISH) {
  5322. /*** FLUSH_BLOCK(s, 1); ***/
  5323. flush_block_only(s, true);
  5324. if (s.strm.avail_out === 0) {
  5325. return BS_FINISH_STARTED;
  5326. }
  5327. /***/
  5328. return BS_FINISH_DONE;
  5329. }
  5330. if (s.last_lit) {
  5331. /*** FLUSH_BLOCK(s, 0); ***/
  5332. flush_block_only(s, false);
  5333. if (s.strm.avail_out === 0) {
  5334. return BS_NEED_MORE;
  5335. }
  5336. /***/
  5337. }
  5338. return BS_BLOCK_DONE;
  5339. }
  5340. /* ===========================================================================
  5341. * For Z_HUFFMAN_ONLY, do not look for matches. Do not maintain a hash table.
  5342. * (It will be regenerated if this run of deflate switches away from Huffman.)
  5343. */
  5344. function deflate_huff(s, flush) {
  5345. var bflush; /* set if current block must be flushed */
  5346. for (;;) {
  5347. /* Make sure that we have a literal to write. */
  5348. if (s.lookahead === 0) {
  5349. fill_window(s);
  5350. if (s.lookahead === 0) {
  5351. if (flush === Z_NO_FLUSH) {
  5352. return BS_NEED_MORE;
  5353. }
  5354. break; /* flush the current block */
  5355. }
  5356. }
  5357. /* Output a literal byte */
  5358. s.match_length = 0;
  5359. //Tracevv((stderr,"%c", s->window[s->strstart]));
  5360. /*** _tr_tally_lit(s, s.window[s.strstart], bflush); ***/
  5361. bflush = trees._tr_tally(s, 0, s.window[s.strstart]);
  5362. s.lookahead--;
  5363. s.strstart++;
  5364. if (bflush) {
  5365. /*** FLUSH_BLOCK(s, 0); ***/
  5366. flush_block_only(s, false);
  5367. if (s.strm.avail_out === 0) {
  5368. return BS_NEED_MORE;
  5369. }
  5370. /***/
  5371. }
  5372. }
  5373. s.insert = 0;
  5374. if (flush === Z_FINISH) {
  5375. /*** FLUSH_BLOCK(s, 1); ***/
  5376. flush_block_only(s, true);
  5377. if (s.strm.avail_out === 0) {
  5378. return BS_FINISH_STARTED;
  5379. }
  5380. /***/
  5381. return BS_FINISH_DONE;
  5382. }
  5383. if (s.last_lit) {
  5384. /*** FLUSH_BLOCK(s, 0); ***/
  5385. flush_block_only(s, false);
  5386. if (s.strm.avail_out === 0) {
  5387. return BS_NEED_MORE;
  5388. }
  5389. /***/
  5390. }
  5391. return BS_BLOCK_DONE;
  5392. }
  5393. /* Values for max_lazy_match, good_match and max_chain_length, depending on
  5394. * the desired pack level (0..9). The values given below have been tuned to
  5395. * exclude worst case performance for pathological files. Better values may be
  5396. * found for specific files.
  5397. */
  5398. var Config = function (good_length, max_lazy, nice_length, max_chain, func) {
  5399. this.good_length = good_length;
  5400. this.max_lazy = max_lazy;
  5401. this.nice_length = nice_length;
  5402. this.max_chain = max_chain;
  5403. this.func = func;
  5404. };
  5405. var configuration_table;
  5406. configuration_table = [
  5407. /* good lazy nice chain */
  5408. new Config(0, 0, 0, 0, deflate_stored), /* 0 store only */
  5409. new Config(4, 4, 8, 4, deflate_fast), /* 1 max speed, no lazy matches */
  5410. new Config(4, 5, 16, 8, deflate_fast), /* 2 */
  5411. new Config(4, 6, 32, 32, deflate_fast), /* 3 */
  5412. new Config(4, 4, 16, 16, deflate_slow), /* 4 lazy matches */
  5413. new Config(8, 16, 32, 32, deflate_slow), /* 5 */
  5414. new Config(8, 16, 128, 128, deflate_slow), /* 6 */
  5415. new Config(8, 32, 128, 256, deflate_slow), /* 7 */
  5416. new Config(32, 128, 258, 1024, deflate_slow), /* 8 */
  5417. new Config(32, 258, 258, 4096, deflate_slow) /* 9 max compression */
  5418. ];
  5419. /* ===========================================================================
  5420. * Initialize the "longest match" routines for a new zlib stream
  5421. */
  5422. function lm_init(s) {
  5423. s.window_size = 2 * s.w_size;
  5424. /*** CLEAR_HASH(s); ***/
  5425. zero(s.head); // Fill with NIL (= 0);
  5426. /* Set the default configuration parameters:
  5427. */
  5428. s.max_lazy_match = configuration_table[s.level].max_lazy;
  5429. s.good_match = configuration_table[s.level].good_length;
  5430. s.nice_match = configuration_table[s.level].nice_length;
  5431. s.max_chain_length = configuration_table[s.level].max_chain;
  5432. s.strstart = 0;
  5433. s.block_start = 0;
  5434. s.lookahead = 0;
  5435. s.insert = 0;
  5436. s.match_length = s.prev_length = MIN_MATCH - 1;
  5437. s.match_available = 0;
  5438. s.ins_h = 0;
  5439. }
  5440. function DeflateState() {
  5441. this.strm = null; /* pointer back to this zlib stream */
  5442. this.status = 0; /* as the name implies */
  5443. this.pending_buf = null; /* output still pending */
  5444. this.pending_buf_size = 0; /* size of pending_buf */
  5445. this.pending_out = 0; /* next pending byte to output to the stream */
  5446. this.pending = 0; /* nb of bytes in the pending buffer */
  5447. this.wrap = 0; /* bit 0 true for zlib, bit 1 true for gzip */
  5448. this.gzhead = null; /* gzip header information to write */
  5449. this.gzindex = 0; /* where in extra, name, or comment */
  5450. this.method = Z_DEFLATED; /* can only be DEFLATED */
  5451. this.last_flush = -1; /* value of flush param for previous deflate call */
  5452. this.w_size = 0; /* LZ77 window size (32K by default) */
  5453. this.w_bits = 0; /* log2(w_size) (8..16) */
  5454. this.w_mask = 0; /* w_size - 1 */
  5455. this.window = null;
  5456. /* Sliding window. Input bytes are read into the second half of the window,
  5457. * and move to the first half later to keep a dictionary of at least wSize
  5458. * bytes. With this organization, matches are limited to a distance of
  5459. * wSize-MAX_MATCH bytes, but this ensures that IO is always
  5460. * performed with a length multiple of the block size.
  5461. */
  5462. this.window_size = 0;
  5463. /* Actual size of window: 2*wSize, except when the user input buffer
  5464. * is directly used as sliding window.
  5465. */
  5466. this.prev = null;
  5467. /* Link to older string with same hash index. To limit the size of this
  5468. * array to 64K, this link is maintained only for the last 32K strings.
  5469. * An index in this array is thus a window index modulo 32K.
  5470. */
  5471. this.head = null; /* Heads of the hash chains or NIL. */
  5472. this.ins_h = 0; /* hash index of string to be inserted */
  5473. this.hash_size = 0; /* number of elements in hash table */
  5474. this.hash_bits = 0; /* log2(hash_size) */
  5475. this.hash_mask = 0; /* hash_size-1 */
  5476. this.hash_shift = 0;
  5477. /* Number of bits by which ins_h must be shifted at each input
  5478. * step. It must be such that after MIN_MATCH steps, the oldest
  5479. * byte no longer takes part in the hash key, that is:
  5480. * hash_shift * MIN_MATCH >= hash_bits
  5481. */
  5482. this.block_start = 0;
  5483. /* Window position at the beginning of the current output block. Gets
  5484. * negative when the window is moved backwards.
  5485. */
  5486. this.match_length = 0; /* length of best match */
  5487. this.prev_match = 0; /* previous match */
  5488. this.match_available = 0; /* set if previous match exists */
  5489. this.strstart = 0; /* start of string to insert */
  5490. this.match_start = 0; /* start of matching string */
  5491. this.lookahead = 0; /* number of valid bytes ahead in window */
  5492. this.prev_length = 0;
  5493. /* Length of the best match at previous step. Matches not greater than this
  5494. * are discarded. This is used in the lazy match evaluation.
  5495. */
  5496. this.max_chain_length = 0;
  5497. /* To speed up deflation, hash chains are never searched beyond this
  5498. * length. A higher limit improves compression ratio but degrades the
  5499. * speed.
  5500. */
  5501. this.max_lazy_match = 0;
  5502. /* Attempt to find a better match only when the current match is strictly
  5503. * smaller than this value. This mechanism is used only for compression
  5504. * levels >= 4.
  5505. */
  5506. // That's alias to max_lazy_match, don't use directly
  5507. //this.max_insert_length = 0;
  5508. /* Insert new strings in the hash table only if the match length is not
  5509. * greater than this length. This saves time but degrades compression.
  5510. * max_insert_length is used only for compression levels <= 3.
  5511. */
  5512. this.level = 0; /* compression level (1..9) */
  5513. this.strategy = 0; /* favor or force Huffman coding*/
  5514. this.good_match = 0;
  5515. /* Use a faster search when the previous match is longer than this */
  5516. this.nice_match = 0; /* Stop searching when current match exceeds this */
  5517. /* used by trees.c: */
  5518. /* Didn't use ct_data typedef below to suppress compiler warning */
  5519. // struct ct_data_s dyn_ltree[HEAP_SIZE]; /* literal and length tree */
  5520. // struct ct_data_s dyn_dtree[2*D_CODES+1]; /* distance tree */
  5521. // struct ct_data_s bl_tree[2*BL_CODES+1]; /* Huffman tree for bit lengths */
  5522. // Use flat array of DOUBLE size, with interleaved fata,
  5523. // because JS does not support effective
  5524. this.dyn_ltree = new utils.Buf16(HEAP_SIZE * 2);
  5525. this.dyn_dtree = new utils.Buf16((2*D_CODES+1) * 2);
  5526. this.bl_tree = new utils.Buf16((2*BL_CODES+1) * 2);
  5527. zero(this.dyn_ltree);
  5528. zero(this.dyn_dtree);
  5529. zero(this.bl_tree);
  5530. this.l_desc = null; /* desc. for literal tree */
  5531. this.d_desc = null; /* desc. for distance tree */
  5532. this.bl_desc = null; /* desc. for bit length tree */
  5533. //ush bl_count[MAX_BITS+1];
  5534. this.bl_count = new utils.Buf16(MAX_BITS+1);
  5535. /* number of codes at each bit length for an optimal tree */
  5536. //int heap[2*L_CODES+1]; /* heap used to build the Huffman trees */
  5537. this.heap = new utils.Buf16(2*L_CODES+1); /* heap used to build the Huffman trees */
  5538. zero(this.heap);
  5539. this.heap_len = 0; /* number of elements in the heap */
  5540. this.heap_max = 0; /* element of largest frequency */
  5541. /* The sons of heap[n] are heap[2*n] and heap[2*n+1]. heap[0] is not used.
  5542. * The same heap array is used to build all trees.
  5543. */
  5544. this.depth = new utils.Buf16(2*L_CODES+1); //uch depth[2*L_CODES+1];
  5545. zero(this.depth);
  5546. /* Depth of each subtree used as tie breaker for trees of equal frequency
  5547. */
  5548. this.l_buf = 0; /* buffer index for literals or lengths */
  5549. this.lit_bufsize = 0;
  5550. /* Size of match buffer for literals/lengths. There are 4 reasons for
  5551. * limiting lit_bufsize to 64K:
  5552. * - frequencies can be kept in 16 bit counters
  5553. * - if compression is not successful for the first block, all input
  5554. * data is still in the window so we can still emit a stored block even
  5555. * when input comes from standard input. (This can also be done for
  5556. * all blocks if lit_bufsize is not greater than 32K.)
  5557. * - if compression is not successful for a file smaller than 64K, we can
  5558. * even emit a stored file instead of a stored block (saving 5 bytes).
  5559. * This is applicable only for zip (not gzip or zlib).
  5560. * - creating new Huffman trees less frequently may not provide fast
  5561. * adaptation to changes in the input data statistics. (Take for
  5562. * example a binary file with poorly compressible code followed by
  5563. * a highly compressible string table.) Smaller buffer sizes give
  5564. * fast adaptation but have of course the overhead of transmitting
  5565. * trees more frequently.
  5566. * - I can't count above 4
  5567. */
  5568. this.last_lit = 0; /* running index in l_buf */
  5569. this.d_buf = 0;
  5570. /* Buffer index for distances. To simplify the code, d_buf and l_buf have
  5571. * the same number of elements. To use different lengths, an extra flag
  5572. * array would be necessary.
  5573. */
  5574. this.opt_len = 0; /* bit length of current block with optimal trees */
  5575. this.static_len = 0; /* bit length of current block with static trees */
  5576. this.matches = 0; /* number of string matches in current block */
  5577. this.insert = 0; /* bytes at end of window left to insert */
  5578. this.bi_buf = 0;
  5579. /* Output buffer. bits are inserted starting at the bottom (least
  5580. * significant bits).
  5581. */
  5582. this.bi_valid = 0;
  5583. /* Number of valid bits in bi_buf. All bits above the last valid bit
  5584. * are always zero.
  5585. */
  5586. // Used for window memory init. We safely ignore it for JS. That makes
  5587. // sense only for pointers and memory check tools.
  5588. //this.high_water = 0;
  5589. /* High water mark offset in window for initialized bytes -- bytes above
  5590. * this are set to zero in order to avoid memory check warnings when
  5591. * longest match routines access bytes past the input. This is then
  5592. * updated to the new high water mark.
  5593. */
  5594. }
  5595. function deflateResetKeep(strm) {
  5596. var s;
  5597. if (!strm || !strm.state) {
  5598. return err(strm, Z_STREAM_ERROR);
  5599. }
  5600. strm.total_in = strm.total_out = 0;
  5601. strm.data_type = Z_UNKNOWN;
  5602. s = strm.state;
  5603. s.pending = 0;
  5604. s.pending_out = 0;
  5605. if (s.wrap < 0) {
  5606. s.wrap = -s.wrap;
  5607. /* was made negative by deflate(..., Z_FINISH); */
  5608. }
  5609. s.status = (s.wrap ? INIT_STATE : BUSY_STATE);
  5610. strm.adler = (s.wrap === 2) ?
  5611. 0 // crc32(0, Z_NULL, 0)
  5612. :
  5613. 1; // adler32(0, Z_NULL, 0)
  5614. s.last_flush = Z_NO_FLUSH;
  5615. trees._tr_init(s);
  5616. return Z_OK;
  5617. }
  5618. function deflateReset(strm) {
  5619. var ret = deflateResetKeep(strm);
  5620. if (ret === Z_OK) {
  5621. lm_init(strm.state);
  5622. }
  5623. return ret;
  5624. }
  5625. function deflateSetHeader(strm, head) {
  5626. if (!strm || !strm.state) { return Z_STREAM_ERROR; }
  5627. if (strm.state.wrap !== 2) { return Z_STREAM_ERROR; }
  5628. strm.state.gzhead = head;
  5629. return Z_OK;
  5630. }
  5631. function deflateInit2(strm, level, method, windowBits, memLevel, strategy) {
  5632. if (!strm) { // === Z_NULL
  5633. return Z_STREAM_ERROR;
  5634. }
  5635. var wrap = 1;
  5636. if (level === Z_DEFAULT_COMPRESSION) {
  5637. level = 6;
  5638. }
  5639. if (windowBits < 0) { /* suppress zlib wrapper */
  5640. wrap = 0;
  5641. windowBits = -windowBits;
  5642. }
  5643. else if (windowBits > 15) {
  5644. wrap = 2; /* write gzip wrapper instead */
  5645. windowBits -= 16;
  5646. }
  5647. if (memLevel < 1 || memLevel > MAX_MEM_LEVEL || method !== Z_DEFLATED ||
  5648. windowBits < 8 || windowBits > 15 || level < 0 || level > 9 ||
  5649. strategy < 0 || strategy > Z_FIXED) {
  5650. return err(strm, Z_STREAM_ERROR);
  5651. }
  5652. if (windowBits === 8) {
  5653. windowBits = 9;
  5654. }
  5655. /* until 256-byte window bug fixed */
  5656. var s = new DeflateState();
  5657. strm.state = s;
  5658. s.strm = strm;
  5659. s.wrap = wrap;
  5660. s.gzhead = null;
  5661. s.w_bits = windowBits;
  5662. s.w_size = 1 << s.w_bits;
  5663. s.w_mask = s.w_size - 1;
  5664. s.hash_bits = memLevel + 7;
  5665. s.hash_size = 1 << s.hash_bits;
  5666. s.hash_mask = s.hash_size - 1;
  5667. s.hash_shift = ~~((s.hash_bits + MIN_MATCH - 1) / MIN_MATCH);
  5668. s.window = new utils.Buf8(s.w_size * 2);
  5669. s.head = new utils.Buf16(s.hash_size);
  5670. s.prev = new utils.Buf16(s.w_size);
  5671. // Don't need mem init magic for JS.
  5672. //s.high_water = 0; /* nothing written to s->window yet */
  5673. s.lit_bufsize = 1 << (memLevel + 6); /* 16K elements by default */
  5674. s.pending_buf_size = s.lit_bufsize * 4;
  5675. s.pending_buf = new utils.Buf8(s.pending_buf_size);
  5676. s.d_buf = s.lit_bufsize >> 1;
  5677. s.l_buf = (1 + 2) * s.lit_bufsize;
  5678. s.level = level;
  5679. s.strategy = strategy;
  5680. s.method = method;
  5681. return deflateReset(strm);
  5682. }
  5683. function deflateInit(strm, level) {
  5684. return deflateInit2(strm, level, Z_DEFLATED, MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY);
  5685. }
  5686. function deflate(strm, flush) {
  5687. var old_flush, s;
  5688. var beg, val; // for gzip header write only
  5689. if (!strm || !strm.state ||
  5690. flush > Z_BLOCK || flush < 0) {
  5691. return strm ? err(strm, Z_STREAM_ERROR) : Z_STREAM_ERROR;
  5692. }
  5693. s = strm.state;
  5694. if (!strm.output ||
  5695. (!strm.input && strm.avail_in !== 0) ||
  5696. (s.status === FINISH_STATE && flush !== Z_FINISH)) {
  5697. return err(strm, (strm.avail_out === 0) ? Z_BUF_ERROR : Z_STREAM_ERROR);
  5698. }
  5699. s.strm = strm; /* just in case */
  5700. old_flush = s.last_flush;
  5701. s.last_flush = flush;
  5702. /* Write the header */
  5703. if (s.status === INIT_STATE) {
  5704. if (s.wrap === 2) { // GZIP header
  5705. strm.adler = 0; //crc32(0L, Z_NULL, 0);
  5706. put_byte(s, 31);
  5707. put_byte(s, 139);
  5708. put_byte(s, 8);
  5709. if (!s.gzhead) { // s->gzhead == Z_NULL
  5710. put_byte(s, 0);
  5711. put_byte(s, 0);
  5712. put_byte(s, 0);
  5713. put_byte(s, 0);
  5714. put_byte(s, 0);
  5715. put_byte(s, s.level === 9 ? 2 :
  5716. (s.strategy >= Z_HUFFMAN_ONLY || s.level < 2 ?
  5717. 4 : 0));
  5718. put_byte(s, OS_CODE);
  5719. s.status = BUSY_STATE;
  5720. }
  5721. else {
  5722. put_byte(s, (s.gzhead.text ? 1 : 0) +
  5723. (s.gzhead.hcrc ? 2 : 0) +
  5724. (!s.gzhead.extra ? 0 : 4) +
  5725. (!s.gzhead.name ? 0 : 8) +
  5726. (!s.gzhead.comment ? 0 : 16)
  5727. );
  5728. put_byte(s, s.gzhead.time & 0xff);
  5729. put_byte(s, (s.gzhead.time >> 8) & 0xff);
  5730. put_byte(s, (s.gzhead.time >> 16) & 0xff);
  5731. put_byte(s, (s.gzhead.time >> 24) & 0xff);
  5732. put_byte(s, s.level === 9 ? 2 :
  5733. (s.strategy >= Z_HUFFMAN_ONLY || s.level < 2 ?
  5734. 4 : 0));
  5735. put_byte(s, s.gzhead.os & 0xff);
  5736. if (s.gzhead.extra && s.gzhead.extra.length) {
  5737. put_byte(s, s.gzhead.extra.length & 0xff);
  5738. put_byte(s, (s.gzhead.extra.length >> 8) & 0xff);
  5739. }
  5740. if (s.gzhead.hcrc) {
  5741. strm.adler = crc32(strm.adler, s.pending_buf, s.pending, 0);
  5742. }
  5743. s.gzindex = 0;
  5744. s.status = EXTRA_STATE;
  5745. }
  5746. }
  5747. else // DEFLATE header
  5748. {
  5749. var header = (Z_DEFLATED + ((s.w_bits - 8) << 4)) << 8;
  5750. var level_flags = -1;
  5751. if (s.strategy >= Z_HUFFMAN_ONLY || s.level < 2) {
  5752. level_flags = 0;
  5753. } else if (s.level < 6) {
  5754. level_flags = 1;
  5755. } else if (s.level === 6) {
  5756. level_flags = 2;
  5757. } else {
  5758. level_flags = 3;
  5759. }
  5760. header |= (level_flags << 6);
  5761. if (s.strstart !== 0) { header |= PRESET_DICT; }
  5762. header += 31 - (header % 31);
  5763. s.status = BUSY_STATE;
  5764. putShortMSB(s, header);
  5765. /* Save the adler32 of the preset dictionary: */
  5766. if (s.strstart !== 0) {
  5767. putShortMSB(s, strm.adler >>> 16);
  5768. putShortMSB(s, strm.adler & 0xffff);
  5769. }
  5770. strm.adler = 1; // adler32(0L, Z_NULL, 0);
  5771. }
  5772. }
  5773. //#ifdef GZIP
  5774. if (s.status === EXTRA_STATE) {
  5775. if (s.gzhead.extra/* != Z_NULL*/) {
  5776. beg = s.pending; /* start of bytes to update crc */
  5777. while (s.gzindex < (s.gzhead.extra.length & 0xffff)) {
  5778. if (s.pending === s.pending_buf_size) {
  5779. if (s.gzhead.hcrc && s.pending > beg) {
  5780. strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg);
  5781. }
  5782. flush_pending(strm);
  5783. beg = s.pending;
  5784. if (s.pending === s.pending_buf_size) {
  5785. break;
  5786. }
  5787. }
  5788. put_byte(s, s.gzhead.extra[s.gzindex] & 0xff);
  5789. s.gzindex++;
  5790. }
  5791. if (s.gzhead.hcrc && s.pending > beg) {
  5792. strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg);
  5793. }
  5794. if (s.gzindex === s.gzhead.extra.length) {
  5795. s.gzindex = 0;
  5796. s.status = NAME_STATE;
  5797. }
  5798. }
  5799. else {
  5800. s.status = NAME_STATE;
  5801. }
  5802. }
  5803. if (s.status === NAME_STATE) {
  5804. if (s.gzhead.name/* != Z_NULL*/) {
  5805. beg = s.pending; /* start of bytes to update crc */
  5806. //int val;
  5807. do {
  5808. if (s.pending === s.pending_buf_size) {
  5809. if (s.gzhead.hcrc && s.pending > beg) {
  5810. strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg);
  5811. }
  5812. flush_pending(strm);
  5813. beg = s.pending;
  5814. if (s.pending === s.pending_buf_size) {
  5815. val = 1;
  5816. break;
  5817. }
  5818. }
  5819. // JS specific: little magic to add zero terminator to end of string
  5820. if (s.gzindex < s.gzhead.name.length) {
  5821. val = s.gzhead.name.charCodeAt(s.gzindex++) & 0xff;
  5822. } else {
  5823. val = 0;
  5824. }
  5825. put_byte(s, val);
  5826. } while (val !== 0);
  5827. if (s.gzhead.hcrc && s.pending > beg){
  5828. strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg);
  5829. }
  5830. if (val === 0) {
  5831. s.gzindex = 0;
  5832. s.status = COMMENT_STATE;
  5833. }
  5834. }
  5835. else {
  5836. s.status = COMMENT_STATE;
  5837. }
  5838. }
  5839. if (s.status === COMMENT_STATE) {
  5840. if (s.gzhead.comment/* != Z_NULL*/) {
  5841. beg = s.pending; /* start of bytes to update crc */
  5842. //int val;
  5843. do {
  5844. if (s.pending === s.pending_buf_size) {
  5845. if (s.gzhead.hcrc && s.pending > beg) {
  5846. strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg);
  5847. }
  5848. flush_pending(strm);
  5849. beg = s.pending;
  5850. if (s.pending === s.pending_buf_size) {
  5851. val = 1;
  5852. break;
  5853. }
  5854. }
  5855. // JS specific: little magic to add zero terminator to end of string
  5856. if (s.gzindex < s.gzhead.comment.length) {
  5857. val = s.gzhead.comment.charCodeAt(s.gzindex++) & 0xff;
  5858. } else {
  5859. val = 0;
  5860. }
  5861. put_byte(s, val);
  5862. } while (val !== 0);
  5863. if (s.gzhead.hcrc && s.pending > beg) {
  5864. strm.adler = crc32(strm.adler, s.pending_buf, s.pending - beg, beg);
  5865. }
  5866. if (val === 0) {
  5867. s.status = HCRC_STATE;
  5868. }
  5869. }
  5870. else {
  5871. s.status = HCRC_STATE;
  5872. }
  5873. }
  5874. if (s.status === HCRC_STATE) {
  5875. if (s.gzhead.hcrc) {
  5876. if (s.pending + 2 > s.pending_buf_size) {
  5877. flush_pending(strm);
  5878. }
  5879. if (s.pending + 2 <= s.pending_buf_size) {
  5880. put_byte(s, strm.adler & 0xff);
  5881. put_byte(s, (strm.adler >> 8) & 0xff);
  5882. strm.adler = 0; //crc32(0L, Z_NULL, 0);
  5883. s.status = BUSY_STATE;
  5884. }
  5885. }
  5886. else {
  5887. s.status = BUSY_STATE;
  5888. }
  5889. }
  5890. //#endif
  5891. /* Flush as much pending output as possible */
  5892. if (s.pending !== 0) {
  5893. flush_pending(strm);
  5894. if (strm.avail_out === 0) {
  5895. /* Since avail_out is 0, deflate will be called again with
  5896. * more output space, but possibly with both pending and
  5897. * avail_in equal to zero. There won't be anything to do,
  5898. * but this is not an error situation so make sure we
  5899. * return OK instead of BUF_ERROR at next call of deflate:
  5900. */
  5901. s.last_flush = -1;
  5902. return Z_OK;
  5903. }
  5904. /* Make sure there is something to do and avoid duplicate consecutive
  5905. * flushes. For repeated and useless calls with Z_FINISH, we keep
  5906. * returning Z_STREAM_END instead of Z_BUF_ERROR.
  5907. */
  5908. } else if (strm.avail_in === 0 && rank(flush) <= rank(old_flush) &&
  5909. flush !== Z_FINISH) {
  5910. return err(strm, Z_BUF_ERROR);
  5911. }
  5912. /* User must not provide more input after the first FINISH: */
  5913. if (s.status === FINISH_STATE && strm.avail_in !== 0) {
  5914. return err(strm, Z_BUF_ERROR);
  5915. }
  5916. /* Start a new block or continue the current one.
  5917. */
  5918. if (strm.avail_in !== 0 || s.lookahead !== 0 ||
  5919. (flush !== Z_NO_FLUSH && s.status !== FINISH_STATE)) {
  5920. var bstate = (s.strategy === Z_HUFFMAN_ONLY) ? deflate_huff(s, flush) :
  5921. (s.strategy === Z_RLE ? deflate_rle(s, flush) :
  5922. configuration_table[s.level].func(s, flush));
  5923. if (bstate === BS_FINISH_STARTED || bstate === BS_FINISH_DONE) {
  5924. s.status = FINISH_STATE;
  5925. }
  5926. if (bstate === BS_NEED_MORE || bstate === BS_FINISH_STARTED) {
  5927. if (strm.avail_out === 0) {
  5928. s.last_flush = -1;
  5929. /* avoid BUF_ERROR next call, see above */
  5930. }
  5931. return Z_OK;
  5932. /* If flush != Z_NO_FLUSH && avail_out == 0, the next call
  5933. * of deflate should use the same flush parameter to make sure
  5934. * that the flush is complete. So we don't have to output an
  5935. * empty block here, this will be done at next call. This also
  5936. * ensures that for a very small output buffer, we emit at most
  5937. * one empty block.
  5938. */
  5939. }
  5940. if (bstate === BS_BLOCK_DONE) {
  5941. if (flush === Z_PARTIAL_FLUSH) {
  5942. trees._tr_align(s);
  5943. }
  5944. else if (flush !== Z_BLOCK) { /* FULL_FLUSH or SYNC_FLUSH */
  5945. trees._tr_stored_block(s, 0, 0, false);
  5946. /* For a full flush, this empty block will be recognized
  5947. * as a special marker by inflate_sync().
  5948. */
  5949. if (flush === Z_FULL_FLUSH) {
  5950. /*** CLEAR_HASH(s); ***/ /* forget history */
  5951. zero(s.head); // Fill with NIL (= 0);
  5952. if (s.lookahead === 0) {
  5953. s.strstart = 0;
  5954. s.block_start = 0;
  5955. s.insert = 0;
  5956. }
  5957. }
  5958. }
  5959. flush_pending(strm);
  5960. if (strm.avail_out === 0) {
  5961. s.last_flush = -1; /* avoid BUF_ERROR at next call, see above */
  5962. return Z_OK;
  5963. }
  5964. }
  5965. }
  5966. //Assert(strm->avail_out > 0, "bug2");
  5967. //if (strm.avail_out <= 0) { throw new error("bug2");}
  5968. if (flush !== Z_FINISH) { return Z_OK; }
  5969. if (s.wrap <= 0) { return Z_STREAM_END; }
  5970. /* Write the trailer */
  5971. if (s.wrap === 2) {
  5972. put_byte(s, strm.adler & 0xff);
  5973. put_byte(s, (strm.adler >> 8) & 0xff);
  5974. put_byte(s, (strm.adler >> 16) & 0xff);
  5975. put_byte(s, (strm.adler >> 24) & 0xff);
  5976. put_byte(s, strm.total_in & 0xff);
  5977. put_byte(s, (strm.total_in >> 8) & 0xff);
  5978. put_byte(s, (strm.total_in >> 16) & 0xff);
  5979. put_byte(s, (strm.total_in >> 24) & 0xff);
  5980. }
  5981. else
  5982. {
  5983. putShortMSB(s, strm.adler >>> 16);
  5984. putShortMSB(s, strm.adler & 0xffff);
  5985. }
  5986. flush_pending(strm);
  5987. /* If avail_out is zero, the application will call deflate again
  5988. * to flush the rest.
  5989. */
  5990. if (s.wrap > 0) { s.wrap = -s.wrap; }
  5991. /* write the trailer only once! */
  5992. return s.pending !== 0 ? Z_OK : Z_STREAM_END;
  5993. }
  5994. function deflateEnd(strm) {
  5995. var status;
  5996. if (!strm/*== Z_NULL*/ || !strm.state/*== Z_NULL*/) {
  5997. return Z_STREAM_ERROR;
  5998. }
  5999. status = strm.state.status;
  6000. if (status !== INIT_STATE &&
  6001. status !== EXTRA_STATE &&
  6002. status !== NAME_STATE &&
  6003. status !== COMMENT_STATE &&
  6004. status !== HCRC_STATE &&
  6005. status !== BUSY_STATE &&
  6006. status !== FINISH_STATE
  6007. ) {
  6008. return err(strm, Z_STREAM_ERROR);
  6009. }
  6010. strm.state = null;
  6011. return status === BUSY_STATE ? err(strm, Z_DATA_ERROR) : Z_OK;
  6012. }
  6013. /* =========================================================================
  6014. * Copy the source state to the destination state
  6015. */
  6016. //function deflateCopy(dest, source) {
  6017. //
  6018. //}
  6019. exports.deflateInit = deflateInit;
  6020. exports.deflateInit2 = deflateInit2;
  6021. exports.deflateReset = deflateReset;
  6022. exports.deflateResetKeep = deflateResetKeep;
  6023. exports.deflateSetHeader = deflateSetHeader;
  6024. exports.deflate = deflate;
  6025. exports.deflateEnd = deflateEnd;
  6026. exports.deflateInfo = 'pako deflate (from Nodeca project)';
  6027. /* Not implemented
  6028. exports.deflateBound = deflateBound;
  6029. exports.deflateCopy = deflateCopy;
  6030. exports.deflateSetDictionary = deflateSetDictionary;
  6031. exports.deflateParams = deflateParams;
  6032. exports.deflatePending = deflatePending;
  6033. exports.deflatePrime = deflatePrime;
  6034. exports.deflateTune = deflateTune;
  6035. */
  6036. },{"../utils/common":27,"./adler32":29,"./crc32":31,"./messages":37,"./trees":38}],33:[function(_dereq_,module,exports){
  6037. 'use strict';
  6038. function GZheader() {
  6039. /* true if compressed data believed to be text */
  6040. this.text = 0;
  6041. /* modification time */
  6042. this.time = 0;
  6043. /* extra flags (not used when writing a gzip file) */
  6044. this.xflags = 0;
  6045. /* operating system */
  6046. this.os = 0;
  6047. /* pointer to extra field or Z_NULL if none */
  6048. this.extra = null;
  6049. /* extra field length (valid if extra != Z_NULL) */
  6050. this.extra_len = 0; // Actually, we don't need it in JS,
  6051. // but leave for few code modifications
  6052. //
  6053. // Setup limits is not necessary because in js we should not preallocate memory
  6054. // for inflate use constant limit in 65536 bytes
  6055. //
  6056. /* space at extra (only when reading header) */
  6057. // this.extra_max = 0;
  6058. /* pointer to zero-terminated file name or Z_NULL */
  6059. this.name = '';
  6060. /* space at name (only when reading header) */
  6061. // this.name_max = 0;
  6062. /* pointer to zero-terminated comment or Z_NULL */
  6063. this.comment = '';
  6064. /* space at comment (only when reading header) */
  6065. // this.comm_max = 0;
  6066. /* true if there was or will be a header crc */
  6067. this.hcrc = 0;
  6068. /* true when done reading gzip header (not used when writing a gzip file) */
  6069. this.done = false;
  6070. }
  6071. module.exports = GZheader;
  6072. },{}],34:[function(_dereq_,module,exports){
  6073. 'use strict';
  6074. // See state defs from inflate.js
  6075. var BAD = 30; /* got a data error -- remain here until reset */
  6076. var TYPE = 12; /* i: waiting for type bits, including last-flag bit */
  6077. /*
  6078. Decode literal, length, and distance codes and write out the resulting
  6079. literal and match bytes until either not enough input or output is
  6080. available, an end-of-block is encountered, or a data error is encountered.
  6081. When large enough input and output buffers are supplied to inflate(), for
  6082. example, a 16K input buffer and a 64K output buffer, more than 95% of the
  6083. inflate execution time is spent in this routine.
  6084. Entry assumptions:
  6085. state.mode === LEN
  6086. strm.avail_in >= 6
  6087. strm.avail_out >= 258
  6088. start >= strm.avail_out
  6089. state.bits < 8
  6090. On return, state.mode is one of:
  6091. LEN -- ran out of enough output space or enough available input
  6092. TYPE -- reached end of block code, inflate() to interpret next block
  6093. BAD -- error in block data
  6094. Notes:
  6095. - The maximum input bits used by a length/distance pair is 15 bits for the
  6096. length code, 5 bits for the length extra, 15 bits for the distance code,
  6097. and 13 bits for the distance extra. This totals 48 bits, or six bytes.
  6098. Therefore if strm.avail_in >= 6, then there is enough input to avoid
  6099. checking for available input while decoding.
  6100. - The maximum bytes that a single length/distance pair can output is 258
  6101. bytes, which is the maximum length that can be coded. inflate_fast()
  6102. requires strm.avail_out >= 258 for each loop to avoid checking for
  6103. output space.
  6104. */
  6105. module.exports = function inflate_fast(strm, start) {
  6106. var state;
  6107. var _in; /* local strm.input */
  6108. var last; /* have enough input while in < last */
  6109. var _out; /* local strm.output */
  6110. var beg; /* inflate()'s initial strm.output */
  6111. var end; /* while out < end, enough space available */
  6112. //#ifdef INFLATE_STRICT
  6113. var dmax; /* maximum distance from zlib header */
  6114. //#endif
  6115. var wsize; /* window size or zero if not using window */
  6116. var whave; /* valid bytes in the window */
  6117. var wnext; /* window write index */
  6118. var window; /* allocated sliding window, if wsize != 0 */
  6119. var hold; /* local strm.hold */
  6120. var bits; /* local strm.bits */
  6121. var lcode; /* local strm.lencode */
  6122. var dcode; /* local strm.distcode */
  6123. var lmask; /* mask for first level of length codes */
  6124. var dmask; /* mask for first level of distance codes */
  6125. var here; /* retrieved table entry */
  6126. var op; /* code bits, operation, extra bits, or */
  6127. /* window position, window bytes to copy */
  6128. var len; /* match length, unused bytes */
  6129. var dist; /* match distance */
  6130. var from; /* where to copy match from */
  6131. var from_source;
  6132. var input, output; // JS specific, because we have no pointers
  6133. /* copy state to local variables */
  6134. state = strm.state;
  6135. //here = state.here;
  6136. _in = strm.next_in;
  6137. input = strm.input;
  6138. last = _in + (strm.avail_in - 5);
  6139. _out = strm.next_out;
  6140. output = strm.output;
  6141. beg = _out - (start - strm.avail_out);
  6142. end = _out + (strm.avail_out - 257);
  6143. //#ifdef INFLATE_STRICT
  6144. dmax = state.dmax;
  6145. //#endif
  6146. wsize = state.wsize;
  6147. whave = state.whave;
  6148. wnext = state.wnext;
  6149. window = state.window;
  6150. hold = state.hold;
  6151. bits = state.bits;
  6152. lcode = state.lencode;
  6153. dcode = state.distcode;
  6154. lmask = (1 << state.lenbits) - 1;
  6155. dmask = (1 << state.distbits) - 1;
  6156. /* decode literals and length/distances until end-of-block or not enough
  6157. input data or output space */
  6158. top:
  6159. do {
  6160. if (bits < 15) {
  6161. hold += input[_in++] << bits;
  6162. bits += 8;
  6163. hold += input[_in++] << bits;
  6164. bits += 8;
  6165. }
  6166. here = lcode[hold & lmask];
  6167. dolen:
  6168. for (;;) { // Goto emulation
  6169. op = here >>> 24/*here.bits*/;
  6170. hold >>>= op;
  6171. bits -= op;
  6172. op = (here >>> 16) & 0xff/*here.op*/;
  6173. if (op === 0) { /* literal */
  6174. //Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ?
  6175. // "inflate: literal '%c'\n" :
  6176. // "inflate: literal 0x%02x\n", here.val));
  6177. output[_out++] = here & 0xffff/*here.val*/;
  6178. }
  6179. else if (op & 16) { /* length base */
  6180. len = here & 0xffff/*here.val*/;
  6181. op &= 15; /* number of extra bits */
  6182. if (op) {
  6183. if (bits < op) {
  6184. hold += input[_in++] << bits;
  6185. bits += 8;
  6186. }
  6187. len += hold & ((1 << op) - 1);
  6188. hold >>>= op;
  6189. bits -= op;
  6190. }
  6191. //Tracevv((stderr, "inflate: length %u\n", len));
  6192. if (bits < 15) {
  6193. hold += input[_in++] << bits;
  6194. bits += 8;
  6195. hold += input[_in++] << bits;
  6196. bits += 8;
  6197. }
  6198. here = dcode[hold & dmask];
  6199. dodist:
  6200. for (;;) { // goto emulation
  6201. op = here >>> 24/*here.bits*/;
  6202. hold >>>= op;
  6203. bits -= op;
  6204. op = (here >>> 16) & 0xff/*here.op*/;
  6205. if (op & 16) { /* distance base */
  6206. dist = here & 0xffff/*here.val*/;
  6207. op &= 15; /* number of extra bits */
  6208. if (bits < op) {
  6209. hold += input[_in++] << bits;
  6210. bits += 8;
  6211. if (bits < op) {
  6212. hold += input[_in++] << bits;
  6213. bits += 8;
  6214. }
  6215. }
  6216. dist += hold & ((1 << op) - 1);
  6217. //#ifdef INFLATE_STRICT
  6218. if (dist > dmax) {
  6219. strm.msg = 'invalid distance too far back';
  6220. state.mode = BAD;
  6221. break top;
  6222. }
  6223. //#endif
  6224. hold >>>= op;
  6225. bits -= op;
  6226. //Tracevv((stderr, "inflate: distance %u\n", dist));
  6227. op = _out - beg; /* max distance in output */
  6228. if (dist > op) { /* see if copy from window */
  6229. op = dist - op; /* distance back in window */
  6230. if (op > whave) {
  6231. if (state.sane) {
  6232. strm.msg = 'invalid distance too far back';
  6233. state.mode = BAD;
  6234. break top;
  6235. }
  6236. // (!) This block is disabled in zlib defailts,
  6237. // don't enable it for binary compatibility
  6238. //#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR
  6239. // if (len <= op - whave) {
  6240. // do {
  6241. // output[_out++] = 0;
  6242. // } while (--len);
  6243. // continue top;
  6244. // }
  6245. // len -= op - whave;
  6246. // do {
  6247. // output[_out++] = 0;
  6248. // } while (--op > whave);
  6249. // if (op === 0) {
  6250. // from = _out - dist;
  6251. // do {
  6252. // output[_out++] = output[from++];
  6253. // } while (--len);
  6254. // continue top;
  6255. // }
  6256. //#endif
  6257. }
  6258. from = 0; // window index
  6259. from_source = window;
  6260. if (wnext === 0) { /* very common case */
  6261. from += wsize - op;
  6262. if (op < len) { /* some from window */
  6263. len -= op;
  6264. do {
  6265. output[_out++] = window[from++];
  6266. } while (--op);
  6267. from = _out - dist; /* rest from output */
  6268. from_source = output;
  6269. }
  6270. }
  6271. else if (wnext < op) { /* wrap around window */
  6272. from += wsize + wnext - op;
  6273. op -= wnext;
  6274. if (op < len) { /* some from end of window */
  6275. len -= op;
  6276. do {
  6277. output[_out++] = window[from++];
  6278. } while (--op);
  6279. from = 0;
  6280. if (wnext < len) { /* some from start of window */
  6281. op = wnext;
  6282. len -= op;
  6283. do {
  6284. output[_out++] = window[from++];
  6285. } while (--op);
  6286. from = _out - dist; /* rest from output */
  6287. from_source = output;
  6288. }
  6289. }
  6290. }
  6291. else { /* contiguous in window */
  6292. from += wnext - op;
  6293. if (op < len) { /* some from window */
  6294. len -= op;
  6295. do {
  6296. output[_out++] = window[from++];
  6297. } while (--op);
  6298. from = _out - dist; /* rest from output */
  6299. from_source = output;
  6300. }
  6301. }
  6302. while (len > 2) {
  6303. output[_out++] = from_source[from++];
  6304. output[_out++] = from_source[from++];
  6305. output[_out++] = from_source[from++];
  6306. len -= 3;
  6307. }
  6308. if (len) {
  6309. output[_out++] = from_source[from++];
  6310. if (len > 1) {
  6311. output[_out++] = from_source[from++];
  6312. }
  6313. }
  6314. }
  6315. else {
  6316. from = _out - dist; /* copy direct from output */
  6317. do { /* minimum length is three */
  6318. output[_out++] = output[from++];
  6319. output[_out++] = output[from++];
  6320. output[_out++] = output[from++];
  6321. len -= 3;
  6322. } while (len > 2);
  6323. if (len) {
  6324. output[_out++] = output[from++];
  6325. if (len > 1) {
  6326. output[_out++] = output[from++];
  6327. }
  6328. }
  6329. }
  6330. }
  6331. else if ((op & 64) === 0) { /* 2nd level distance code */
  6332. here = dcode[(here & 0xffff)/*here.val*/ + (hold & ((1 << op) - 1))];
  6333. continue dodist;
  6334. }
  6335. else {
  6336. strm.msg = 'invalid distance code';
  6337. state.mode = BAD;
  6338. break top;
  6339. }
  6340. break; // need to emulate goto via "continue"
  6341. }
  6342. }
  6343. else if ((op & 64) === 0) { /* 2nd level length code */
  6344. here = lcode[(here & 0xffff)/*here.val*/ + (hold & ((1 << op) - 1))];
  6345. continue dolen;
  6346. }
  6347. else if (op & 32) { /* end-of-block */
  6348. //Tracevv((stderr, "inflate: end of block\n"));
  6349. state.mode = TYPE;
  6350. break top;
  6351. }
  6352. else {
  6353. strm.msg = 'invalid literal/length code';
  6354. state.mode = BAD;
  6355. break top;
  6356. }
  6357. break; // need to emulate goto via "continue"
  6358. }
  6359. } while (_in < last && _out < end);
  6360. /* return unused bytes (on entry, bits < 8, so in won't go too far back) */
  6361. len = bits >> 3;
  6362. _in -= len;
  6363. bits -= len << 3;
  6364. hold &= (1 << bits) - 1;
  6365. /* update state and return */
  6366. strm.next_in = _in;
  6367. strm.next_out = _out;
  6368. strm.avail_in = (_in < last ? 5 + (last - _in) : 5 - (_in - last));
  6369. strm.avail_out = (_out < end ? 257 + (end - _out) : 257 - (_out - end));
  6370. state.hold = hold;
  6371. state.bits = bits;
  6372. return;
  6373. };
  6374. },{}],35:[function(_dereq_,module,exports){
  6375. 'use strict';
  6376. var utils = _dereq_('../utils/common');
  6377. var adler32 = _dereq_('./adler32');
  6378. var crc32 = _dereq_('./crc32');
  6379. var inflate_fast = _dereq_('./inffast');
  6380. var inflate_table = _dereq_('./inftrees');
  6381. var CODES = 0;
  6382. var LENS = 1;
  6383. var DISTS = 2;
  6384. /* Public constants ==========================================================*/
  6385. /* ===========================================================================*/
  6386. /* Allowed flush values; see deflate() and inflate() below for details */
  6387. //var Z_NO_FLUSH = 0;
  6388. //var Z_PARTIAL_FLUSH = 1;
  6389. //var Z_SYNC_FLUSH = 2;
  6390. //var Z_FULL_FLUSH = 3;
  6391. var Z_FINISH = 4;
  6392. var Z_BLOCK = 5;
  6393. var Z_TREES = 6;
  6394. /* Return codes for the compression/decompression functions. Negative values
  6395. * are errors, positive values are used for special but normal events.
  6396. */
  6397. var Z_OK = 0;
  6398. var Z_STREAM_END = 1;
  6399. var Z_NEED_DICT = 2;
  6400. //var Z_ERRNO = -1;
  6401. var Z_STREAM_ERROR = -2;
  6402. var Z_DATA_ERROR = -3;
  6403. var Z_MEM_ERROR = -4;
  6404. var Z_BUF_ERROR = -5;
  6405. //var Z_VERSION_ERROR = -6;
  6406. /* The deflate compression method */
  6407. var Z_DEFLATED = 8;
  6408. /* STATES ====================================================================*/
  6409. /* ===========================================================================*/
  6410. var HEAD = 1; /* i: waiting for magic header */
  6411. var FLAGS = 2; /* i: waiting for method and flags (gzip) */
  6412. var TIME = 3; /* i: waiting for modification time (gzip) */
  6413. var OS = 4; /* i: waiting for extra flags and operating system (gzip) */
  6414. var EXLEN = 5; /* i: waiting for extra length (gzip) */
  6415. var EXTRA = 6; /* i: waiting for extra bytes (gzip) */
  6416. var NAME = 7; /* i: waiting for end of file name (gzip) */
  6417. var COMMENT = 8; /* i: waiting for end of comment (gzip) */
  6418. var HCRC = 9; /* i: waiting for header crc (gzip) */
  6419. var DICTID = 10; /* i: waiting for dictionary check value */
  6420. var DICT = 11; /* waiting for inflateSetDictionary() call */
  6421. var TYPE = 12; /* i: waiting for type bits, including last-flag bit */
  6422. var TYPEDO = 13; /* i: same, but skip check to exit inflate on new block */
  6423. var STORED = 14; /* i: waiting for stored size (length and complement) */
  6424. var COPY_ = 15; /* i/o: same as COPY below, but only first time in */
  6425. var COPY = 16; /* i/o: waiting for input or output to copy stored block */
  6426. var TABLE = 17; /* i: waiting for dynamic block table lengths */
  6427. var LENLENS = 18; /* i: waiting for code length code lengths */
  6428. var CODELENS = 19; /* i: waiting for length/lit and distance code lengths */
  6429. var LEN_ = 20; /* i: same as LEN below, but only first time in */
  6430. var LEN = 21; /* i: waiting for length/lit/eob code */
  6431. var LENEXT = 22; /* i: waiting for length extra bits */
  6432. var DIST = 23; /* i: waiting for distance code */
  6433. var DISTEXT = 24; /* i: waiting for distance extra bits */
  6434. var MATCH = 25; /* o: waiting for output space to copy string */
  6435. var LIT = 26; /* o: waiting for output space to write literal */
  6436. var CHECK = 27; /* i: waiting for 32-bit check value */
  6437. var LENGTH = 28; /* i: waiting for 32-bit length (gzip) */
  6438. var DONE = 29; /* finished check, done -- remain here until reset */
  6439. var BAD = 30; /* got a data error -- remain here until reset */
  6440. var MEM = 31; /* got an inflate() memory error -- remain here until reset */
  6441. var SYNC = 32; /* looking for synchronization bytes to restart inflate() */
  6442. /* ===========================================================================*/
  6443. var ENOUGH_LENS = 852;
  6444. var ENOUGH_DISTS = 592;
  6445. //var ENOUGH = (ENOUGH_LENS+ENOUGH_DISTS);
  6446. var MAX_WBITS = 15;
  6447. /* 32K LZ77 window */
  6448. var DEF_WBITS = MAX_WBITS;
  6449. function ZSWAP32(q) {
  6450. return (((q >>> 24) & 0xff) +
  6451. ((q >>> 8) & 0xff00) +
  6452. ((q & 0xff00) << 8) +
  6453. ((q & 0xff) << 24));
  6454. }
  6455. function InflateState() {
  6456. this.mode = 0; /* current inflate mode */
  6457. this.last = false; /* true if processing last block */
  6458. this.wrap = 0; /* bit 0 true for zlib, bit 1 true for gzip */
  6459. this.havedict = false; /* true if dictionary provided */
  6460. this.flags = 0; /* gzip header method and flags (0 if zlib) */
  6461. this.dmax = 0; /* zlib header max distance (INFLATE_STRICT) */
  6462. this.check = 0; /* protected copy of check value */
  6463. this.total = 0; /* protected copy of output count */
  6464. // TODO: may be {}
  6465. this.head = null; /* where to save gzip header information */
  6466. /* sliding window */
  6467. this.wbits = 0; /* log base 2 of requested window size */
  6468. this.wsize = 0; /* window size or zero if not using window */
  6469. this.whave = 0; /* valid bytes in the window */
  6470. this.wnext = 0; /* window write index */
  6471. this.window = null; /* allocated sliding window, if needed */
  6472. /* bit accumulator */
  6473. this.hold = 0; /* input bit accumulator */
  6474. this.bits = 0; /* number of bits in "in" */
  6475. /* for string and stored block copying */
  6476. this.length = 0; /* literal or length of data to copy */
  6477. this.offset = 0; /* distance back to copy string from */
  6478. /* for table and code decoding */
  6479. this.extra = 0; /* extra bits needed */
  6480. /* fixed and dynamic code tables */
  6481. this.lencode = null; /* starting table for length/literal codes */
  6482. this.distcode = null; /* starting table for distance codes */
  6483. this.lenbits = 0; /* index bits for lencode */
  6484. this.distbits = 0; /* index bits for distcode */
  6485. /* dynamic table building */
  6486. this.ncode = 0; /* number of code length code lengths */
  6487. this.nlen = 0; /* number of length code lengths */
  6488. this.ndist = 0; /* number of distance code lengths */
  6489. this.have = 0; /* number of code lengths in lens[] */
  6490. this.next = null; /* next available space in codes[] */
  6491. this.lens = new utils.Buf16(320); /* temporary storage for code lengths */
  6492. this.work = new utils.Buf16(288); /* work area for code table building */
  6493. /*
  6494. because we don't have pointers in js, we use lencode and distcode directly
  6495. as buffers so we don't need codes
  6496. */
  6497. //this.codes = new utils.Buf32(ENOUGH); /* space for code tables */
  6498. this.lendyn = null; /* dynamic table for length/literal codes (JS specific) */
  6499. this.distdyn = null; /* dynamic table for distance codes (JS specific) */
  6500. this.sane = 0; /* if false, allow invalid distance too far */
  6501. this.back = 0; /* bits back of last unprocessed length/lit */
  6502. this.was = 0; /* initial length of match */
  6503. }
  6504. function inflateResetKeep(strm) {
  6505. var state;
  6506. if (!strm || !strm.state) { return Z_STREAM_ERROR; }
  6507. state = strm.state;
  6508. strm.total_in = strm.total_out = state.total = 0;
  6509. strm.msg = ''; /*Z_NULL*/
  6510. if (state.wrap) { /* to support ill-conceived Java test suite */
  6511. strm.adler = state.wrap & 1;
  6512. }
  6513. state.mode = HEAD;
  6514. state.last = 0;
  6515. state.havedict = 0;
  6516. state.dmax = 32768;
  6517. state.head = null/*Z_NULL*/;
  6518. state.hold = 0;
  6519. state.bits = 0;
  6520. //state.lencode = state.distcode = state.next = state.codes;
  6521. state.lencode = state.lendyn = new utils.Buf32(ENOUGH_LENS);
  6522. state.distcode = state.distdyn = new utils.Buf32(ENOUGH_DISTS);
  6523. state.sane = 1;
  6524. state.back = -1;
  6525. //Tracev((stderr, "inflate: reset\n"));
  6526. return Z_OK;
  6527. }
  6528. function inflateReset(strm) {
  6529. var state;
  6530. if (!strm || !strm.state) { return Z_STREAM_ERROR; }
  6531. state = strm.state;
  6532. state.wsize = 0;
  6533. state.whave = 0;
  6534. state.wnext = 0;
  6535. return inflateResetKeep(strm);
  6536. }
  6537. function inflateReset2(strm, windowBits) {
  6538. var wrap;
  6539. var state;
  6540. /* get the state */
  6541. if (!strm || !strm.state) { return Z_STREAM_ERROR; }
  6542. state = strm.state;
  6543. /* extract wrap request from windowBits parameter */
  6544. if (windowBits < 0) {
  6545. wrap = 0;
  6546. windowBits = -windowBits;
  6547. }
  6548. else {
  6549. wrap = (windowBits >> 4) + 1;
  6550. if (windowBits < 48) {
  6551. windowBits &= 15;
  6552. }
  6553. }
  6554. /* set number of window bits, free window if different */
  6555. if (windowBits && (windowBits < 8 || windowBits > 15)) {
  6556. return Z_STREAM_ERROR;
  6557. }
  6558. if (state.window !== null && state.wbits !== windowBits) {
  6559. state.window = null;
  6560. }
  6561. /* update state and reset the rest of it */
  6562. state.wrap = wrap;
  6563. state.wbits = windowBits;
  6564. return inflateReset(strm);
  6565. }
  6566. function inflateInit2(strm, windowBits) {
  6567. var ret;
  6568. var state;
  6569. if (!strm) { return Z_STREAM_ERROR; }
  6570. //strm.msg = Z_NULL; /* in case we return an error */
  6571. state = new InflateState();
  6572. //if (state === Z_NULL) return Z_MEM_ERROR;
  6573. //Tracev((stderr, "inflate: allocated\n"));
  6574. strm.state = state;
  6575. state.window = null/*Z_NULL*/;
  6576. ret = inflateReset2(strm, windowBits);
  6577. if (ret !== Z_OK) {
  6578. strm.state = null/*Z_NULL*/;
  6579. }
  6580. return ret;
  6581. }
  6582. function inflateInit(strm) {
  6583. return inflateInit2(strm, DEF_WBITS);
  6584. }
  6585. /*
  6586. Return state with length and distance decoding tables and index sizes set to
  6587. fixed code decoding. Normally this returns fixed tables from inffixed.h.
  6588. If BUILDFIXED is defined, then instead this routine builds the tables the
  6589. first time it's called, and returns those tables the first time and
  6590. thereafter. This reduces the size of the code by about 2K bytes, in
  6591. exchange for a little execution time. However, BUILDFIXED should not be
  6592. used for threaded applications, since the rewriting of the tables and virgin
  6593. may not be thread-safe.
  6594. */
  6595. var virgin = true;
  6596. var lenfix, distfix; // We have no pointers in JS, so keep tables separate
  6597. function fixedtables(state) {
  6598. /* build fixed huffman tables if first call (may not be thread safe) */
  6599. if (virgin) {
  6600. var sym;
  6601. lenfix = new utils.Buf32(512);
  6602. distfix = new utils.Buf32(32);
  6603. /* literal/length table */
  6604. sym = 0;
  6605. while (sym < 144) { state.lens[sym++] = 8; }
  6606. while (sym < 256) { state.lens[sym++] = 9; }
  6607. while (sym < 280) { state.lens[sym++] = 7; }
  6608. while (sym < 288) { state.lens[sym++] = 8; }
  6609. inflate_table(LENS, state.lens, 0, 288, lenfix, 0, state.work, {bits: 9});
  6610. /* distance table */
  6611. sym = 0;
  6612. while (sym < 32) { state.lens[sym++] = 5; }
  6613. inflate_table(DISTS, state.lens, 0, 32, distfix, 0, state.work, {bits: 5});
  6614. /* do this just once */
  6615. virgin = false;
  6616. }
  6617. state.lencode = lenfix;
  6618. state.lenbits = 9;
  6619. state.distcode = distfix;
  6620. state.distbits = 5;
  6621. }
  6622. /*
  6623. Update the window with the last wsize (normally 32K) bytes written before
  6624. returning. If window does not exist yet, create it. This is only called
  6625. when a window is already in use, or when output has been written during this
  6626. inflate call, but the end of the deflate stream has not been reached yet.
  6627. It is also called to create a window for dictionary data when a dictionary
  6628. is loaded.
  6629. Providing output buffers larger than 32K to inflate() should provide a speed
  6630. advantage, since only the last 32K of output is copied to the sliding window
  6631. upon return from inflate(), and since all distances after the first 32K of
  6632. output will fall in the output data, making match copies simpler and faster.
  6633. The advantage may be dependent on the size of the processor's data caches.
  6634. */
  6635. function updatewindow(strm, src, end, copy) {
  6636. var dist;
  6637. var state = strm.state;
  6638. /* if it hasn't been done already, allocate space for the window */
  6639. if (state.window === null) {
  6640. state.wsize = 1 << state.wbits;
  6641. state.wnext = 0;
  6642. state.whave = 0;
  6643. state.window = new utils.Buf8(state.wsize);
  6644. }
  6645. /* copy state->wsize or less output bytes into the circular window */
  6646. if (copy >= state.wsize) {
  6647. utils.arraySet(state.window,src, end - state.wsize, state.wsize, 0);
  6648. state.wnext = 0;
  6649. state.whave = state.wsize;
  6650. }
  6651. else {
  6652. dist = state.wsize - state.wnext;
  6653. if (dist > copy) {
  6654. dist = copy;
  6655. }
  6656. //zmemcpy(state->window + state->wnext, end - copy, dist);
  6657. utils.arraySet(state.window,src, end - copy, dist, state.wnext);
  6658. copy -= dist;
  6659. if (copy) {
  6660. //zmemcpy(state->window, end - copy, copy);
  6661. utils.arraySet(state.window,src, end - copy, copy, 0);
  6662. state.wnext = copy;
  6663. state.whave = state.wsize;
  6664. }
  6665. else {
  6666. state.wnext += dist;
  6667. if (state.wnext === state.wsize) { state.wnext = 0; }
  6668. if (state.whave < state.wsize) { state.whave += dist; }
  6669. }
  6670. }
  6671. return 0;
  6672. }
  6673. function inflate(strm, flush) {
  6674. var state;
  6675. var input, output; // input/output buffers
  6676. var next; /* next input INDEX */
  6677. var put; /* next output INDEX */
  6678. var have, left; /* available input and output */
  6679. var hold; /* bit buffer */
  6680. var bits; /* bits in bit buffer */
  6681. var _in, _out; /* save starting available input and output */
  6682. var copy; /* number of stored or match bytes to copy */
  6683. var from; /* where to copy match bytes from */
  6684. var from_source;
  6685. var here = 0; /* current decoding table entry */
  6686. var here_bits, here_op, here_val; // paked "here" denormalized (JS specific)
  6687. //var last; /* parent table entry */
  6688. var last_bits, last_op, last_val; // paked "last" denormalized (JS specific)
  6689. var len; /* length to copy for repeats, bits to drop */
  6690. var ret; /* return code */
  6691. var hbuf = new utils.Buf8(4); /* buffer for gzip header crc calculation */
  6692. var opts;
  6693. var n; // temporary var for NEED_BITS
  6694. var order = /* permutation of code lengths */
  6695. [16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15];
  6696. if (!strm || !strm.state || !strm.output ||
  6697. (!strm.input && strm.avail_in !== 0)) {
  6698. return Z_STREAM_ERROR;
  6699. }
  6700. state = strm.state;
  6701. if (state.mode === TYPE) { state.mode = TYPEDO; } /* skip check */
  6702. //--- LOAD() ---
  6703. put = strm.next_out;
  6704. output = strm.output;
  6705. left = strm.avail_out;
  6706. next = strm.next_in;
  6707. input = strm.input;
  6708. have = strm.avail_in;
  6709. hold = state.hold;
  6710. bits = state.bits;
  6711. //---
  6712. _in = have;
  6713. _out = left;
  6714. ret = Z_OK;
  6715. inf_leave: // goto emulation
  6716. for (;;) {
  6717. switch (state.mode) {
  6718. case HEAD:
  6719. if (state.wrap === 0) {
  6720. state.mode = TYPEDO;
  6721. break;
  6722. }
  6723. //=== NEEDBITS(16);
  6724. while (bits < 16) {
  6725. if (have === 0) { break inf_leave; }
  6726. have--;
  6727. hold += input[next++] << bits;
  6728. bits += 8;
  6729. }
  6730. //===//
  6731. if ((state.wrap & 2) && hold === 0x8b1f) { /* gzip header */
  6732. state.check = 0/*crc32(0L, Z_NULL, 0)*/;
  6733. //=== CRC2(state.check, hold);
  6734. hbuf[0] = hold & 0xff;
  6735. hbuf[1] = (hold >>> 8) & 0xff;
  6736. state.check = crc32(state.check, hbuf, 2, 0);
  6737. //===//
  6738. //=== INITBITS();
  6739. hold = 0;
  6740. bits = 0;
  6741. //===//
  6742. state.mode = FLAGS;
  6743. break;
  6744. }
  6745. state.flags = 0; /* expect zlib header */
  6746. if (state.head) {
  6747. state.head.done = false;
  6748. }
  6749. if (!(state.wrap & 1) || /* check if zlib header allowed */
  6750. (((hold & 0xff)/*BITS(8)*/ << 8) + (hold >> 8)) % 31) {
  6751. strm.msg = 'incorrect header check';
  6752. state.mode = BAD;
  6753. break;
  6754. }
  6755. if ((hold & 0x0f)/*BITS(4)*/ !== Z_DEFLATED) {
  6756. strm.msg = 'unknown compression method';
  6757. state.mode = BAD;
  6758. break;
  6759. }
  6760. //--- DROPBITS(4) ---//
  6761. hold >>>= 4;
  6762. bits -= 4;
  6763. //---//
  6764. len = (hold & 0x0f)/*BITS(4)*/ + 8;
  6765. if (state.wbits === 0) {
  6766. state.wbits = len;
  6767. }
  6768. else if (len > state.wbits) {
  6769. strm.msg = 'invalid window size';
  6770. state.mode = BAD;
  6771. break;
  6772. }
  6773. state.dmax = 1 << len;
  6774. //Tracev((stderr, "inflate: zlib header ok\n"));
  6775. strm.adler = state.check = 1/*adler32(0L, Z_NULL, 0)*/;
  6776. state.mode = hold & 0x200 ? DICTID : TYPE;
  6777. //=== INITBITS();
  6778. hold = 0;
  6779. bits = 0;
  6780. //===//
  6781. break;
  6782. case FLAGS:
  6783. //=== NEEDBITS(16); */
  6784. while (bits < 16) {
  6785. if (have === 0) { break inf_leave; }
  6786. have--;
  6787. hold += input[next++] << bits;
  6788. bits += 8;
  6789. }
  6790. //===//
  6791. state.flags = hold;
  6792. if ((state.flags & 0xff) !== Z_DEFLATED) {
  6793. strm.msg = 'unknown compression method';
  6794. state.mode = BAD;
  6795. break;
  6796. }
  6797. if (state.flags & 0xe000) {
  6798. strm.msg = 'unknown header flags set';
  6799. state.mode = BAD;
  6800. break;
  6801. }
  6802. if (state.head) {
  6803. state.head.text = ((hold >> 8) & 1);
  6804. }
  6805. if (state.flags & 0x0200) {
  6806. //=== CRC2(state.check, hold);
  6807. hbuf[0] = hold & 0xff;
  6808. hbuf[1] = (hold >>> 8) & 0xff;
  6809. state.check = crc32(state.check, hbuf, 2, 0);
  6810. //===//
  6811. }
  6812. //=== INITBITS();
  6813. hold = 0;
  6814. bits = 0;
  6815. //===//
  6816. state.mode = TIME;
  6817. /* falls through */
  6818. case TIME:
  6819. //=== NEEDBITS(32); */
  6820. while (bits < 32) {
  6821. if (have === 0) { break inf_leave; }
  6822. have--;
  6823. hold += input[next++] << bits;
  6824. bits += 8;
  6825. }
  6826. //===//
  6827. if (state.head) {
  6828. state.head.time = hold;
  6829. }
  6830. if (state.flags & 0x0200) {
  6831. //=== CRC4(state.check, hold)
  6832. hbuf[0] = hold & 0xff;
  6833. hbuf[1] = (hold >>> 8) & 0xff;
  6834. hbuf[2] = (hold >>> 16) & 0xff;
  6835. hbuf[3] = (hold >>> 24) & 0xff;
  6836. state.check = crc32(state.check, hbuf, 4, 0);
  6837. //===
  6838. }
  6839. //=== INITBITS();
  6840. hold = 0;
  6841. bits = 0;
  6842. //===//
  6843. state.mode = OS;
  6844. /* falls through */
  6845. case OS:
  6846. //=== NEEDBITS(16); */
  6847. while (bits < 16) {
  6848. if (have === 0) { break inf_leave; }
  6849. have--;
  6850. hold += input[next++] << bits;
  6851. bits += 8;
  6852. }
  6853. //===//
  6854. if (state.head) {
  6855. state.head.xflags = (hold & 0xff);
  6856. state.head.os = (hold >> 8);
  6857. }
  6858. if (state.flags & 0x0200) {
  6859. //=== CRC2(state.check, hold);
  6860. hbuf[0] = hold & 0xff;
  6861. hbuf[1] = (hold >>> 8) & 0xff;
  6862. state.check = crc32(state.check, hbuf, 2, 0);
  6863. //===//
  6864. }
  6865. //=== INITBITS();
  6866. hold = 0;
  6867. bits = 0;
  6868. //===//
  6869. state.mode = EXLEN;
  6870. /* falls through */
  6871. case EXLEN:
  6872. if (state.flags & 0x0400) {
  6873. //=== NEEDBITS(16); */
  6874. while (bits < 16) {
  6875. if (have === 0) { break inf_leave; }
  6876. have--;
  6877. hold += input[next++] << bits;
  6878. bits += 8;
  6879. }
  6880. //===//
  6881. state.length = hold;
  6882. if (state.head) {
  6883. state.head.extra_len = hold;
  6884. }
  6885. if (state.flags & 0x0200) {
  6886. //=== CRC2(state.check, hold);
  6887. hbuf[0] = hold & 0xff;
  6888. hbuf[1] = (hold >>> 8) & 0xff;
  6889. state.check = crc32(state.check, hbuf, 2, 0);
  6890. //===//
  6891. }
  6892. //=== INITBITS();
  6893. hold = 0;
  6894. bits = 0;
  6895. //===//
  6896. }
  6897. else if (state.head) {
  6898. state.head.extra = null/*Z_NULL*/;
  6899. }
  6900. state.mode = EXTRA;
  6901. /* falls through */
  6902. case EXTRA:
  6903. if (state.flags & 0x0400) {
  6904. copy = state.length;
  6905. if (copy > have) { copy = have; }
  6906. if (copy) {
  6907. if (state.head) {
  6908. len = state.head.extra_len - state.length;
  6909. if (!state.head.extra) {
  6910. // Use untyped array for more conveniend processing later
  6911. state.head.extra = new Array(state.head.extra_len);
  6912. }
  6913. utils.arraySet(
  6914. state.head.extra,
  6915. input,
  6916. next,
  6917. // extra field is limited to 65536 bytes
  6918. // - no need for additional size check
  6919. copy,
  6920. /*len + copy > state.head.extra_max - len ? state.head.extra_max : copy,*/
  6921. len
  6922. );
  6923. //zmemcpy(state.head.extra + len, next,
  6924. // len + copy > state.head.extra_max ?
  6925. // state.head.extra_max - len : copy);
  6926. }
  6927. if (state.flags & 0x0200) {
  6928. state.check = crc32(state.check, input, copy, next);
  6929. }
  6930. have -= copy;
  6931. next += copy;
  6932. state.length -= copy;
  6933. }
  6934. if (state.length) { break inf_leave; }
  6935. }
  6936. state.length = 0;
  6937. state.mode = NAME;
  6938. /* falls through */
  6939. case NAME:
  6940. if (state.flags & 0x0800) {
  6941. if (have === 0) { break inf_leave; }
  6942. copy = 0;
  6943. do {
  6944. // TODO: 2 or 1 bytes?
  6945. len = input[next + copy++];
  6946. /* use constant limit because in js we should not preallocate memory */
  6947. if (state.head && len &&
  6948. (state.length < 65536 /*state.head.name_max*/)) {
  6949. state.head.name += String.fromCharCode(len);
  6950. }
  6951. } while (len && copy < have);
  6952. if (state.flags & 0x0200) {
  6953. state.check = crc32(state.check, input, copy, next);
  6954. }
  6955. have -= copy;
  6956. next += copy;
  6957. if (len) { break inf_leave; }
  6958. }
  6959. else if (state.head) {
  6960. state.head.name = null;
  6961. }
  6962. state.length = 0;
  6963. state.mode = COMMENT;
  6964. /* falls through */
  6965. case COMMENT:
  6966. if (state.flags & 0x1000) {
  6967. if (have === 0) { break inf_leave; }
  6968. copy = 0;
  6969. do {
  6970. len = input[next + copy++];
  6971. /* use constant limit because in js we should not preallocate memory */
  6972. if (state.head && len &&
  6973. (state.length < 65536 /*state.head.comm_max*/)) {
  6974. state.head.comment += String.fromCharCode(len);
  6975. }
  6976. } while (len && copy < have);
  6977. if (state.flags & 0x0200) {
  6978. state.check = crc32(state.check, input, copy, next);
  6979. }
  6980. have -= copy;
  6981. next += copy;
  6982. if (len) { break inf_leave; }
  6983. }
  6984. else if (state.head) {
  6985. state.head.comment = null;
  6986. }
  6987. state.mode = HCRC;
  6988. /* falls through */
  6989. case HCRC:
  6990. if (state.flags & 0x0200) {
  6991. //=== NEEDBITS(16); */
  6992. while (bits < 16) {
  6993. if (have === 0) { break inf_leave; }
  6994. have--;
  6995. hold += input[next++] << bits;
  6996. bits += 8;
  6997. }
  6998. //===//
  6999. if (hold !== (state.check & 0xffff)) {
  7000. strm.msg = 'header crc mismatch';
  7001. state.mode = BAD;
  7002. break;
  7003. }
  7004. //=== INITBITS();
  7005. hold = 0;
  7006. bits = 0;
  7007. //===//
  7008. }
  7009. if (state.head) {
  7010. state.head.hcrc = ((state.flags >> 9) & 1);
  7011. state.head.done = true;
  7012. }
  7013. strm.adler = state.check = 0 /*crc32(0L, Z_NULL, 0)*/;
  7014. state.mode = TYPE;
  7015. break;
  7016. case DICTID:
  7017. //=== NEEDBITS(32); */
  7018. while (bits < 32) {
  7019. if (have === 0) { break inf_leave; }
  7020. have--;
  7021. hold += input[next++] << bits;
  7022. bits += 8;
  7023. }
  7024. //===//
  7025. strm.adler = state.check = ZSWAP32(hold);
  7026. //=== INITBITS();
  7027. hold = 0;
  7028. bits = 0;
  7029. //===//
  7030. state.mode = DICT;
  7031. /* falls through */
  7032. case DICT:
  7033. if (state.havedict === 0) {
  7034. //--- RESTORE() ---
  7035. strm.next_out = put;
  7036. strm.avail_out = left;
  7037. strm.next_in = next;
  7038. strm.avail_in = have;
  7039. state.hold = hold;
  7040. state.bits = bits;
  7041. //---
  7042. return Z_NEED_DICT;
  7043. }
  7044. strm.adler = state.check = 1/*adler32(0L, Z_NULL, 0)*/;
  7045. state.mode = TYPE;
  7046. /* falls through */
  7047. case TYPE:
  7048. if (flush === Z_BLOCK || flush === Z_TREES) { break inf_leave; }
  7049. /* falls through */
  7050. case TYPEDO:
  7051. if (state.last) {
  7052. //--- BYTEBITS() ---//
  7053. hold >>>= bits & 7;
  7054. bits -= bits & 7;
  7055. //---//
  7056. state.mode = CHECK;
  7057. break;
  7058. }
  7059. //=== NEEDBITS(3); */
  7060. while (bits < 3) {
  7061. if (have === 0) { break inf_leave; }
  7062. have--;
  7063. hold += input[next++] << bits;
  7064. bits += 8;
  7065. }
  7066. //===//
  7067. state.last = (hold & 0x01)/*BITS(1)*/;
  7068. //--- DROPBITS(1) ---//
  7069. hold >>>= 1;
  7070. bits -= 1;
  7071. //---//
  7072. switch ((hold & 0x03)/*BITS(2)*/) {
  7073. case 0: /* stored block */
  7074. //Tracev((stderr, "inflate: stored block%s\n",
  7075. // state.last ? " (last)" : ""));
  7076. state.mode = STORED;
  7077. break;
  7078. case 1: /* fixed block */
  7079. fixedtables(state);
  7080. //Tracev((stderr, "inflate: fixed codes block%s\n",
  7081. // state.last ? " (last)" : ""));
  7082. state.mode = LEN_; /* decode codes */
  7083. if (flush === Z_TREES) {
  7084. //--- DROPBITS(2) ---//
  7085. hold >>>= 2;
  7086. bits -= 2;
  7087. //---//
  7088. break inf_leave;
  7089. }
  7090. break;
  7091. case 2: /* dynamic block */
  7092. //Tracev((stderr, "inflate: dynamic codes block%s\n",
  7093. // state.last ? " (last)" : ""));
  7094. state.mode = TABLE;
  7095. break;
  7096. case 3:
  7097. strm.msg = 'invalid block type';
  7098. state.mode = BAD;
  7099. }
  7100. //--- DROPBITS(2) ---//
  7101. hold >>>= 2;
  7102. bits -= 2;
  7103. //---//
  7104. break;
  7105. case STORED:
  7106. //--- BYTEBITS() ---// /* go to byte boundary */
  7107. hold >>>= bits & 7;
  7108. bits -= bits & 7;
  7109. //---//
  7110. //=== NEEDBITS(32); */
  7111. while (bits < 32) {
  7112. if (have === 0) { break inf_leave; }
  7113. have--;
  7114. hold += input[next++] << bits;
  7115. bits += 8;
  7116. }
  7117. //===//
  7118. if ((hold & 0xffff) !== ((hold >>> 16) ^ 0xffff)) {
  7119. strm.msg = 'invalid stored block lengths';
  7120. state.mode = BAD;
  7121. break;
  7122. }
  7123. state.length = hold & 0xffff;
  7124. //Tracev((stderr, "inflate: stored length %u\n",
  7125. // state.length));
  7126. //=== INITBITS();
  7127. hold = 0;
  7128. bits = 0;
  7129. //===//
  7130. state.mode = COPY_;
  7131. if (flush === Z_TREES) { break inf_leave; }
  7132. /* falls through */
  7133. case COPY_:
  7134. state.mode = COPY;
  7135. /* falls through */
  7136. case COPY:
  7137. copy = state.length;
  7138. if (copy) {
  7139. if (copy > have) { copy = have; }
  7140. if (copy > left) { copy = left; }
  7141. if (copy === 0) { break inf_leave; }
  7142. //--- zmemcpy(put, next, copy); ---
  7143. utils.arraySet(output, input, next, copy, put);
  7144. //---//
  7145. have -= copy;
  7146. next += copy;
  7147. left -= copy;
  7148. put += copy;
  7149. state.length -= copy;
  7150. break;
  7151. }
  7152. //Tracev((stderr, "inflate: stored end\n"));
  7153. state.mode = TYPE;
  7154. break;
  7155. case TABLE:
  7156. //=== NEEDBITS(14); */
  7157. while (bits < 14) {
  7158. if (have === 0) { break inf_leave; }
  7159. have--;
  7160. hold += input[next++] << bits;
  7161. bits += 8;
  7162. }
  7163. //===//
  7164. state.nlen = (hold & 0x1f)/*BITS(5)*/ + 257;
  7165. //--- DROPBITS(5) ---//
  7166. hold >>>= 5;
  7167. bits -= 5;
  7168. //---//
  7169. state.ndist = (hold & 0x1f)/*BITS(5)*/ + 1;
  7170. //--- DROPBITS(5) ---//
  7171. hold >>>= 5;
  7172. bits -= 5;
  7173. //---//
  7174. state.ncode = (hold & 0x0f)/*BITS(4)*/ + 4;
  7175. //--- DROPBITS(4) ---//
  7176. hold >>>= 4;
  7177. bits -= 4;
  7178. //---//
  7179. //#ifndef PKZIP_BUG_WORKAROUND
  7180. if (state.nlen > 286 || state.ndist > 30) {
  7181. strm.msg = 'too many length or distance symbols';
  7182. state.mode = BAD;
  7183. break;
  7184. }
  7185. //#endif
  7186. //Tracev((stderr, "inflate: table sizes ok\n"));
  7187. state.have = 0;
  7188. state.mode = LENLENS;
  7189. /* falls through */
  7190. case LENLENS:
  7191. while (state.have < state.ncode) {
  7192. //=== NEEDBITS(3);
  7193. while (bits < 3) {
  7194. if (have === 0) { break inf_leave; }
  7195. have--;
  7196. hold += input[next++] << bits;
  7197. bits += 8;
  7198. }
  7199. //===//
  7200. state.lens[order[state.have++]] = (hold & 0x07);//BITS(3);
  7201. //--- DROPBITS(3) ---//
  7202. hold >>>= 3;
  7203. bits -= 3;
  7204. //---//
  7205. }
  7206. while (state.have < 19) {
  7207. state.lens[order[state.have++]] = 0;
  7208. }
  7209. // We have separate tables & no pointers. 2 commented lines below not needed.
  7210. //state.next = state.codes;
  7211. //state.lencode = state.next;
  7212. // Switch to use dynamic table
  7213. state.lencode = state.lendyn;
  7214. state.lenbits = 7;
  7215. opts = {bits: state.lenbits};
  7216. ret = inflate_table(CODES, state.lens, 0, 19, state.lencode, 0, state.work, opts);
  7217. state.lenbits = opts.bits;
  7218. if (ret) {
  7219. strm.msg = 'invalid code lengths set';
  7220. state.mode = BAD;
  7221. break;
  7222. }
  7223. //Tracev((stderr, "inflate: code lengths ok\n"));
  7224. state.have = 0;
  7225. state.mode = CODELENS;
  7226. /* falls through */
  7227. case CODELENS:
  7228. while (state.have < state.nlen + state.ndist) {
  7229. for (;;) {
  7230. here = state.lencode[hold & ((1 << state.lenbits) - 1)];/*BITS(state.lenbits)*/
  7231. here_bits = here >>> 24;
  7232. here_op = (here >>> 16) & 0xff;
  7233. here_val = here & 0xffff;
  7234. if ((here_bits) <= bits) { break; }
  7235. //--- PULLBYTE() ---//
  7236. if (have === 0) { break inf_leave; }
  7237. have--;
  7238. hold += input[next++] << bits;
  7239. bits += 8;
  7240. //---//
  7241. }
  7242. if (here_val < 16) {
  7243. //--- DROPBITS(here.bits) ---//
  7244. hold >>>= here_bits;
  7245. bits -= here_bits;
  7246. //---//
  7247. state.lens[state.have++] = here_val;
  7248. }
  7249. else {
  7250. if (here_val === 16) {
  7251. //=== NEEDBITS(here.bits + 2);
  7252. n = here_bits + 2;
  7253. while (bits < n) {
  7254. if (have === 0) { break inf_leave; }
  7255. have--;
  7256. hold += input[next++] << bits;
  7257. bits += 8;
  7258. }
  7259. //===//
  7260. //--- DROPBITS(here.bits) ---//
  7261. hold >>>= here_bits;
  7262. bits -= here_bits;
  7263. //---//
  7264. if (state.have === 0) {
  7265. strm.msg = 'invalid bit length repeat';
  7266. state.mode = BAD;
  7267. break;
  7268. }
  7269. len = state.lens[state.have - 1];
  7270. copy = 3 + (hold & 0x03);//BITS(2);
  7271. //--- DROPBITS(2) ---//
  7272. hold >>>= 2;
  7273. bits -= 2;
  7274. //---//
  7275. }
  7276. else if (here_val === 17) {
  7277. //=== NEEDBITS(here.bits + 3);
  7278. n = here_bits + 3;
  7279. while (bits < n) {
  7280. if (have === 0) { break inf_leave; }
  7281. have--;
  7282. hold += input[next++] << bits;
  7283. bits += 8;
  7284. }
  7285. //===//
  7286. //--- DROPBITS(here.bits) ---//
  7287. hold >>>= here_bits;
  7288. bits -= here_bits;
  7289. //---//
  7290. len = 0;
  7291. copy = 3 + (hold & 0x07);//BITS(3);
  7292. //--- DROPBITS(3) ---//
  7293. hold >>>= 3;
  7294. bits -= 3;
  7295. //---//
  7296. }
  7297. else {
  7298. //=== NEEDBITS(here.bits + 7);
  7299. n = here_bits + 7;
  7300. while (bits < n) {
  7301. if (have === 0) { break inf_leave; }
  7302. have--;
  7303. hold += input[next++] << bits;
  7304. bits += 8;
  7305. }
  7306. //===//
  7307. //--- DROPBITS(here.bits) ---//
  7308. hold >>>= here_bits;
  7309. bits -= here_bits;
  7310. //---//
  7311. len = 0;
  7312. copy = 11 + (hold & 0x7f);//BITS(7);
  7313. //--- DROPBITS(7) ---//
  7314. hold >>>= 7;
  7315. bits -= 7;
  7316. //---//
  7317. }
  7318. if (state.have + copy > state.nlen + state.ndist) {
  7319. strm.msg = 'invalid bit length repeat';
  7320. state.mode = BAD;
  7321. break;
  7322. }
  7323. while (copy--) {
  7324. state.lens[state.have++] = len;
  7325. }
  7326. }
  7327. }
  7328. /* handle error breaks in while */
  7329. if (state.mode === BAD) { break; }
  7330. /* check for end-of-block code (better have one) */
  7331. if (state.lens[256] === 0) {
  7332. strm.msg = 'invalid code -- missing end-of-block';
  7333. state.mode = BAD;
  7334. break;
  7335. }
  7336. /* build code tables -- note: do not change the lenbits or distbits
  7337. values here (9 and 6) without reading the comments in inftrees.h
  7338. concerning the ENOUGH constants, which depend on those values */
  7339. state.lenbits = 9;
  7340. opts = {bits: state.lenbits};
  7341. ret = inflate_table(LENS, state.lens, 0, state.nlen, state.lencode, 0, state.work, opts);
  7342. // We have separate tables & no pointers. 2 commented lines below not needed.
  7343. // state.next_index = opts.table_index;
  7344. state.lenbits = opts.bits;
  7345. // state.lencode = state.next;
  7346. if (ret) {
  7347. strm.msg = 'invalid literal/lengths set';
  7348. state.mode = BAD;
  7349. break;
  7350. }
  7351. state.distbits = 6;
  7352. //state.distcode.copy(state.codes);
  7353. // Switch to use dynamic table
  7354. state.distcode = state.distdyn;
  7355. opts = {bits: state.distbits};
  7356. ret = inflate_table(DISTS, state.lens, state.nlen, state.ndist, state.distcode, 0, state.work, opts);
  7357. // We have separate tables & no pointers. 2 commented lines below not needed.
  7358. // state.next_index = opts.table_index;
  7359. state.distbits = opts.bits;
  7360. // state.distcode = state.next;
  7361. if (ret) {
  7362. strm.msg = 'invalid distances set';
  7363. state.mode = BAD;
  7364. break;
  7365. }
  7366. //Tracev((stderr, 'inflate: codes ok\n'));
  7367. state.mode = LEN_;
  7368. if (flush === Z_TREES) { break inf_leave; }
  7369. /* falls through */
  7370. case LEN_:
  7371. state.mode = LEN;
  7372. /* falls through */
  7373. case LEN:
  7374. if (have >= 6 && left >= 258) {
  7375. //--- RESTORE() ---
  7376. strm.next_out = put;
  7377. strm.avail_out = left;
  7378. strm.next_in = next;
  7379. strm.avail_in = have;
  7380. state.hold = hold;
  7381. state.bits = bits;
  7382. //---
  7383. inflate_fast(strm, _out);
  7384. //--- LOAD() ---
  7385. put = strm.next_out;
  7386. output = strm.output;
  7387. left = strm.avail_out;
  7388. next = strm.next_in;
  7389. input = strm.input;
  7390. have = strm.avail_in;
  7391. hold = state.hold;
  7392. bits = state.bits;
  7393. //---
  7394. if (state.mode === TYPE) {
  7395. state.back = -1;
  7396. }
  7397. break;
  7398. }
  7399. state.back = 0;
  7400. for (;;) {
  7401. here = state.lencode[hold & ((1 << state.lenbits) -1)]; /*BITS(state.lenbits)*/
  7402. here_bits = here >>> 24;
  7403. here_op = (here >>> 16) & 0xff;
  7404. here_val = here & 0xffff;
  7405. if (here_bits <= bits) { break; }
  7406. //--- PULLBYTE() ---//
  7407. if (have === 0) { break inf_leave; }
  7408. have--;
  7409. hold += input[next++] << bits;
  7410. bits += 8;
  7411. //---//
  7412. }
  7413. if (here_op && (here_op & 0xf0) === 0) {
  7414. last_bits = here_bits;
  7415. last_op = here_op;
  7416. last_val = here_val;
  7417. for (;;) {
  7418. here = state.lencode[last_val +
  7419. ((hold & ((1 << (last_bits + last_op)) -1))/*BITS(last.bits + last.op)*/ >> last_bits)];
  7420. here_bits = here >>> 24;
  7421. here_op = (here >>> 16) & 0xff;
  7422. here_val = here & 0xffff;
  7423. if ((last_bits + here_bits) <= bits) { break; }
  7424. //--- PULLBYTE() ---//
  7425. if (have === 0) { break inf_leave; }
  7426. have--;
  7427. hold += input[next++] << bits;
  7428. bits += 8;
  7429. //---//
  7430. }
  7431. //--- DROPBITS(last.bits) ---//
  7432. hold >>>= last_bits;
  7433. bits -= last_bits;
  7434. //---//
  7435. state.back += last_bits;
  7436. }
  7437. //--- DROPBITS(here.bits) ---//
  7438. hold >>>= here_bits;
  7439. bits -= here_bits;
  7440. //---//
  7441. state.back += here_bits;
  7442. state.length = here_val;
  7443. if (here_op === 0) {
  7444. //Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ?
  7445. // "inflate: literal '%c'\n" :
  7446. // "inflate: literal 0x%02x\n", here.val));
  7447. state.mode = LIT;
  7448. break;
  7449. }
  7450. if (here_op & 32) {
  7451. //Tracevv((stderr, "inflate: end of block\n"));
  7452. state.back = -1;
  7453. state.mode = TYPE;
  7454. break;
  7455. }
  7456. if (here_op & 64) {
  7457. strm.msg = 'invalid literal/length code';
  7458. state.mode = BAD;
  7459. break;
  7460. }
  7461. state.extra = here_op & 15;
  7462. state.mode = LENEXT;
  7463. /* falls through */
  7464. case LENEXT:
  7465. if (state.extra) {
  7466. //=== NEEDBITS(state.extra);
  7467. n = state.extra;
  7468. while (bits < n) {
  7469. if (have === 0) { break inf_leave; }
  7470. have--;
  7471. hold += input[next++] << bits;
  7472. bits += 8;
  7473. }
  7474. //===//
  7475. state.length += hold & ((1 << state.extra) -1)/*BITS(state.extra)*/;
  7476. //--- DROPBITS(state.extra) ---//
  7477. hold >>>= state.extra;
  7478. bits -= state.extra;
  7479. //---//
  7480. state.back += state.extra;
  7481. }
  7482. //Tracevv((stderr, "inflate: length %u\n", state.length));
  7483. state.was = state.length;
  7484. state.mode = DIST;
  7485. /* falls through */
  7486. case DIST:
  7487. for (;;) {
  7488. here = state.distcode[hold & ((1 << state.distbits) -1)];/*BITS(state.distbits)*/
  7489. here_bits = here >>> 24;
  7490. here_op = (here >>> 16) & 0xff;
  7491. here_val = here & 0xffff;
  7492. if ((here_bits) <= bits) { break; }
  7493. //--- PULLBYTE() ---//
  7494. if (have === 0) { break inf_leave; }
  7495. have--;
  7496. hold += input[next++] << bits;
  7497. bits += 8;
  7498. //---//
  7499. }
  7500. if ((here_op & 0xf0) === 0) {
  7501. last_bits = here_bits;
  7502. last_op = here_op;
  7503. last_val = here_val;
  7504. for (;;) {
  7505. here = state.distcode[last_val +
  7506. ((hold & ((1 << (last_bits + last_op)) -1))/*BITS(last.bits + last.op)*/ >> last_bits)];
  7507. here_bits = here >>> 24;
  7508. here_op = (here >>> 16) & 0xff;
  7509. here_val = here & 0xffff;
  7510. if ((last_bits + here_bits) <= bits) { break; }
  7511. //--- PULLBYTE() ---//
  7512. if (have === 0) { break inf_leave; }
  7513. have--;
  7514. hold += input[next++] << bits;
  7515. bits += 8;
  7516. //---//
  7517. }
  7518. //--- DROPBITS(last.bits) ---//
  7519. hold >>>= last_bits;
  7520. bits -= last_bits;
  7521. //---//
  7522. state.back += last_bits;
  7523. }
  7524. //--- DROPBITS(here.bits) ---//
  7525. hold >>>= here_bits;
  7526. bits -= here_bits;
  7527. //---//
  7528. state.back += here_bits;
  7529. if (here_op & 64) {
  7530. strm.msg = 'invalid distance code';
  7531. state.mode = BAD;
  7532. break;
  7533. }
  7534. state.offset = here_val;
  7535. state.extra = (here_op) & 15;
  7536. state.mode = DISTEXT;
  7537. /* falls through */
  7538. case DISTEXT:
  7539. if (state.extra) {
  7540. //=== NEEDBITS(state.extra);
  7541. n = state.extra;
  7542. while (bits < n) {
  7543. if (have === 0) { break inf_leave; }
  7544. have--;
  7545. hold += input[next++] << bits;
  7546. bits += 8;
  7547. }
  7548. //===//
  7549. state.offset += hold & ((1 << state.extra) -1)/*BITS(state.extra)*/;
  7550. //--- DROPBITS(state.extra) ---//
  7551. hold >>>= state.extra;
  7552. bits -= state.extra;
  7553. //---//
  7554. state.back += state.extra;
  7555. }
  7556. //#ifdef INFLATE_STRICT
  7557. if (state.offset > state.dmax) {
  7558. strm.msg = 'invalid distance too far back';
  7559. state.mode = BAD;
  7560. break;
  7561. }
  7562. //#endif
  7563. //Tracevv((stderr, "inflate: distance %u\n", state.offset));
  7564. state.mode = MATCH;
  7565. /* falls through */
  7566. case MATCH:
  7567. if (left === 0) { break inf_leave; }
  7568. copy = _out - left;
  7569. if (state.offset > copy) { /* copy from window */
  7570. copy = state.offset - copy;
  7571. if (copy > state.whave) {
  7572. if (state.sane) {
  7573. strm.msg = 'invalid distance too far back';
  7574. state.mode = BAD;
  7575. break;
  7576. }
  7577. // (!) This block is disabled in zlib defailts,
  7578. // don't enable it for binary compatibility
  7579. //#ifdef INFLATE_ALLOW_INVALID_DISTANCE_TOOFAR_ARRR
  7580. // Trace((stderr, "inflate.c too far\n"));
  7581. // copy -= state.whave;
  7582. // if (copy > state.length) { copy = state.length; }
  7583. // if (copy > left) { copy = left; }
  7584. // left -= copy;
  7585. // state.length -= copy;
  7586. // do {
  7587. // output[put++] = 0;
  7588. // } while (--copy);
  7589. // if (state.length === 0) { state.mode = LEN; }
  7590. // break;
  7591. //#endif
  7592. }
  7593. if (copy > state.wnext) {
  7594. copy -= state.wnext;
  7595. from = state.wsize - copy;
  7596. }
  7597. else {
  7598. from = state.wnext - copy;
  7599. }
  7600. if (copy > state.length) { copy = state.length; }
  7601. from_source = state.window;
  7602. }
  7603. else { /* copy from output */
  7604. from_source = output;
  7605. from = put - state.offset;
  7606. copy = state.length;
  7607. }
  7608. if (copy > left) { copy = left; }
  7609. left -= copy;
  7610. state.length -= copy;
  7611. do {
  7612. output[put++] = from_source[from++];
  7613. } while (--copy);
  7614. if (state.length === 0) { state.mode = LEN; }
  7615. break;
  7616. case LIT:
  7617. if (left === 0) { break inf_leave; }
  7618. output[put++] = state.length;
  7619. left--;
  7620. state.mode = LEN;
  7621. break;
  7622. case CHECK:
  7623. if (state.wrap) {
  7624. //=== NEEDBITS(32);
  7625. while (bits < 32) {
  7626. if (have === 0) { break inf_leave; }
  7627. have--;
  7628. // Use '|' insdead of '+' to make sure that result is signed
  7629. hold |= input[next++] << bits;
  7630. bits += 8;
  7631. }
  7632. //===//
  7633. _out -= left;
  7634. strm.total_out += _out;
  7635. state.total += _out;
  7636. if (_out) {
  7637. strm.adler = state.check =
  7638. /*UPDATE(state.check, put - _out, _out);*/
  7639. (state.flags ? crc32(state.check, output, _out, put - _out) : adler32(state.check, output, _out, put - _out));
  7640. }
  7641. _out = left;
  7642. // NB: crc32 stored as signed 32-bit int, ZSWAP32 returns signed too
  7643. if ((state.flags ? hold : ZSWAP32(hold)) !== state.check) {
  7644. strm.msg = 'incorrect data check';
  7645. state.mode = BAD;
  7646. break;
  7647. }
  7648. //=== INITBITS();
  7649. hold = 0;
  7650. bits = 0;
  7651. //===//
  7652. //Tracev((stderr, "inflate: check matches trailer\n"));
  7653. }
  7654. state.mode = LENGTH;
  7655. /* falls through */
  7656. case LENGTH:
  7657. if (state.wrap && state.flags) {
  7658. //=== NEEDBITS(32);
  7659. while (bits < 32) {
  7660. if (have === 0) { break inf_leave; }
  7661. have--;
  7662. hold += input[next++] << bits;
  7663. bits += 8;
  7664. }
  7665. //===//
  7666. if (hold !== (state.total & 0xffffffff)) {
  7667. strm.msg = 'incorrect length check';
  7668. state.mode = BAD;
  7669. break;
  7670. }
  7671. //=== INITBITS();
  7672. hold = 0;
  7673. bits = 0;
  7674. //===//
  7675. //Tracev((stderr, "inflate: length matches trailer\n"));
  7676. }
  7677. state.mode = DONE;
  7678. /* falls through */
  7679. case DONE:
  7680. ret = Z_STREAM_END;
  7681. break inf_leave;
  7682. case BAD:
  7683. ret = Z_DATA_ERROR;
  7684. break inf_leave;
  7685. case MEM:
  7686. return Z_MEM_ERROR;
  7687. case SYNC:
  7688. /* falls through */
  7689. default:
  7690. return Z_STREAM_ERROR;
  7691. }
  7692. }
  7693. // inf_leave <- here is real place for "goto inf_leave", emulated via "break inf_leave"
  7694. /*
  7695. Return from inflate(), updating the total counts and the check value.
  7696. If there was no progress during the inflate() call, return a buffer
  7697. error. Call updatewindow() to create and/or update the window state.
  7698. Note: a memory error from inflate() is non-recoverable.
  7699. */
  7700. //--- RESTORE() ---
  7701. strm.next_out = put;
  7702. strm.avail_out = left;
  7703. strm.next_in = next;
  7704. strm.avail_in = have;
  7705. state.hold = hold;
  7706. state.bits = bits;
  7707. //---
  7708. if (state.wsize || (_out !== strm.avail_out && state.mode < BAD &&
  7709. (state.mode < CHECK || flush !== Z_FINISH))) {
  7710. if (updatewindow(strm, strm.output, strm.next_out, _out - strm.avail_out)) {
  7711. state.mode = MEM;
  7712. return Z_MEM_ERROR;
  7713. }
  7714. }
  7715. _in -= strm.avail_in;
  7716. _out -= strm.avail_out;
  7717. strm.total_in += _in;
  7718. strm.total_out += _out;
  7719. state.total += _out;
  7720. if (state.wrap && _out) {
  7721. strm.adler = state.check = /*UPDATE(state.check, strm.next_out - _out, _out);*/
  7722. (state.flags ? crc32(state.check, output, _out, strm.next_out - _out) : adler32(state.check, output, _out, strm.next_out - _out));
  7723. }
  7724. strm.data_type = state.bits + (state.last ? 64 : 0) +
  7725. (state.mode === TYPE ? 128 : 0) +
  7726. (state.mode === LEN_ || state.mode === COPY_ ? 256 : 0);
  7727. if (((_in === 0 && _out === 0) || flush === Z_FINISH) && ret === Z_OK) {
  7728. ret = Z_BUF_ERROR;
  7729. }
  7730. return ret;
  7731. }
  7732. function inflateEnd(strm) {
  7733. if (!strm || !strm.state /*|| strm->zfree == (free_func)0*/) {
  7734. return Z_STREAM_ERROR;
  7735. }
  7736. var state = strm.state;
  7737. if (state.window) {
  7738. state.window = null;
  7739. }
  7740. strm.state = null;
  7741. return Z_OK;
  7742. }
  7743. function inflateGetHeader(strm, head) {
  7744. var state;
  7745. /* check state */
  7746. if (!strm || !strm.state) { return Z_STREAM_ERROR; }
  7747. state = strm.state;
  7748. if ((state.wrap & 2) === 0) { return Z_STREAM_ERROR; }
  7749. /* save header structure */
  7750. state.head = head;
  7751. head.done = false;
  7752. return Z_OK;
  7753. }
  7754. exports.inflateReset = inflateReset;
  7755. exports.inflateReset2 = inflateReset2;
  7756. exports.inflateResetKeep = inflateResetKeep;
  7757. exports.inflateInit = inflateInit;
  7758. exports.inflateInit2 = inflateInit2;
  7759. exports.inflate = inflate;
  7760. exports.inflateEnd = inflateEnd;
  7761. exports.inflateGetHeader = inflateGetHeader;
  7762. exports.inflateInfo = 'pako inflate (from Nodeca project)';
  7763. /* Not implemented
  7764. exports.inflateCopy = inflateCopy;
  7765. exports.inflateGetDictionary = inflateGetDictionary;
  7766. exports.inflateMark = inflateMark;
  7767. exports.inflatePrime = inflatePrime;
  7768. exports.inflateSetDictionary = inflateSetDictionary;
  7769. exports.inflateSync = inflateSync;
  7770. exports.inflateSyncPoint = inflateSyncPoint;
  7771. exports.inflateUndermine = inflateUndermine;
  7772. */
  7773. },{"../utils/common":27,"./adler32":29,"./crc32":31,"./inffast":34,"./inftrees":36}],36:[function(_dereq_,module,exports){
  7774. 'use strict';
  7775. var utils = _dereq_('../utils/common');
  7776. var MAXBITS = 15;
  7777. var ENOUGH_LENS = 852;
  7778. var ENOUGH_DISTS = 592;
  7779. //var ENOUGH = (ENOUGH_LENS+ENOUGH_DISTS);
  7780. var CODES = 0;
  7781. var LENS = 1;
  7782. var DISTS = 2;
  7783. var lbase = [ /* Length codes 257..285 base */
  7784. 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31,
  7785. 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0
  7786. ];
  7787. var lext = [ /* Length codes 257..285 extra */
  7788. 16, 16, 16, 16, 16, 16, 16, 16, 17, 17, 17, 17, 18, 18, 18, 18,
  7789. 19, 19, 19, 19, 20, 20, 20, 20, 21, 21, 21, 21, 16, 72, 78
  7790. ];
  7791. var dbase = [ /* Distance codes 0..29 base */
  7792. 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193,
  7793. 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145,
  7794. 8193, 12289, 16385, 24577, 0, 0
  7795. ];
  7796. var dext = [ /* Distance codes 0..29 extra */
  7797. 16, 16, 16, 16, 17, 17, 18, 18, 19, 19, 20, 20, 21, 21, 22, 22,
  7798. 23, 23, 24, 24, 25, 25, 26, 26, 27, 27,
  7799. 28, 28, 29, 29, 64, 64
  7800. ];
  7801. module.exports = function inflate_table(type, lens, lens_index, codes, table, table_index, work, opts)
  7802. {
  7803. var bits = opts.bits;
  7804. //here = opts.here; /* table entry for duplication */
  7805. var len = 0; /* a code's length in bits */
  7806. var sym = 0; /* index of code symbols */
  7807. var min = 0, max = 0; /* minimum and maximum code lengths */
  7808. var root = 0; /* number of index bits for root table */
  7809. var curr = 0; /* number of index bits for current table */
  7810. var drop = 0; /* code bits to drop for sub-table */
  7811. var left = 0; /* number of prefix codes available */
  7812. var used = 0; /* code entries in table used */
  7813. var huff = 0; /* Huffman code */
  7814. var incr; /* for incrementing code, index */
  7815. var fill; /* index for replicating entries */
  7816. var low; /* low bits for current root entry */
  7817. var mask; /* mask for low root bits */
  7818. var next; /* next available space in table */
  7819. var base = null; /* base value table to use */
  7820. var base_index = 0;
  7821. // var shoextra; /* extra bits table to use */
  7822. var end; /* use base and extra for symbol > end */
  7823. var count = new utils.Buf16(MAXBITS+1); //[MAXBITS+1]; /* number of codes of each length */
  7824. var offs = new utils.Buf16(MAXBITS+1); //[MAXBITS+1]; /* offsets in table for each length */
  7825. var extra = null;
  7826. var extra_index = 0;
  7827. var here_bits, here_op, here_val;
  7828. /*
  7829. Process a set of code lengths to create a canonical Huffman code. The
  7830. code lengths are lens[0..codes-1]. Each length corresponds to the
  7831. symbols 0..codes-1. The Huffman code is generated by first sorting the
  7832. symbols by length from short to long, and retaining the symbol order
  7833. for codes with equal lengths. Then the code starts with all zero bits
  7834. for the first code of the shortest length, and the codes are integer
  7835. increments for the same length, and zeros are appended as the length
  7836. increases. For the deflate format, these bits are stored backwards
  7837. from their more natural integer increment ordering, and so when the
  7838. decoding tables are built in the large loop below, the integer codes
  7839. are incremented backwards.
  7840. This routine assumes, but does not check, that all of the entries in
  7841. lens[] are in the range 0..MAXBITS. The caller must assure this.
  7842. 1..MAXBITS is interpreted as that code length. zero means that that
  7843. symbol does not occur in this code.
  7844. The codes are sorted by computing a count of codes for each length,
  7845. creating from that a table of starting indices for each length in the
  7846. sorted table, and then entering the symbols in order in the sorted
  7847. table. The sorted table is work[], with that space being provided by
  7848. the caller.
  7849. The length counts are used for other purposes as well, i.e. finding
  7850. the minimum and maximum length codes, determining if there are any
  7851. codes at all, checking for a valid set of lengths, and looking ahead
  7852. at length counts to determine sub-table sizes when building the
  7853. decoding tables.
  7854. */
  7855. /* accumulate lengths for codes (assumes lens[] all in 0..MAXBITS) */
  7856. for (len = 0; len <= MAXBITS; len++) {
  7857. count[len] = 0;
  7858. }
  7859. for (sym = 0; sym < codes; sym++) {
  7860. count[lens[lens_index + sym]]++;
  7861. }
  7862. /* bound code lengths, force root to be within code lengths */
  7863. root = bits;
  7864. for (max = MAXBITS; max >= 1; max--) {
  7865. if (count[max] !== 0) { break; }
  7866. }
  7867. if (root > max) {
  7868. root = max;
  7869. }
  7870. if (max === 0) { /* no symbols to code at all */
  7871. //table.op[opts.table_index] = 64; //here.op = (var char)64; /* invalid code marker */
  7872. //table.bits[opts.table_index] = 1; //here.bits = (var char)1;
  7873. //table.val[opts.table_index++] = 0; //here.val = (var short)0;
  7874. table[table_index++] = (1 << 24) | (64 << 16) | 0;
  7875. //table.op[opts.table_index] = 64;
  7876. //table.bits[opts.table_index] = 1;
  7877. //table.val[opts.table_index++] = 0;
  7878. table[table_index++] = (1 << 24) | (64 << 16) | 0;
  7879. opts.bits = 1;
  7880. return 0; /* no symbols, but wait for decoding to report error */
  7881. }
  7882. for (min = 1; min < max; min++) {
  7883. if (count[min] !== 0) { break; }
  7884. }
  7885. if (root < min) {
  7886. root = min;
  7887. }
  7888. /* check for an over-subscribed or incomplete set of lengths */
  7889. left = 1;
  7890. for (len = 1; len <= MAXBITS; len++) {
  7891. left <<= 1;
  7892. left -= count[len];
  7893. if (left < 0) {
  7894. return -1;
  7895. } /* over-subscribed */
  7896. }
  7897. if (left > 0 && (type === CODES || max !== 1)) {
  7898. return -1; /* incomplete set */
  7899. }
  7900. /* generate offsets into symbol table for each length for sorting */
  7901. offs[1] = 0;
  7902. for (len = 1; len < MAXBITS; len++) {
  7903. offs[len + 1] = offs[len] + count[len];
  7904. }
  7905. /* sort symbols by length, by symbol order within each length */
  7906. for (sym = 0; sym < codes; sym++) {
  7907. if (lens[lens_index + sym] !== 0) {
  7908. work[offs[lens[lens_index + sym]]++] = sym;
  7909. }
  7910. }
  7911. /*
  7912. Create and fill in decoding tables. In this loop, the table being
  7913. filled is at next and has curr index bits. The code being used is huff
  7914. with length len. That code is converted to an index by dropping drop
  7915. bits off of the bottom. For codes where len is less than drop + curr,
  7916. those top drop + curr - len bits are incremented through all values to
  7917. fill the table with replicated entries.
  7918. root is the number of index bits for the root table. When len exceeds
  7919. root, sub-tables are created pointed to by the root entry with an index
  7920. of the low root bits of huff. This is saved in low to check for when a
  7921. new sub-table should be started. drop is zero when the root table is
  7922. being filled, and drop is root when sub-tables are being filled.
  7923. When a new sub-table is needed, it is necessary to look ahead in the
  7924. code lengths to determine what size sub-table is needed. The length
  7925. counts are used for this, and so count[] is decremented as codes are
  7926. entered in the tables.
  7927. used keeps track of how many table entries have been allocated from the
  7928. provided *table space. It is checked for LENS and DIST tables against
  7929. the constants ENOUGH_LENS and ENOUGH_DISTS to guard against changes in
  7930. the initial root table size constants. See the comments in inftrees.h
  7931. for more information.
  7932. sym increments through all symbols, and the loop terminates when
  7933. all codes of length max, i.e. all codes, have been processed. This
  7934. routine permits incomplete codes, so another loop after this one fills
  7935. in the rest of the decoding tables with invalid code markers.
  7936. */
  7937. /* set up for code type */
  7938. // poor man optimization - use if-else instead of switch,
  7939. // to avoid deopts in old v8
  7940. if (type === CODES) {
  7941. base = extra = work; /* dummy value--not used */
  7942. end = 19;
  7943. } else if (type === LENS) {
  7944. base = lbase;
  7945. base_index -= 257;
  7946. extra = lext;
  7947. extra_index -= 257;
  7948. end = 256;
  7949. } else { /* DISTS */
  7950. base = dbase;
  7951. extra = dext;
  7952. end = -1;
  7953. }
  7954. /* initialize opts for loop */
  7955. huff = 0; /* starting code */
  7956. sym = 0; /* starting code symbol */
  7957. len = min; /* starting code length */
  7958. next = table_index; /* current table to fill in */
  7959. curr = root; /* current table index bits */
  7960. drop = 0; /* current bits to drop from code for index */
  7961. low = -1; /* trigger new sub-table when len > root */
  7962. used = 1 << root; /* use root table entries */
  7963. mask = used - 1; /* mask for comparing low */
  7964. /* check available table space */
  7965. if ((type === LENS && used > ENOUGH_LENS) ||
  7966. (type === DISTS && used > ENOUGH_DISTS)) {
  7967. return 1;
  7968. }
  7969. var i=0;
  7970. /* process all codes and make table entries */
  7971. for (;;) {
  7972. i++;
  7973. /* create table entry */
  7974. here_bits = len - drop;
  7975. if (work[sym] < end) {
  7976. here_op = 0;
  7977. here_val = work[sym];
  7978. }
  7979. else if (work[sym] > end) {
  7980. here_op = extra[extra_index + work[sym]];
  7981. here_val = base[base_index + work[sym]];
  7982. }
  7983. else {
  7984. here_op = 32 + 64; /* end of block */
  7985. here_val = 0;
  7986. }
  7987. /* replicate for those indices with low len bits equal to huff */
  7988. incr = 1 << (len - drop);
  7989. fill = 1 << curr;
  7990. min = fill; /* save offset to next table */
  7991. do {
  7992. fill -= incr;
  7993. table[next + (huff >> drop) + fill] = (here_bits << 24) | (here_op << 16) | here_val |0;
  7994. } while (fill !== 0);
  7995. /* backwards increment the len-bit code huff */
  7996. incr = 1 << (len - 1);
  7997. while (huff & incr) {
  7998. incr >>= 1;
  7999. }
  8000. if (incr !== 0) {
  8001. huff &= incr - 1;
  8002. huff += incr;
  8003. } else {
  8004. huff = 0;
  8005. }
  8006. /* go to next symbol, update count, len */
  8007. sym++;
  8008. if (--count[len] === 0) {
  8009. if (len === max) { break; }
  8010. len = lens[lens_index + work[sym]];
  8011. }
  8012. /* create new sub-table if needed */
  8013. if (len > root && (huff & mask) !== low) {
  8014. /* if first time, transition to sub-tables */
  8015. if (drop === 0) {
  8016. drop = root;
  8017. }
  8018. /* increment past last table */
  8019. next += min; /* here min is 1 << curr */
  8020. /* determine length of next table */
  8021. curr = len - drop;
  8022. left = 1 << curr;
  8023. while (curr + drop < max) {
  8024. left -= count[curr + drop];
  8025. if (left <= 0) { break; }
  8026. curr++;
  8027. left <<= 1;
  8028. }
  8029. /* check for enough space */
  8030. used += 1 << curr;
  8031. if ((type === LENS && used > ENOUGH_LENS) ||
  8032. (type === DISTS && used > ENOUGH_DISTS)) {
  8033. return 1;
  8034. }
  8035. /* point entry in root table to sub-table */
  8036. low = huff & mask;
  8037. /*table.op[low] = curr;
  8038. table.bits[low] = root;
  8039. table.val[low] = next - opts.table_index;*/
  8040. table[low] = (root << 24) | (curr << 16) | (next - table_index) |0;
  8041. }
  8042. }
  8043. /* fill in remaining table entry if code is incomplete (guaranteed to have
  8044. at most one remaining entry, since if the code is incomplete, the
  8045. maximum code length that was allowed to get this far is one bit) */
  8046. if (huff !== 0) {
  8047. //table.op[next + huff] = 64; /* invalid code marker */
  8048. //table.bits[next + huff] = len - drop;
  8049. //table.val[next + huff] = 0;
  8050. table[next + huff] = ((len - drop) << 24) | (64 << 16) |0;
  8051. }
  8052. /* set return parameters */
  8053. //opts.table_index += used;
  8054. opts.bits = root;
  8055. return 0;
  8056. };
  8057. },{"../utils/common":27}],37:[function(_dereq_,module,exports){
  8058. 'use strict';
  8059. module.exports = {
  8060. '2': 'need dictionary', /* Z_NEED_DICT 2 */
  8061. '1': 'stream end', /* Z_STREAM_END 1 */
  8062. '0': '', /* Z_OK 0 */
  8063. '-1': 'file error', /* Z_ERRNO (-1) */
  8064. '-2': 'stream error', /* Z_STREAM_ERROR (-2) */
  8065. '-3': 'data error', /* Z_DATA_ERROR (-3) */
  8066. '-4': 'insufficient memory', /* Z_MEM_ERROR (-4) */
  8067. '-5': 'buffer error', /* Z_BUF_ERROR (-5) */
  8068. '-6': 'incompatible version' /* Z_VERSION_ERROR (-6) */
  8069. };
  8070. },{}],38:[function(_dereq_,module,exports){
  8071. 'use strict';
  8072. var utils = _dereq_('../utils/common');
  8073. /* Public constants ==========================================================*/
  8074. /* ===========================================================================*/
  8075. //var Z_FILTERED = 1;
  8076. //var Z_HUFFMAN_ONLY = 2;
  8077. //var Z_RLE = 3;
  8078. var Z_FIXED = 4;
  8079. //var Z_DEFAULT_STRATEGY = 0;
  8080. /* Possible values of the data_type field (though see inflate()) */
  8081. var Z_BINARY = 0;
  8082. var Z_TEXT = 1;
  8083. //var Z_ASCII = 1; // = Z_TEXT
  8084. var Z_UNKNOWN = 2;
  8085. /*============================================================================*/
  8086. function zero(buf) { var len = buf.length; while (--len >= 0) { buf[len] = 0; } }
  8087. // From zutil.h
  8088. var STORED_BLOCK = 0;
  8089. var STATIC_TREES = 1;
  8090. var DYN_TREES = 2;
  8091. /* The three kinds of block type */
  8092. var MIN_MATCH = 3;
  8093. var MAX_MATCH = 258;
  8094. /* The minimum and maximum match lengths */
  8095. // From deflate.h
  8096. /* ===========================================================================
  8097. * Internal compression state.
  8098. */
  8099. var LENGTH_CODES = 29;
  8100. /* number of length codes, not counting the special END_BLOCK code */
  8101. var LITERALS = 256;
  8102. /* number of literal bytes 0..255 */
  8103. var L_CODES = LITERALS + 1 + LENGTH_CODES;
  8104. /* number of Literal or Length codes, including the END_BLOCK code */
  8105. var D_CODES = 30;
  8106. /* number of distance codes */
  8107. var BL_CODES = 19;
  8108. /* number of codes used to transfer the bit lengths */
  8109. var HEAP_SIZE = 2*L_CODES + 1;
  8110. /* maximum heap size */
  8111. var MAX_BITS = 15;
  8112. /* All codes must not exceed MAX_BITS bits */
  8113. var Buf_size = 16;
  8114. /* size of bit buffer in bi_buf */
  8115. /* ===========================================================================
  8116. * Constants
  8117. */
  8118. var MAX_BL_BITS = 7;
  8119. /* Bit length codes must not exceed MAX_BL_BITS bits */
  8120. var END_BLOCK = 256;
  8121. /* end of block literal code */
  8122. var REP_3_6 = 16;
  8123. /* repeat previous bit length 3-6 times (2 bits of repeat count) */
  8124. var REPZ_3_10 = 17;
  8125. /* repeat a zero length 3-10 times (3 bits of repeat count) */
  8126. var REPZ_11_138 = 18;
  8127. /* repeat a zero length 11-138 times (7 bits of repeat count) */
  8128. var extra_lbits = /* extra bits for each length code */
  8129. [0,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,0];
  8130. var extra_dbits = /* extra bits for each distance code */
  8131. [0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13];
  8132. var extra_blbits = /* extra bits for each bit length code */
  8133. [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,2,3,7];
  8134. var bl_order =
  8135. [16,17,18,0,8,7,9,6,10,5,11,4,12,3,13,2,14,1,15];
  8136. /* The lengths of the bit length codes are sent in order of decreasing
  8137. * probability, to avoid transmitting the lengths for unused bit length codes.
  8138. */
  8139. /* ===========================================================================
  8140. * Local data. These are initialized only once.
  8141. */
  8142. // We pre-fill arrays with 0 to avoid uninitialized gaps
  8143. var DIST_CODE_LEN = 512; /* see definition of array dist_code below */
  8144. // !!!! Use flat array insdead of structure, Freq = i*2, Len = i*2+1
  8145. var static_ltree = new Array((L_CODES+2) * 2);
  8146. zero(static_ltree);
  8147. /* The static literal tree. Since the bit lengths are imposed, there is no
  8148. * need for the L_CODES extra codes used during heap construction. However
  8149. * The codes 286 and 287 are needed to build a canonical tree (see _tr_init
  8150. * below).
  8151. */
  8152. var static_dtree = new Array(D_CODES * 2);
  8153. zero(static_dtree);
  8154. /* The static distance tree. (Actually a trivial tree since all codes use
  8155. * 5 bits.)
  8156. */
  8157. var _dist_code = new Array(DIST_CODE_LEN);
  8158. zero(_dist_code);
  8159. /* Distance codes. The first 256 values correspond to the distances
  8160. * 3 .. 258, the last 256 values correspond to the top 8 bits of
  8161. * the 15 bit distances.
  8162. */
  8163. var _length_code = new Array(MAX_MATCH-MIN_MATCH+1);
  8164. zero(_length_code);
  8165. /* length code for each normalized match length (0 == MIN_MATCH) */
  8166. var base_length = new Array(LENGTH_CODES);
  8167. zero(base_length);
  8168. /* First normalized length for each code (0 = MIN_MATCH) */
  8169. var base_dist = new Array(D_CODES);
  8170. zero(base_dist);
  8171. /* First normalized distance for each code (0 = distance of 1) */
  8172. var StaticTreeDesc = function (static_tree, extra_bits, extra_base, elems, max_length) {
  8173. this.static_tree = static_tree; /* static tree or NULL */
  8174. this.extra_bits = extra_bits; /* extra bits for each code or NULL */
  8175. this.extra_base = extra_base; /* base index for extra_bits */
  8176. this.elems = elems; /* max number of elements in the tree */
  8177. this.max_length = max_length; /* max bit length for the codes */
  8178. // show if `static_tree` has data or dummy - needed for monomorphic objects
  8179. this.has_stree = static_tree && static_tree.length;
  8180. };
  8181. var static_l_desc;
  8182. var static_d_desc;
  8183. var static_bl_desc;
  8184. var TreeDesc = function(dyn_tree, stat_desc) {
  8185. this.dyn_tree = dyn_tree; /* the dynamic tree */
  8186. this.max_code = 0; /* largest code with non zero frequency */
  8187. this.stat_desc = stat_desc; /* the corresponding static tree */
  8188. };
  8189. function d_code(dist) {
  8190. return dist < 256 ? _dist_code[dist] : _dist_code[256 + (dist >>> 7)];
  8191. }
  8192. /* ===========================================================================
  8193. * Output a short LSB first on the stream.
  8194. * IN assertion: there is enough room in pendingBuf.
  8195. */
  8196. function put_short (s, w) {
  8197. // put_byte(s, (uch)((w) & 0xff));
  8198. // put_byte(s, (uch)((ush)(w) >> 8));
  8199. s.pending_buf[s.pending++] = (w) & 0xff;
  8200. s.pending_buf[s.pending++] = (w >>> 8) & 0xff;
  8201. }
  8202. /* ===========================================================================
  8203. * Send a value on a given number of bits.
  8204. * IN assertion: length <= 16 and value fits in length bits.
  8205. */
  8206. function send_bits(s, value, length) {
  8207. if (s.bi_valid > (Buf_size - length)) {
  8208. s.bi_buf |= (value << s.bi_valid) & 0xffff;
  8209. put_short(s, s.bi_buf);
  8210. s.bi_buf = value >> (Buf_size - s.bi_valid);
  8211. s.bi_valid += length - Buf_size;
  8212. } else {
  8213. s.bi_buf |= (value << s.bi_valid) & 0xffff;
  8214. s.bi_valid += length;
  8215. }
  8216. }
  8217. function send_code(s, c, tree) {
  8218. send_bits(s, tree[c*2]/*.Code*/, tree[c*2 + 1]/*.Len*/);
  8219. }
  8220. /* ===========================================================================
  8221. * Reverse the first len bits of a code, using straightforward code (a faster
  8222. * method would use a table)
  8223. * IN assertion: 1 <= len <= 15
  8224. */
  8225. function bi_reverse(code, len) {
  8226. var res = 0;
  8227. do {
  8228. res |= code & 1;
  8229. code >>>= 1;
  8230. res <<= 1;
  8231. } while (--len > 0);
  8232. return res >>> 1;
  8233. }
  8234. /* ===========================================================================
  8235. * Flush the bit buffer, keeping at most 7 bits in it.
  8236. */
  8237. function bi_flush(s) {
  8238. if (s.bi_valid === 16) {
  8239. put_short(s, s.bi_buf);
  8240. s.bi_buf = 0;
  8241. s.bi_valid = 0;
  8242. } else if (s.bi_valid >= 8) {
  8243. s.pending_buf[s.pending++] = s.bi_buf & 0xff;
  8244. s.bi_buf >>= 8;
  8245. s.bi_valid -= 8;
  8246. }
  8247. }
  8248. /* ===========================================================================
  8249. * Compute the optimal bit lengths for a tree and update the total bit length
  8250. * for the current block.
  8251. * IN assertion: the fields freq and dad are set, heap[heap_max] and
  8252. * above are the tree nodes sorted by increasing frequency.
  8253. * OUT assertions: the field len is set to the optimal bit length, the
  8254. * array bl_count contains the frequencies for each bit length.
  8255. * The length opt_len is updated; static_len is also updated if stree is
  8256. * not null.
  8257. */
  8258. function gen_bitlen(s, desc)
  8259. // deflate_state *s;
  8260. // tree_desc *desc; /* the tree descriptor */
  8261. {
  8262. var tree = desc.dyn_tree;
  8263. var max_code = desc.max_code;
  8264. var stree = desc.stat_desc.static_tree;
  8265. var has_stree = desc.stat_desc.has_stree;
  8266. var extra = desc.stat_desc.extra_bits;
  8267. var base = desc.stat_desc.extra_base;
  8268. var max_length = desc.stat_desc.max_length;
  8269. var h; /* heap index */
  8270. var n, m; /* iterate over the tree elements */
  8271. var bits; /* bit length */
  8272. var xbits; /* extra bits */
  8273. var f; /* frequency */
  8274. var overflow = 0; /* number of elements with bit length too large */
  8275. for (bits = 0; bits <= MAX_BITS; bits++) {
  8276. s.bl_count[bits] = 0;
  8277. }
  8278. /* In a first pass, compute the optimal bit lengths (which may
  8279. * overflow in the case of the bit length tree).
  8280. */
  8281. tree[s.heap[s.heap_max]*2 + 1]/*.Len*/ = 0; /* root of the heap */
  8282. for (h = s.heap_max+1; h < HEAP_SIZE; h++) {
  8283. n = s.heap[h];
  8284. bits = tree[tree[n*2 +1]/*.Dad*/ * 2 + 1]/*.Len*/ + 1;
  8285. if (bits > max_length) {
  8286. bits = max_length;
  8287. overflow++;
  8288. }
  8289. tree[n*2 + 1]/*.Len*/ = bits;
  8290. /* We overwrite tree[n].Dad which is no longer needed */
  8291. if (n > max_code) { continue; } /* not a leaf node */
  8292. s.bl_count[bits]++;
  8293. xbits = 0;
  8294. if (n >= base) {
  8295. xbits = extra[n-base];
  8296. }
  8297. f = tree[n * 2]/*.Freq*/;
  8298. s.opt_len += f * (bits + xbits);
  8299. if (has_stree) {
  8300. s.static_len += f * (stree[n*2 + 1]/*.Len*/ + xbits);
  8301. }
  8302. }
  8303. if (overflow === 0) { return; }
  8304. // Trace((stderr,"\nbit length overflow\n"));
  8305. /* This happens for example on obj2 and pic of the Calgary corpus */
  8306. /* Find the first bit length which could increase: */
  8307. do {
  8308. bits = max_length-1;
  8309. while (s.bl_count[bits] === 0) { bits--; }
  8310. s.bl_count[bits]--; /* move one leaf down the tree */
  8311. s.bl_count[bits+1] += 2; /* move one overflow item as its brother */
  8312. s.bl_count[max_length]--;
  8313. /* The brother of the overflow item also moves one step up,
  8314. * but this does not affect bl_count[max_length]
  8315. */
  8316. overflow -= 2;
  8317. } while (overflow > 0);
  8318. /* Now recompute all bit lengths, scanning in increasing frequency.
  8319. * h is still equal to HEAP_SIZE. (It is simpler to reconstruct all
  8320. * lengths instead of fixing only the wrong ones. This idea is taken
  8321. * from 'ar' written by Haruhiko Okumura.)
  8322. */
  8323. for (bits = max_length; bits !== 0; bits--) {
  8324. n = s.bl_count[bits];
  8325. while (n !== 0) {
  8326. m = s.heap[--h];
  8327. if (m > max_code) { continue; }
  8328. if (tree[m*2 + 1]/*.Len*/ !== bits) {
  8329. // Trace((stderr,"code %d bits %d->%d\n", m, tree[m].Len, bits));
  8330. s.opt_len += (bits - tree[m*2 + 1]/*.Len*/)*tree[m*2]/*.Freq*/;
  8331. tree[m*2 + 1]/*.Len*/ = bits;
  8332. }
  8333. n--;
  8334. }
  8335. }
  8336. }
  8337. /* ===========================================================================
  8338. * Generate the codes for a given tree and bit counts (which need not be
  8339. * optimal).
  8340. * IN assertion: the array bl_count contains the bit length statistics for
  8341. * the given tree and the field len is set for all tree elements.
  8342. * OUT assertion: the field code is set for all tree elements of non
  8343. * zero code length.
  8344. */
  8345. function gen_codes(tree, max_code, bl_count)
  8346. // ct_data *tree; /* the tree to decorate */
  8347. // int max_code; /* largest code with non zero frequency */
  8348. // ushf *bl_count; /* number of codes at each bit length */
  8349. {
  8350. var next_code = new Array(MAX_BITS+1); /* next code value for each bit length */
  8351. var code = 0; /* running code value */
  8352. var bits; /* bit index */
  8353. var n; /* code index */
  8354. /* The distribution counts are first used to generate the code values
  8355. * without bit reversal.
  8356. */
  8357. for (bits = 1; bits <= MAX_BITS; bits++) {
  8358. next_code[bits] = code = (code + bl_count[bits-1]) << 1;
  8359. }
  8360. /* Check that the bit counts in bl_count are consistent. The last code
  8361. * must be all ones.
  8362. */
  8363. //Assert (code + bl_count[MAX_BITS]-1 == (1<<MAX_BITS)-1,
  8364. // "inconsistent bit counts");
  8365. //Tracev((stderr,"\ngen_codes: max_code %d ", max_code));
  8366. for (n = 0; n <= max_code; n++) {
  8367. var len = tree[n*2 + 1]/*.Len*/;
  8368. if (len === 0) { continue; }
  8369. /* Now reverse the bits */
  8370. tree[n*2]/*.Code*/ = bi_reverse(next_code[len]++, len);
  8371. //Tracecv(tree != static_ltree, (stderr,"\nn %3d %c l %2d c %4x (%x) ",
  8372. // n, (isgraph(n) ? n : ' '), len, tree[n].Code, next_code[len]-1));
  8373. }
  8374. }
  8375. /* ===========================================================================
  8376. * Initialize the various 'constant' tables.
  8377. */
  8378. function tr_static_init() {
  8379. var n; /* iterates over tree elements */
  8380. var bits; /* bit counter */
  8381. var length; /* length value */
  8382. var code; /* code value */
  8383. var dist; /* distance index */
  8384. var bl_count = new Array(MAX_BITS+1);
  8385. /* number of codes at each bit length for an optimal tree */
  8386. // do check in _tr_init()
  8387. //if (static_init_done) return;
  8388. /* For some embedded targets, global variables are not initialized: */
  8389. /*#ifdef NO_INIT_GLOBAL_POINTERS
  8390. static_l_desc.static_tree = static_ltree;
  8391. static_l_desc.extra_bits = extra_lbits;
  8392. static_d_desc.static_tree = static_dtree;
  8393. static_d_desc.extra_bits = extra_dbits;
  8394. static_bl_desc.extra_bits = extra_blbits;
  8395. #endif*/
  8396. /* Initialize the mapping length (0..255) -> length code (0..28) */
  8397. length = 0;
  8398. for (code = 0; code < LENGTH_CODES-1; code++) {
  8399. base_length[code] = length;
  8400. for (n = 0; n < (1<<extra_lbits[code]); n++) {
  8401. _length_code[length++] = code;
  8402. }
  8403. }
  8404. //Assert (length == 256, "tr_static_init: length != 256");
  8405. /* Note that the length 255 (match length 258) can be represented
  8406. * in two different ways: code 284 + 5 bits or code 285, so we
  8407. * overwrite length_code[255] to use the best encoding:
  8408. */
  8409. _length_code[length-1] = code;
  8410. /* Initialize the mapping dist (0..32K) -> dist code (0..29) */
  8411. dist = 0;
  8412. for (code = 0 ; code < 16; code++) {
  8413. base_dist[code] = dist;
  8414. for (n = 0; n < (1<<extra_dbits[code]); n++) {
  8415. _dist_code[dist++] = code;
  8416. }
  8417. }
  8418. //Assert (dist == 256, "tr_static_init: dist != 256");
  8419. dist >>= 7; /* from now on, all distances are divided by 128 */
  8420. for ( ; code < D_CODES; code++) {
  8421. base_dist[code] = dist << 7;
  8422. for (n = 0; n < (1<<(extra_dbits[code]-7)); n++) {
  8423. _dist_code[256 + dist++] = code;
  8424. }
  8425. }
  8426. //Assert (dist == 256, "tr_static_init: 256+dist != 512");
  8427. /* Construct the codes of the static literal tree */
  8428. for (bits = 0; bits <= MAX_BITS; bits++) {
  8429. bl_count[bits] = 0;
  8430. }
  8431. n = 0;
  8432. while (n <= 143) {
  8433. static_ltree[n*2 + 1]/*.Len*/ = 8;
  8434. n++;
  8435. bl_count[8]++;
  8436. }
  8437. while (n <= 255) {
  8438. static_ltree[n*2 + 1]/*.Len*/ = 9;
  8439. n++;
  8440. bl_count[9]++;
  8441. }
  8442. while (n <= 279) {
  8443. static_ltree[n*2 + 1]/*.Len*/ = 7;
  8444. n++;
  8445. bl_count[7]++;
  8446. }
  8447. while (n <= 287) {
  8448. static_ltree[n*2 + 1]/*.Len*/ = 8;
  8449. n++;
  8450. bl_count[8]++;
  8451. }
  8452. /* Codes 286 and 287 do not exist, but we must include them in the
  8453. * tree construction to get a canonical Huffman tree (longest code
  8454. * all ones)
  8455. */
  8456. gen_codes(static_ltree, L_CODES+1, bl_count);
  8457. /* The static distance tree is trivial: */
  8458. for (n = 0; n < D_CODES; n++) {
  8459. static_dtree[n*2 + 1]/*.Len*/ = 5;
  8460. static_dtree[n*2]/*.Code*/ = bi_reverse(n, 5);
  8461. }
  8462. // Now data ready and we can init static trees
  8463. static_l_desc = new StaticTreeDesc(static_ltree, extra_lbits, LITERALS+1, L_CODES, MAX_BITS);
  8464. static_d_desc = new StaticTreeDesc(static_dtree, extra_dbits, 0, D_CODES, MAX_BITS);
  8465. static_bl_desc =new StaticTreeDesc(new Array(0), extra_blbits, 0, BL_CODES, MAX_BL_BITS);
  8466. //static_init_done = true;
  8467. }
  8468. /* ===========================================================================
  8469. * Initialize a new block.
  8470. */
  8471. function init_block(s) {
  8472. var n; /* iterates over tree elements */
  8473. /* Initialize the trees. */
  8474. for (n = 0; n < L_CODES; n++) { s.dyn_ltree[n*2]/*.Freq*/ = 0; }
  8475. for (n = 0; n < D_CODES; n++) { s.dyn_dtree[n*2]/*.Freq*/ = 0; }
  8476. for (n = 0; n < BL_CODES; n++) { s.bl_tree[n*2]/*.Freq*/ = 0; }
  8477. s.dyn_ltree[END_BLOCK*2]/*.Freq*/ = 1;
  8478. s.opt_len = s.static_len = 0;
  8479. s.last_lit = s.matches = 0;
  8480. }
  8481. /* ===========================================================================
  8482. * Flush the bit buffer and align the output on a byte boundary
  8483. */
  8484. function bi_windup(s)
  8485. {
  8486. if (s.bi_valid > 8) {
  8487. put_short(s, s.bi_buf);
  8488. } else if (s.bi_valid > 0) {
  8489. //put_byte(s, (Byte)s->bi_buf);
  8490. s.pending_buf[s.pending++] = s.bi_buf;
  8491. }
  8492. s.bi_buf = 0;
  8493. s.bi_valid = 0;
  8494. }
  8495. /* ===========================================================================
  8496. * Copy a stored block, storing first the length and its
  8497. * one's complement if requested.
  8498. */
  8499. function copy_block(s, buf, len, header)
  8500. //DeflateState *s;
  8501. //charf *buf; /* the input data */
  8502. //unsigned len; /* its length */
  8503. //int header; /* true if block header must be written */
  8504. {
  8505. bi_windup(s); /* align on byte boundary */
  8506. if (header) {
  8507. put_short(s, len);
  8508. put_short(s, ~len);
  8509. }
  8510. // while (len--) {
  8511. // put_byte(s, *buf++);
  8512. // }
  8513. utils.arraySet(s.pending_buf, s.window, buf, len, s.pending);
  8514. s.pending += len;
  8515. }
  8516. /* ===========================================================================
  8517. * Compares to subtrees, using the tree depth as tie breaker when
  8518. * the subtrees have equal frequency. This minimizes the worst case length.
  8519. */
  8520. function smaller(tree, n, m, depth) {
  8521. var _n2 = n*2;
  8522. var _m2 = m*2;
  8523. return (tree[_n2]/*.Freq*/ < tree[_m2]/*.Freq*/ ||
  8524. (tree[_n2]/*.Freq*/ === tree[_m2]/*.Freq*/ && depth[n] <= depth[m]));
  8525. }
  8526. /* ===========================================================================
  8527. * Restore the heap property by moving down the tree starting at node k,
  8528. * exchanging a node with the smallest of its two sons if necessary, stopping
  8529. * when the heap property is re-established (each father smaller than its
  8530. * two sons).
  8531. */
  8532. function pqdownheap(s, tree, k)
  8533. // deflate_state *s;
  8534. // ct_data *tree; /* the tree to restore */
  8535. // int k; /* node to move down */
  8536. {
  8537. var v = s.heap[k];
  8538. var j = k << 1; /* left son of k */
  8539. while (j <= s.heap_len) {
  8540. /* Set j to the smallest of the two sons: */
  8541. if (j < s.heap_len &&
  8542. smaller(tree, s.heap[j+1], s.heap[j], s.depth)) {
  8543. j++;
  8544. }
  8545. /* Exit if v is smaller than both sons */
  8546. if (smaller(tree, v, s.heap[j], s.depth)) { break; }
  8547. /* Exchange v with the smallest son */
  8548. s.heap[k] = s.heap[j];
  8549. k = j;
  8550. /* And continue down the tree, setting j to the left son of k */
  8551. j <<= 1;
  8552. }
  8553. s.heap[k] = v;
  8554. }
  8555. // inlined manually
  8556. // var SMALLEST = 1;
  8557. /* ===========================================================================
  8558. * Send the block data compressed using the given Huffman trees
  8559. */
  8560. function compress_block(s, ltree, dtree)
  8561. // deflate_state *s;
  8562. // const ct_data *ltree; /* literal tree */
  8563. // const ct_data *dtree; /* distance tree */
  8564. {
  8565. var dist; /* distance of matched string */
  8566. var lc; /* match length or unmatched char (if dist == 0) */
  8567. var lx = 0; /* running index in l_buf */
  8568. var code; /* the code to send */
  8569. var extra; /* number of extra bits to send */
  8570. if (s.last_lit !== 0) {
  8571. do {
  8572. dist = (s.pending_buf[s.d_buf + lx*2] << 8) | (s.pending_buf[s.d_buf + lx*2 + 1]);
  8573. lc = s.pending_buf[s.l_buf + lx];
  8574. lx++;
  8575. if (dist === 0) {
  8576. send_code(s, lc, ltree); /* send a literal byte */
  8577. //Tracecv(isgraph(lc), (stderr," '%c' ", lc));
  8578. } else {
  8579. /* Here, lc is the match length - MIN_MATCH */
  8580. code = _length_code[lc];
  8581. send_code(s, code+LITERALS+1, ltree); /* send the length code */
  8582. extra = extra_lbits[code];
  8583. if (extra !== 0) {
  8584. lc -= base_length[code];
  8585. send_bits(s, lc, extra); /* send the extra length bits */
  8586. }
  8587. dist--; /* dist is now the match distance - 1 */
  8588. code = d_code(dist);
  8589. //Assert (code < D_CODES, "bad d_code");
  8590. send_code(s, code, dtree); /* send the distance code */
  8591. extra = extra_dbits[code];
  8592. if (extra !== 0) {
  8593. dist -= base_dist[code];
  8594. send_bits(s, dist, extra); /* send the extra distance bits */
  8595. }
  8596. } /* literal or match pair ? */
  8597. /* Check that the overlay between pending_buf and d_buf+l_buf is ok: */
  8598. //Assert((uInt)(s->pending) < s->lit_bufsize + 2*lx,
  8599. // "pendingBuf overflow");
  8600. } while (lx < s.last_lit);
  8601. }
  8602. send_code(s, END_BLOCK, ltree);
  8603. }
  8604. /* ===========================================================================
  8605. * Construct one Huffman tree and assigns the code bit strings and lengths.
  8606. * Update the total bit length for the current block.
  8607. * IN assertion: the field freq is set for all tree elements.
  8608. * OUT assertions: the fields len and code are set to the optimal bit length
  8609. * and corresponding code. The length opt_len is updated; static_len is
  8610. * also updated if stree is not null. The field max_code is set.
  8611. */
  8612. function build_tree(s, desc)
  8613. // deflate_state *s;
  8614. // tree_desc *desc; /* the tree descriptor */
  8615. {
  8616. var tree = desc.dyn_tree;
  8617. var stree = desc.stat_desc.static_tree;
  8618. var has_stree = desc.stat_desc.has_stree;
  8619. var elems = desc.stat_desc.elems;
  8620. var n, m; /* iterate over heap elements */
  8621. var max_code = -1; /* largest code with non zero frequency */
  8622. var node; /* new node being created */
  8623. /* Construct the initial heap, with least frequent element in
  8624. * heap[SMALLEST]. The sons of heap[n] are heap[2*n] and heap[2*n+1].
  8625. * heap[0] is not used.
  8626. */
  8627. s.heap_len = 0;
  8628. s.heap_max = HEAP_SIZE;
  8629. for (n = 0; n < elems; n++) {
  8630. if (tree[n * 2]/*.Freq*/ !== 0) {
  8631. s.heap[++s.heap_len] = max_code = n;
  8632. s.depth[n] = 0;
  8633. } else {
  8634. tree[n*2 + 1]/*.Len*/ = 0;
  8635. }
  8636. }
  8637. /* The pkzip format requires that at least one distance code exists,
  8638. * and that at least one bit should be sent even if there is only one
  8639. * possible code. So to avoid special checks later on we force at least
  8640. * two codes of non zero frequency.
  8641. */
  8642. while (s.heap_len < 2) {
  8643. node = s.heap[++s.heap_len] = (max_code < 2 ? ++max_code : 0);
  8644. tree[node * 2]/*.Freq*/ = 1;
  8645. s.depth[node] = 0;
  8646. s.opt_len--;
  8647. if (has_stree) {
  8648. s.static_len -= stree[node*2 + 1]/*.Len*/;
  8649. }
  8650. /* node is 0 or 1 so it does not have extra bits */
  8651. }
  8652. desc.max_code = max_code;
  8653. /* The elements heap[heap_len/2+1 .. heap_len] are leaves of the tree,
  8654. * establish sub-heaps of increasing lengths:
  8655. */
  8656. for (n = (s.heap_len >> 1/*int /2*/); n >= 1; n--) { pqdownheap(s, tree, n); }
  8657. /* Construct the Huffman tree by repeatedly combining the least two
  8658. * frequent nodes.
  8659. */
  8660. node = elems; /* next internal node of the tree */
  8661. do {
  8662. //pqremove(s, tree, n); /* n = node of least frequency */
  8663. /*** pqremove ***/
  8664. n = s.heap[1/*SMALLEST*/];
  8665. s.heap[1/*SMALLEST*/] = s.heap[s.heap_len--];
  8666. pqdownheap(s, tree, 1/*SMALLEST*/);
  8667. /***/
  8668. m = s.heap[1/*SMALLEST*/]; /* m = node of next least frequency */
  8669. s.heap[--s.heap_max] = n; /* keep the nodes sorted by frequency */
  8670. s.heap[--s.heap_max] = m;
  8671. /* Create a new node father of n and m */
  8672. tree[node * 2]/*.Freq*/ = tree[n * 2]/*.Freq*/ + tree[m * 2]/*.Freq*/;
  8673. s.depth[node] = (s.depth[n] >= s.depth[m] ? s.depth[n] : s.depth[m]) + 1;
  8674. tree[n*2 + 1]/*.Dad*/ = tree[m*2 + 1]/*.Dad*/ = node;
  8675. /* and insert the new node in the heap */
  8676. s.heap[1/*SMALLEST*/] = node++;
  8677. pqdownheap(s, tree, 1/*SMALLEST*/);
  8678. } while (s.heap_len >= 2);
  8679. s.heap[--s.heap_max] = s.heap[1/*SMALLEST*/];
  8680. /* At this point, the fields freq and dad are set. We can now
  8681. * generate the bit lengths.
  8682. */
  8683. gen_bitlen(s, desc);
  8684. /* The field len is now set, we can generate the bit codes */
  8685. gen_codes(tree, max_code, s.bl_count);
  8686. }
  8687. /* ===========================================================================
  8688. * Scan a literal or distance tree to determine the frequencies of the codes
  8689. * in the bit length tree.
  8690. */
  8691. function scan_tree(s, tree, max_code)
  8692. // deflate_state *s;
  8693. // ct_data *tree; /* the tree to be scanned */
  8694. // int max_code; /* and its largest code of non zero frequency */
  8695. {
  8696. var n; /* iterates over all tree elements */
  8697. var prevlen = -1; /* last emitted length */
  8698. var curlen; /* length of current code */
  8699. var nextlen = tree[0*2 + 1]/*.Len*/; /* length of next code */
  8700. var count = 0; /* repeat count of the current code */
  8701. var max_count = 7; /* max repeat count */
  8702. var min_count = 4; /* min repeat count */
  8703. if (nextlen === 0) {
  8704. max_count = 138;
  8705. min_count = 3;
  8706. }
  8707. tree[(max_code+1)*2 + 1]/*.Len*/ = 0xffff; /* guard */
  8708. for (n = 0; n <= max_code; n++) {
  8709. curlen = nextlen;
  8710. nextlen = tree[(n+1)*2 + 1]/*.Len*/;
  8711. if (++count < max_count && curlen === nextlen) {
  8712. continue;
  8713. } else if (count < min_count) {
  8714. s.bl_tree[curlen * 2]/*.Freq*/ += count;
  8715. } else if (curlen !== 0) {
  8716. if (curlen !== prevlen) { s.bl_tree[curlen * 2]/*.Freq*/++; }
  8717. s.bl_tree[REP_3_6*2]/*.Freq*/++;
  8718. } else if (count <= 10) {
  8719. s.bl_tree[REPZ_3_10*2]/*.Freq*/++;
  8720. } else {
  8721. s.bl_tree[REPZ_11_138*2]/*.Freq*/++;
  8722. }
  8723. count = 0;
  8724. prevlen = curlen;
  8725. if (nextlen === 0) {
  8726. max_count = 138;
  8727. min_count = 3;
  8728. } else if (curlen === nextlen) {
  8729. max_count = 6;
  8730. min_count = 3;
  8731. } else {
  8732. max_count = 7;
  8733. min_count = 4;
  8734. }
  8735. }
  8736. }
  8737. /* ===========================================================================
  8738. * Send a literal or distance tree in compressed form, using the codes in
  8739. * bl_tree.
  8740. */
  8741. function send_tree(s, tree, max_code)
  8742. // deflate_state *s;
  8743. // ct_data *tree; /* the tree to be scanned */
  8744. // int max_code; /* and its largest code of non zero frequency */
  8745. {
  8746. var n; /* iterates over all tree elements */
  8747. var prevlen = -1; /* last emitted length */
  8748. var curlen; /* length of current code */
  8749. var nextlen = tree[0*2 + 1]/*.Len*/; /* length of next code */
  8750. var count = 0; /* repeat count of the current code */
  8751. var max_count = 7; /* max repeat count */
  8752. var min_count = 4; /* min repeat count */
  8753. /* tree[max_code+1].Len = -1; */ /* guard already set */
  8754. if (nextlen === 0) {
  8755. max_count = 138;
  8756. min_count = 3;
  8757. }
  8758. for (n = 0; n <= max_code; n++) {
  8759. curlen = nextlen;
  8760. nextlen = tree[(n+1)*2 + 1]/*.Len*/;
  8761. if (++count < max_count && curlen === nextlen) {
  8762. continue;
  8763. } else if (count < min_count) {
  8764. do { send_code(s, curlen, s.bl_tree); } while (--count !== 0);
  8765. } else if (curlen !== 0) {
  8766. if (curlen !== prevlen) {
  8767. send_code(s, curlen, s.bl_tree);
  8768. count--;
  8769. }
  8770. //Assert(count >= 3 && count <= 6, " 3_6?");
  8771. send_code(s, REP_3_6, s.bl_tree);
  8772. send_bits(s, count-3, 2);
  8773. } else if (count <= 10) {
  8774. send_code(s, REPZ_3_10, s.bl_tree);
  8775. send_bits(s, count-3, 3);
  8776. } else {
  8777. send_code(s, REPZ_11_138, s.bl_tree);
  8778. send_bits(s, count-11, 7);
  8779. }
  8780. count = 0;
  8781. prevlen = curlen;
  8782. if (nextlen === 0) {
  8783. max_count = 138;
  8784. min_count = 3;
  8785. } else if (curlen === nextlen) {
  8786. max_count = 6;
  8787. min_count = 3;
  8788. } else {
  8789. max_count = 7;
  8790. min_count = 4;
  8791. }
  8792. }
  8793. }
  8794. /* ===========================================================================
  8795. * Construct the Huffman tree for the bit lengths and return the index in
  8796. * bl_order of the last bit length code to send.
  8797. */
  8798. function build_bl_tree(s) {
  8799. var max_blindex; /* index of last bit length code of non zero freq */
  8800. /* Determine the bit length frequencies for literal and distance trees */
  8801. scan_tree(s, s.dyn_ltree, s.l_desc.max_code);
  8802. scan_tree(s, s.dyn_dtree, s.d_desc.max_code);
  8803. /* Build the bit length tree: */
  8804. build_tree(s, s.bl_desc);
  8805. /* opt_len now includes the length of the tree representations, except
  8806. * the lengths of the bit lengths codes and the 5+5+4 bits for the counts.
  8807. */
  8808. /* Determine the number of bit length codes to send. The pkzip format
  8809. * requires that at least 4 bit length codes be sent. (appnote.txt says
  8810. * 3 but the actual value used is 4.)
  8811. */
  8812. for (max_blindex = BL_CODES-1; max_blindex >= 3; max_blindex--) {
  8813. if (s.bl_tree[bl_order[max_blindex]*2 + 1]/*.Len*/ !== 0) {
  8814. break;
  8815. }
  8816. }
  8817. /* Update opt_len to include the bit length tree and counts */
  8818. s.opt_len += 3*(max_blindex+1) + 5+5+4;
  8819. //Tracev((stderr, "\ndyn trees: dyn %ld, stat %ld",
  8820. // s->opt_len, s->static_len));
  8821. return max_blindex;
  8822. }
  8823. /* ===========================================================================
  8824. * Send the header for a block using dynamic Huffman trees: the counts, the
  8825. * lengths of the bit length codes, the literal tree and the distance tree.
  8826. * IN assertion: lcodes >= 257, dcodes >= 1, blcodes >= 4.
  8827. */
  8828. function send_all_trees(s, lcodes, dcodes, blcodes)
  8829. // deflate_state *s;
  8830. // int lcodes, dcodes, blcodes; /* number of codes for each tree */
  8831. {
  8832. var rank; /* index in bl_order */
  8833. //Assert (lcodes >= 257 && dcodes >= 1 && blcodes >= 4, "not enough codes");
  8834. //Assert (lcodes <= L_CODES && dcodes <= D_CODES && blcodes <= BL_CODES,
  8835. // "too many codes");
  8836. //Tracev((stderr, "\nbl counts: "));
  8837. send_bits(s, lcodes-257, 5); /* not +255 as stated in appnote.txt */
  8838. send_bits(s, dcodes-1, 5);
  8839. send_bits(s, blcodes-4, 4); /* not -3 as stated in appnote.txt */
  8840. for (rank = 0; rank < blcodes; rank++) {
  8841. //Tracev((stderr, "\nbl code %2d ", bl_order[rank]));
  8842. send_bits(s, s.bl_tree[bl_order[rank]*2 + 1]/*.Len*/, 3);
  8843. }
  8844. //Tracev((stderr, "\nbl tree: sent %ld", s->bits_sent));
  8845. send_tree(s, s.dyn_ltree, lcodes-1); /* literal tree */
  8846. //Tracev((stderr, "\nlit tree: sent %ld", s->bits_sent));
  8847. send_tree(s, s.dyn_dtree, dcodes-1); /* distance tree */
  8848. //Tracev((stderr, "\ndist tree: sent %ld", s->bits_sent));
  8849. }
  8850. /* ===========================================================================
  8851. * Check if the data type is TEXT or BINARY, using the following algorithm:
  8852. * - TEXT if the two conditions below are satisfied:
  8853. * a) There are no non-portable control characters belonging to the
  8854. * "black list" (0..6, 14..25, 28..31).
  8855. * b) There is at least one printable character belonging to the
  8856. * "white list" (9 {TAB}, 10 {LF}, 13 {CR}, 32..255).
  8857. * - BINARY otherwise.
  8858. * - The following partially-portable control characters form a
  8859. * "gray list" that is ignored in this detection algorithm:
  8860. * (7 {BEL}, 8 {BS}, 11 {VT}, 12 {FF}, 26 {SUB}, 27 {ESC}).
  8861. * IN assertion: the fields Freq of dyn_ltree are set.
  8862. */
  8863. function detect_data_type(s) {
  8864. /* black_mask is the bit mask of black-listed bytes
  8865. * set bits 0..6, 14..25, and 28..31
  8866. * 0xf3ffc07f = binary 11110011111111111100000001111111
  8867. */
  8868. var black_mask = 0xf3ffc07f;
  8869. var n;
  8870. /* Check for non-textual ("black-listed") bytes. */
  8871. for (n = 0; n <= 31; n++, black_mask >>>= 1) {
  8872. if ((black_mask & 1) && (s.dyn_ltree[n*2]/*.Freq*/ !== 0)) {
  8873. return Z_BINARY;
  8874. }
  8875. }
  8876. /* Check for textual ("white-listed") bytes. */
  8877. if (s.dyn_ltree[9 * 2]/*.Freq*/ !== 0 || s.dyn_ltree[10 * 2]/*.Freq*/ !== 0 ||
  8878. s.dyn_ltree[13 * 2]/*.Freq*/ !== 0) {
  8879. return Z_TEXT;
  8880. }
  8881. for (n = 32; n < LITERALS; n++) {
  8882. if (s.dyn_ltree[n * 2]/*.Freq*/ !== 0) {
  8883. return Z_TEXT;
  8884. }
  8885. }
  8886. /* There are no "black-listed" or "white-listed" bytes:
  8887. * this stream either is empty or has tolerated ("gray-listed") bytes only.
  8888. */
  8889. return Z_BINARY;
  8890. }
  8891. var static_init_done = false;
  8892. /* ===========================================================================
  8893. * Initialize the tree data structures for a new zlib stream.
  8894. */
  8895. function _tr_init(s)
  8896. {
  8897. if (!static_init_done) {
  8898. tr_static_init();
  8899. static_init_done = true;
  8900. }
  8901. s.l_desc = new TreeDesc(s.dyn_ltree, static_l_desc);
  8902. s.d_desc = new TreeDesc(s.dyn_dtree, static_d_desc);
  8903. s.bl_desc = new TreeDesc(s.bl_tree, static_bl_desc);
  8904. s.bi_buf = 0;
  8905. s.bi_valid = 0;
  8906. /* Initialize the first block of the first file: */
  8907. init_block(s);
  8908. }
  8909. /* ===========================================================================
  8910. * Send a stored block
  8911. */
  8912. function _tr_stored_block(s, buf, stored_len, last)
  8913. //DeflateState *s;
  8914. //charf *buf; /* input block */
  8915. //ulg stored_len; /* length of input block */
  8916. //int last; /* one if this is the last block for a file */
  8917. {
  8918. send_bits(s, (STORED_BLOCK<<1)+(last ? 1 : 0), 3); /* send block type */
  8919. copy_block(s, buf, stored_len, true); /* with header */
  8920. }
  8921. /* ===========================================================================
  8922. * Send one empty static block to give enough lookahead for inflate.
  8923. * This takes 10 bits, of which 7 may remain in the bit buffer.
  8924. */
  8925. function _tr_align(s) {
  8926. send_bits(s, STATIC_TREES<<1, 3);
  8927. send_code(s, END_BLOCK, static_ltree);
  8928. bi_flush(s);
  8929. }
  8930. /* ===========================================================================
  8931. * Determine the best encoding for the current block: dynamic trees, static
  8932. * trees or store, and output the encoded block to the zip file.
  8933. */
  8934. function _tr_flush_block(s, buf, stored_len, last)
  8935. //DeflateState *s;
  8936. //charf *buf; /* input block, or NULL if too old */
  8937. //ulg stored_len; /* length of input block */
  8938. //int last; /* one if this is the last block for a file */
  8939. {
  8940. var opt_lenb, static_lenb; /* opt_len and static_len in bytes */
  8941. var max_blindex = 0; /* index of last bit length code of non zero freq */
  8942. /* Build the Huffman trees unless a stored block is forced */
  8943. if (s.level > 0) {
  8944. /* Check if the file is binary or text */
  8945. if (s.strm.data_type === Z_UNKNOWN) {
  8946. s.strm.data_type = detect_data_type(s);
  8947. }
  8948. /* Construct the literal and distance trees */
  8949. build_tree(s, s.l_desc);
  8950. // Tracev((stderr, "\nlit data: dyn %ld, stat %ld", s->opt_len,
  8951. // s->static_len));
  8952. build_tree(s, s.d_desc);
  8953. // Tracev((stderr, "\ndist data: dyn %ld, stat %ld", s->opt_len,
  8954. // s->static_len));
  8955. /* At this point, opt_len and static_len are the total bit lengths of
  8956. * the compressed block data, excluding the tree representations.
  8957. */
  8958. /* Build the bit length tree for the above two trees, and get the index
  8959. * in bl_order of the last bit length code to send.
  8960. */
  8961. max_blindex = build_bl_tree(s);
  8962. /* Determine the best encoding. Compute the block lengths in bytes. */
  8963. opt_lenb = (s.opt_len+3+7) >>> 3;
  8964. static_lenb = (s.static_len+3+7) >>> 3;
  8965. // Tracev((stderr, "\nopt %lu(%lu) stat %lu(%lu) stored %lu lit %u ",
  8966. // opt_lenb, s->opt_len, static_lenb, s->static_len, stored_len,
  8967. // s->last_lit));
  8968. if (static_lenb <= opt_lenb) { opt_lenb = static_lenb; }
  8969. } else {
  8970. // Assert(buf != (char*)0, "lost buf");
  8971. opt_lenb = static_lenb = stored_len + 5; /* force a stored block */
  8972. }
  8973. if ((stored_len+4 <= opt_lenb) && (buf !== -1)) {
  8974. /* 4: two words for the lengths */
  8975. /* The test buf != NULL is only necessary if LIT_BUFSIZE > WSIZE.
  8976. * Otherwise we can't have processed more than WSIZE input bytes since
  8977. * the last block flush, because compression would have been
  8978. * successful. If LIT_BUFSIZE <= WSIZE, it is never too late to
  8979. * transform a block into a stored block.
  8980. */
  8981. _tr_stored_block(s, buf, stored_len, last);
  8982. } else if (s.strategy === Z_FIXED || static_lenb === opt_lenb) {
  8983. send_bits(s, (STATIC_TREES<<1) + (last ? 1 : 0), 3);
  8984. compress_block(s, static_ltree, static_dtree);
  8985. } else {
  8986. send_bits(s, (DYN_TREES<<1) + (last ? 1 : 0), 3);
  8987. send_all_trees(s, s.l_desc.max_code+1, s.d_desc.max_code+1, max_blindex+1);
  8988. compress_block(s, s.dyn_ltree, s.dyn_dtree);
  8989. }
  8990. // Assert (s->compressed_len == s->bits_sent, "bad compressed size");
  8991. /* The above check is made mod 2^32, for files larger than 512 MB
  8992. * and uLong implemented on 32 bits.
  8993. */
  8994. init_block(s);
  8995. if (last) {
  8996. bi_windup(s);
  8997. }
  8998. // Tracev((stderr,"\ncomprlen %lu(%lu) ", s->compressed_len>>3,
  8999. // s->compressed_len-7*last));
  9000. }
  9001. /* ===========================================================================
  9002. * Save the match info and tally the frequency counts. Return true if
  9003. * the current block must be flushed.
  9004. */
  9005. function _tr_tally(s, dist, lc)
  9006. // deflate_state *s;
  9007. // unsigned dist; /* distance of matched string */
  9008. // unsigned lc; /* match length-MIN_MATCH or unmatched char (if dist==0) */
  9009. {
  9010. //var out_length, in_length, dcode;
  9011. s.pending_buf[s.d_buf + s.last_lit * 2] = (dist >>> 8) & 0xff;
  9012. s.pending_buf[s.d_buf + s.last_lit * 2 + 1] = dist & 0xff;
  9013. s.pending_buf[s.l_buf + s.last_lit] = lc & 0xff;
  9014. s.last_lit++;
  9015. if (dist === 0) {
  9016. /* lc is the unmatched char */
  9017. s.dyn_ltree[lc*2]/*.Freq*/++;
  9018. } else {
  9019. s.matches++;
  9020. /* Here, lc is the match length - MIN_MATCH */
  9021. dist--; /* dist = match distance - 1 */
  9022. //Assert((ush)dist < (ush)MAX_DIST(s) &&
  9023. // (ush)lc <= (ush)(MAX_MATCH-MIN_MATCH) &&
  9024. // (ush)d_code(dist) < (ush)D_CODES, "_tr_tally: bad match");
  9025. s.dyn_ltree[(_length_code[lc]+LITERALS+1) * 2]/*.Freq*/++;
  9026. s.dyn_dtree[d_code(dist) * 2]/*.Freq*/++;
  9027. }
  9028. // (!) This block is disabled in zlib defailts,
  9029. // don't enable it for binary compatibility
  9030. //#ifdef TRUNCATE_BLOCK
  9031. // /* Try to guess if it is profitable to stop the current block here */
  9032. // if ((s.last_lit & 0x1fff) === 0 && s.level > 2) {
  9033. // /* Compute an upper bound for the compressed length */
  9034. // out_length = s.last_lit*8;
  9035. // in_length = s.strstart - s.block_start;
  9036. //
  9037. // for (dcode = 0; dcode < D_CODES; dcode++) {
  9038. // out_length += s.dyn_dtree[dcode*2]/*.Freq*/ * (5 + extra_dbits[dcode]);
  9039. // }
  9040. // out_length >>>= 3;
  9041. // //Tracev((stderr,"\nlast_lit %u, in %ld, out ~%ld(%ld%%) ",
  9042. // // s->last_lit, in_length, out_length,
  9043. // // 100L - out_length*100L/in_length));
  9044. // if (s.matches < (s.last_lit>>1)/*int /2*/ && out_length < (in_length>>1)/*int /2*/) {
  9045. // return true;
  9046. // }
  9047. // }
  9048. //#endif
  9049. return (s.last_lit === s.lit_bufsize-1);
  9050. /* We avoid equality with lit_bufsize because of wraparound at 64K
  9051. * on 16 bit machines and because stored blocks are restricted to
  9052. * 64K-1 bytes.
  9053. */
  9054. }
  9055. exports._tr_init = _tr_init;
  9056. exports._tr_stored_block = _tr_stored_block;
  9057. exports._tr_flush_block = _tr_flush_block;
  9058. exports._tr_tally = _tr_tally;
  9059. exports._tr_align = _tr_align;
  9060. },{"../utils/common":27}],39:[function(_dereq_,module,exports){
  9061. 'use strict';
  9062. function ZStream() {
  9063. /* next input byte */
  9064. this.input = null; // JS specific, because we have no pointers
  9065. this.next_in = 0;
  9066. /* number of bytes available at input */
  9067. this.avail_in = 0;
  9068. /* total number of input bytes read so far */
  9069. this.total_in = 0;
  9070. /* next output byte should be put there */
  9071. this.output = null; // JS specific, because we have no pointers
  9072. this.next_out = 0;
  9073. /* remaining free space at output */
  9074. this.avail_out = 0;
  9075. /* total number of bytes output so far */
  9076. this.total_out = 0;
  9077. /* last error message, NULL if no error */
  9078. this.msg = ''/*Z_NULL*/;
  9079. /* not visible by applications */
  9080. this.state = null;
  9081. /* best guess about the data type: binary or text */
  9082. this.data_type = 2/*Z_UNKNOWN*/;
  9083. /* adler32 value of the uncompressed data */
  9084. this.adler = 0;
  9085. }
  9086. module.exports = ZStream;
  9087. },{}]},{},[9])
  9088. (9)
  9089. }));/*---------split--------*//*! xlsx.js (C) 2013-present SheetJS -- http://sheetjs.com */
  9090. /* vim: set ts=2: */
  9091. /*exported XLSX */
  9092. /*global global, exports, module, require:false, process:false, Buffer:false, ArrayBuffer:false */
  9093. var XLSX = {};
  9094. function make_xlsx_lib(XLSX){
  9095. XLSX.version = '0.14.3';
  9096. var current_codepage = 1200, current_ansi = 1252;
  9097. /*global cptable:true, window */
  9098. if(typeof module !== "undefined" && typeof require !== 'undefined') {
  9099. if(typeof cptable === 'undefined') {
  9100. if(typeof global !== 'undefined') global.cptable = undefined;
  9101. else if(typeof window !== 'undefined') window.cptable = undefined;
  9102. }
  9103. }
  9104. var VALID_ANSI = [ 874, 932, 936, 949, 950 ];
  9105. for(var i = 0; i <= 8; ++i) VALID_ANSI.push(1250 + i);
  9106. /* ECMA-376 Part I 18.4.1 charset to codepage mapping */
  9107. var CS2CP = ({
  9108. 0: 1252, /* ANSI */
  9109. 1: 65001, /* DEFAULT */
  9110. 2: 65001, /* SYMBOL */
  9111. 77: 10000, /* MAC */
  9112. 128: 932, /* SHIFTJIS */
  9113. 129: 949, /* HANGUL */
  9114. 130: 1361, /* JOHAB */
  9115. 134: 936, /* GB2312 */
  9116. 136: 950, /* CHINESEBIG5 */
  9117. 161: 1253, /* GREEK */
  9118. 162: 1254, /* TURKISH */
  9119. 163: 1258, /* VIETNAMESE */
  9120. 177: 1255, /* HEBREW */
  9121. 178: 1256, /* ARABIC */
  9122. 186: 1257, /* BALTIC */
  9123. 204: 1251, /* RUSSIAN */
  9124. 222: 874, /* THAI */
  9125. 238: 1250, /* EASTEUROPE */
  9126. 255: 1252, /* OEM */
  9127. 69: 6969 /* MISC */
  9128. });
  9129. var set_ansi = function(cp) { if(VALID_ANSI.indexOf(cp) == -1) return; current_ansi = CS2CP[0] = cp; };
  9130. function reset_ansi() { set_ansi(1252); }
  9131. var set_cp = function(cp) { current_codepage = cp; set_ansi(cp); };
  9132. function reset_cp() { set_cp(1200); reset_ansi(); }
  9133. function char_codes(data) { var o = []; for(var i = 0, len = data.length; i < len; ++i) o[i] = data.charCodeAt(i); return o; }
  9134. function utf16leread(data) {
  9135. var o = [];
  9136. for(var i = 0; i < (data.length>>1); ++i) o[i] = String.fromCharCode(data.charCodeAt(2*i) + (data.charCodeAt(2*i+1)<<8));
  9137. return o.join("");
  9138. }
  9139. function utf16beread(data) {
  9140. var o = [];
  9141. for(var i = 0; i < (data.length>>1); ++i) o[i] = String.fromCharCode(data.charCodeAt(2*i+1) + (data.charCodeAt(2*i)<<8));
  9142. return o.join("");
  9143. }
  9144. var debom = function(data) {
  9145. var c1 = data.charCodeAt(0), c2 = data.charCodeAt(1);
  9146. if(c1 == 0xFF && c2 == 0xFE) return utf16leread(data.slice(2));
  9147. if(c1 == 0xFE && c2 == 0xFF) return utf16beread(data.slice(2));
  9148. if(c1 == 0xFEFF) return data.slice(1);
  9149. return data;
  9150. };
  9151. var _getchar = function _gc1(x) { return String.fromCharCode(x); };
  9152. if(typeof cptable !== 'undefined') {
  9153. set_cp = function(cp) { current_codepage = cp; };
  9154. debom = function(data) {
  9155. if(data.charCodeAt(0) === 0xFF && data.charCodeAt(1) === 0xFE) { return cptable.utils.decode(1200, char_codes(data.slice(2))); }
  9156. return data;
  9157. };
  9158. _getchar = function _gc2(x) {
  9159. if(current_codepage === 1200) return String.fromCharCode(x);
  9160. return cptable.utils.decode(current_codepage, [x&255,x>>8])[0];
  9161. };
  9162. }
  9163. var DENSE = null;
  9164. var DIF_XL = true;
  9165. var Base64 = (function make_b64(){
  9166. var map = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  9167. return {
  9168. encode: function(input) {
  9169. var o = "";
  9170. var c1=0, c2=0, c3=0, e1=0, e2=0, e3=0, e4=0;
  9171. for(var i = 0; i < input.length; ) {
  9172. c1 = input.charCodeAt(i++);
  9173. e1 = (c1 >> 2);
  9174. c2 = input.charCodeAt(i++);
  9175. e2 = ((c1 & 3) << 4) | (c2 >> 4);
  9176. c3 = input.charCodeAt(i++);
  9177. e3 = ((c2 & 15) << 2) | (c3 >> 6);
  9178. e4 = (c3 & 63);
  9179. if (isNaN(c2)) { e3 = e4 = 64; }
  9180. else if (isNaN(c3)) { e4 = 64; }
  9181. o += map.charAt(e1) + map.charAt(e2) + map.charAt(e3) + map.charAt(e4);
  9182. }
  9183. return o;
  9184. },
  9185. decode: function b64_decode(input) {
  9186. var o = "";
  9187. var c1=0, c2=0, c3=0, e1=0, e2=0, e3=0, e4=0;
  9188. input = input.replace(/[^\w\+\/\=]/g, "");
  9189. for(var i = 0; i < input.length;) {
  9190. e1 = map.indexOf(input.charAt(i++));
  9191. e2 = map.indexOf(input.charAt(i++));
  9192. c1 = (e1 << 2) | (e2 >> 4);
  9193. o += String.fromCharCode(c1);
  9194. e3 = map.indexOf(input.charAt(i++));
  9195. c2 = ((e2 & 15) << 4) | (e3 >> 2);
  9196. if (e3 !== 64) { o += String.fromCharCode(c2); }
  9197. e4 = map.indexOf(input.charAt(i++));
  9198. c3 = ((e3 & 3) << 6) | e4;
  9199. if (e4 !== 64) { o += String.fromCharCode(c3); }
  9200. }
  9201. return o;
  9202. }
  9203. };
  9204. })();
  9205. var has_buf = (typeof Buffer !== 'undefined' && typeof process !== 'undefined' && typeof process.versions !== 'undefined' && !!process.versions.node);
  9206. var Buffer_from = function(){};
  9207. if(typeof Buffer !== 'undefined') {
  9208. var nbfs = !Buffer.from;
  9209. if(!nbfs) try { Buffer.from("foo", "utf8"); } catch(e) { nbfs = true; }
  9210. Buffer_from = nbfs ? function(buf, enc) { return (enc) ? new Buffer(buf, enc) : new Buffer(buf); } : Buffer.from.bind(Buffer);
  9211. // $FlowIgnore
  9212. if(!Buffer.alloc) Buffer.alloc = function(n) { return new Buffer(n); };
  9213. // $FlowIgnore
  9214. if(!Buffer.allocUnsafe) Buffer.allocUnsafe = function(n) { return new Buffer(n); };
  9215. }
  9216. function new_raw_buf(len) {
  9217. /* jshint -W056 */
  9218. return has_buf ? Buffer.alloc(len) : new Array(len);
  9219. /* jshint +W056 */
  9220. }
  9221. function new_unsafe_buf(len) {
  9222. /* jshint -W056 */
  9223. return has_buf ? Buffer.allocUnsafe(len) : new Array(len);
  9224. /* jshint +W056 */
  9225. }
  9226. var s2a = function s2a(s) {
  9227. // $FlowIgnore
  9228. if(has_buf) return Buffer_from(s, "binary");
  9229. return s.split("").map(function(x){ return x.charCodeAt(0) & 0xff; });
  9230. };
  9231. function s2ab(s) {
  9232. if(typeof ArrayBuffer === 'undefined') return s2a(s);
  9233. var buf = new ArrayBuffer(s.length), view = new Uint8Array(buf);
  9234. for (var i=0; i!=s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF;
  9235. return buf;
  9236. }
  9237. function a2s(data) {
  9238. if(Array.isArray(data)) return data.map(_chr).join("");
  9239. var o = []; for(var i = 0; i < data.length; ++i) o[i] = _chr(data[i]); return o.join("");
  9240. }
  9241. function a2u(data) {
  9242. if(typeof Uint8Array === 'undefined') throw new Error("Unsupported");
  9243. return new Uint8Array(data);
  9244. }
  9245. function ab2a(data) {
  9246. if(typeof ArrayBuffer == 'undefined') throw new Error("Unsupported");
  9247. if(data instanceof ArrayBuffer) return ab2a(new Uint8Array(data));
  9248. var o = new Array(data.length);
  9249. for(var i = 0; i < data.length; ++i) o[i] = data[i];
  9250. return o;
  9251. }
  9252. var bconcat = function(bufs) { return [].concat.apply([], bufs); };
  9253. var chr0 = /\u0000/g, chr1 = /[\u0001-\u0006]/g;
  9254. /* ssf.js (C) 2013-present SheetJS -- http://sheetjs.com */
  9255. /*jshint -W041 */
  9256. var SSF = ({});
  9257. var make_ssf = function make_ssf(SSF){
  9258. SSF.version = '0.10.2';
  9259. function _strrev(x) { var o = "", i = x.length-1; while(i>=0) o += x.charAt(i--); return o; }
  9260. function fill(c,l) { var o = ""; while(o.length < l) o+=c; return o; }
  9261. function pad0(v,d){var t=""+v; return t.length>=d?t:fill('0',d-t.length)+t;}
  9262. function pad_(v,d){var t=""+v;return t.length>=d?t:fill(' ',d-t.length)+t;}
  9263. function rpad_(v,d){var t=""+v; return t.length>=d?t:t+fill(' ',d-t.length);}
  9264. function pad0r1(v,d){var t=""+Math.round(v); return t.length>=d?t:fill('0',d-t.length)+t;}
  9265. function pad0r2(v,d){var t=""+v; return t.length>=d?t:fill('0',d-t.length)+t;}
  9266. var p2_32 = Math.pow(2,32);
  9267. function pad0r(v,d){if(v>p2_32||v<-p2_32) return pad0r1(v,d); var i = Math.round(v); return pad0r2(i,d); }
  9268. function isgeneral(s, i) { i = i || 0; return s.length >= 7 + i && (s.charCodeAt(i)|32) === 103 && (s.charCodeAt(i+1)|32) === 101 && (s.charCodeAt(i+2)|32) === 110 && (s.charCodeAt(i+3)|32) === 101 && (s.charCodeAt(i+4)|32) === 114 && (s.charCodeAt(i+5)|32) === 97 && (s.charCodeAt(i+6)|32) === 108; }
  9269. var days = [
  9270. ['Sun', 'Sunday'],
  9271. ['Mon', 'Monday'],
  9272. ['Tue', 'Tuesday'],
  9273. ['Wed', 'Wednesday'],
  9274. ['Thu', 'Thursday'],
  9275. ['Fri', 'Friday'],
  9276. ['Sat', 'Saturday']
  9277. ];
  9278. var months = [
  9279. ['J', 'Jan', 'January'],
  9280. ['F', 'Feb', 'February'],
  9281. ['M', 'Mar', 'March'],
  9282. ['A', 'Apr', 'April'],
  9283. ['M', 'May', 'May'],
  9284. ['J', 'Jun', 'June'],
  9285. ['J', 'Jul', 'July'],
  9286. ['A', 'Aug', 'August'],
  9287. ['S', 'Sep', 'September'],
  9288. ['O', 'Oct', 'October'],
  9289. ['N', 'Nov', 'November'],
  9290. ['D', 'Dec', 'December']
  9291. ];
  9292. function init_table(t) {
  9293. t[0]= 'General';
  9294. t[1]= '0';
  9295. t[2]= '0.00';
  9296. t[3]= '#,##0';
  9297. t[4]= '#,##0.00';
  9298. t[9]= '0%';
  9299. t[10]= '0.00%';
  9300. t[11]= '0.00E+00';
  9301. t[12]= '# ?/?';
  9302. t[13]= '# ??/??';
  9303. t[14]= 'm/d/yy';
  9304. t[15]= 'd-mmm-yy';
  9305. t[16]= 'd-mmm';
  9306. t[17]= 'mmm-yy';
  9307. t[18]= 'h:mm AM/PM';
  9308. t[19]= 'h:mm:ss AM/PM';
  9309. t[20]= 'h:mm';
  9310. t[21]= 'h:mm:ss';
  9311. t[22]= 'm/d/yy h:mm';
  9312. t[37]= '#,##0 ;(#,##0)';
  9313. t[38]= '#,##0 ;[Red](#,##0)';
  9314. t[39]= '#,##0.00;(#,##0.00)';
  9315. t[40]= '#,##0.00;[Red](#,##0.00)';
  9316. t[45]= 'mm:ss';
  9317. t[46]= '[h]:mm:ss';
  9318. t[47]= 'mmss.0';
  9319. t[48]= '##0.0E+0';
  9320. t[49]= '@';
  9321. t[56]= '"上午/下午 "hh"時"mm"分"ss"秒 "';
  9322. t[65535]= 'General';
  9323. }
  9324. var table_fmt = {};
  9325. init_table(table_fmt);
  9326. function frac(x, D, mixed) {
  9327. var sgn = x < 0 ? -1 : 1;
  9328. var B = x * sgn;
  9329. var P_2 = 0, P_1 = 1, P = 0;
  9330. var Q_2 = 1, Q_1 = 0, Q = 0;
  9331. var A = Math.floor(B);
  9332. while(Q_1 < D) {
  9333. A = Math.floor(B);
  9334. P = A * P_1 + P_2;
  9335. Q = A * Q_1 + Q_2;
  9336. if((B - A) < 0.00000005) break;
  9337. B = 1 / (B - A);
  9338. P_2 = P_1; P_1 = P;
  9339. Q_2 = Q_1; Q_1 = Q;
  9340. }
  9341. if(Q > D) { if(Q_1 > D) { Q = Q_2; P = P_2; } else { Q = Q_1; P = P_1; } }
  9342. if(!mixed) return [0, sgn * P, Q];
  9343. var q = Math.floor(sgn * P/Q);
  9344. return [q, sgn*P - q*Q, Q];
  9345. }
  9346. function parse_date_code(v,opts,b2) {
  9347. if(v > 2958465 || v < 0) return null;
  9348. var date = (v|0), time = Math.floor(86400 * (v - date)), dow=0;
  9349. var dout=[];
  9350. var out={D:date, T:time, u:86400*(v-date)-time,y:0,m:0,d:0,H:0,M:0,S:0,q:0};
  9351. if(Math.abs(out.u) < 1e-6) out.u = 0;
  9352. if(opts && opts.date1904) date += 1462;
  9353. if(out.u > 0.9999) {
  9354. out.u = 0;
  9355. if(++time == 86400) { out.T = time = 0; ++date; ++out.D; }
  9356. }
  9357. if(date === 60) {dout = b2 ? [1317,10,29] : [1900,2,29]; dow=3;}
  9358. else if(date === 0) {dout = b2 ? [1317,8,29] : [1900,1,0]; dow=6;}
  9359. else {
  9360. if(date > 60) --date;
  9361. /* 1 = Jan 1 1900 in Gregorian */
  9362. var d = new Date(1900, 0, 1);
  9363. d.setDate(d.getDate() + date - 1);
  9364. dout = [d.getFullYear(), d.getMonth()+1,d.getDate()];
  9365. dow = d.getDay();
  9366. if(date < 60) dow = (dow + 6) % 7;
  9367. if(b2) dow = fix_hijri(d, dout);
  9368. }
  9369. out.y = dout[0]; out.m = dout[1]; out.d = dout[2];
  9370. out.S = time % 60; time = Math.floor(time / 60);
  9371. out.M = time % 60; time = Math.floor(time / 60);
  9372. out.H = time;
  9373. out.q = dow;
  9374. return out;
  9375. }
  9376. SSF.parse_date_code = parse_date_code;
  9377. var basedate = new Date(1899, 11, 31, 0, 0, 0);
  9378. var dnthresh = basedate.getTime();
  9379. var base1904 = new Date(1900, 2, 1, 0, 0, 0);
  9380. function datenum_local(v, date1904) {
  9381. var epoch = v.getTime();
  9382. if(date1904) epoch -= 1461*24*60*60*1000;
  9383. else if(v >= base1904) epoch += 24*60*60*1000;
  9384. return (epoch - (dnthresh + (v.getTimezoneOffset() - basedate.getTimezoneOffset()) * 60000)) / (24 * 60 * 60 * 1000);
  9385. }
  9386. function general_fmt_int(v) { return v.toString(10); }
  9387. SSF._general_int = general_fmt_int;
  9388. var general_fmt_num = (function make_general_fmt_num() {
  9389. var gnr1 = /\.(\d*[1-9])0+$/, gnr2 = /\.0*$/, gnr4 = /\.(\d*[1-9])0+/, gnr5 = /\.0*[Ee]/, gnr6 = /(E[+-])(\d)$/;
  9390. function gfn2(v) {
  9391. var w = (v<0?12:11);
  9392. var o = gfn5(v.toFixed(12)); if(o.length <= w) return o;
  9393. o = v.toPrecision(10); if(o.length <= w) return o;
  9394. return v.toExponential(5);
  9395. }
  9396. function gfn3(v) {
  9397. var o = v.toFixed(11).replace(gnr1,".$1");
  9398. if(o.length > (v<0?12:11)) o = v.toPrecision(6);
  9399. return o;
  9400. }
  9401. function gfn4(o) {
  9402. for(var i = 0; i != o.length; ++i) if((o.charCodeAt(i) | 0x20) === 101) return o.replace(gnr4,".$1").replace(gnr5,"E").replace("e","E").replace(gnr6,"$10$2");
  9403. return o;
  9404. }
  9405. function gfn5(o) {
  9406. return o.indexOf(".") > -1 ? o.replace(gnr2,"").replace(gnr1,".$1") : o;
  9407. }
  9408. return function general_fmt_num(v) {
  9409. var V = Math.floor(Math.log(Math.abs(v))*Math.LOG10E), o;
  9410. if(V >= -4 && V <= -1) o = v.toPrecision(10+V);
  9411. else if(Math.abs(V) <= 9) o = gfn2(v);
  9412. else if(V === 10) o = v.toFixed(10).substr(0,12);
  9413. else o = gfn3(v);
  9414. return gfn5(gfn4(o));
  9415. };})();
  9416. SSF._general_num = general_fmt_num;
  9417. function general_fmt(v, opts) {
  9418. switch(typeof v) {
  9419. case 'string': return v;
  9420. case 'boolean': return v ? "TRUE" : "FALSE";
  9421. case 'number': return (v|0) === v ? general_fmt_int(v) : general_fmt_num(v);
  9422. case 'undefined': return "";
  9423. case 'object':
  9424. if(v == null) return "";
  9425. if(v instanceof Date) return format(14, datenum_local(v, opts && opts.date1904), opts);
  9426. }
  9427. throw new Error("unsupported value in General format: " + v);
  9428. }
  9429. SSF._general = general_fmt;
  9430. function fix_hijri() { return 0; }
  9431. /*jshint -W086 */
  9432. function write_date(type, fmt, val, ss0) {
  9433. var o="", ss=0, tt=0, y = val.y, out, outl = 0;
  9434. switch(type) {
  9435. case 98: /* 'b' buddhist year */
  9436. y = val.y + 543;
  9437. /* falls through */
  9438. case 121: /* 'y' year */
  9439. switch(fmt.length) {
  9440. case 1: case 2: out = y % 100; outl = 2; break;
  9441. default: out = y % 10000; outl = 4; break;
  9442. } break;
  9443. case 109: /* 'm' month */
  9444. switch(fmt.length) {
  9445. case 1: case 2: out = val.m; outl = fmt.length; break;
  9446. case 3: return months[val.m-1][1];
  9447. case 5: return months[val.m-1][0];
  9448. default: return months[val.m-1][2];
  9449. } break;
  9450. case 100: /* 'd' day */
  9451. switch(fmt.length) {
  9452. case 1: case 2: out = val.d; outl = fmt.length; break;
  9453. case 3: return days[val.q][0];
  9454. default: return days[val.q][1];
  9455. } break;
  9456. case 104: /* 'h' 12-hour */
  9457. switch(fmt.length) {
  9458. case 1: case 2: out = 1+(val.H+11)%12; outl = fmt.length; break;
  9459. default: throw 'bad hour format: ' + fmt;
  9460. } break;
  9461. case 72: /* 'H' 24-hour */
  9462. switch(fmt.length) {
  9463. case 1: case 2: out = val.H; outl = fmt.length; break;
  9464. default: throw 'bad hour format: ' + fmt;
  9465. } break;
  9466. case 77: /* 'M' minutes */
  9467. switch(fmt.length) {
  9468. case 1: case 2: out = val.M; outl = fmt.length; break;
  9469. default: throw 'bad minute format: ' + fmt;
  9470. } break;
  9471. case 115: /* 's' seconds */
  9472. if(fmt != 's' && fmt != 'ss' && fmt != '.0' && fmt != '.00' && fmt != '.000') throw 'bad second format: ' + fmt;
  9473. if(val.u === 0 && (fmt == "s" || fmt == "ss")) return pad0(val.S, fmt.length);
  9474. if(ss0 >= 2) tt = ss0 === 3 ? 1000 : 100;
  9475. else tt = ss0 === 1 ? 10 : 1;
  9476. ss = Math.round((tt)*(val.S + val.u));
  9477. if(ss >= 60*tt) ss = 0;
  9478. if(fmt === 's') return ss === 0 ? "0" : ""+ss/tt;
  9479. o = pad0(ss,2 + ss0);
  9480. if(fmt === 'ss') return o.substr(0,2);
  9481. return "." + o.substr(2,fmt.length-1);
  9482. case 90: /* 'Z' absolute time */
  9483. switch(fmt) {
  9484. case '[h]': case '[hh]': out = val.D*24+val.H; break;
  9485. case '[m]': case '[mm]': out = (val.D*24+val.H)*60+val.M; break;
  9486. case '[s]': case '[ss]': out = ((val.D*24+val.H)*60+val.M)*60+Math.round(val.S+val.u); break;
  9487. default: throw 'bad abstime format: ' + fmt;
  9488. } outl = fmt.length === 3 ? 1 : 2; break;
  9489. case 101: /* 'e' era */
  9490. out = y; outl = 1;
  9491. }
  9492. if(outl > 0) return pad0(out, outl); else return "";
  9493. }
  9494. /*jshint +W086 */
  9495. function commaify(s) {
  9496. var w = 3;
  9497. if(s.length <= w) return s;
  9498. var j = (s.length % w), o = s.substr(0,j);
  9499. for(; j!=s.length; j+=w) o+=(o.length > 0 ? "," : "") + s.substr(j,w);
  9500. return o;
  9501. }
  9502. var write_num = (function make_write_num(){
  9503. var pct1 = /%/g;
  9504. function write_num_pct(type, fmt, val){
  9505. var sfmt = fmt.replace(pct1,""), mul = fmt.length - sfmt.length;
  9506. return write_num(type, sfmt, val * Math.pow(10,2*mul)) + fill("%",mul);
  9507. }
  9508. function write_num_cm(type, fmt, val){
  9509. var idx = fmt.length - 1;
  9510. while(fmt.charCodeAt(idx-1) === 44) --idx;
  9511. return write_num(type, fmt.substr(0,idx), val / Math.pow(10,3*(fmt.length-idx)));
  9512. }
  9513. function write_num_exp(fmt, val){
  9514. var o;
  9515. var idx = fmt.indexOf("E") - fmt.indexOf(".") - 1;
  9516. if(fmt.match(/^#+0.0E\+0$/)) {
  9517. if(val == 0) return "0.0E+0";
  9518. else if(val < 0) return "-" + write_num_exp(fmt, -val);
  9519. var period = fmt.indexOf("."); if(period === -1) period=fmt.indexOf('E');
  9520. var ee = Math.floor(Math.log(val)*Math.LOG10E)%period;
  9521. if(ee < 0) ee += period;
  9522. o = (val/Math.pow(10,ee)).toPrecision(idx+1+(period+ee)%period);
  9523. if(o.indexOf("e") === -1) {
  9524. var fakee = Math.floor(Math.log(val)*Math.LOG10E);
  9525. if(o.indexOf(".") === -1) o = o.charAt(0) + "." + o.substr(1) + "E+" + (fakee - o.length+ee);
  9526. else o += "E+" + (fakee - ee);
  9527. while(o.substr(0,2) === "0.") {
  9528. o = o.charAt(0) + o.substr(2,period) + "." + o.substr(2+period);
  9529. o = o.replace(/^0+([1-9])/,"$1").replace(/^0+\./,"0.");
  9530. }
  9531. o = o.replace(/\+-/,"-");
  9532. }
  9533. o = o.replace(/^([+-]?)(\d*)\.(\d*)[Ee]/,function($$,$1,$2,$3) { return $1 + $2 + $3.substr(0,(period+ee)%period) + "." + $3.substr(ee) + "E"; });
  9534. } else o = val.toExponential(idx);
  9535. if(fmt.match(/E\+00$/) && o.match(/e[+-]\d$/)) o = o.substr(0,o.length-1) + "0" + o.charAt(o.length-1);
  9536. if(fmt.match(/E\-/) && o.match(/e\+/)) o = o.replace(/e\+/,"e");
  9537. return o.replace("e","E");
  9538. }
  9539. var frac1 = /# (\?+)( ?)\/( ?)(\d+)/;
  9540. function write_num_f1(r, aval, sign) {
  9541. var den = parseInt(r[4],10), rr = Math.round(aval * den), base = Math.floor(rr/den);
  9542. var myn = (rr - base*den), myd = den;
  9543. return sign + (base === 0 ? "" : ""+base) + " " + (myn === 0 ? fill(" ", r[1].length + 1 + r[4].length) : pad_(myn,r[1].length) + r[2] + "/" + r[3] + pad0(myd,r[4].length));
  9544. }
  9545. function write_num_f2(r, aval, sign) {
  9546. return sign + (aval === 0 ? "" : ""+aval) + fill(" ", r[1].length + 2 + r[4].length);
  9547. }
  9548. var dec1 = /^#*0*\.([0#]+)/;
  9549. var closeparen = /\).*[0#]/;
  9550. var phone = /\(###\) ###\\?-####/;
  9551. function hashq(str) {
  9552. var o = "", cc;
  9553. for(var i = 0; i != str.length; ++i) switch((cc=str.charCodeAt(i))) {
  9554. case 35: break;
  9555. case 63: o+= " "; break;
  9556. case 48: o+= "0"; break;
  9557. default: o+= String.fromCharCode(cc);
  9558. }
  9559. return o;
  9560. }
  9561. function rnd(val, d) { var dd = Math.pow(10,d); return ""+(Math.round(val * dd)/dd); }
  9562. function dec(val, d) {
  9563. if (d < ('' + Math.round((val-Math.floor(val))*Math.pow(10,d))).length) {
  9564. return 0;
  9565. }
  9566. return Math.round((val-Math.floor(val))*Math.pow(10,d));
  9567. }
  9568. function carry(val, d) {
  9569. if (d < ('' + Math.round((val-Math.floor(val))*Math.pow(10,d))).length) {
  9570. return 1;
  9571. }
  9572. return 0;
  9573. }
  9574. function flr(val) { if(val < 2147483647 && val > -2147483648) return ""+(val >= 0 ? (val|0) : (val-1|0)); return ""+Math.floor(val); }
  9575. function write_num_flt(type, fmt, val) {
  9576. if(type.charCodeAt(0) === 40 && !fmt.match(closeparen)) {
  9577. var ffmt = fmt.replace(/\( */,"").replace(/ \)/,"").replace(/\)/,"");
  9578. if(val >= 0) return write_num_flt('n', ffmt, val);
  9579. return '(' + write_num_flt('n', ffmt, -val) + ')';
  9580. }
  9581. if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm(type, fmt, val);
  9582. if(fmt.indexOf('%') !== -1) return write_num_pct(type, fmt, val);
  9583. if(fmt.indexOf('E') !== -1) return write_num_exp(fmt, val);
  9584. if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
  9585. var o;
  9586. var r, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
  9587. if(fmt.match(/^00+$/)) return sign + pad0r(aval,fmt.length);
  9588. if(fmt.match(/^[#?]+$/)) {
  9589. o = pad0r(val,0); if(o === "0") o = "";
  9590. return o.length > fmt.length ? o : hashq(fmt.substr(0,fmt.length-o.length)) + o;
  9591. }
  9592. if((r = fmt.match(frac1))) return write_num_f1(r, aval, sign);
  9593. if(fmt.match(/^#+0+$/)) return sign + pad0r(aval,fmt.length - fmt.indexOf("0"));
  9594. if((r = fmt.match(dec1))) {
  9595. o = rnd(val, r[1].length).replace(/^([^\.]+)$/,"$1."+hashq(r[1])).replace(/\.$/,"."+hashq(r[1])).replace(/\.(\d*)$/,function($$, $1) { return "." + $1 + fill("0", hashq(r[1]).length-$1.length); });
  9596. return fmt.indexOf("0.") !== -1 ? o : o.replace(/^0\./,".");
  9597. }
  9598. fmt = fmt.replace(/^#+([0.])/, "$1");
  9599. if((r = fmt.match(/^(0*)\.(#*)$/))) {
  9600. return sign + rnd(aval, r[2].length).replace(/\.(\d*[1-9])0*$/,".$1").replace(/^(-?\d*)$/,"$1.").replace(/^0\./,r[1].length?"0.":".");
  9601. }
  9602. if((r = fmt.match(/^#{1,3},##0(\.?)$/))) return sign + commaify(pad0r(aval,0));
  9603. if((r = fmt.match(/^#,##0\.([#0]*0)$/))) {
  9604. return val < 0 ? "-" + write_num_flt(type, fmt, -val) : commaify(""+(Math.floor(val) + carry(val, r[1].length))) + "." + pad0(dec(val, r[1].length),r[1].length);
  9605. }
  9606. if((r = fmt.match(/^#,#*,#0/))) return write_num_flt(type,fmt.replace(/^#,#*,/,""),val);
  9607. if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
  9608. o = _strrev(write_num_flt(type, fmt.replace(/[\\-]/g,""), val));
  9609. ri = 0;
  9610. return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
  9611. }
  9612. if(fmt.match(phone)) {
  9613. o = write_num_flt(type, "##########", val);
  9614. return "(" + o.substr(0,3) + ") " + o.substr(3, 3) + "-" + o.substr(6);
  9615. }
  9616. var oa = "";
  9617. if((r = fmt.match(/^([#0?]+)( ?)\/( ?)([#0?]+)/))) {
  9618. ri = Math.min(r[4].length,7);
  9619. ff = frac(aval, Math.pow(10,ri)-1, false);
  9620. o = "" + sign;
  9621. oa = write_num("n", r[1], ff[1]);
  9622. if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
  9623. o += oa + r[2] + "/" + r[3];
  9624. oa = rpad_(ff[2],ri);
  9625. if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
  9626. o += oa;
  9627. return o;
  9628. }
  9629. if((r = fmt.match(/^# ([#0?]+)( ?)\/( ?)([#0?]+)/))) {
  9630. ri = Math.min(Math.max(r[1].length, r[4].length),7);
  9631. ff = frac(aval, Math.pow(10,ri)-1, true);
  9632. return sign + (ff[0]||(ff[1] ? "" : "0")) + " " + (ff[1] ? pad_(ff[1],ri) + r[2] + "/" + r[3] + rpad_(ff[2],ri): fill(" ", 2*ri+1 + r[2].length + r[3].length));
  9633. }
  9634. if((r = fmt.match(/^[#0?]+$/))) {
  9635. o = pad0r(val, 0);
  9636. if(fmt.length <= o.length) return o;
  9637. return hashq(fmt.substr(0,fmt.length-o.length)) + o;
  9638. }
  9639. if((r = fmt.match(/^([#0?]+)\.([#0]+)$/))) {
  9640. o = "" + val.toFixed(Math.min(r[2].length,10)).replace(/([^0])0+$/,"$1");
  9641. ri = o.indexOf(".");
  9642. var lres = fmt.indexOf(".") - ri, rres = fmt.length - o.length - lres;
  9643. return hashq(fmt.substr(0,lres) + o + fmt.substr(fmt.length-rres));
  9644. }
  9645. if((r = fmt.match(/^00,000\.([#0]*0)$/))) {
  9646. ri = dec(val, r[1].length);
  9647. return val < 0 ? "-" + write_num_flt(type, fmt, -val) : commaify(flr(val)).replace(/^\d,\d{3}$/,"0$&").replace(/^\d*$/,function($$) { return "00," + ($$.length < 3 ? pad0(0,3-$$.length) : "") + $$; }) + "." + pad0(ri,r[1].length);
  9648. }
  9649. switch(fmt) {
  9650. case "###,##0.00": return write_num_flt(type, "#,##0.00", val);
  9651. case "###,###":
  9652. case "##,###":
  9653. case "#,###": var x = commaify(pad0r(aval,0)); return x !== "0" ? sign + x : "";
  9654. case "###,###.00": return write_num_flt(type, "###,##0.00",val).replace(/^0\./,".");
  9655. case "#,###.00": return write_num_flt(type, "#,##0.00",val).replace(/^0\./,".");
  9656. default:
  9657. }
  9658. throw new Error("unsupported format |" + fmt + "|");
  9659. }
  9660. function write_num_cm2(type, fmt, val){
  9661. var idx = fmt.length - 1;
  9662. while(fmt.charCodeAt(idx-1) === 44) --idx;
  9663. return write_num(type, fmt.substr(0,idx), val / Math.pow(10,3*(fmt.length-idx)));
  9664. }
  9665. function write_num_pct2(type, fmt, val){
  9666. var sfmt = fmt.replace(pct1,""), mul = fmt.length - sfmt.length;
  9667. return write_num(type, sfmt, val * Math.pow(10,2*mul)) + fill("%",mul);
  9668. }
  9669. function write_num_exp2(fmt, val){
  9670. var o;
  9671. var idx = fmt.indexOf("E") - fmt.indexOf(".") - 1;
  9672. if(fmt.match(/^#+0.0E\+0$/)) {
  9673. if(val == 0) return "0.0E+0";
  9674. else if(val < 0) return "-" + write_num_exp2(fmt, -val);
  9675. var period = fmt.indexOf("."); if(period === -1) period=fmt.indexOf('E');
  9676. var ee = Math.floor(Math.log(val)*Math.LOG10E)%period;
  9677. if(ee < 0) ee += period;
  9678. o = (val/Math.pow(10,ee)).toPrecision(idx+1+(period+ee)%period);
  9679. if(!o.match(/[Ee]/)) {
  9680. var fakee = Math.floor(Math.log(val)*Math.LOG10E);
  9681. if(o.indexOf(".") === -1) o = o.charAt(0) + "." + o.substr(1) + "E+" + (fakee - o.length+ee);
  9682. else o += "E+" + (fakee - ee);
  9683. o = o.replace(/\+-/,"-");
  9684. }
  9685. o = o.replace(/^([+-]?)(\d*)\.(\d*)[Ee]/,function($$,$1,$2,$3) { return $1 + $2 + $3.substr(0,(period+ee)%period) + "." + $3.substr(ee) + "E"; });
  9686. } else o = val.toExponential(idx);
  9687. if(fmt.match(/E\+00$/) && o.match(/e[+-]\d$/)) o = o.substr(0,o.length-1) + "0" + o.charAt(o.length-1);
  9688. if(fmt.match(/E\-/) && o.match(/e\+/)) o = o.replace(/e\+/,"e");
  9689. return o.replace("e","E");
  9690. }
  9691. function write_num_int(type, fmt, val) {
  9692. if(type.charCodeAt(0) === 40 && !fmt.match(closeparen)) {
  9693. var ffmt = fmt.replace(/\( */,"").replace(/ \)/,"").replace(/\)/,"");
  9694. if(val >= 0) return write_num_int('n', ffmt, val);
  9695. return '(' + write_num_int('n', ffmt, -val) + ')';
  9696. }
  9697. if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm2(type, fmt, val);
  9698. if(fmt.indexOf('%') !== -1) return write_num_pct2(type, fmt, val);
  9699. if(fmt.indexOf('E') !== -1) return write_num_exp2(fmt, val);
  9700. if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
  9701. var o;
  9702. var r, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
  9703. if(fmt.match(/^00+$/)) return sign + pad0(aval,fmt.length);
  9704. if(fmt.match(/^[#?]+$/)) {
  9705. o = (""+val); if(val === 0) o = "";
  9706. return o.length > fmt.length ? o : hashq(fmt.substr(0,fmt.length-o.length)) + o;
  9707. }
  9708. if((r = fmt.match(frac1))) return write_num_f2(r, aval, sign);
  9709. if(fmt.match(/^#+0+$/)) return sign + pad0(aval,fmt.length - fmt.indexOf("0"));
  9710. if((r = fmt.match(dec1))) {
  9711. o = (""+val).replace(/^([^\.]+)$/,"$1."+hashq(r[1])).replace(/\.$/,"."+hashq(r[1]));
  9712. o = o.replace(/\.(\d*)$/,function($$, $1) {
  9713. return "." + $1 + fill("0", hashq(r[1]).length-$1.length); });
  9714. return fmt.indexOf("0.") !== -1 ? o : o.replace(/^0\./,".");
  9715. }
  9716. fmt = fmt.replace(/^#+([0.])/, "$1");
  9717. if((r = fmt.match(/^(0*)\.(#*)$/))) {
  9718. return sign + (""+aval).replace(/\.(\d*[1-9])0*$/,".$1").replace(/^(-?\d*)$/,"$1.").replace(/^0\./,r[1].length?"0.":".");
  9719. }
  9720. if((r = fmt.match(/^#{1,3},##0(\.?)$/))) return sign + commaify((""+aval));
  9721. if((r = fmt.match(/^#,##0\.([#0]*0)$/))) {
  9722. return val < 0 ? "-" + write_num_int(type, fmt, -val) : commaify((""+val)) + "." + fill('0',r[1].length);
  9723. }
  9724. if((r = fmt.match(/^#,#*,#0/))) return write_num_int(type,fmt.replace(/^#,#*,/,""),val);
  9725. if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
  9726. o = _strrev(write_num_int(type, fmt.replace(/[\\-]/g,""), val));
  9727. ri = 0;
  9728. return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
  9729. }
  9730. if(fmt.match(phone)) {
  9731. o = write_num_int(type, "##########", val);
  9732. return "(" + o.substr(0,3) + ") " + o.substr(3, 3) + "-" + o.substr(6);
  9733. }
  9734. var oa = "";
  9735. if((r = fmt.match(/^([#0?]+)( ?)\/( ?)([#0?]+)/))) {
  9736. ri = Math.min(r[4].length,7);
  9737. ff = frac(aval, Math.pow(10,ri)-1, false);
  9738. o = "" + sign;
  9739. oa = write_num("n", r[1], ff[1]);
  9740. if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
  9741. o += oa + r[2] + "/" + r[3];
  9742. oa = rpad_(ff[2],ri);
  9743. if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
  9744. o += oa;
  9745. return o;
  9746. }
  9747. if((r = fmt.match(/^# ([#0?]+)( ?)\/( ?)([#0?]+)/))) {
  9748. ri = Math.min(Math.max(r[1].length, r[4].length),7);
  9749. ff = frac(aval, Math.pow(10,ri)-1, true);
  9750. return sign + (ff[0]||(ff[1] ? "" : "0")) + " " + (ff[1] ? pad_(ff[1],ri) + r[2] + "/" + r[3] + rpad_(ff[2],ri): fill(" ", 2*ri+1 + r[2].length + r[3].length));
  9751. }
  9752. if((r = fmt.match(/^[#0?]+$/))) {
  9753. o = "" + val;
  9754. if(fmt.length <= o.length) return o;
  9755. return hashq(fmt.substr(0,fmt.length-o.length)) + o;
  9756. }
  9757. if((r = fmt.match(/^([#0]+)\.([#0]+)$/))) {
  9758. o = "" + val.toFixed(Math.min(r[2].length,10)).replace(/([^0])0+$/,"$1");
  9759. ri = o.indexOf(".");
  9760. var lres = fmt.indexOf(".") - ri, rres = fmt.length - o.length - lres;
  9761. return hashq(fmt.substr(0,lres) + o + fmt.substr(fmt.length-rres));
  9762. }
  9763. if((r = fmt.match(/^00,000\.([#0]*0)$/))) {
  9764. return val < 0 ? "-" + write_num_int(type, fmt, -val) : commaify(""+val).replace(/^\d,\d{3}$/,"0$&").replace(/^\d*$/,function($$) { return "00," + ($$.length < 3 ? pad0(0,3-$$.length) : "") + $$; }) + "." + pad0(0,r[1].length);
  9765. }
  9766. switch(fmt) {
  9767. case "###,###":
  9768. case "##,###":
  9769. case "#,###": var x = commaify(""+aval); return x !== "0" ? sign + x : "";
  9770. default:
  9771. if(fmt.match(/\.[0#?]*$/)) return write_num_int(type, fmt.slice(0,fmt.lastIndexOf(".")), val) + hashq(fmt.slice(fmt.lastIndexOf(".")));
  9772. }
  9773. throw new Error("unsupported format |" + fmt + "|");
  9774. }
  9775. return function write_num(type, fmt, val) {
  9776. return (val|0) === val ? write_num_int(type, fmt, val) : write_num_flt(type, fmt, val);
  9777. };})();
  9778. function split_fmt(fmt) {
  9779. var out = [];
  9780. var in_str = false/*, cc*/;
  9781. for(var i = 0, j = 0; i < fmt.length; ++i) switch((/*cc=*/fmt.charCodeAt(i))) {
  9782. case 34: /* '"' */
  9783. in_str = !in_str; break;
  9784. case 95: case 42: case 92: /* '_' '*' '\\' */
  9785. ++i; break;
  9786. case 59: /* ';' */
  9787. out[out.length] = fmt.substr(j,i-j);
  9788. j = i+1;
  9789. }
  9790. out[out.length] = fmt.substr(j);
  9791. if(in_str === true) throw new Error("Format |" + fmt + "| unterminated string ");
  9792. return out;
  9793. }
  9794. SSF._split = split_fmt;
  9795. var abstime = /\[[HhMmSs]*\]/;
  9796. function fmt_is_date(fmt) {
  9797. var i = 0, /*cc = 0,*/ c = "", o = "";
  9798. while(i < fmt.length) {
  9799. switch((c = fmt.charAt(i))) {
  9800. case 'G': if(isgeneral(fmt, i)) i+= 6; i++; break;
  9801. case '"': for(;(/*cc=*/fmt.charCodeAt(++i)) !== 34 && i < fmt.length;) ++i; ++i; break;
  9802. case '\\': i+=2; break;
  9803. case '_': i+=2; break;
  9804. case '@': ++i; break;
  9805. case 'B': case 'b':
  9806. if(fmt.charAt(i+1) === "1" || fmt.charAt(i+1) === "2") return true;
  9807. /* falls through */
  9808. case 'M': case 'D': case 'Y': case 'H': case 'S': case 'E':
  9809. /* falls through */
  9810. case 'm': case 'd': case 'y': case 'h': case 's': case 'e': case 'g': return true;
  9811. case 'A': case 'a':
  9812. if(fmt.substr(i, 3).toUpperCase() === "A/P") return true;
  9813. if(fmt.substr(i, 5).toUpperCase() === "AM/PM") return true;
  9814. ++i; break;
  9815. case '[':
  9816. o = c;
  9817. while(fmt.charAt(i++) !== ']' && i < fmt.length) o += fmt.charAt(i);
  9818. if(o.match(abstime)) return true;
  9819. break;
  9820. case '.':
  9821. /* falls through */
  9822. case '0': case '#':
  9823. while(i < fmt.length && ("0#?.,E+-%".indexOf(c=fmt.charAt(++i)) > -1 || (c=='\\' && fmt.charAt(i+1) == "-" && "0#".indexOf(fmt.charAt(i+2))>-1))){/* empty */}
  9824. break;
  9825. case '?': while(fmt.charAt(++i) === c){/* empty */} break;
  9826. case '*': ++i; if(fmt.charAt(i) == ' ' || fmt.charAt(i) == '*') ++i; break;
  9827. case '(': case ')': ++i; break;
  9828. case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9':
  9829. while(i < fmt.length && "0123456789".indexOf(fmt.charAt(++i)) > -1){/* empty */} break;
  9830. case ' ': ++i; break;
  9831. default: ++i; break;
  9832. }
  9833. }
  9834. return false;
  9835. }
  9836. SSF.is_date = fmt_is_date;
  9837. function eval_fmt(fmt, v, opts, flen) {
  9838. var out = [], o = "", i = 0, c = "", lst='t', dt, j, cc;
  9839. var hr='H';
  9840. /* Tokenize */
  9841. while(i < fmt.length) {
  9842. switch((c = fmt.charAt(i))) {
  9843. case 'G': /* General */
  9844. if(!isgeneral(fmt, i)) throw new Error('unrecognized character ' + c + ' in ' +fmt);
  9845. out[out.length] = {t:'G', v:'General'}; i+=7; break;
  9846. case '"': /* Literal text */
  9847. for(o="";(cc=fmt.charCodeAt(++i)) !== 34 && i < fmt.length;) o += String.fromCharCode(cc);
  9848. out[out.length] = {t:'t', v:o}; ++i; break;
  9849. case '\\': var w = fmt.charAt(++i), t = (w === "(" || w === ")") ? w : 't';
  9850. out[out.length] = {t:t, v:w}; ++i; break;
  9851. case '_': out[out.length] = {t:'t', v:" "}; i+=2; break;
  9852. case '@': /* Text Placeholder */
  9853. out[out.length] = {t:'T', v:v}; ++i; break;
  9854. case 'B': case 'b':
  9855. if(fmt.charAt(i+1) === "1" || fmt.charAt(i+1) === "2") {
  9856. if(dt==null) { dt=parse_date_code(v, opts, fmt.charAt(i+1) === "2"); if(dt==null) return ""; }
  9857. out[out.length] = {t:'X', v:fmt.substr(i,2)}; lst = c; i+=2; break;
  9858. }
  9859. /* falls through */
  9860. case 'M': case 'D': case 'Y': case 'H': case 'S': case 'E':
  9861. c = c.toLowerCase();
  9862. /* falls through */
  9863. case 'm': case 'd': case 'y': case 'h': case 's': case 'e': case 'g':
  9864. if(v < 0) return "";
  9865. if(dt==null) { dt=parse_date_code(v, opts); if(dt==null) return ""; }
  9866. o = c; while(++i < fmt.length && fmt.charAt(i).toLowerCase() === c) o+=c;
  9867. if(c === 'm' && lst.toLowerCase() === 'h') c = 'M';
  9868. if(c === 'h') c = hr;
  9869. out[out.length] = {t:c, v:o}; lst = c; break;
  9870. case 'A': case 'a':
  9871. var q={t:c, v:c};
  9872. if(dt==null) dt=parse_date_code(v, opts);
  9873. if(fmt.substr(i, 3).toUpperCase() === "A/P") { if(dt!=null) q.v = dt.H >= 12 ? "P" : "A"; q.t = 'T'; hr='h';i+=3;}
  9874. else if(fmt.substr(i,5).toUpperCase() === "AM/PM") { if(dt!=null) q.v = dt.H >= 12 ? "PM" : "AM"; q.t = 'T'; i+=5; hr='h'; }
  9875. else { q.t = "t"; ++i; }
  9876. if(dt==null && q.t === 'T') return "";
  9877. out[out.length] = q; lst = c; break;
  9878. case '[':
  9879. o = c;
  9880. while(fmt.charAt(i++) !== ']' && i < fmt.length) o += fmt.charAt(i);
  9881. if(o.slice(-1) !== ']') throw 'unterminated "[" block: |' + o + '|';
  9882. if(o.match(abstime)) {
  9883. if(dt==null) { dt=parse_date_code(v, opts); if(dt==null) return ""; }
  9884. out[out.length] = {t:'Z', v:o.toLowerCase()};
  9885. lst = o.charAt(1);
  9886. } else if(o.indexOf("$") > -1) {
  9887. o = (o.match(/\$([^-\[\]]*)/)||[])[1]||"$";
  9888. if(!fmt_is_date(fmt)) out[out.length] = {t:'t',v:o};
  9889. }
  9890. break;
  9891. /* Numbers */
  9892. case '.':
  9893. if(dt != null) {
  9894. o = c; while(++i < fmt.length && (c=fmt.charAt(i)) === "0") o += c;
  9895. out[out.length] = {t:'s', v:o}; break;
  9896. }
  9897. /* falls through */
  9898. case '0': case '#':
  9899. o = c; while((++i < fmt.length && "0#?.,E+-%".indexOf(c=fmt.charAt(i)) > -1) || (c=='\\' && fmt.charAt(i+1) == "-" && i < fmt.length - 2 && "0#".indexOf(fmt.charAt(i+2))>-1)) o += c;
  9900. out[out.length] = {t:'n', v:o}; break;
  9901. case '?':
  9902. o = c; while(fmt.charAt(++i) === c) o+=c;
  9903. out[out.length] = {t:c, v:o}; lst = c; break;
  9904. case '*': ++i; if(fmt.charAt(i) == ' ' || fmt.charAt(i) == '*') ++i; break; // **
  9905. case '(': case ')': out[out.length] = {t:(flen===1?'t':c), v:c}; ++i; break;
  9906. case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9':
  9907. o = c; while(i < fmt.length && "0123456789".indexOf(fmt.charAt(++i)) > -1) o+=fmt.charAt(i);
  9908. out[out.length] = {t:'D', v:o}; break;
  9909. case ' ': out[out.length] = {t:c, v:c}; ++i; break;
  9910. default:
  9911. if(",$-+/():!^&'~{}<>=€acfijklopqrtuvwxzP".indexOf(c) === -1) throw new Error('unrecognized character ' + c + ' in ' + fmt);
  9912. out[out.length] = {t:'t', v:c}; ++i; break;
  9913. }
  9914. }
  9915. var bt = 0, ss0 = 0, ssm;
  9916. for(i=out.length-1, lst='t'; i >= 0; --i) {
  9917. switch(out[i].t) {
  9918. case 'h': case 'H': out[i].t = hr; lst='h'; if(bt < 1) bt = 1; break;
  9919. case 's':
  9920. if((ssm=out[i].v.match(/\.0+$/))) ss0=Math.max(ss0,ssm[0].length-1);
  9921. if(bt < 3) bt = 3;
  9922. /* falls through */
  9923. case 'd': case 'y': case 'M': case 'e': lst=out[i].t; break;
  9924. case 'm': if(lst === 's') { out[i].t = 'M'; if(bt < 2) bt = 2; } break;
  9925. case 'X': /*if(out[i].v === "B2");*/
  9926. break;
  9927. case 'Z':
  9928. if(bt < 1 && out[i].v.match(/[Hh]/)) bt = 1;
  9929. if(bt < 2 && out[i].v.match(/[Mm]/)) bt = 2;
  9930. if(bt < 3 && out[i].v.match(/[Ss]/)) bt = 3;
  9931. }
  9932. }
  9933. switch(bt) {
  9934. case 0: break;
  9935. case 1:
  9936. if(dt.u >= 0.5) { dt.u = 0; ++dt.S; }
  9937. if(dt.S >= 60) { dt.S = 0; ++dt.M; }
  9938. if(dt.M >= 60) { dt.M = 0; ++dt.H; }
  9939. break;
  9940. case 2:
  9941. if(dt.u >= 0.5) { dt.u = 0; ++dt.S; }
  9942. if(dt.S >= 60) { dt.S = 0; ++dt.M; }
  9943. break;
  9944. }
  9945. /* replace fields */
  9946. var nstr = "", jj;
  9947. for(i=0; i < out.length; ++i) {
  9948. switch(out[i].t) {
  9949. case 't': case 'T': case ' ': case 'D': break;
  9950. case 'X': out[i].v = ""; out[i].t = ";"; break;
  9951. case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'e': case 'b': case 'Z':
  9952. out[i].v = write_date(out[i].t.charCodeAt(0), out[i].v, dt, ss0);
  9953. out[i].t = 't'; break;
  9954. case 'n': case '(': case '?':
  9955. jj = i+1;
  9956. while(out[jj] != null && (
  9957. (c=out[jj].t) === "?" || c === "D" ||
  9958. ((c === " " || c === "t") && out[jj+1] != null && (out[jj+1].t === '?' || out[jj+1].t === "t" && out[jj+1].v === '/')) ||
  9959. (out[i].t === '(' && (c === ' ' || c === 'n' || c === ')')) ||
  9960. (c === 't' && (out[jj].v === '/' || out[jj].v === ' ' && out[jj+1] != null && out[jj+1].t == '?'))
  9961. )) {
  9962. out[i].v += out[jj].v;
  9963. out[jj] = {v:"", t:";"}; ++jj;
  9964. }
  9965. nstr += out[i].v;
  9966. i = jj-1; break;
  9967. case 'G': out[i].t = 't'; out[i].v = general_fmt(v,opts); break;
  9968. }
  9969. }
  9970. var vv = "", myv, ostr;
  9971. if(nstr.length > 0) {
  9972. if(nstr.charCodeAt(0) == 40) /* '(' */ {
  9973. myv = (v<0&&nstr.charCodeAt(0) === 45 ? -v : v);
  9974. ostr = write_num('(', nstr, myv);
  9975. } else {
  9976. myv = (v<0 && flen > 1 ? -v : v);
  9977. ostr = write_num('n', nstr, myv);
  9978. if(myv < 0 && out[0] && out[0].t == 't') {
  9979. ostr = ostr.substr(1);
  9980. out[0].v = "-" + out[0].v;
  9981. }
  9982. }
  9983. jj=ostr.length-1;
  9984. var decpt = out.length;
  9985. for(i=0; i < out.length; ++i) if(out[i] != null && out[i].t != 't' && out[i].v.indexOf(".") > -1) { decpt = i; break; }
  9986. var lasti=out.length;
  9987. if(decpt === out.length && ostr.indexOf("E") === -1) {
  9988. for(i=out.length-1; i>= 0;--i) {
  9989. if(out[i] == null || 'n?('.indexOf(out[i].t) === -1) continue;
  9990. if(jj>=out[i].v.length-1) { jj -= out[i].v.length; out[i].v = ostr.substr(jj+1, out[i].v.length); }
  9991. else if(jj < 0) out[i].v = "";
  9992. else { out[i].v = ostr.substr(0, jj+1); jj = -1; }
  9993. out[i].t = 't';
  9994. lasti = i;
  9995. }
  9996. if(jj>=0 && lasti<out.length) out[lasti].v = ostr.substr(0,jj+1) + out[lasti].v;
  9997. }
  9998. else if(decpt !== out.length && ostr.indexOf("E") === -1) {
  9999. jj = ostr.indexOf(".")-1;
  10000. for(i=decpt; i>= 0; --i) {
  10001. if(out[i] == null || 'n?('.indexOf(out[i].t) === -1) continue;
  10002. j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")-1:out[i].v.length-1;
  10003. vv = out[i].v.substr(j+1);
  10004. for(; j>=0; --j) {
  10005. if(jj>=0 && (out[i].v.charAt(j) === "0" || out[i].v.charAt(j) === "#")) vv = ostr.charAt(jj--) + vv;
  10006. }
  10007. out[i].v = vv;
  10008. out[i].t = 't';
  10009. lasti = i;
  10010. }
  10011. if(jj>=0 && lasti<out.length) out[lasti].v = ostr.substr(0,jj+1) + out[lasti].v;
  10012. jj = ostr.indexOf(".")+1;
  10013. for(i=decpt; i<out.length; ++i) {
  10014. if(out[i] == null || ('n?('.indexOf(out[i].t) === -1 && i !== decpt)) continue;
  10015. j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")+1:0;
  10016. vv = out[i].v.substr(0,j);
  10017. for(; j<out[i].v.length; ++j) {
  10018. if(jj<ostr.length) vv += ostr.charAt(jj++);
  10019. }
  10020. out[i].v = vv;
  10021. out[i].t = 't';
  10022. lasti = i;
  10023. }
  10024. }
  10025. }
  10026. for(i=0; i<out.length; ++i) if(out[i] != null && 'n(?'.indexOf(out[i].t)>-1) {
  10027. myv = (flen >1 && v < 0 && i>0 && out[i-1].v === "-" ? -v:v);
  10028. out[i].v = write_num(out[i].t, out[i].v, myv);
  10029. out[i].t = 't';
  10030. }
  10031. var retval = "";
  10032. for(i=0; i !== out.length; ++i) if(out[i] != null) retval += out[i].v;
  10033. return retval;
  10034. }
  10035. SSF._eval = eval_fmt;
  10036. var cfregex = /\[[=<>]/;
  10037. var cfregex2 = /\[(=|>[=]?|<[>=]?)(-?\d+(?:\.\d*)?)\]/;
  10038. function chkcond(v, rr) {
  10039. if(rr == null) return false;
  10040. var thresh = parseFloat(rr[2]);
  10041. switch(rr[1]) {
  10042. case "=": if(v == thresh) return true; break;
  10043. case ">": if(v > thresh) return true; break;
  10044. case "<": if(v < thresh) return true; break;
  10045. case "<>": if(v != thresh) return true; break;
  10046. case ">=": if(v >= thresh) return true; break;
  10047. case "<=": if(v <= thresh) return true; break;
  10048. }
  10049. return false;
  10050. }
  10051. function choose_fmt(f, v) {
  10052. var fmt = split_fmt(f);
  10053. var l = fmt.length, lat = fmt[l-1].indexOf("@");
  10054. if(l<4 && lat>-1) --l;
  10055. if(fmt.length > 4) throw new Error("cannot find right format for |" + fmt.join("|") + "|");
  10056. if(typeof v !== "number") return [4, fmt.length === 4 || lat>-1?fmt[fmt.length-1]:"@"];
  10057. switch(fmt.length) {
  10058. case 1: fmt = lat>-1 ? ["General", "General", "General", fmt[0]] : [fmt[0], fmt[0], fmt[0], "@"]; break;
  10059. case 2: fmt = lat>-1 ? [fmt[0], fmt[0], fmt[0], fmt[1]] : [fmt[0], fmt[1], fmt[0], "@"]; break;
  10060. case 3: fmt = lat>-1 ? [fmt[0], fmt[1], fmt[0], fmt[2]] : [fmt[0], fmt[1], fmt[2], "@"]; break;
  10061. case 4: break;
  10062. }
  10063. var ff = v > 0 ? fmt[0] : v < 0 ? fmt[1] : fmt[2];
  10064. if(fmt[0].indexOf("[") === -1 && fmt[1].indexOf("[") === -1) return [l, ff];
  10065. if(fmt[0].match(cfregex) != null || fmt[1].match(cfregex) != null) {
  10066. var m1 = fmt[0].match(cfregex2);
  10067. var m2 = fmt[1].match(cfregex2);
  10068. return chkcond(v, m1) ? [l, fmt[0]] : chkcond(v, m2) ? [l, fmt[1]] : [l, fmt[m1 != null && m2 != null ? 2 : 1]];
  10069. }
  10070. return [l, ff];
  10071. }
  10072. function format(fmt,v,o) {
  10073. if(o == null) o = {};
  10074. var sfmt = "";
  10075. switch(typeof fmt) {
  10076. case "string":
  10077. if(fmt == "m/d/yy" && o.dateNF) sfmt = o.dateNF;
  10078. else sfmt = fmt;
  10079. break;
  10080. case "number":
  10081. if(fmt == 14 && o.dateNF) sfmt = o.dateNF;
  10082. else sfmt = (o.table != null ? (o.table) : table_fmt)[fmt];
  10083. break;
  10084. }
  10085. if(isgeneral(sfmt,0)) return general_fmt(v, o);
  10086. if(v instanceof Date) v = datenum_local(v, o.date1904);
  10087. var f = choose_fmt(sfmt, v);
  10088. if(isgeneral(f[1])) return general_fmt(v, o);
  10089. if(v === true) v = "TRUE"; else if(v === false) v = "FALSE";
  10090. else if(v === "" || v == null) return "";
  10091. return eval_fmt(f[1], v, o, f[0]);
  10092. }
  10093. function load_entry(fmt, idx) {
  10094. if(typeof idx != 'number') {
  10095. idx = +idx || -1;
  10096. for(var i = 0; i < 0x0188; ++i) {
  10097. if(table_fmt[i] == undefined) { if(idx < 0) idx = i; continue; }
  10098. if(table_fmt[i] == fmt) { idx = i; break; }
  10099. }
  10100. if(idx < 0) idx = 0x187;
  10101. }
  10102. table_fmt[idx] = fmt;
  10103. return idx;
  10104. }
  10105. SSF.load = load_entry;
  10106. SSF._table = table_fmt;
  10107. SSF.get_table = function get_table() { return table_fmt; };
  10108. SSF.load_table = function load_table(tbl) {
  10109. for(var i=0; i!=0x0188; ++i)
  10110. if(tbl[i] !== undefined) load_entry(tbl[i], i);
  10111. };
  10112. SSF.init_table = init_table;
  10113. SSF.format = format;
  10114. };
  10115. make_ssf(SSF);
  10116. /* map from xlml named formats to SSF TODO: localize */
  10117. var XLMLFormatMap/*{[string]:string}*/ = ({
  10118. "General Number": "General",
  10119. "General Date": SSF._table[22],
  10120. "Long Date": "dddd, mmmm dd, yyyy",
  10121. "Medium Date": SSF._table[15],
  10122. "Short Date": SSF._table[14],
  10123. "Long Time": SSF._table[19],
  10124. "Medium Time": SSF._table[18],
  10125. "Short Time": SSF._table[20],
  10126. "Currency": '"$"#,##0.00_);[Red]\\("$"#,##0.00\\)',
  10127. "Fixed": SSF._table[2],
  10128. "Standard": SSF._table[4],
  10129. "Percent": SSF._table[10],
  10130. "Scientific": SSF._table[11],
  10131. "Yes/No": '"Yes";"Yes";"No";@',
  10132. "True/False": '"True";"True";"False";@',
  10133. "On/Off": '"Yes";"Yes";"No";@'
  10134. });
  10135. var SSFImplicit/*{[number]:string}*/ = ({
  10136. "5": '"$"#,##0_);\\("$"#,##0\\)',
  10137. "6": '"$"#,##0_);[Red]\\("$"#,##0\\)',
  10138. "7": '"$"#,##0.00_);\\("$"#,##0.00\\)',
  10139. "8": '"$"#,##0.00_);[Red]\\("$"#,##0.00\\)',
  10140. "23": 'General', "24": 'General', "25": 'General', "26": 'General',
  10141. "27": 'm/d/yy', "28": 'm/d/yy', "29": 'm/d/yy', "30": 'm/d/yy', "31": 'm/d/yy',
  10142. "32": 'h:mm:ss', "33": 'h:mm:ss', "34": 'h:mm:ss', "35": 'h:mm:ss',
  10143. "36": 'm/d/yy',
  10144. "41": '_(* #,##0_);_(* \(#,##0\);_(* "-"_);_(@_)',
  10145. "42": '_("$"* #,##0_);_("$"* \(#,##0\);_("$"* "-"_);_(@_)',
  10146. "43": '_(* #,##0.00_);_(* \(#,##0.00\);_(* "-"??_);_(@_)',
  10147. "44": '_("$"* #,##0.00_);_("$"* \(#,##0.00\);_("$"* "-"??_);_(@_)',
  10148. "50": 'm/d/yy', "51": 'm/d/yy', "52": 'm/d/yy', "53": 'm/d/yy', "54": 'm/d/yy',
  10149. "55": 'm/d/yy', "56": 'm/d/yy', "57": 'm/d/yy', "58": 'm/d/yy',
  10150. "59": '0',
  10151. "60": '0.00',
  10152. "61": '#,##0',
  10153. "62": '#,##0.00',
  10154. "63": '"$"#,##0_);\\("$"#,##0\\)',
  10155. "64": '"$"#,##0_);[Red]\\("$"#,##0\\)',
  10156. "65": '"$"#,##0.00_);\\("$"#,##0.00\\)',
  10157. "66": '"$"#,##0.00_);[Red]\\("$"#,##0.00\\)',
  10158. "67": '0%',
  10159. "68": '0.00%',
  10160. "69": '# ?/?',
  10161. "70": '# ??/??',
  10162. "71": 'm/d/yy',
  10163. "72": 'm/d/yy',
  10164. "73": 'd-mmm-yy',
  10165. "74": 'd-mmm',
  10166. "75": 'mmm-yy',
  10167. "76": 'h:mm',
  10168. "77": 'h:mm:ss',
  10169. "78": 'm/d/yy h:mm',
  10170. "79": 'mm:ss',
  10171. "80": '[h]:mm:ss',
  10172. "81": 'mmss.0'
  10173. });
  10174. /* dateNF parse TODO: move to SSF */
  10175. var dateNFregex = /[dD]+|[mM]+|[yYeE]+|[Hh]+|[Ss]+/g;
  10176. function dateNF_regex(dateNF) {
  10177. var fmt = typeof dateNF == "number" ? SSF._table[dateNF] : dateNF;
  10178. fmt = fmt.replace(dateNFregex, "(\\d+)");
  10179. return new RegExp("^" + fmt + "$");
  10180. }
  10181. function dateNF_fix(str, dateNF, match) {
  10182. var Y = -1, m = -1, d = -1, H = -1, M = -1, S = -1;
  10183. (dateNF.match(dateNFregex)||[]).forEach(function(n, i) {
  10184. var v = parseInt(match[i+1], 10);
  10185. switch(n.toLowerCase().charAt(0)) {
  10186. case 'y': Y = v; break; case 'd': d = v; break;
  10187. case 'h': H = v; break; case 's': S = v; break;
  10188. case 'm': if(H >= 0) M = v; else m = v; break;
  10189. }
  10190. });
  10191. if(S >= 0 && M == -1 && m >= 0) { M = m; m = -1; }
  10192. var datestr = (("" + (Y>=0?Y: new Date().getFullYear())).slice(-4) + "-" + ("00" + (m>=1?m:1)).slice(-2) + "-" + ("00" + (d>=1?d:1)).slice(-2));
  10193. if(datestr.length == 7) datestr = "0" + datestr;
  10194. if(datestr.length == 8) datestr = "20" + datestr;
  10195. var timestr = (("00" + (H>=0?H:0)).slice(-2) + ":" + ("00" + (M>=0?M:0)).slice(-2) + ":" + ("00" + (S>=0?S:0)).slice(-2));
  10196. if(H == -1 && M == -1 && S == -1) return datestr;
  10197. if(Y == -1 && m == -1 && d == -1) return timestr;
  10198. return datestr + "T" + timestr;
  10199. }
  10200. var DO_NOT_EXPORT_CFB = true;
  10201. /* cfb.js (C) 2013-present SheetJS -- http://sheetjs.com */
  10202. /* vim: set ts=2: */
  10203. /*jshint eqnull:true */
  10204. /*exported CFB */
  10205. /*global module, require:false, process:false, Buffer:false, Uint8Array:false, Uint16Array:false */
  10206. /* crc32.js (C) 2014-present SheetJS -- http://sheetjs.com */
  10207. /* vim: set ts=2: */
  10208. /*exported CRC32 */
  10209. var CRC32;
  10210. (function (factory) {
  10211. /*jshint ignore:start */
  10212. /*eslint-disable */
  10213. factory(CRC32 = {});
  10214. /*eslint-enable */
  10215. /*jshint ignore:end */
  10216. }(function(CRC32) {
  10217. CRC32.version = '1.2.0';
  10218. /* see perf/crc32table.js */
  10219. /*global Int32Array */
  10220. function signed_crc_table() {
  10221. var c = 0, table = new Array(256);
  10222. for(var n =0; n != 256; ++n){
  10223. c = n;
  10224. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10225. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10226. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10227. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10228. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10229. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10230. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10231. c = ((c&1) ? (-306674912 ^ (c >>> 1)) : (c >>> 1));
  10232. table[n] = c;
  10233. }
  10234. return typeof Int32Array !== 'undefined' ? new Int32Array(table) : table;
  10235. }
  10236. var T = signed_crc_table();
  10237. function crc32_bstr(bstr, seed) {
  10238. var C = seed ^ -1, L = bstr.length - 1;
  10239. for(var i = 0; i < L;) {
  10240. C = (C>>>8) ^ T[(C^bstr.charCodeAt(i++))&0xFF];
  10241. C = (C>>>8) ^ T[(C^bstr.charCodeAt(i++))&0xFF];
  10242. }
  10243. if(i === L) C = (C>>>8) ^ T[(C ^ bstr.charCodeAt(i))&0xFF];
  10244. return C ^ -1;
  10245. }
  10246. function crc32_buf(buf, seed) {
  10247. if(buf.length > 10000) return crc32_buf_8(buf, seed);
  10248. var C = seed ^ -1, L = buf.length - 3;
  10249. for(var i = 0; i < L;) {
  10250. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10251. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10252. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10253. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10254. }
  10255. while(i < L+3) C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10256. return C ^ -1;
  10257. }
  10258. function crc32_buf_8(buf, seed) {
  10259. var C = seed ^ -1, L = buf.length - 7;
  10260. for(var i = 0; i < L;) {
  10261. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10262. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10263. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10264. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10265. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10266. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10267. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10268. C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10269. }
  10270. while(i < L+7) C = (C>>>8) ^ T[(C^buf[i++])&0xFF];
  10271. return C ^ -1;
  10272. }
  10273. function crc32_str(str, seed) {
  10274. var C = seed ^ -1;
  10275. for(var i = 0, L=str.length, c, d; i < L;) {
  10276. c = str.charCodeAt(i++);
  10277. if(c < 0x80) {
  10278. C = (C>>>8) ^ T[(C ^ c)&0xFF];
  10279. } else if(c < 0x800) {
  10280. C = (C>>>8) ^ T[(C ^ (192|((c>>6)&31)))&0xFF];
  10281. C = (C>>>8) ^ T[(C ^ (128|(c&63)))&0xFF];
  10282. } else if(c >= 0xD800 && c < 0xE000) {
  10283. c = (c&1023)+64; d = str.charCodeAt(i++)&1023;
  10284. C = (C>>>8) ^ T[(C ^ (240|((c>>8)&7)))&0xFF];
  10285. C = (C>>>8) ^ T[(C ^ (128|((c>>2)&63)))&0xFF];
  10286. C = (C>>>8) ^ T[(C ^ (128|((d>>6)&15)|((c&3)<<4)))&0xFF];
  10287. C = (C>>>8) ^ T[(C ^ (128|(d&63)))&0xFF];
  10288. } else {
  10289. C = (C>>>8) ^ T[(C ^ (224|((c>>12)&15)))&0xFF];
  10290. C = (C>>>8) ^ T[(C ^ (128|((c>>6)&63)))&0xFF];
  10291. C = (C>>>8) ^ T[(C ^ (128|(c&63)))&0xFF];
  10292. }
  10293. }
  10294. return C ^ -1;
  10295. }
  10296. CRC32.table = T;
  10297. CRC32.bstr = crc32_bstr;
  10298. CRC32.buf = crc32_buf;
  10299. CRC32.str = crc32_str;
  10300. }));
  10301. /* [MS-CFB] v20171201 */
  10302. var CFB = (function _CFB(){
  10303. var exports = {};
  10304. exports.version = '1.1.0';
  10305. /* [MS-CFB] 2.6.4 */
  10306. function namecmp(l, r) {
  10307. var L = l.split("/"), R = r.split("/");
  10308. for(var i = 0, c = 0, Z = Math.min(L.length, R.length); i < Z; ++i) {
  10309. if((c = L[i].length - R[i].length)) return c;
  10310. if(L[i] != R[i]) return L[i] < R[i] ? -1 : 1;
  10311. }
  10312. return L.length - R.length;
  10313. }
  10314. function dirname(p) {
  10315. if(p.charAt(p.length - 1) == "/") return (p.slice(0,-1).indexOf("/") === -1) ? p : dirname(p.slice(0, -1));
  10316. var c = p.lastIndexOf("/");
  10317. return (c === -1) ? p : p.slice(0, c+1);
  10318. }
  10319. function filename(p) {
  10320. if(p.charAt(p.length - 1) == "/") return filename(p.slice(0, -1));
  10321. var c = p.lastIndexOf("/");
  10322. return (c === -1) ? p : p.slice(c+1);
  10323. }
  10324. /* -------------------------------------------------------------------------- */
  10325. /* DOS Date format:
  10326. high|YYYYYYYm.mmmddddd.HHHHHMMM.MMMSSSSS|low
  10327. add 1980 to stored year
  10328. stored second should be doubled
  10329. */
  10330. /* write JS date to buf as a DOS date */
  10331. function write_dos_date(buf, date) {
  10332. if(typeof date === "string") date = new Date(date);
  10333. var hms = date.getHours();
  10334. hms = hms << 6 | date.getMinutes();
  10335. hms = hms << 5 | (date.getSeconds()>>>1);
  10336. buf.write_shift(2, hms);
  10337. var ymd = (date.getFullYear() - 1980);
  10338. ymd = ymd << 4 | (date.getMonth()+1);
  10339. ymd = ymd << 5 | date.getDate();
  10340. buf.write_shift(2, ymd);
  10341. }
  10342. /* read four bytes from buf and interpret as a DOS date */
  10343. function parse_dos_date(buf) {
  10344. var hms = buf.read_shift(2) & 0xFFFF;
  10345. var ymd = buf.read_shift(2) & 0xFFFF;
  10346. var val = new Date();
  10347. var d = ymd & 0x1F; ymd >>>= 5;
  10348. var m = ymd & 0x0F; ymd >>>= 4;
  10349. val.setMilliseconds(0);
  10350. val.setFullYear(ymd + 1980);
  10351. val.setMonth(m-1);
  10352. val.setDate(d);
  10353. var S = hms & 0x1F; hms >>>= 5;
  10354. var M = hms & 0x3F; hms >>>= 6;
  10355. val.setHours(hms);
  10356. val.setMinutes(M);
  10357. val.setSeconds(S<<1);
  10358. return val;
  10359. }
  10360. function parse_extra_field(blob) {
  10361. prep_blob(blob, 0);
  10362. var o = {};
  10363. var flags = 0;
  10364. while(blob.l <= blob.length - 4) {
  10365. var type = blob.read_shift(2);
  10366. var sz = blob.read_shift(2), tgt = blob.l + sz;
  10367. var p = {};
  10368. switch(type) {
  10369. /* UNIX-style Timestamps */
  10370. case 0x5455: {
  10371. flags = blob.read_shift(1);
  10372. if(flags & 1) p.mtime = blob.read_shift(4);
  10373. /* for some reason, CD flag corresponds to LFH */
  10374. if(sz > 5) {
  10375. if(flags & 2) p.atime = blob.read_shift(4);
  10376. if(flags & 4) p.ctime = blob.read_shift(4);
  10377. }
  10378. if(p.mtime) p.mt = new Date(p.mtime*1000);
  10379. }
  10380. break;
  10381. }
  10382. blob.l = tgt;
  10383. o[type] = p;
  10384. }
  10385. return o;
  10386. }
  10387. var fs;
  10388. function get_fs() { return fs || (fs = require('fs')); }
  10389. function parse(file, options) {
  10390. if(file[0] == 0x50 && file[1] == 0x4b) return parse_zip(file, options);
  10391. if(file.length < 512) throw new Error("CFB file size " + file.length + " < 512");
  10392. var mver = 3;
  10393. var ssz = 512;
  10394. var nmfs = 0; // number of mini FAT sectors
  10395. var difat_sec_cnt = 0;
  10396. var dir_start = 0;
  10397. var minifat_start = 0;
  10398. var difat_start = 0;
  10399. var fat_addrs = []; // locations of FAT sectors
  10400. /* [MS-CFB] 2.2 Compound File Header */
  10401. var blob = file.slice(0,512);
  10402. prep_blob(blob, 0);
  10403. /* major version */
  10404. var mv = check_get_mver(blob);
  10405. mver = mv[0];
  10406. switch(mver) {
  10407. case 3: ssz = 512; break; case 4: ssz = 4096; break;
  10408. case 0: if(mv[1] == 0) return parse_zip(file, options);
  10409. /* falls through */
  10410. default: throw new Error("Major Version: Expected 3 or 4 saw " + mver);
  10411. }
  10412. /* reprocess header */
  10413. if(ssz !== 512) { blob = file.slice(0,ssz); prep_blob(blob, 28 /* blob.l */); }
  10414. /* Save header for final object */
  10415. var header = file.slice(0,ssz);
  10416. check_shifts(blob, mver);
  10417. // Number of Directory Sectors
  10418. var dir_cnt = blob.read_shift(4, 'i');
  10419. if(mver === 3 && dir_cnt !== 0) throw new Error('# Directory Sectors: Expected 0 saw ' + dir_cnt);
  10420. // Number of FAT Sectors
  10421. blob.l += 4;
  10422. // First Directory Sector Location
  10423. dir_start = blob.read_shift(4, 'i');
  10424. // Transaction Signature
  10425. blob.l += 4;
  10426. // Mini Stream Cutoff Size
  10427. blob.chk('00100000', 'Mini Stream Cutoff Size: ');
  10428. // First Mini FAT Sector Location
  10429. minifat_start = blob.read_shift(4, 'i');
  10430. // Number of Mini FAT Sectors
  10431. nmfs = blob.read_shift(4, 'i');
  10432. // First DIFAT sector location
  10433. difat_start = blob.read_shift(4, 'i');
  10434. // Number of DIFAT Sectors
  10435. difat_sec_cnt = blob.read_shift(4, 'i');
  10436. // Grab FAT Sector Locations
  10437. for(var q = -1, j = 0; j < 109; ++j) { /* 109 = (512 - blob.l)>>>2; */
  10438. q = blob.read_shift(4, 'i');
  10439. if(q<0) break;
  10440. fat_addrs[j] = q;
  10441. }
  10442. /** Break the file up into sectors */
  10443. var sectors = sectorify(file, ssz);
  10444. sleuth_fat(difat_start, difat_sec_cnt, sectors, ssz, fat_addrs);
  10445. /** Chains */
  10446. var sector_list = make_sector_list(sectors, dir_start, fat_addrs, ssz);
  10447. sector_list[dir_start].name = "!Directory";
  10448. if(nmfs > 0 && minifat_start !== ENDOFCHAIN) sector_list[minifat_start].name = "!MiniFAT";
  10449. sector_list[fat_addrs[0]].name = "!FAT";
  10450. sector_list.fat_addrs = fat_addrs;
  10451. sector_list.ssz = ssz;
  10452. /* [MS-CFB] 2.6.1 Compound File Directory Entry */
  10453. var files = {}, Paths = [], FileIndex = [], FullPaths = [];
  10454. read_directory(dir_start, sector_list, sectors, Paths, nmfs, files, FileIndex, minifat_start);
  10455. build_full_paths(FileIndex, FullPaths, Paths);
  10456. Paths.shift();
  10457. var o = {
  10458. FileIndex: FileIndex,
  10459. FullPaths: FullPaths
  10460. };
  10461. // $FlowIgnore
  10462. if(options && options.raw) o.raw = {header: header, sectors: sectors};
  10463. return o;
  10464. } // parse
  10465. /* [MS-CFB] 2.2 Compound File Header -- read up to major version */
  10466. function check_get_mver(blob) {
  10467. if(blob[blob.l] == 0x50 && blob[blob.l + 1] == 0x4b) return [0, 0];
  10468. // header signature 8
  10469. blob.chk(HEADER_SIGNATURE, 'Header Signature: ');
  10470. // clsid 16
  10471. blob.chk(HEADER_CLSID, 'CLSID: ');
  10472. // minor version 2
  10473. var mver = blob.read_shift(2, 'u');
  10474. return [blob.read_shift(2,'u'), mver];
  10475. }
  10476. function check_shifts(blob, mver) {
  10477. var shift = 0x09;
  10478. // Byte Orders
  10479. //blob.chk('feff', 'Byte Orders: '); // note: some writers put 0xffff
  10480. blob.l += 2;
  10481. // Sector Shift
  10482. switch((shift = blob.read_shift(2))) {
  10483. case 0x09: if(mver != 3) throw new Error('Sector Shift: Expected 9 saw ' + shift); break;
  10484. case 0x0c: if(mver != 4) throw new Error('Sector Shift: Expected 12 saw ' + shift); break;
  10485. default: throw new Error('Sector Shift: Expected 9 or 12 saw ' + shift);
  10486. }
  10487. // Mini Sector Shift
  10488. blob.chk('0600', 'Mini Sector Shift: ');
  10489. // Reserved
  10490. blob.chk('000000000000', 'Reserved: ');
  10491. }
  10492. /** Break the file up into sectors */
  10493. function sectorify(file, ssz) {
  10494. var nsectors = Math.ceil(file.length/ssz)-1;
  10495. var sectors = [];
  10496. for(var i=1; i < nsectors; ++i) sectors[i-1] = file.slice(i*ssz,(i+1)*ssz);
  10497. sectors[nsectors-1] = file.slice(nsectors*ssz);
  10498. return sectors;
  10499. }
  10500. /* [MS-CFB] 2.6.4 Red-Black Tree */
  10501. function build_full_paths(FI, FP, Paths) {
  10502. var i = 0, L = 0, R = 0, C = 0, j = 0, pl = Paths.length;
  10503. var dad = [], q = [];
  10504. for(; i < pl; ++i) { dad[i]=q[i]=i; FP[i]=Paths[i]; }
  10505. for(; j < q.length; ++j) {
  10506. i = q[j];
  10507. L = FI[i].L; R = FI[i].R; C = FI[i].C;
  10508. if(dad[i] === i) {
  10509. if(L !== -1 /*NOSTREAM*/ && dad[L] !== L) dad[i] = dad[L];
  10510. if(R !== -1 && dad[R] !== R) dad[i] = dad[R];
  10511. }
  10512. if(C !== -1 /*NOSTREAM*/) dad[C] = i;
  10513. if(L !== -1) { dad[L] = dad[i]; if(q.lastIndexOf(L) < j) q.push(L); }
  10514. if(R !== -1) { dad[R] = dad[i]; if(q.lastIndexOf(R) < j) q.push(R); }
  10515. }
  10516. for(i=1; i < pl; ++i) if(dad[i] === i) {
  10517. if(R !== -1 /*NOSTREAM*/ && dad[R] !== R) dad[i] = dad[R];
  10518. else if(L !== -1 && dad[L] !== L) dad[i] = dad[L];
  10519. }
  10520. for(i=1; i < pl; ++i) {
  10521. if(FI[i].type === 0 /* unknown */) continue;
  10522. j = dad[i];
  10523. if(j === 0) FP[i] = FP[0] + "/" + FP[i];
  10524. else while(j !== 0 && j !== dad[j]) {
  10525. FP[i] = FP[j] + "/" + FP[i];
  10526. j = dad[j];
  10527. }
  10528. dad[i] = 0;
  10529. }
  10530. FP[0] += "/";
  10531. for(i=1; i < pl; ++i) {
  10532. if(FI[i].type !== 2 /* stream */) FP[i] += "/";
  10533. }
  10534. }
  10535. function get_mfat_entry(entry, payload, mini) {
  10536. var start = entry.start, size = entry.size;
  10537. //return (payload.slice(start*MSSZ, start*MSSZ + size));
  10538. var o = [];
  10539. var idx = start;
  10540. while(mini && size > 0 && idx >= 0) {
  10541. o.push(payload.slice(idx * MSSZ, idx * MSSZ + MSSZ));
  10542. size -= MSSZ;
  10543. idx = __readInt32LE(mini, idx * 4);
  10544. }
  10545. if(o.length === 0) return (new_buf(0));
  10546. return (bconcat(o).slice(0, entry.size));
  10547. }
  10548. /** Chase down the rest of the DIFAT chain to build a comprehensive list
  10549. DIFAT chains by storing the next sector number as the last 32 bits */
  10550. function sleuth_fat(idx, cnt, sectors, ssz, fat_addrs) {
  10551. var q = ENDOFCHAIN;
  10552. if(idx === ENDOFCHAIN) {
  10553. if(cnt !== 0) throw new Error("DIFAT chain shorter than expected");
  10554. } else if(idx !== -1 /*FREESECT*/) {
  10555. var sector = sectors[idx], m = (ssz>>>2)-1;
  10556. if(!sector) return;
  10557. for(var i = 0; i < m; ++i) {
  10558. if((q = __readInt32LE(sector,i*4)) === ENDOFCHAIN) break;
  10559. fat_addrs.push(q);
  10560. }
  10561. sleuth_fat(__readInt32LE(sector,ssz-4),cnt - 1, sectors, ssz, fat_addrs);
  10562. }
  10563. }
  10564. /** Follow the linked list of sectors for a given starting point */
  10565. function get_sector_list(sectors, start, fat_addrs, ssz, chkd) {
  10566. var buf = [], buf_chain = [];
  10567. if(!chkd) chkd = [];
  10568. var modulus = ssz - 1, j = 0, jj = 0;
  10569. for(j=start; j>=0;) {
  10570. chkd[j] = true;
  10571. buf[buf.length] = j;
  10572. buf_chain.push(sectors[j]);
  10573. var addr = fat_addrs[Math.floor(j*4/ssz)];
  10574. jj = ((j*4) & modulus);
  10575. if(ssz < 4 + jj) throw new Error("FAT boundary crossed: " + j + " 4 "+ssz);
  10576. if(!sectors[addr]) break;
  10577. j = __readInt32LE(sectors[addr], jj);
  10578. }
  10579. return {nodes: buf, data:__toBuffer([buf_chain])};
  10580. }
  10581. /** Chase down the sector linked lists */
  10582. function make_sector_list(sectors, dir_start, fat_addrs, ssz) {
  10583. var sl = sectors.length, sector_list = ([]);
  10584. var chkd = [], buf = [], buf_chain = [];
  10585. var modulus = ssz - 1, i=0, j=0, k=0, jj=0;
  10586. for(i=0; i < sl; ++i) {
  10587. buf = ([]);
  10588. k = (i + dir_start); if(k >= sl) k-=sl;
  10589. if(chkd[k]) continue;
  10590. buf_chain = [];
  10591. for(j=k; j>=0;) {
  10592. chkd[j] = true;
  10593. buf[buf.length] = j;
  10594. buf_chain.push(sectors[j]);
  10595. var addr = fat_addrs[Math.floor(j*4/ssz)];
  10596. jj = ((j*4) & modulus);
  10597. if(ssz < 4 + jj) throw new Error("FAT boundary crossed: " + j + " 4 "+ssz);
  10598. if(!sectors[addr]) break;
  10599. j = __readInt32LE(sectors[addr], jj);
  10600. }
  10601. sector_list[k] = ({nodes: buf, data:__toBuffer([buf_chain])});
  10602. }
  10603. return sector_list;
  10604. }
  10605. /* [MS-CFB] 2.6.1 Compound File Directory Entry */
  10606. function read_directory(dir_start, sector_list, sectors, Paths, nmfs, files, FileIndex, mini) {
  10607. var minifat_store = 0, pl = (Paths.length?2:0);
  10608. var sector = sector_list[dir_start].data;
  10609. var i = 0, namelen = 0, name;
  10610. for(; i < sector.length; i+= 128) {
  10611. var blob = sector.slice(i, i+128);
  10612. prep_blob(blob, 64);
  10613. namelen = blob.read_shift(2);
  10614. name = __utf16le(blob,0,namelen-pl);
  10615. Paths.push(name);
  10616. var o = ({
  10617. name: name,
  10618. type: blob.read_shift(1),
  10619. color: blob.read_shift(1),
  10620. L: blob.read_shift(4, 'i'),
  10621. R: blob.read_shift(4, 'i'),
  10622. C: blob.read_shift(4, 'i'),
  10623. clsid: blob.read_shift(16),
  10624. state: blob.read_shift(4, 'i'),
  10625. start: 0,
  10626. size: 0
  10627. });
  10628. var ctime = blob.read_shift(2) + blob.read_shift(2) + blob.read_shift(2) + blob.read_shift(2);
  10629. if(ctime !== 0) o.ct = read_date(blob, blob.l-8);
  10630. var mtime = blob.read_shift(2) + blob.read_shift(2) + blob.read_shift(2) + blob.read_shift(2);
  10631. if(mtime !== 0) o.mt = read_date(blob, blob.l-8);
  10632. o.start = blob.read_shift(4, 'i');
  10633. o.size = blob.read_shift(4, 'i');
  10634. if(o.size < 0 && o.start < 0) { o.size = o.type = 0; o.start = ENDOFCHAIN; o.name = ""; }
  10635. if(o.type === 5) { /* root */
  10636. minifat_store = o.start;
  10637. if(nmfs > 0 && minifat_store !== ENDOFCHAIN) sector_list[minifat_store].name = "!StreamData";
  10638. /*minifat_size = o.size;*/
  10639. } else if(o.size >= 4096 /* MSCSZ */) {
  10640. o.storage = 'fat';
  10641. if(sector_list[o.start] === undefined) sector_list[o.start] = get_sector_list(sectors, o.start, sector_list.fat_addrs, sector_list.ssz);
  10642. sector_list[o.start].name = o.name;
  10643. o.content = (sector_list[o.start].data.slice(0,o.size));
  10644. } else {
  10645. o.storage = 'minifat';
  10646. if(o.size < 0) o.size = 0;
  10647. else if(minifat_store !== ENDOFCHAIN && o.start !== ENDOFCHAIN && sector_list[minifat_store]) {
  10648. o.content = get_mfat_entry(o, sector_list[minifat_store].data, (sector_list[mini]||{}).data);
  10649. }
  10650. }
  10651. if(o.content) prep_blob(o.content, 0);
  10652. files[name] = o;
  10653. FileIndex.push(o);
  10654. }
  10655. }
  10656. function read_date(blob, offset) {
  10657. return new Date(( ( (__readUInt32LE(blob,offset+4)/1e7)*Math.pow(2,32)+__readUInt32LE(blob,offset)/1e7 ) - 11644473600)*1000);
  10658. }
  10659. function read_file(filename, options) {
  10660. get_fs();
  10661. return parse(fs.readFileSync(filename), options);
  10662. }
  10663. function read(blob, options) {
  10664. switch(options && options.type || "base64") {
  10665. case "file": return read_file(blob, options);
  10666. case "base64": return parse(s2a(Base64.decode(blob)), options);
  10667. case "binary": return parse(s2a(blob), options);
  10668. }
  10669. return parse(blob, options);
  10670. }
  10671. function init_cfb(cfb, opts) {
  10672. var o = opts || {}, root = o.root || "Root Entry";
  10673. if(!cfb.FullPaths) cfb.FullPaths = [];
  10674. if(!cfb.FileIndex) cfb.FileIndex = [];
  10675. if(cfb.FullPaths.length !== cfb.FileIndex.length) throw new Error("inconsistent CFB structure");
  10676. if(cfb.FullPaths.length === 0) {
  10677. cfb.FullPaths[0] = root + "/";
  10678. cfb.FileIndex[0] = ({ name: root, type: 5 });
  10679. }
  10680. if(o.CLSID) cfb.FileIndex[0].clsid = o.CLSID;
  10681. seed_cfb(cfb);
  10682. }
  10683. function seed_cfb(cfb) {
  10684. var nm = "\u0001Sh33tJ5";
  10685. if(CFB.find(cfb, "/" + nm)) return;
  10686. var p = new_buf(4); p[0] = 55; p[1] = p[3] = 50; p[2] = 54;
  10687. cfb.FileIndex.push(({ name: nm, type: 2, content:p, size:4, L:69, R:69, C:69 }));
  10688. cfb.FullPaths.push(cfb.FullPaths[0] + nm);
  10689. rebuild_cfb(cfb);
  10690. }
  10691. function rebuild_cfb(cfb, f) {
  10692. init_cfb(cfb);
  10693. var gc = false, s = false;
  10694. for(var i = cfb.FullPaths.length - 1; i >= 0; --i) {
  10695. var _file = cfb.FileIndex[i];
  10696. switch(_file.type) {
  10697. case 0:
  10698. if(s) gc = true;
  10699. else { cfb.FileIndex.pop(); cfb.FullPaths.pop(); }
  10700. break;
  10701. case 1: case 2: case 5:
  10702. s = true;
  10703. if(isNaN(_file.R * _file.L * _file.C)) gc = true;
  10704. if(_file.R > -1 && _file.L > -1 && _file.R == _file.L) gc = true;
  10705. break;
  10706. default: gc = true; break;
  10707. }
  10708. }
  10709. if(!gc && !f) return;
  10710. var now = new Date(1987, 1, 19), j = 0;
  10711. var data = [];
  10712. for(i = 0; i < cfb.FullPaths.length; ++i) {
  10713. if(cfb.FileIndex[i].type === 0) continue;
  10714. data.push([cfb.FullPaths[i], cfb.FileIndex[i]]);
  10715. }
  10716. for(i = 0; i < data.length; ++i) {
  10717. var dad = dirname(data[i][0]);
  10718. s = false;
  10719. for(j = 0; j < data.length; ++j) if(data[j][0] === dad) s = true;
  10720. if(!s) data.push([dad, ({
  10721. name: filename(dad).replace("/",""),
  10722. type: 1,
  10723. clsid: HEADER_CLSID,
  10724. ct: now, mt: now,
  10725. content: null
  10726. })]);
  10727. }
  10728. data.sort(function(x,y) { return namecmp(x[0], y[0]); });
  10729. cfb.FullPaths = []; cfb.FileIndex = [];
  10730. for(i = 0; i < data.length; ++i) { cfb.FullPaths[i] = data[i][0]; cfb.FileIndex[i] = data[i][1]; }
  10731. for(i = 0; i < data.length; ++i) {
  10732. var elt = cfb.FileIndex[i];
  10733. var nm = cfb.FullPaths[i];
  10734. elt.name = filename(nm).replace("/","");
  10735. elt.L = elt.R = elt.C = -(elt.color = 1);
  10736. elt.size = elt.content ? elt.content.length : 0;
  10737. elt.start = 0;
  10738. elt.clsid = (elt.clsid || HEADER_CLSID);
  10739. if(i === 0) {
  10740. elt.C = data.length > 1 ? 1 : -1;
  10741. elt.size = 0;
  10742. elt.type = 5;
  10743. } else if(nm.slice(-1) == "/") {
  10744. for(j=i+1;j < data.length; ++j) if(dirname(cfb.FullPaths[j])==nm) break;
  10745. elt.C = j >= data.length ? -1 : j;
  10746. for(j=i+1;j < data.length; ++j) if(dirname(cfb.FullPaths[j])==dirname(nm)) break;
  10747. elt.R = j >= data.length ? -1 : j;
  10748. elt.type = 1;
  10749. } else {
  10750. if(dirname(cfb.FullPaths[i+1]||"") == dirname(nm)) elt.R = i + 1;
  10751. elt.type = 2;
  10752. }
  10753. }
  10754. }
  10755. function _write(cfb, options) {
  10756. var _opts = options || {};
  10757. rebuild_cfb(cfb);
  10758. if(_opts.fileType == 'zip') return write_zip(cfb, _opts);
  10759. var L = (function(cfb){
  10760. var mini_size = 0, fat_size = 0;
  10761. for(var i = 0; i < cfb.FileIndex.length; ++i) {
  10762. var file = cfb.FileIndex[i];
  10763. if(!file.content) continue;
  10764. var flen = file.content.length;
  10765. if(flen > 0){
  10766. if(flen < 0x1000) mini_size += (flen + 0x3F) >> 6;
  10767. else fat_size += (flen + 0x01FF) >> 9;
  10768. }
  10769. }
  10770. var dir_cnt = (cfb.FullPaths.length +3) >> 2;
  10771. var mini_cnt = (mini_size + 7) >> 3;
  10772. var mfat_cnt = (mini_size + 0x7F) >> 7;
  10773. var fat_base = mini_cnt + fat_size + dir_cnt + mfat_cnt;
  10774. var fat_cnt = (fat_base + 0x7F) >> 7;
  10775. var difat_cnt = fat_cnt <= 109 ? 0 : Math.ceil((fat_cnt-109)/0x7F);
  10776. while(((fat_base + fat_cnt + difat_cnt + 0x7F) >> 7) > fat_cnt) difat_cnt = ++fat_cnt <= 109 ? 0 : Math.ceil((fat_cnt-109)/0x7F);
  10777. var L = [1, difat_cnt, fat_cnt, mfat_cnt, dir_cnt, fat_size, mini_size, 0];
  10778. cfb.FileIndex[0].size = mini_size << 6;
  10779. L[7] = (cfb.FileIndex[0].start=L[0]+L[1]+L[2]+L[3]+L[4]+L[5])+((L[6]+7) >> 3);
  10780. return L;
  10781. })(cfb);
  10782. var o = new_buf(L[7] << 9);
  10783. var i = 0, T = 0;
  10784. {
  10785. for(i = 0; i < 8; ++i) o.write_shift(1, HEADER_SIG[i]);
  10786. for(i = 0; i < 8; ++i) o.write_shift(2, 0);
  10787. o.write_shift(2, 0x003E);
  10788. o.write_shift(2, 0x0003);
  10789. o.write_shift(2, 0xFFFE);
  10790. o.write_shift(2, 0x0009);
  10791. o.write_shift(2, 0x0006);
  10792. for(i = 0; i < 3; ++i) o.write_shift(2, 0);
  10793. o.write_shift(4, 0);
  10794. o.write_shift(4, L[2]);
  10795. o.write_shift(4, L[0] + L[1] + L[2] + L[3] - 1);
  10796. o.write_shift(4, 0);
  10797. o.write_shift(4, 1<<12);
  10798. o.write_shift(4, L[3] ? L[0] + L[1] + L[2] - 1: ENDOFCHAIN);
  10799. o.write_shift(4, L[3]);
  10800. o.write_shift(-4, L[1] ? L[0] - 1: ENDOFCHAIN);
  10801. o.write_shift(4, L[1]);
  10802. for(i = 0; i < 109; ++i) o.write_shift(-4, i < L[2] ? L[1] + i : -1);
  10803. }
  10804. if(L[1]) {
  10805. for(T = 0; T < L[1]; ++T) {
  10806. for(; i < 236 + T * 127; ++i) o.write_shift(-4, i < L[2] ? L[1] + i : -1);
  10807. o.write_shift(-4, T === L[1] - 1 ? ENDOFCHAIN : T + 1);
  10808. }
  10809. }
  10810. var chainit = function(w) {
  10811. for(T += w; i<T-1; ++i) o.write_shift(-4, i+1);
  10812. if(w) { ++i; o.write_shift(-4, ENDOFCHAIN); }
  10813. };
  10814. T = i = 0;
  10815. for(T+=L[1]; i<T; ++i) o.write_shift(-4, consts.DIFSECT);
  10816. for(T+=L[2]; i<T; ++i) o.write_shift(-4, consts.FATSECT);
  10817. chainit(L[3]);
  10818. chainit(L[4]);
  10819. var j = 0, flen = 0;
  10820. var file = cfb.FileIndex[0];
  10821. for(; j < cfb.FileIndex.length; ++j) {
  10822. file = cfb.FileIndex[j];
  10823. if(!file.content) continue;
  10824. flen = file.content.length;
  10825. if(flen < 0x1000) continue;
  10826. file.start = T;
  10827. chainit((flen + 0x01FF) >> 9);
  10828. }
  10829. chainit((L[6] + 7) >> 3);
  10830. while(o.l & 0x1FF) o.write_shift(-4, consts.ENDOFCHAIN);
  10831. T = i = 0;
  10832. for(j = 0; j < cfb.FileIndex.length; ++j) {
  10833. file = cfb.FileIndex[j];
  10834. if(!file.content) continue;
  10835. flen = file.content.length;
  10836. if(!flen || flen >= 0x1000) continue;
  10837. file.start = T;
  10838. chainit((flen + 0x3F) >> 6);
  10839. }
  10840. while(o.l & 0x1FF) o.write_shift(-4, consts.ENDOFCHAIN);
  10841. for(i = 0; i < L[4]<<2; ++i) {
  10842. var nm = cfb.FullPaths[i];
  10843. if(!nm || nm.length === 0) {
  10844. for(j = 0; j < 17; ++j) o.write_shift(4, 0);
  10845. for(j = 0; j < 3; ++j) o.write_shift(4, -1);
  10846. for(j = 0; j < 12; ++j) o.write_shift(4, 0);
  10847. continue;
  10848. }
  10849. file = cfb.FileIndex[i];
  10850. if(i === 0) file.start = file.size ? file.start - 1 : ENDOFCHAIN;
  10851. var _nm = (i === 0 && _opts.root) || file.name;
  10852. flen = 2*(_nm.length+1);
  10853. o.write_shift(64, _nm, "utf16le");
  10854. o.write_shift(2, flen);
  10855. o.write_shift(1, file.type);
  10856. o.write_shift(1, file.color);
  10857. o.write_shift(-4, file.L);
  10858. o.write_shift(-4, file.R);
  10859. o.write_shift(-4, file.C);
  10860. if(!file.clsid) for(j = 0; j < 4; ++j) o.write_shift(4, 0);
  10861. else o.write_shift(16, file.clsid, "hex");
  10862. o.write_shift(4, file.state || 0);
  10863. o.write_shift(4, 0); o.write_shift(4, 0);
  10864. o.write_shift(4, 0); o.write_shift(4, 0);
  10865. o.write_shift(4, file.start);
  10866. o.write_shift(4, file.size); o.write_shift(4, 0);
  10867. }
  10868. for(i = 1; i < cfb.FileIndex.length; ++i) {
  10869. file = cfb.FileIndex[i];
  10870. if(file.size >= 0x1000) {
  10871. o.l = (file.start+1) << 9;
  10872. for(j = 0; j < file.size; ++j) o.write_shift(1, file.content[j]);
  10873. for(; j & 0x1FF; ++j) o.write_shift(1, 0);
  10874. }
  10875. }
  10876. for(i = 1; i < cfb.FileIndex.length; ++i) {
  10877. file = cfb.FileIndex[i];
  10878. if(file.size > 0 && file.size < 0x1000) {
  10879. for(j = 0; j < file.size; ++j) o.write_shift(1, file.content[j]);
  10880. for(; j & 0x3F; ++j) o.write_shift(1, 0);
  10881. }
  10882. }
  10883. while(o.l < o.length) o.write_shift(1, 0);
  10884. return o;
  10885. }
  10886. /* [MS-CFB] 2.6.4 (Unicode 3.0.1 case conversion) */
  10887. function find(cfb, path) {
  10888. var UCFullPaths = cfb.FullPaths.map(function(x) { return x.toUpperCase(); });
  10889. var UCPaths = UCFullPaths.map(function(x) { var y = x.split("/"); return y[y.length - (x.slice(-1) == "/" ? 2 : 1)]; });
  10890. var k = false;
  10891. if(path.charCodeAt(0) === 47 /* "/" */) { k = true; path = UCFullPaths[0].slice(0, -1) + path; }
  10892. else k = path.indexOf("/") !== -1;
  10893. var UCPath = path.toUpperCase();
  10894. var w = k === true ? UCFullPaths.indexOf(UCPath) : UCPaths.indexOf(UCPath);
  10895. if(w !== -1) return cfb.FileIndex[w];
  10896. var m = !UCPath.match(chr1);
  10897. UCPath = UCPath.replace(chr0,'');
  10898. if(m) UCPath = UCPath.replace(chr1,'!');
  10899. for(w = 0; w < UCFullPaths.length; ++w) {
  10900. if((m ? UCFullPaths[w].replace(chr1,'!') : UCFullPaths[w]).replace(chr0,'') == UCPath) return cfb.FileIndex[w];
  10901. if((m ? UCPaths[w].replace(chr1,'!') : UCPaths[w]).replace(chr0,'') == UCPath) return cfb.FileIndex[w];
  10902. }
  10903. return null;
  10904. }
  10905. /** CFB Constants */
  10906. var MSSZ = 64; /* Mini Sector Size = 1<<6 */
  10907. //var MSCSZ = 4096; /* Mini Stream Cutoff Size */
  10908. /* 2.1 Compound File Sector Numbers and Types */
  10909. var ENDOFCHAIN = -2;
  10910. /* 2.2 Compound File Header */
  10911. var HEADER_SIGNATURE = 'd0cf11e0a1b11ae1';
  10912. var HEADER_SIG = [0xD0, 0xCF, 0x11, 0xE0, 0xA1, 0xB1, 0x1A, 0xE1];
  10913. var HEADER_CLSID = '00000000000000000000000000000000';
  10914. var consts = {
  10915. /* 2.1 Compund File Sector Numbers and Types */
  10916. MAXREGSECT: -6,
  10917. DIFSECT: -4,
  10918. FATSECT: -3,
  10919. ENDOFCHAIN: ENDOFCHAIN,
  10920. FREESECT: -1,
  10921. /* 2.2 Compound File Header */
  10922. HEADER_SIGNATURE: HEADER_SIGNATURE,
  10923. HEADER_MINOR_VERSION: '3e00',
  10924. MAXREGSID: -6,
  10925. NOSTREAM: -1,
  10926. HEADER_CLSID: HEADER_CLSID,
  10927. /* 2.6.1 Compound File Directory Entry */
  10928. EntryTypes: ['unknown','storage','stream','lockbytes','property','root']
  10929. };
  10930. function write_file(cfb, filename, options) {
  10931. get_fs();
  10932. var o = _write(cfb, options);
  10933. fs.writeFileSync(filename, o);
  10934. }
  10935. function a2s(o) {
  10936. var out = new Array(o.length);
  10937. for(var i = 0; i < o.length; ++i) out[i] = String.fromCharCode(o[i]);
  10938. return out.join("");
  10939. }
  10940. function write(cfb, options) {
  10941. var o = _write(cfb, options);
  10942. switch(options && options.type) {
  10943. case "file": get_fs(); fs.writeFileSync(options.filename, (o)); return o;
  10944. case "binary": return a2s(o);
  10945. case "base64": return Base64.encode(a2s(o));
  10946. }
  10947. return o;
  10948. }
  10949. /* node < 8.1 zlib does not expose bytesRead, so default to pure JS */
  10950. var _zlib;
  10951. function use_zlib(zlib) { try {
  10952. var InflateRaw = zlib.InflateRaw;
  10953. var InflRaw = new InflateRaw();
  10954. InflRaw._processChunk(new Uint8Array([3, 0]), InflRaw._finishFlushFlag);
  10955. if(InflRaw.bytesRead) _zlib = zlib;
  10956. else throw new Error("zlib does not expose bytesRead");
  10957. } catch(e) {console.error("cannot use native zlib: " + (e.message || e)); } }
  10958. function _inflateRawSync(payload, usz) {
  10959. if(!_zlib) return _inflate(payload, usz);
  10960. var InflateRaw = _zlib.InflateRaw;
  10961. var InflRaw = new InflateRaw();
  10962. var out = InflRaw._processChunk(payload.slice(payload.l), InflRaw._finishFlushFlag);
  10963. payload.l += InflRaw.bytesRead;
  10964. return out;
  10965. }
  10966. function _deflateRawSync(payload) {
  10967. return _zlib ? _zlib.deflateRawSync(payload) : _deflate(payload);
  10968. }
  10969. var CLEN_ORDER = [ 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 ];
  10970. /* LEN_ID = [ 257, 258, 259, 260, 261, 262, 263, 264, 265, 266, 267, 268, 269, 270, 271, 272, 273, 274, 275, 276, 277, 278, 279, 280, 281, 282, 283, 284, 285 ]; */
  10971. var LEN_LN = [ 3, 4, 5, 6, 7, 8, 9, 10, 11, 13 , 15, 17, 19, 23, 27, 31, 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258 ];
  10972. /* DST_ID = [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29 ]; */
  10973. var DST_LN = [ 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, 8193, 12289, 16385, 24577 ];
  10974. function bit_swap_8(n) { var t = (((((n<<1)|(n<<11)) & 0x22110) | (((n<<5)|(n<<15)) & 0x88440))); return ((t>>16) | (t>>8) |t)&0xFF; }
  10975. var use_typed_arrays = typeof Uint8Array !== 'undefined';
  10976. var bitswap8 = use_typed_arrays ? new Uint8Array(1<<8) : [];
  10977. for(var q = 0; q < (1<<8); ++q) bitswap8[q] = bit_swap_8(q);
  10978. function bit_swap_n(n, b) {
  10979. var rev = bitswap8[n & 0xFF];
  10980. if(b <= 8) return rev >>> (8-b);
  10981. rev = (rev << 8) | bitswap8[(n>>8)&0xFF];
  10982. if(b <= 16) return rev >>> (16-b);
  10983. rev = (rev << 8) | bitswap8[(n>>16)&0xFF];
  10984. return rev >>> (24-b);
  10985. }
  10986. /* helpers for unaligned bit reads */
  10987. function read_bits_2(buf, bl) { var w = (bl&7), h = (bl>>>3); return ((buf[h]|(w <= 6 ? 0 : buf[h+1]<<8))>>>w)& 0x03; }
  10988. function read_bits_3(buf, bl) { var w = (bl&7), h = (bl>>>3); return ((buf[h]|(w <= 5 ? 0 : buf[h+1]<<8))>>>w)& 0x07; }
  10989. function read_bits_4(buf, bl) { var w = (bl&7), h = (bl>>>3); return ((buf[h]|(w <= 4 ? 0 : buf[h+1]<<8))>>>w)& 0x0F; }
  10990. function read_bits_5(buf, bl) { var w = (bl&7), h = (bl>>>3); return ((buf[h]|(w <= 3 ? 0 : buf[h+1]<<8))>>>w)& 0x1F; }
  10991. function read_bits_7(buf, bl) { var w = (bl&7), h = (bl>>>3); return ((buf[h]|(w <= 1 ? 0 : buf[h+1]<<8))>>>w)& 0x7F; }
  10992. /* works up to n = 3 * 8 + 1 = 25 */
  10993. function read_bits_n(buf, bl, n) {
  10994. var w = (bl&7), h = (bl>>>3), f = ((1<<n)-1);
  10995. var v = buf[h] >>> w;
  10996. if(n < 8 - w) return v & f;
  10997. v |= buf[h+1]<<(8-w);
  10998. if(n < 16 - w) return v & f;
  10999. v |= buf[h+2]<<(16-w);
  11000. if(n < 24 - w) return v & f;
  11001. v |= buf[h+3]<<(24-w);
  11002. return v & f;
  11003. }
  11004. /* until ArrayBuffer#realloc is a thing, fake a realloc */
  11005. function realloc(b, sz) {
  11006. var L = b.length, M = 2*L > sz ? 2*L : sz + 5, i = 0;
  11007. if(L >= sz) return b;
  11008. if(has_buf) {
  11009. var o = new_unsafe_buf(M);
  11010. // $FlowIgnore
  11011. if(b.copy) b.copy(o);
  11012. else for(; i < b.length; ++i) o[i] = b[i];
  11013. return o;
  11014. } else if(use_typed_arrays) {
  11015. var a = new Uint8Array(M);
  11016. if(a.set) a.set(b);
  11017. else for(; i < b.length; ++i) a[i] = b[i];
  11018. return a;
  11019. }
  11020. b.length = M;
  11021. return b;
  11022. }
  11023. /* zero-filled arrays for older browsers */
  11024. function zero_fill_array(n) {
  11025. var o = new Array(n);
  11026. for(var i = 0; i < n; ++i) o[i] = 0;
  11027. return o;
  11028. }var _deflate = (function() {
  11029. var _deflateRaw = (function() {
  11030. return function deflateRaw(data, out) {
  11031. var boff = 0;
  11032. while(boff < data.length) {
  11033. var L = Math.min(0xFFFF, data.length - boff);
  11034. var h = boff + L == data.length;
  11035. /* TODO: this is only type 0 stored */
  11036. out.write_shift(1, +h);
  11037. out.write_shift(2, L);
  11038. out.write_shift(2, (~L) & 0xFFFF);
  11039. while(L-- > 0) out[out.l++] = data[boff++];
  11040. }
  11041. return out.l;
  11042. };
  11043. })();
  11044. return function(data) {
  11045. var buf = new_buf(50+Math.floor(data.length*1.1));
  11046. var off = _deflateRaw(data, buf);
  11047. return buf.slice(0, off);
  11048. };
  11049. })();
  11050. /* modified inflate function also moves original read head */
  11051. /* build tree (used for literals and lengths) */
  11052. function build_tree(clens, cmap, MAX) {
  11053. var maxlen = 1, w = 0, i = 0, j = 0, ccode = 0, L = clens.length;
  11054. var bl_count = use_typed_arrays ? new Uint16Array(32) : zero_fill_array(32);
  11055. for(i = 0; i < 32; ++i) bl_count[i] = 0;
  11056. for(i = L; i < MAX; ++i) clens[i] = 0;
  11057. L = clens.length;
  11058. var ctree = use_typed_arrays ? new Uint16Array(L) : zero_fill_array(L); // []
  11059. /* build code tree */
  11060. for(i = 0; i < L; ++i) {
  11061. bl_count[(w = clens[i])]++;
  11062. if(maxlen < w) maxlen = w;
  11063. ctree[i] = 0;
  11064. }
  11065. bl_count[0] = 0;
  11066. for(i = 1; i <= maxlen; ++i) bl_count[i+16] = (ccode = (ccode + bl_count[i-1])<<1);
  11067. for(i = 0; i < L; ++i) {
  11068. ccode = clens[i];
  11069. if(ccode != 0) ctree[i] = bl_count[ccode+16]++;
  11070. }
  11071. /* cmap[maxlen + 4 bits] = (off&15) + (lit<<4) reverse mapping */
  11072. var cleni = 0;
  11073. for(i = 0; i < L; ++i) {
  11074. cleni = clens[i];
  11075. if(cleni != 0) {
  11076. ccode = bit_swap_n(ctree[i], maxlen)>>(maxlen-cleni);
  11077. for(j = (1<<(maxlen + 4 - cleni)) - 1; j>=0; --j)
  11078. cmap[ccode|(j<<cleni)] = (cleni&15) | (i<<4);
  11079. }
  11080. }
  11081. return maxlen;
  11082. }
  11083. var fix_lmap = use_typed_arrays ? new Uint16Array(512) : zero_fill_array(512);
  11084. var fix_dmap = use_typed_arrays ? new Uint16Array(32) : zero_fill_array(32);
  11085. if(!use_typed_arrays) {
  11086. for(var i = 0; i < 512; ++i) fix_lmap[i] = 0;
  11087. for(i = 0; i < 32; ++i) fix_dmap[i] = 0;
  11088. }
  11089. (function() {
  11090. var dlens = [];
  11091. var i = 0;
  11092. for(;i<32; i++) dlens.push(5);
  11093. build_tree(dlens, fix_dmap, 32);
  11094. var clens = [];
  11095. i = 0;
  11096. for(; i<=143; i++) clens.push(8);
  11097. for(; i<=255; i++) clens.push(9);
  11098. for(; i<=279; i++) clens.push(7);
  11099. for(; i<=287; i++) clens.push(8);
  11100. build_tree(clens, fix_lmap, 288);
  11101. })();
  11102. var dyn_lmap = use_typed_arrays ? new Uint16Array(32768) : zero_fill_array(32768);
  11103. var dyn_dmap = use_typed_arrays ? new Uint16Array(32768) : zero_fill_array(32768);
  11104. var dyn_cmap = use_typed_arrays ? new Uint16Array(128) : zero_fill_array(128);
  11105. var dyn_len_1 = 1, dyn_len_2 = 1;
  11106. /* 5.5.3 Expanding Huffman Codes */
  11107. function dyn(data, boff) {
  11108. /* nomenclature from RFC1951 refers to bit values; these are offset by the implicit constant */
  11109. var _HLIT = read_bits_5(data, boff) + 257; boff += 5;
  11110. var _HDIST = read_bits_5(data, boff) + 1; boff += 5;
  11111. var _HCLEN = read_bits_4(data, boff) + 4; boff += 4;
  11112. var w = 0;
  11113. /* grab and store code lengths */
  11114. var clens = use_typed_arrays ? new Uint8Array(19) : zero_fill_array(19);
  11115. var ctree = [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ];
  11116. var maxlen = 1;
  11117. var bl_count = use_typed_arrays ? new Uint8Array(8) : zero_fill_array(8);
  11118. var next_code = use_typed_arrays ? new Uint8Array(8) : zero_fill_array(8);
  11119. var L = clens.length; /* 19 */
  11120. for(var i = 0; i < _HCLEN; ++i) {
  11121. clens[CLEN_ORDER[i]] = w = read_bits_3(data, boff);
  11122. if(maxlen < w) maxlen = w;
  11123. bl_count[w]++;
  11124. boff += 3;
  11125. }
  11126. /* build code tree */
  11127. var ccode = 0;
  11128. bl_count[0] = 0;
  11129. for(i = 1; i <= maxlen; ++i) next_code[i] = ccode = (ccode + bl_count[i-1])<<1;
  11130. for(i = 0; i < L; ++i) if((ccode = clens[i]) != 0) ctree[i] = next_code[ccode]++;
  11131. /* cmap[7 bits from stream] = (off&7) + (lit<<3) */
  11132. var cleni = 0;
  11133. for(i = 0; i < L; ++i) {
  11134. cleni = clens[i];
  11135. if(cleni != 0) {
  11136. ccode = bitswap8[ctree[i]]>>(8-cleni);
  11137. for(var j = (1<<(7-cleni))-1; j>=0; --j) dyn_cmap[ccode|(j<<cleni)] = (cleni&7) | (i<<3);
  11138. }
  11139. }
  11140. /* read literal and dist codes at once */
  11141. var hcodes = [];
  11142. maxlen = 1;
  11143. for(; hcodes.length < _HLIT + _HDIST;) {
  11144. ccode = dyn_cmap[read_bits_7(data, boff)];
  11145. boff += ccode & 7;
  11146. switch((ccode >>>= 3)) {
  11147. case 16:
  11148. w = 3 + read_bits_2(data, boff); boff += 2;
  11149. ccode = hcodes[hcodes.length - 1];
  11150. while(w-- > 0) hcodes.push(ccode);
  11151. break;
  11152. case 17:
  11153. w = 3 + read_bits_3(data, boff); boff += 3;
  11154. while(w-- > 0) hcodes.push(0);
  11155. break;
  11156. case 18:
  11157. w = 11 + read_bits_7(data, boff); boff += 7;
  11158. while(w -- > 0) hcodes.push(0);
  11159. break;
  11160. default:
  11161. hcodes.push(ccode);
  11162. if(maxlen < ccode) maxlen = ccode;
  11163. break;
  11164. }
  11165. }
  11166. /* build literal / length trees */
  11167. var h1 = hcodes.slice(0, _HLIT), h2 = hcodes.slice(_HLIT);
  11168. for(i = _HLIT; i < 286; ++i) h1[i] = 0;
  11169. for(i = _HDIST; i < 30; ++i) h2[i] = 0;
  11170. dyn_len_1 = build_tree(h1, dyn_lmap, 286);
  11171. dyn_len_2 = build_tree(h2, dyn_dmap, 30);
  11172. return boff;
  11173. }
  11174. /* return [ data, bytesRead ] */
  11175. function inflate(data, usz) {
  11176. /* shortcircuit for empty buffer [0x03, 0x00] */
  11177. if(data[0] == 3 && !(data[1] & 0x3)) { return [new_raw_buf(usz), 2]; }
  11178. /* bit offset */
  11179. var boff = 0;
  11180. /* header includes final bit and type bits */
  11181. var header = 0;
  11182. var outbuf = new_unsafe_buf(usz ? usz : (1<<18));
  11183. var woff = 0;
  11184. var OL = outbuf.length>>>0;
  11185. var max_len_1 = 0, max_len_2 = 0;
  11186. while((header&1) == 0) {
  11187. header = read_bits_3(data, boff); boff += 3;
  11188. if((header >>> 1) == 0) {
  11189. /* Stored block */
  11190. if(boff & 7) boff += 8 - (boff&7);
  11191. /* 2 bytes sz, 2 bytes bit inverse */
  11192. var sz = data[boff>>>3] | data[(boff>>>3)+1]<<8;
  11193. boff += 32;
  11194. /* push sz bytes */
  11195. if(!usz && OL < woff + sz) { outbuf = realloc(outbuf, woff + sz); OL = outbuf.length; }
  11196. if(typeof data.copy === 'function') {
  11197. // $FlowIgnore
  11198. data.copy(outbuf, woff, boff>>>3, (boff>>>3)+sz);
  11199. woff += sz; boff += 8*sz;
  11200. } else while(sz-- > 0) { outbuf[woff++] = data[boff>>>3]; boff += 8; }
  11201. continue;
  11202. } else if((header >>> 1) == 1) {
  11203. /* Fixed Huffman */
  11204. max_len_1 = 9; max_len_2 = 5;
  11205. } else {
  11206. /* Dynamic Huffman */
  11207. boff = dyn(data, boff);
  11208. max_len_1 = dyn_len_1; max_len_2 = dyn_len_2;
  11209. }
  11210. if(!usz && (OL < woff + 32767)) { outbuf = realloc(outbuf, woff + 32767); OL = outbuf.length; }
  11211. for(;;) { // while(true) is apparently out of vogue in modern JS circles
  11212. /* ingest code and move read head */
  11213. var bits = read_bits_n(data, boff, max_len_1);
  11214. var code = (header>>>1) == 1 ? fix_lmap[bits] : dyn_lmap[bits];
  11215. boff += code & 15; code >>>= 4;
  11216. /* 0-255 are literals, 256 is end of block token, 257+ are copy tokens */
  11217. if(((code>>>8)&0xFF) === 0) outbuf[woff++] = code;
  11218. else if(code == 256) break;
  11219. else {
  11220. code -= 257;
  11221. var len_eb = (code < 8) ? 0 : ((code-4)>>2); if(len_eb > 5) len_eb = 0;
  11222. var tgt = woff + LEN_LN[code];
  11223. /* length extra bits */
  11224. if(len_eb > 0) {
  11225. tgt += read_bits_n(data, boff, len_eb);
  11226. boff += len_eb;
  11227. }
  11228. /* dist code */
  11229. bits = read_bits_n(data, boff, max_len_2);
  11230. code = (header>>>1) == 1 ? fix_dmap[bits] : dyn_dmap[bits];
  11231. boff += code & 15; code >>>= 4;
  11232. var dst_eb = (code < 4 ? 0 : (code-2)>>1);
  11233. var dst = DST_LN[code];
  11234. /* dist extra bits */
  11235. if(dst_eb > 0) {
  11236. dst += read_bits_n(data, boff, dst_eb);
  11237. boff += dst_eb;
  11238. }
  11239. /* in the common case, manual byte copy is faster than TA set / Buffer copy */
  11240. if(!usz && OL < tgt) { outbuf = realloc(outbuf, tgt); OL = outbuf.length; }
  11241. while(woff < tgt) { outbuf[woff] = outbuf[woff - dst]; ++woff; }
  11242. }
  11243. }
  11244. }
  11245. return [usz ? outbuf : outbuf.slice(0, woff), (boff+7)>>>3];
  11246. }
  11247. function _inflate(payload, usz) {
  11248. var data = payload.slice(payload.l||0);
  11249. var out = inflate(data, usz);
  11250. payload.l += out[1];
  11251. return out[0];
  11252. }
  11253. function warn_or_throw(wrn, msg) {
  11254. if(wrn) { if(typeof console !== 'undefined') console.error(msg); }
  11255. else throw new Error(msg);
  11256. }
  11257. function parse_zip(file, options) {
  11258. var blob = file;
  11259. prep_blob(blob, 0);
  11260. var FileIndex = [], FullPaths = [];
  11261. var o = {
  11262. FileIndex: FileIndex,
  11263. FullPaths: FullPaths
  11264. };
  11265. init_cfb(o, { root: options.root });
  11266. /* find end of central directory, start just after signature */
  11267. var i = blob.length - 4;
  11268. while((blob[i] != 0x50 || blob[i+1] != 0x4b || blob[i+2] != 0x05 || blob[i+3] != 0x06) && i >= 0) --i;
  11269. blob.l = i + 4;
  11270. /* parse end of central directory */
  11271. blob.l += 4;
  11272. var fcnt = blob.read_shift(2);
  11273. blob.l += 6;
  11274. var start_cd = blob.read_shift(4);
  11275. /* parse central directory */
  11276. blob.l = start_cd;
  11277. for(i = 0; i < fcnt; ++i) {
  11278. /* trust local file header instead of CD entry */
  11279. blob.l += 20;
  11280. var csz = blob.read_shift(4);
  11281. var usz = blob.read_shift(4);
  11282. var namelen = blob.read_shift(2);
  11283. var efsz = blob.read_shift(2);
  11284. var fcsz = blob.read_shift(2);
  11285. blob.l += 8;
  11286. var offset = blob.read_shift(4);
  11287. var EF = parse_extra_field(blob.slice(blob.l+namelen, blob.l+namelen+efsz));
  11288. blob.l += namelen + efsz + fcsz;
  11289. var L = blob.l;
  11290. blob.l = offset + 4;
  11291. parse_local_file(blob, csz, usz, o, EF);
  11292. blob.l = L;
  11293. }
  11294. return o;
  11295. }
  11296. /* head starts just after local file header signature */
  11297. function parse_local_file(blob, csz, usz, o, EF) {
  11298. /* [local file header] */
  11299. blob.l += 2;
  11300. var flags = blob.read_shift(2);
  11301. var meth = blob.read_shift(2);
  11302. var date = parse_dos_date(blob);
  11303. if(flags & 0x2041) throw new Error("Unsupported ZIP encryption");
  11304. var crc32 = blob.read_shift(4);
  11305. var _csz = blob.read_shift(4);
  11306. var _usz = blob.read_shift(4);
  11307. var namelen = blob.read_shift(2);
  11308. var efsz = blob.read_shift(2);
  11309. // TODO: flags & (1<<11) // UTF8
  11310. var name = ""; for(var i = 0; i < namelen; ++i) name += String.fromCharCode(blob[blob.l++]);
  11311. if(efsz) {
  11312. var ef = parse_extra_field(blob.slice(blob.l, blob.l + efsz));
  11313. if((ef[0x5455]||{}).mt) date = ef[0x5455].mt;
  11314. if(((EF||{})[0x5455]||{}).mt) date = EF[0x5455].mt;
  11315. }
  11316. blob.l += efsz;
  11317. /* [encryption header] */
  11318. /* [file data] */
  11319. var data = blob.slice(blob.l, blob.l + _csz);
  11320. switch(meth) {
  11321. case 8: data = _inflateRawSync(blob, _usz); break;
  11322. case 0: break;
  11323. default: throw new Error("Unsupported ZIP Compression method " + meth);
  11324. }
  11325. /* [data descriptor] */
  11326. var wrn = false;
  11327. if(flags & 8) {
  11328. crc32 = blob.read_shift(4);
  11329. if(crc32 == 0x08074b50) { crc32 = blob.read_shift(4); wrn = true; }
  11330. _csz = blob.read_shift(4);
  11331. _usz = blob.read_shift(4);
  11332. }
  11333. if(_csz != csz) warn_or_throw(wrn, "Bad compressed size: " + csz + " != " + _csz);
  11334. if(_usz != usz) warn_or_throw(wrn, "Bad uncompressed size: " + usz + " != " + _usz);
  11335. var _crc32 = CRC32.buf(data, 0);
  11336. if(crc32 != _crc32) warn_or_throw(wrn, "Bad CRC32 checksum: " + crc32 + " != " + _crc32);
  11337. cfb_add(o, name, data, {unsafe: true, mt: date});
  11338. }
  11339. function write_zip(cfb, options) {
  11340. var _opts = options || {};
  11341. var out = [], cdirs = [];
  11342. var o = new_buf(1);
  11343. var method = (_opts.compression ? 8 : 0), flags = 0;
  11344. var desc = false;
  11345. if(desc) flags |= 8;
  11346. var i = 0, j = 0;
  11347. var start_cd = 0, fcnt = 0;
  11348. var root = cfb.FullPaths[0], fp = root, fi = cfb.FileIndex[0];
  11349. var crcs = [];
  11350. var sz_cd = 0;
  11351. for(i = 1; i < cfb.FullPaths.length; ++i) {
  11352. fp = cfb.FullPaths[i].slice(root.length); fi = cfb.FileIndex[i];
  11353. if(!fi.size || !fi.content || fp == "\u0001Sh33tJ5") continue;
  11354. var start = start_cd;
  11355. /* TODO: CP437 filename */
  11356. var namebuf = new_buf(fp.length);
  11357. for(j = 0; j < fp.length; ++j) namebuf.write_shift(1, fp.charCodeAt(j) & 0x7F);
  11358. namebuf = namebuf.slice(0, namebuf.l);
  11359. crcs[fcnt] = CRC32.buf(fi.content, 0);
  11360. var outbuf = fi.content;
  11361. if(method == 8) outbuf = _deflateRawSync(outbuf);
  11362. /* local file header */
  11363. o = new_buf(30);
  11364. o.write_shift(4, 0x04034b50);
  11365. o.write_shift(2, 20);
  11366. o.write_shift(2, flags);
  11367. o.write_shift(2, method);
  11368. /* TODO: last mod file time/date */
  11369. if(fi.mt) write_dos_date(o, fi.mt);
  11370. else o.write_shift(4, 0);
  11371. o.write_shift(-4, (flags & 8) ? 0 : crcs[fcnt]);
  11372. o.write_shift(4, (flags & 8) ? 0 : outbuf.length);
  11373. o.write_shift(4, (flags & 8) ? 0 : fi.content.length);
  11374. o.write_shift(2, namebuf.length);
  11375. o.write_shift(2, 0);
  11376. start_cd += o.length;
  11377. out.push(o);
  11378. start_cd += namebuf.length;
  11379. out.push(namebuf);
  11380. /* TODO: encryption header ? */
  11381. start_cd += outbuf.length;
  11382. out.push(outbuf);
  11383. /* data descriptor */
  11384. if(flags & 8) {
  11385. o = new_buf(12);
  11386. o.write_shift(-4, crcs[fcnt]);
  11387. o.write_shift(4, outbuf.length);
  11388. o.write_shift(4, fi.content.length);
  11389. start_cd += o.l;
  11390. out.push(o);
  11391. }
  11392. /* central directory */
  11393. o = new_buf(46);
  11394. o.write_shift(4, 0x02014b50);
  11395. o.write_shift(2, 0);
  11396. o.write_shift(2, 20);
  11397. o.write_shift(2, flags);
  11398. o.write_shift(2, method);
  11399. o.write_shift(4, 0); /* TODO: last mod file time/date */
  11400. o.write_shift(-4, crcs[fcnt]);
  11401. o.write_shift(4, outbuf.length);
  11402. o.write_shift(4, fi.content.length);
  11403. o.write_shift(2, namebuf.length);
  11404. o.write_shift(2, 0);
  11405. o.write_shift(2, 0);
  11406. o.write_shift(2, 0);
  11407. o.write_shift(2, 0);
  11408. o.write_shift(4, 0);
  11409. o.write_shift(4, start);
  11410. sz_cd += o.l;
  11411. cdirs.push(o);
  11412. sz_cd += namebuf.length;
  11413. cdirs.push(namebuf);
  11414. ++fcnt;
  11415. }
  11416. /* end of central directory */
  11417. o = new_buf(22);
  11418. o.write_shift(4, 0x06054b50);
  11419. o.write_shift(2, 0);
  11420. o.write_shift(2, 0);
  11421. o.write_shift(2, fcnt);
  11422. o.write_shift(2, fcnt);
  11423. o.write_shift(4, sz_cd);
  11424. o.write_shift(4, start_cd);
  11425. o.write_shift(2, 0);
  11426. return bconcat(([bconcat((out)), bconcat(cdirs), o]));
  11427. }
  11428. function cfb_new(opts) {
  11429. var o = ({});
  11430. init_cfb(o, opts);
  11431. return o;
  11432. }
  11433. function cfb_add(cfb, name, content, opts) {
  11434. var unsafe = opts && opts.unsafe;
  11435. if(!unsafe) init_cfb(cfb);
  11436. var file = !unsafe && CFB.find(cfb, name);
  11437. if(!file) {
  11438. var fpath = cfb.FullPaths[0];
  11439. if(name.slice(0, fpath.length) == fpath) fpath = name;
  11440. else {
  11441. if(fpath.slice(-1) != "/") fpath += "/";
  11442. fpath = (fpath + name).replace("//","/");
  11443. }
  11444. file = ({name: filename(name), type: 2});
  11445. cfb.FileIndex.push(file);
  11446. cfb.FullPaths.push(fpath);
  11447. if(!unsafe) CFB.utils.cfb_gc(cfb);
  11448. }
  11449. file.content = (content);
  11450. file.size = content ? content.length : 0;
  11451. if(opts) {
  11452. if(opts.CLSID) file.clsid = opts.CLSID;
  11453. if(opts.mt) file.mt = opts.mt;
  11454. if(opts.ct) file.ct = opts.ct;
  11455. }
  11456. return file;
  11457. }
  11458. function cfb_del(cfb, name) {
  11459. init_cfb(cfb);
  11460. var file = CFB.find(cfb, name);
  11461. if(file) for(var j = 0; j < cfb.FileIndex.length; ++j) if(cfb.FileIndex[j] == file) {
  11462. cfb.FileIndex.splice(j, 1);
  11463. cfb.FullPaths.splice(j, 1);
  11464. return true;
  11465. }
  11466. return false;
  11467. }
  11468. function cfb_mov(cfb, old_name, new_name) {
  11469. init_cfb(cfb);
  11470. var file = CFB.find(cfb, old_name);
  11471. if(file) for(var j = 0; j < cfb.FileIndex.length; ++j) if(cfb.FileIndex[j] == file) {
  11472. cfb.FileIndex[j].name = filename(new_name);
  11473. cfb.FullPaths[j] = new_name;
  11474. return true;
  11475. }
  11476. return false;
  11477. }
  11478. function cfb_gc(cfb) { rebuild_cfb(cfb, true); }
  11479. exports.find = find;
  11480. exports.read = read;
  11481. exports.parse = parse;
  11482. exports.write = write;
  11483. exports.writeFile = write_file;
  11484. exports.utils = {
  11485. cfb_new: cfb_new,
  11486. cfb_add: cfb_add,
  11487. cfb_del: cfb_del,
  11488. cfb_mov: cfb_mov,
  11489. cfb_gc: cfb_gc,
  11490. ReadShift: ReadShift,
  11491. CheckField: CheckField,
  11492. prep_blob: prep_blob,
  11493. bconcat: bconcat,
  11494. use_zlib: use_zlib,
  11495. _deflateRaw: _deflate,
  11496. _inflateRaw: _inflate,
  11497. consts: consts
  11498. };
  11499. return exports;
  11500. })();
  11501. if(typeof require !== 'undefined' && typeof module !== 'undefined' && typeof DO_NOT_EXPORT_CFB === 'undefined') { module.exports = CFB; }
  11502. var _fs;
  11503. if(typeof require !== 'undefined') try { _fs = require('fs'); } catch(e) {}
  11504. /* normalize data for blob ctor */
  11505. function blobify(data) {
  11506. if(typeof data === "string") return s2ab(data);
  11507. if(Array.isArray(data)) return a2u(data);
  11508. return data;
  11509. }
  11510. /* write or download file */
  11511. function write_dl(fname, payload, enc) {
  11512. /*global IE_SaveFile, Blob, navigator, saveAs, URL, document, File, chrome */
  11513. if(typeof _fs !== 'undefined' && _fs.writeFileSync) return enc ? _fs.writeFileSync(fname, payload, enc) : _fs.writeFileSync(fname, payload);
  11514. var data = (enc == "utf8") ? utf8write(payload) : payload;
  11515. if(typeof IE_SaveFile !== 'undefined') return IE_SaveFile(data, fname);
  11516. if(typeof Blob !== 'undefined') {
  11517. var blob = new Blob([blobify(data)], {type:"application/octet-stream"});
  11518. if(typeof navigator !== 'undefined' && navigator.msSaveBlob) return navigator.msSaveBlob(blob, fname);
  11519. if(typeof saveAs !== 'undefined') return saveAs(blob, fname);
  11520. if(typeof URL !== 'undefined' && typeof document !== 'undefined' && document.createElement && URL.createObjectURL) {
  11521. var url = URL.createObjectURL(blob);
  11522. if(typeof chrome === 'object' && typeof (chrome.downloads||{}).download == "function") {
  11523. if(URL.revokeObjectURL && typeof setTimeout !== 'undefined') setTimeout(function() { URL.revokeObjectURL(url); }, 60000);
  11524. return chrome.downloads.download({ url: url, filename: fname, saveAs: true});
  11525. }
  11526. var a = document.createElement("a");
  11527. if(a.download != null) {
  11528. a.download = fname; a.href = url; document.body.appendChild(a); a.click();
  11529. document.body.removeChild(a);
  11530. if(URL.revokeObjectURL && typeof setTimeout !== 'undefined') setTimeout(function() { URL.revokeObjectURL(url); }, 60000);
  11531. return url;
  11532. }
  11533. }
  11534. }
  11535. // $FlowIgnore
  11536. if(typeof $ !== 'undefined' && typeof File !== 'undefined' && typeof Folder !== 'undefined') try { // extendscript
  11537. // $FlowIgnore
  11538. var out = File(fname); out.open("w"); out.encoding = "binary";
  11539. if(Array.isArray(payload)) payload = a2s(payload);
  11540. out.write(payload); out.close(); return payload;
  11541. } catch(e) { if(!e.message || !e.message.match(/onstruct/)) throw e; }
  11542. throw new Error("cannot save file " + fname);
  11543. }
  11544. /* read binary data from file */
  11545. function read_binary(path) {
  11546. if(typeof _fs !== 'undefined') return _fs.readFileSync(path);
  11547. // $FlowIgnore
  11548. if(typeof $ !== 'undefined' && typeof File !== 'undefined' && typeof Folder !== 'undefined') try { // extendscript
  11549. // $FlowIgnore
  11550. var infile = File(path); infile.open("r"); infile.encoding = "binary";
  11551. var data = infile.read(); infile.close();
  11552. return data;
  11553. } catch(e) { if(!e.message || !e.message.match(/onstruct/)) throw e; }
  11554. throw new Error("Cannot access file " + path);
  11555. }
  11556. function keys(o) {
  11557. var ks = Object.keys(o), o2 = [];
  11558. for(var i = 0; i < ks.length; ++i) if(o.hasOwnProperty(ks[i])) o2.push(ks[i]);
  11559. return o2;
  11560. }
  11561. function evert_key(obj, key) {
  11562. var o = ([]), K = keys(obj);
  11563. for(var i = 0; i !== K.length; ++i) if(o[obj[K[i]][key]] == null) o[obj[K[i]][key]] = K[i];
  11564. return o;
  11565. }
  11566. function evert(obj) {
  11567. var o = ([]), K = keys(obj);
  11568. for(var i = 0; i !== K.length; ++i) o[obj[K[i]]] = K[i];
  11569. return o;
  11570. }
  11571. function evert_num(obj) {
  11572. var o = ([]), K = keys(obj);
  11573. for(var i = 0; i !== K.length; ++i) o[obj[K[i]]] = parseInt(K[i],10);
  11574. return o;
  11575. }
  11576. function evert_arr(obj) {
  11577. var o = ([]), K = keys(obj);
  11578. for(var i = 0; i !== K.length; ++i) {
  11579. if(o[obj[K[i]]] == null) o[obj[K[i]]] = [];
  11580. o[obj[K[i]]].push(K[i]);
  11581. }
  11582. return o;
  11583. }
  11584. var basedate = new Date(1899, 11, 30, 0, 0, 0); // 2209161600000
  11585. var dnthresh = basedate.getTime() + (new Date().getTimezoneOffset() - basedate.getTimezoneOffset()) * 60000;
  11586. function datenum(v, date1904) {
  11587. var epoch = v.getTime();
  11588. if(date1904) epoch -= 1462*24*60*60*1000;
  11589. return (epoch - dnthresh) / (24 * 60 * 60 * 1000);
  11590. }
  11591. function numdate(v) {
  11592. var out = new Date();
  11593. out.setTime(v * 24 * 60 * 60 * 1000 + dnthresh);
  11594. return out;
  11595. }
  11596. /* ISO 8601 Duration */
  11597. function parse_isodur(s) {
  11598. var sec = 0, mt = 0, time = false;
  11599. var m = s.match(/P([0-9\.]+Y)?([0-9\.]+M)?([0-9\.]+D)?T([0-9\.]+H)?([0-9\.]+M)?([0-9\.]+S)?/);
  11600. if(!m) throw new Error("|" + s + "| is not an ISO8601 Duration");
  11601. for(var i = 1; i != m.length; ++i) {
  11602. if(!m[i]) continue;
  11603. mt = 1;
  11604. if(i > 3) time = true;
  11605. switch(m[i].slice(m[i].length-1)) {
  11606. case 'Y':
  11607. throw new Error("Unsupported ISO Duration Field: " + m[i].slice(m[i].length-1));
  11608. case 'D': mt *= 24;
  11609. /* falls through */
  11610. case 'H': mt *= 60;
  11611. /* falls through */
  11612. case 'M':
  11613. if(!time) throw new Error("Unsupported ISO Duration Field: M");
  11614. else mt *= 60;
  11615. /* falls through */
  11616. case 'S': break;
  11617. }
  11618. sec += mt * parseInt(m[i], 10);
  11619. }
  11620. return sec;
  11621. }
  11622. var good_pd_date = new Date('2017-02-19T19:06:09.000Z');
  11623. if(isNaN(good_pd_date.getFullYear())) good_pd_date = new Date('2/19/17');
  11624. var good_pd = good_pd_date.getFullYear() == 2017;
  11625. /* parses a date as a local date */
  11626. function parseDate(str, fixdate) {
  11627. var d = new Date(str);
  11628. if(good_pd) {
  11629. if(fixdate > 0) d.setTime(d.getTime() + d.getTimezoneOffset() * 60 * 1000);
  11630. else if(fixdate < 0) d.setTime(d.getTime() - d.getTimezoneOffset() * 60 * 1000);
  11631. return d;
  11632. }
  11633. if(str instanceof Date) return str;
  11634. if(good_pd_date.getFullYear() == 1917 && !isNaN(d.getFullYear())) {
  11635. var s = d.getFullYear();
  11636. if(str.indexOf("" + s) > -1) return d;
  11637. d.setFullYear(d.getFullYear() + 100); return d;
  11638. }
  11639. var n = str.match(/\d+/g)||["2017","2","19","0","0","0"];
  11640. var out = new Date(+n[0], +n[1] - 1, +n[2], (+n[3]||0), (+n[4]||0), (+n[5]||0));
  11641. if(str.indexOf("Z") > -1) out = new Date(out.getTime() - out.getTimezoneOffset() * 60 * 1000);
  11642. return out;
  11643. }
  11644. function cc2str(arr) {
  11645. var o = "";
  11646. for(var i = 0; i != arr.length; ++i) o += String.fromCharCode(arr[i]);
  11647. return o;
  11648. }
  11649. function dup(o) {
  11650. if(typeof JSON != 'undefined' && !Array.isArray(o)) return JSON.parse(JSON.stringify(o));
  11651. if(typeof o != 'object' || o == null) return o;
  11652. if(o instanceof Date) return new Date(o.getTime());
  11653. var out = {};
  11654. for(var k in o) if(o.hasOwnProperty(k)) out[k] = dup(o[k]);
  11655. return out;
  11656. }
  11657. function fill(c,l) { var o = ""; while(o.length < l) o+=c; return o; }
  11658. /* TODO: stress test */
  11659. function fuzzynum(s) {
  11660. var v = Number(s);
  11661. if(!isNaN(v)) return v;
  11662. var wt = 1;
  11663. var ss = s.replace(/([\d]),([\d])/g,"$1$2").replace(/[$]/g,"").replace(/[%]/g, function() { wt *= 100; return "";});
  11664. if(!isNaN(v = Number(ss))) return v / wt;
  11665. ss = ss.replace(/[(](.*)[)]/,function($$, $1) { wt = -wt; return $1;});
  11666. if(!isNaN(v = Number(ss))) return v / wt;
  11667. return v;
  11668. }
  11669. function fuzzydate(s) {
  11670. var o = new Date(s), n = new Date(NaN);
  11671. var y = o.getYear(), m = o.getMonth(), d = o.getDate();
  11672. if(isNaN(d)) return n;
  11673. if(y < 0 || y > 8099) return n;
  11674. if((m > 0 || d > 1) && y != 101) return o;
  11675. if(s.toLowerCase().match(/jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec/)) return o;
  11676. if(s.match(/[^-0-9:,\/\\]/)) return n;
  11677. return o;
  11678. }
  11679. var safe_split_regex = "abacaba".split(/(:?b)/i).length == 5;
  11680. function split_regex(str, re, def) {
  11681. if(safe_split_regex || typeof re == "string") return str.split(re);
  11682. var p = str.split(re), o = [p[0]];
  11683. for(var i = 1; i < p.length; ++i) { o.push(def); o.push(p[i]); }
  11684. return o;
  11685. }
  11686. function getdatastr(data) {
  11687. if(!data) return null;
  11688. if(data.data) return debom(data.data);
  11689. if(data.asNodeBuffer && has_buf) return debom(data.asNodeBuffer().toString('binary'));
  11690. if(data.asBinary) return debom(data.asBinary());
  11691. if(data._data && data._data.getContent) return debom(cc2str(Array.prototype.slice.call(data._data.getContent(),0)));
  11692. return null;
  11693. }
  11694. function getdatabin(data) {
  11695. if(!data) return null;
  11696. if(data.data) return char_codes(data.data);
  11697. if(data.asNodeBuffer && has_buf) return data.asNodeBuffer();
  11698. if(data._data && data._data.getContent) {
  11699. var o = data._data.getContent();
  11700. if(typeof o == "string") return char_codes(o);
  11701. return Array.prototype.slice.call(o);
  11702. }
  11703. return null;
  11704. }
  11705. function getdata(data) { return (data && data.name.slice(-4) === ".bin") ? getdatabin(data) : getdatastr(data); }
  11706. /* Part 2 Section 10.1.2 "Mapping Content Types" Names are case-insensitive */
  11707. /* OASIS does not comment on filename case sensitivity */
  11708. function safegetzipfile(zip, file) {
  11709. var k = keys(zip.files);
  11710. var f = file.toLowerCase(), g = f.replace(/\//g,'\\');
  11711. for(var i=0; i<k.length; ++i) {
  11712. var n = k[i].toLowerCase();
  11713. if(f == n || g == n) return zip.files[k[i]];
  11714. }
  11715. return null;
  11716. }
  11717. function getzipfile(zip, file) {
  11718. var o = safegetzipfile(zip, file);
  11719. if(o == null) throw new Error("Cannot find file " + file + " in zip");
  11720. return o;
  11721. }
  11722. function getzipdata(zip, file, safe) {
  11723. if(!safe) return getdata(getzipfile(zip, file));
  11724. if(!file) return null;
  11725. try { return getzipdata(zip, file); } catch(e) { return null; }
  11726. }
  11727. function getzipstr(zip, file, safe) {
  11728. if(!safe) return getdatastr(getzipfile(zip, file));
  11729. if(!file) return null;
  11730. try { return getzipstr(zip, file); } catch(e) { return null; }
  11731. }
  11732. function zipentries(zip) {
  11733. var k = keys(zip.files), o = [];
  11734. for(var i = 0; i < k.length; ++i) if(k[i].slice(-1) != '/') o.push(k[i]);
  11735. return o.sort();
  11736. }
  11737. var jszip;
  11738. /*global JSZipSync:true */
  11739. if(typeof JSZipSync !== 'undefined') jszip = JSZipSync;
  11740. if(typeof exports !== 'undefined') {
  11741. if(typeof module !== 'undefined' && module.exports) {
  11742. if(typeof jszip === 'undefined') jszip = undefined;
  11743. }
  11744. }
  11745. function resolve_path(path, base) {
  11746. var result = base.split('/');
  11747. if(base.slice(-1) != "/") result.pop(); // folder path
  11748. var target = path.split('/');
  11749. while (target.length !== 0) {
  11750. var step = target.shift();
  11751. if (step === '..') result.pop();
  11752. else if (step !== '.') result.push(step);
  11753. }
  11754. return result.join('/');
  11755. }
  11756. var XML_HEADER = '<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\r\n';
  11757. var attregexg=/([^"\s?>\/]+)\s*=\s*((?:")([^"]*)(?:")|(?:')([^']*)(?:')|([^'">\s]+))/g;
  11758. var tagregex=/<[\/\?]?[a-zA-Z0-9:]+(?:\s+[^"\s?>\/]+\s*=\s*(?:"[^"]*"|'[^']*'|[^'">\s=]+))*\s?[\/\?]?>/g;
  11759. if(!(XML_HEADER.match(tagregex))) tagregex = /<[^>]*>/g;
  11760. var nsregex=/<\w*:/, nsregex2 = /<(\/?)\w+:/;
  11761. function parsexmltag(tag, skip_root) {
  11762. var z = ({});
  11763. var eq = 0, c = 0;
  11764. for(; eq !== tag.length; ++eq) if((c = tag.charCodeAt(eq)) === 32 || c === 10 || c === 13) break;
  11765. if(!skip_root) z[0] = tag.slice(0, eq);
  11766. if(eq === tag.length) return z;
  11767. var m = tag.match(attregexg), j=0, v="", i=0, q="", cc="", quot = 1;
  11768. if(m) for(i = 0; i != m.length; ++i) {
  11769. cc = m[i];
  11770. for(c=0; c != cc.length; ++c) if(cc.charCodeAt(c) === 61) break;
  11771. q = cc.slice(0,c).trim();
  11772. while(cc.charCodeAt(c+1) == 32) ++c;
  11773. quot = ((eq=cc.charCodeAt(c+1)) == 34 || eq == 39) ? 1 : 0;
  11774. v = cc.slice(c+1+quot, cc.length-quot);
  11775. for(j=0;j!=q.length;++j) if(q.charCodeAt(j) === 58) break;
  11776. if(j===q.length) {
  11777. if(q.indexOf("_") > 0) q = q.slice(0, q.indexOf("_")); // from ods
  11778. z[q] = v;
  11779. z[q.toLowerCase()] = v;
  11780. }
  11781. else {
  11782. var k = (j===5 && q.slice(0,5)==="xmlns"?"xmlns":"")+q.slice(j+1);
  11783. if(z[k] && q.slice(j-3,j) == "ext") continue; // from ods
  11784. z[k] = v;
  11785. z[k.toLowerCase()] = v;
  11786. }
  11787. }
  11788. return z;
  11789. }
  11790. function strip_ns(x) { return x.replace(nsregex2, "<$1"); }
  11791. var encodings = {
  11792. '&quot;': '"',
  11793. '&apos;': "'",
  11794. '&gt;': '>',
  11795. '&lt;': '<',
  11796. '&amp;': '&'
  11797. };
  11798. var rencoding = evert(encodings);
  11799. //var rencstr = "&<>'\"".split("");
  11800. // TODO: CP remap (need to read file version to determine OS)
  11801. var unescapexml = (function() {
  11802. /* 22.4.2.4 bstr (Basic String) */
  11803. var encregex = /&(?:quot|apos|gt|lt|amp|#x?([\da-fA-F]+));/g, coderegex = /_x([\da-fA-F]{4})_/g;
  11804. return function unescapexml(text) {
  11805. var s = text + '', i = s.indexOf("<![CDATA[");
  11806. if(i == -1) return s.replace(encregex, function($$, $1) { return encodings[$$]||String.fromCharCode(parseInt($1,$$.indexOf("x")>-1?16:10))||$$; }).replace(coderegex,function(m,c) {return String.fromCharCode(parseInt(c,16));});
  11807. var j = s.indexOf("]]>");
  11808. return unescapexml(s.slice(0, i)) + s.slice(i+9,j) + unescapexml(s.slice(j+3));
  11809. };
  11810. })();
  11811. var decregex=/[&<>'"]/g, charegex = /[\u0000-\u0008\u000b-\u001f]/g;
  11812. function escapexml(text){
  11813. var s = text + '';
  11814. return s.replace(decregex, function(y) { return rencoding[y]; }).replace(charegex,function(s) { return "_x" + ("000"+s.charCodeAt(0).toString(16)).slice(-4) + "_";});
  11815. }
  11816. function escapexmltag(text){ return escapexml(text).replace(/ /g,"_x0020_"); }
  11817. var htmlcharegex = /[\u0000-\u001f]/g;
  11818. function escapehtml(text){
  11819. var s = text + '';
  11820. return s.replace(decregex, function(y) { return rencoding[y]; }).replace(/\n/g, "<br/>").replace(htmlcharegex,function(s) { return "&#x" + ("000"+s.charCodeAt(0).toString(16)).slice(-4) + ";"; });
  11821. }
  11822. function escapexlml(text){
  11823. var s = text + '';
  11824. return s.replace(decregex, function(y) { return rencoding[y]; }).replace(htmlcharegex,function(s) { return "&#x" + (s.charCodeAt(0).toString(16)).toUpperCase() + ";"; });
  11825. }
  11826. /* TODO: handle codepages */
  11827. var xlml_fixstr = (function() {
  11828. var entregex = /&#(\d+);/g;
  11829. function entrepl($$,$1) { return String.fromCharCode(parseInt($1,10)); }
  11830. return function xlml_fixstr(str) { return str.replace(entregex,entrepl); };
  11831. })();
  11832. var xlml_unfixstr = (function() {
  11833. return function xlml_unfixstr(str) { return str.replace(/(\r\n|[\r\n])/g,"\&#10;"); };
  11834. })();
  11835. function parsexmlbool(value) {
  11836. switch(value) {
  11837. case 1: case true: case '1': case 'true': case 'TRUE': return true;
  11838. /* case '0': case 'false': case 'FALSE':*/
  11839. default: return false;
  11840. }
  11841. }
  11842. var utf8read = function utf8reada(orig) {
  11843. var out = "", i = 0, c = 0, d = 0, e = 0, f = 0, w = 0;
  11844. while (i < orig.length) {
  11845. c = orig.charCodeAt(i++);
  11846. if (c < 128) { out += String.fromCharCode(c); continue; }
  11847. d = orig.charCodeAt(i++);
  11848. if (c>191 && c<224) { f = ((c & 31) << 6); f |= (d & 63); out += String.fromCharCode(f); continue; }
  11849. e = orig.charCodeAt(i++);
  11850. if (c < 240) { out += String.fromCharCode(((c & 15) << 12) | ((d & 63) << 6) | (e & 63)); continue; }
  11851. f = orig.charCodeAt(i++);
  11852. w = (((c & 7) << 18) | ((d & 63) << 12) | ((e & 63) << 6) | (f & 63))-65536;
  11853. out += String.fromCharCode(0xD800 + ((w>>>10)&1023));
  11854. out += String.fromCharCode(0xDC00 + (w&1023));
  11855. }
  11856. return out;
  11857. };
  11858. var utf8write = function(orig) {
  11859. var out = [], i = 0, c = 0, d = 0;
  11860. while(i < orig.length) {
  11861. c = orig.charCodeAt(i++);
  11862. switch(true) {
  11863. case c < 128: out.push(String.fromCharCode(c)); break;
  11864. case c < 2048:
  11865. out.push(String.fromCharCode(192 + (c >> 6)));
  11866. out.push(String.fromCharCode(128 + (c & 63)));
  11867. break;
  11868. case c >= 55296 && c < 57344:
  11869. c -= 55296; d = orig.charCodeAt(i++) - 56320 + (c<<10);
  11870. out.push(String.fromCharCode(240 + ((d >>18) & 7)));
  11871. out.push(String.fromCharCode(144 + ((d >>12) & 63)));
  11872. out.push(String.fromCharCode(128 + ((d >> 6) & 63)));
  11873. out.push(String.fromCharCode(128 + (d & 63)));
  11874. break;
  11875. default:
  11876. out.push(String.fromCharCode(224 + (c >> 12)));
  11877. out.push(String.fromCharCode(128 + ((c >> 6) & 63)));
  11878. out.push(String.fromCharCode(128 + (c & 63)));
  11879. }
  11880. }
  11881. return out.join("");
  11882. };
  11883. if(has_buf) {
  11884. var utf8readb = function utf8readb(data) {
  11885. var out = Buffer.alloc(2*data.length), w, i, j = 1, k = 0, ww=0, c;
  11886. for(i = 0; i < data.length; i+=j) {
  11887. j = 1;
  11888. if((c=data.charCodeAt(i)) < 128) w = c;
  11889. else if(c < 224) { w = (c&31)*64+(data.charCodeAt(i+1)&63); j=2; }
  11890. else if(c < 240) { w=(c&15)*4096+(data.charCodeAt(i+1)&63)*64+(data.charCodeAt(i+2)&63); j=3; }
  11891. else { j = 4;
  11892. w = (c & 7)*262144+(data.charCodeAt(i+1)&63)*4096+(data.charCodeAt(i+2)&63)*64+(data.charCodeAt(i+3)&63);
  11893. w -= 65536; ww = 0xD800 + ((w>>>10)&1023); w = 0xDC00 + (w&1023);
  11894. }
  11895. if(ww !== 0) { out[k++] = ww&255; out[k++] = ww>>>8; ww = 0; }
  11896. out[k++] = w%256; out[k++] = w>>>8;
  11897. }
  11898. return out.slice(0,k).toString('ucs2');
  11899. };
  11900. var corpus = "foo bar baz\u00e2\u0098\u0083\u00f0\u009f\u008d\u00a3";
  11901. if(utf8read(corpus) == utf8readb(corpus)) utf8read = utf8readb;
  11902. // $FlowIgnore
  11903. var utf8readc = function utf8readc(data) { return Buffer_from(data, 'binary').toString('utf8'); };
  11904. if(utf8read(corpus) == utf8readc(corpus)) utf8read = utf8readc;
  11905. // $FlowIgnore
  11906. utf8write = function(data) { return Buffer_from(data, 'utf8').toString("binary"); };
  11907. }
  11908. // matches <foo>...</foo> extracts content
  11909. var matchtag = (function() {
  11910. var mtcache = ({});
  11911. return function matchtag(f,g) {
  11912. var t = f+"|"+(g||"");
  11913. if(mtcache[t]) return mtcache[t];
  11914. return (mtcache[t] = new RegExp('<(?:\\w+:)?'+f+'(?: xml:space="preserve")?(?:[^>]*)>([\\s\\S]*?)</(?:\\w+:)?'+f+'>',((g||""))));
  11915. };
  11916. })();
  11917. var htmldecode = (function() {
  11918. var entities = [
  11919. ['nbsp', ' '], ['middot', '·'],
  11920. ['quot', '"'], ['apos', "'"], ['gt', '>'], ['lt', '<'], ['amp', '&']
  11921. ].map(function(x) { return [new RegExp('&' + x[0] + ';', "g"), x[1]]; });
  11922. return function htmldecode(str) {
  11923. var o = str.replace(/^[\t\n\r ]+/, "").replace(/[\t\n\r ]+$/,"").replace(/[\t\n\r ]+/g, " ").replace(/<\s*[bB][rR]\s*\/?>/g,"\n").replace(/<[^>]*>/g,"");
  11924. for(var i = 0; i < entities.length; ++i) o = o.replace(entities[i][0], entities[i][1]);
  11925. return o;
  11926. };
  11927. })();
  11928. var vtregex = (function(){ var vt_cache = {};
  11929. return function vt_regex(bt) {
  11930. if(vt_cache[bt] !== undefined) return vt_cache[bt];
  11931. return (vt_cache[bt] = new RegExp("<(?:vt:)?" + bt + ">([\\s\\S]*?)</(?:vt:)?" + bt + ">", 'g') );
  11932. };})();
  11933. var vtvregex = /<\/?(?:vt:)?variant>/g, vtmregex = /<(?:vt:)([^>]*)>([\s\S]*)</;
  11934. function parseVector(data, opts) {
  11935. var h = parsexmltag(data);
  11936. var matches = data.match(vtregex(h.baseType))||[];
  11937. var res = [];
  11938. if(matches.length != h.size) {
  11939. if(opts.WTF) throw new Error("unexpected vector length " + matches.length + " != " + h.size);
  11940. return res;
  11941. }
  11942. matches.forEach(function(x) {
  11943. var v = x.replace(vtvregex,"").match(vtmregex);
  11944. if(v) res.push({v:utf8read(v[2]), t:v[1]});
  11945. });
  11946. return res;
  11947. }
  11948. var wtregex = /(^\s|\s$|\n)/;
  11949. function writetag(f,g) { return '<' + f + (g.match(wtregex)?' xml:space="preserve"' : "") + '>' + g + '</' + f + '>'; }
  11950. function wxt_helper(h) { return keys(h).map(function(k) { return " " + k + '="' + h[k] + '"';}).join(""); }
  11951. function writextag(f,g,h) { return '<' + f + ((h != null) ? wxt_helper(h) : "") + ((g != null) ? (g.match(wtregex)?' xml:space="preserve"' : "") + '>' + g + '</' + f : "/") + '>';}
  11952. function write_w3cdtf(d, t) { try { return d.toISOString().replace(/\.\d*/,""); } catch(e) { if(t) throw e; } return ""; }
  11953. function write_vt(s) {
  11954. switch(typeof s) {
  11955. case 'string': return writextag('vt:lpwstr', s);
  11956. case 'number': return writextag((s|0)==s?'vt:i4':'vt:r8', String(s));
  11957. case 'boolean': return writextag('vt:bool',s?'true':'false');
  11958. }
  11959. if(s instanceof Date) return writextag('vt:filetime', write_w3cdtf(s));
  11960. throw new Error("Unable to serialize " + s);
  11961. }
  11962. var XMLNS = ({
  11963. 'dc': 'http://purl.org/dc/elements/1.1/',
  11964. 'dcterms': 'http://purl.org/dc/terms/',
  11965. 'dcmitype': 'http://purl.org/dc/dcmitype/',
  11966. 'mx': 'http://schemas.microsoft.com/office/mac/excel/2008/main',
  11967. 'r': 'http://schemas.openxmlformats.org/officeDocument/2006/relationships',
  11968. 'sjs': 'http://schemas.openxmlformats.org/package/2006/sheetjs/core-properties',
  11969. 'vt': 'http://schemas.openxmlformats.org/officeDocument/2006/docPropsVTypes',
  11970. 'xsi': 'http://www.w3.org/2001/XMLSchema-instance',
  11971. 'xsd': 'http://www.w3.org/2001/XMLSchema'
  11972. });
  11973. XMLNS.main = [
  11974. 'http://schemas.openxmlformats.org/spreadsheetml/2006/main',
  11975. 'http://purl.oclc.org/ooxml/spreadsheetml/main',
  11976. 'http://schemas.microsoft.com/office/excel/2006/main',
  11977. 'http://schemas.microsoft.com/office/excel/2006/2'
  11978. ];
  11979. var XLMLNS = ({
  11980. 'o': 'urn:schemas-microsoft-com:office:office',
  11981. 'x': 'urn:schemas-microsoft-com:office:excel',
  11982. 'ss': 'urn:schemas-microsoft-com:office:spreadsheet',
  11983. 'dt': 'uuid:C2F41010-65B3-11d1-A29F-00AA00C14882',
  11984. 'mv': 'http://macVmlSchemaUri',
  11985. 'v': 'urn:schemas-microsoft-com:vml',
  11986. 'html': 'http://www.w3.org/TR/REC-html40'
  11987. });
  11988. function read_double_le(b, idx) {
  11989. var s = 1 - 2 * (b[idx + 7] >>> 7);
  11990. var e = ((b[idx + 7] & 0x7f) << 4) + ((b[idx + 6] >>> 4) & 0x0f);
  11991. var m = (b[idx+6]&0x0f);
  11992. for(var i = 5; i >= 0; --i) m = m * 256 + b[idx + i];
  11993. if(e == 0x7ff) return m == 0 ? (s * Infinity) : NaN;
  11994. if(e == 0) e = -1022;
  11995. else { e -= 1023; m += Math.pow(2,52); }
  11996. return s * Math.pow(2, e - 52) * m;
  11997. }
  11998. function write_double_le(b, v, idx) {
  11999. var bs = ((((v < 0) || (1/v == -Infinity)) ? 1 : 0) << 7), e = 0, m = 0;
  12000. var av = bs ? (-v) : v;
  12001. if(!isFinite(av)) { e = 0x7ff; m = isNaN(v) ? 0x6969 : 0; }
  12002. else if(av == 0) e = m = 0;
  12003. else {
  12004. e = Math.floor(Math.log(av) / Math.LN2);
  12005. m = av * Math.pow(2, 52 - e);
  12006. if((e <= -1023) && (!isFinite(m) || (m < Math.pow(2,52)))) { e = -1022; }
  12007. else { m -= Math.pow(2,52); e+=1023; }
  12008. }
  12009. for(var i = 0; i <= 5; ++i, m/=256) b[idx + i] = m & 0xff;
  12010. b[idx + 6] = ((e & 0x0f) << 4) | (m & 0xf);
  12011. b[idx + 7] = (e >> 4) | bs;
  12012. }
  12013. var __toBuffer = function(bufs) { var x=[],w=10240; for(var i=0;i<bufs[0].length;++i) if(bufs[0][i]) for(var j=0,L=bufs[0][i].length;j<L;j+=w) x.push.apply(x, bufs[0][i].slice(j,j+w)); return x; };
  12014. var ___toBuffer = __toBuffer;
  12015. var __utf16le = function(b,s,e) { var ss=[]; for(var i=s; i<e; i+=2) ss.push(String.fromCharCode(__readUInt16LE(b,i))); return ss.join("").replace(chr0,''); };
  12016. var ___utf16le = __utf16le;
  12017. var __hexlify = function(b,s,l) { var ss=[]; for(var i=s; i<s+l; ++i) ss.push(("0" + b[i].toString(16)).slice(-2)); return ss.join(""); };
  12018. var ___hexlify = __hexlify;
  12019. var __utf8 = function(b,s,e) { var ss=[]; for(var i=s; i<e; i++) ss.push(String.fromCharCode(__readUInt8(b,i))); return ss.join(""); };
  12020. var ___utf8 = __utf8;
  12021. var __lpstr = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? __utf8(b, i+4,i+4+len-1) : "";};
  12022. var ___lpstr = __lpstr;
  12023. var __cpstr = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? __utf8(b, i+4,i+4+len-1) : "";};
  12024. var ___cpstr = __cpstr;
  12025. var __lpwstr = function(b,i) { var len = 2*__readUInt32LE(b,i); return len > 0 ? __utf8(b, i+4,i+4+len-1) : "";};
  12026. var ___lpwstr = __lpwstr;
  12027. var __lpp4, ___lpp4;
  12028. __lpp4 = ___lpp4 = function lpp4_(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? __utf16le(b, i+4,i+4+len) : "";};
  12029. var __8lpp4 = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? __utf8(b, i+4,i+4+len) : "";};
  12030. var ___8lpp4 = __8lpp4;
  12031. var __double, ___double;
  12032. __double = ___double = function(b, idx) { return read_double_le(b, idx);};
  12033. var is_buf = function is_buf_a(a) { return Array.isArray(a); };
  12034. if(has_buf) {
  12035. __utf16le = function(b,s,e) { if(!Buffer.isBuffer(b)) return ___utf16le(b,s,e); return b.toString('utf16le',s,e).replace(chr0,'')/*.replace(chr1,'!')*/; };
  12036. __hexlify = function(b,s,l) { return Buffer.isBuffer(b) ? b.toString('hex',s,s+l) : ___hexlify(b,s,l); };
  12037. __lpstr = function lpstr_b(b, i) { if(!Buffer.isBuffer(b)) return ___lpstr(b, i); var len = b.readUInt32LE(i); return len > 0 ? b.toString('utf8',i+4,i+4+len-1) : "";};
  12038. __cpstr = function cpstr_b(b, i) { if(!Buffer.isBuffer(b)) return ___cpstr(b, i); var len = b.readUInt32LE(i); return len > 0 ? b.toString('utf8',i+4,i+4+len-1) : "";};
  12039. __lpwstr = function lpwstr_b(b, i) { if(!Buffer.isBuffer(b)) return ___lpwstr(b, i); var len = 2*b.readUInt32LE(i); return b.toString('utf16le',i+4,i+4+len-1);};
  12040. __lpp4 = function lpp4_b(b, i) { if(!Buffer.isBuffer(b)) return ___lpp4(b, i); var len = b.readUInt32LE(i); return b.toString('utf16le',i+4,i+4+len);};
  12041. __8lpp4 = function lpp4_8b(b, i) { if(!Buffer.isBuffer(b)) return ___8lpp4(b, i); var len = b.readUInt32LE(i); return b.toString('utf8',i+4,i+4+len);};
  12042. __utf8 = function utf8_b(b, s, e) { return (Buffer.isBuffer(b)) ? b.toString('utf8',s,e) : ___utf8(b,s,e); };
  12043. __toBuffer = function(bufs) { return (bufs[0].length > 0 && Buffer.isBuffer(bufs[0][0])) ? Buffer.concat(bufs[0]) : ___toBuffer(bufs);};
  12044. bconcat = function(bufs) { return Buffer.isBuffer(bufs[0]) ? Buffer.concat(bufs) : [].concat.apply([], bufs); };
  12045. __double = function double_(b, i) { if(Buffer.isBuffer(b)) return b.readDoubleLE(i); return ___double(b,i); };
  12046. is_buf = function is_buf_b(a) { return Buffer.isBuffer(a) || Array.isArray(a); };
  12047. }
  12048. /* from js-xls */
  12049. if(typeof cptable !== 'undefined') {
  12050. __utf16le = function(b,s,e) { return cptable.utils.decode(1200, b.slice(s,e)).replace(chr0, ''); };
  12051. __utf8 = function(b,s,e) { return cptable.utils.decode(65001, b.slice(s,e)); };
  12052. __lpstr = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? cptable.utils.decode(current_ansi, b.slice(i+4, i+4+len-1)) : "";};
  12053. __cpstr = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? cptable.utils.decode(current_codepage, b.slice(i+4, i+4+len-1)) : "";};
  12054. __lpwstr = function(b,i) { var len = 2*__readUInt32LE(b,i); return len > 0 ? cptable.utils.decode(1200, b.slice(i+4,i+4+len-1)) : "";};
  12055. __lpp4 = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? cptable.utils.decode(1200, b.slice(i+4,i+4+len)) : "";};
  12056. __8lpp4 = function(b,i) { var len = __readUInt32LE(b,i); return len > 0 ? cptable.utils.decode(65001, b.slice(i+4,i+4+len)) : "";};
  12057. }
  12058. var __readUInt8 = function(b, idx) { return b[idx]; };
  12059. var __readUInt16LE = function(b, idx) { return (b[idx+1]*(1<<8))+b[idx]; };
  12060. var __readInt16LE = function(b, idx) { var u = (b[idx+1]*(1<<8))+b[idx]; return (u < 0x8000) ? u : ((0xffff - u + 1) * -1); };
  12061. var __readUInt32LE = function(b, idx) { return b[idx+3]*(1<<24)+(b[idx+2]<<16)+(b[idx+1]<<8)+b[idx]; };
  12062. var __readInt32LE = function(b, idx) { return (b[idx+3]<<24)|(b[idx+2]<<16)|(b[idx+1]<<8)|b[idx]; };
  12063. var __readInt32BE = function(b, idx) { return (b[idx]<<24)|(b[idx+1]<<16)|(b[idx+2]<<8)|b[idx+3]; };
  12064. function ReadShift(size, t) {
  12065. var o="", oI, oR, oo=[], w, vv, i, loc;
  12066. switch(t) {
  12067. case 'dbcs':
  12068. loc = this.l;
  12069. if(has_buf && Buffer.isBuffer(this)) o = this.slice(this.l, this.l+2*size).toString("utf16le");
  12070. else for(i = 0; i < size; ++i) { o+=String.fromCharCode(__readUInt16LE(this, loc)); loc+=2; }
  12071. size *= 2;
  12072. break;
  12073. case 'utf8': o = __utf8(this, this.l, this.l + size); break;
  12074. case 'utf16le': size *= 2; o = __utf16le(this, this.l, this.l + size); break;
  12075. case 'wstr':
  12076. if(typeof cptable !== 'undefined') o = cptable.utils.decode(current_codepage, this.slice(this.l, this.l+2*size));
  12077. else return ReadShift.call(this, size, 'dbcs');
  12078. size = 2 * size; break;
  12079. /* [MS-OLEDS] 2.1.4 LengthPrefixedAnsiString */
  12080. case 'lpstr-ansi': o = __lpstr(this, this.l); size = 4 + __readUInt32LE(this, this.l); break;
  12081. case 'lpstr-cp': o = __cpstr(this, this.l); size = 4 + __readUInt32LE(this, this.l); break;
  12082. /* [MS-OLEDS] 2.1.5 LengthPrefixedUnicodeString */
  12083. case 'lpwstr': o = __lpwstr(this, this.l); size = 4 + 2 * __readUInt32LE(this, this.l); break;
  12084. /* [MS-OFFCRYPTO] 2.1.2 Length-Prefixed Padded Unicode String (UNICODE-LP-P4) */
  12085. case 'lpp4': size = 4 + __readUInt32LE(this, this.l); o = __lpp4(this, this.l); if(size & 0x02) size += 2; break;
  12086. /* [MS-OFFCRYPTO] 2.1.3 Length-Prefixed UTF-8 String (UTF-8-LP-P4) */
  12087. case '8lpp4': size = 4 + __readUInt32LE(this, this.l); o = __8lpp4(this, this.l); if(size & 0x03) size += 4 - (size & 0x03); break;
  12088. case 'cstr': size = 0; o = "";
  12089. while((w=__readUInt8(this, this.l + size++))!==0) oo.push(_getchar(w));
  12090. o = oo.join(""); break;
  12091. case '_wstr': size = 0; o = "";
  12092. while((w=__readUInt16LE(this,this.l +size))!==0){oo.push(_getchar(w));size+=2;}
  12093. size+=2; o = oo.join(""); break;
  12094. /* sbcs and dbcs support continue records in the SST way TODO codepages */
  12095. case 'dbcs-cont': o = ""; loc = this.l;
  12096. for(i = 0; i < size; ++i) {
  12097. if(this.lens && this.lens.indexOf(loc) !== -1) {
  12098. w = __readUInt8(this, loc);
  12099. this.l = loc + 1;
  12100. vv = ReadShift.call(this, size-i, w ? 'dbcs-cont' : 'sbcs-cont');
  12101. return oo.join("") + vv;
  12102. }
  12103. oo.push(_getchar(__readUInt16LE(this, loc)));
  12104. loc+=2;
  12105. } o = oo.join(""); size *= 2; break;
  12106. case 'cpstr':
  12107. if(typeof cptable !== 'undefined') {
  12108. o = cptable.utils.decode(current_codepage, this.slice(this.l, this.l + size));
  12109. break;
  12110. }
  12111. /* falls through */
  12112. case 'sbcs-cont': o = ""; loc = this.l;
  12113. for(i = 0; i != size; ++i) {
  12114. if(this.lens && this.lens.indexOf(loc) !== -1) {
  12115. w = __readUInt8(this, loc);
  12116. this.l = loc + 1;
  12117. vv = ReadShift.call(this, size-i, w ? 'dbcs-cont' : 'sbcs-cont');
  12118. return oo.join("") + vv;
  12119. }
  12120. oo.push(_getchar(__readUInt8(this, loc)));
  12121. loc+=1;
  12122. } o = oo.join(""); break;
  12123. default:
  12124. switch(size) {
  12125. case 1: oI = __readUInt8(this, this.l); this.l++; return oI;
  12126. case 2: oI = (t === 'i' ? __readInt16LE : __readUInt16LE)(this, this.l); this.l += 2; return oI;
  12127. case 4: case -4:
  12128. if(t === 'i' || ((this[this.l+3] & 0x80)===0)) { oI = ((size > 0) ? __readInt32LE : __readInt32BE)(this, this.l); this.l += 4; return oI; }
  12129. else { oR = __readUInt32LE(this, this.l); this.l += 4; } return oR;
  12130. case 8: case -8:
  12131. if(t === 'f') {
  12132. if(size == 8) oR = __double(this, this.l);
  12133. else oR = __double([this[this.l+7],this[this.l+6],this[this.l+5],this[this.l+4],this[this.l+3],this[this.l+2],this[this.l+1],this[this.l+0]], 0);
  12134. this.l += 8; return oR;
  12135. } else size = 8;
  12136. /* falls through */
  12137. case 16: o = __hexlify(this, this.l, size); break;
  12138. }}
  12139. this.l+=size; return o;
  12140. }
  12141. var __writeUInt32LE = function(b, val, idx) { b[idx] = (val & 0xFF); b[idx+1] = ((val >>> 8) & 0xFF); b[idx+2] = ((val >>> 16) & 0xFF); b[idx+3] = ((val >>> 24) & 0xFF); };
  12142. var __writeInt32LE = function(b, val, idx) { b[idx] = (val & 0xFF); b[idx+1] = ((val >> 8) & 0xFF); b[idx+2] = ((val >> 16) & 0xFF); b[idx+3] = ((val >> 24) & 0xFF); };
  12143. var __writeUInt16LE = function(b, val, idx) { b[idx] = (val & 0xFF); b[idx+1] = ((val >>> 8) & 0xFF); };
  12144. function WriteShift(t, val, f) {
  12145. var size = 0, i = 0;
  12146. if(f === 'dbcs') {
  12147. for(i = 0; i != val.length; ++i) __writeUInt16LE(this, val.charCodeAt(i), this.l + 2 * i);
  12148. size = 2 * val.length;
  12149. } else if(f === 'sbcs') {
  12150. /* TODO: codepage */
  12151. val = val.replace(/[^\x00-\x7F]/g, "_");
  12152. for(i = 0; i != val.length; ++i) this[this.l + i] = (val.charCodeAt(i) & 0xFF);
  12153. size = val.length;
  12154. } else if(f === 'hex') {
  12155. for(; i < t; ++i) {
  12156. this[this.l++] = (parseInt(val.slice(2*i, 2*i+2), 16)||0);
  12157. } return this;
  12158. } else if(f === 'utf16le') {
  12159. var end = Math.min(this.l + t, this.length);
  12160. for(i = 0; i < Math.min(val.length, t); ++i) {
  12161. var cc = val.charCodeAt(i);
  12162. this[this.l++] = (cc & 0xff);
  12163. this[this.l++] = (cc >> 8);
  12164. }
  12165. while(this.l < end) this[this.l++] = 0;
  12166. return this;
  12167. } else switch(t) {
  12168. case 1: size = 1; this[this.l] = val&0xFF; break;
  12169. case 2: size = 2; this[this.l] = val&0xFF; val >>>= 8; this[this.l+1] = val&0xFF; break;
  12170. case 3: size = 3; this[this.l] = val&0xFF; val >>>= 8; this[this.l+1] = val&0xFF; val >>>= 8; this[this.l+2] = val&0xFF; break;
  12171. case 4: size = 4; __writeUInt32LE(this, val, this.l); break;
  12172. case 8: size = 8; if(f === 'f') { write_double_le(this, val, this.l); break; }
  12173. /* falls through */
  12174. case 16: break;
  12175. case -4: size = 4; __writeInt32LE(this, val, this.l); break;
  12176. }
  12177. this.l += size; return this;
  12178. }
  12179. function CheckField(hexstr, fld) {
  12180. var m = __hexlify(this,this.l,hexstr.length>>1);
  12181. if(m !== hexstr) throw new Error(fld + 'Expected ' + hexstr + ' saw ' + m);
  12182. this.l += hexstr.length>>1;
  12183. }
  12184. function prep_blob(blob, pos) {
  12185. blob.l = pos;
  12186. blob.read_shift = ReadShift;
  12187. blob.chk = CheckField;
  12188. blob.write_shift = WriteShift;
  12189. }
  12190. function parsenoop(blob, length) { blob.l += length; }
  12191. function new_buf(sz) {
  12192. var o = new_raw_buf(sz);
  12193. prep_blob(o, 0);
  12194. return o;
  12195. }
  12196. /* [MS-XLSB] 2.1.4 Record */
  12197. function recordhopper(data, cb, opts) {
  12198. if(!data) return;
  12199. var tmpbyte, cntbyte, length;
  12200. prep_blob(data, data.l || 0);
  12201. var L = data.length, RT = 0, tgt = 0;
  12202. while(data.l < L) {
  12203. RT = data.read_shift(1);
  12204. if(RT & 0x80) RT = (RT & 0x7F) + ((data.read_shift(1) & 0x7F)<<7);
  12205. var R = XLSBRecordEnum[RT] || XLSBRecordEnum[0xFFFF];
  12206. tmpbyte = data.read_shift(1);
  12207. length = tmpbyte & 0x7F;
  12208. for(cntbyte = 1; cntbyte <4 && (tmpbyte & 0x80); ++cntbyte) length += ((tmpbyte = data.read_shift(1)) & 0x7F)<<(7*cntbyte);
  12209. tgt = data.l + length;
  12210. var d = (R.f||parsenoop)(data, length, opts);
  12211. data.l = tgt;
  12212. if(cb(d, R.n, RT)) return;
  12213. }
  12214. }
  12215. /* control buffer usage for fixed-length buffers */
  12216. function buf_array() {
  12217. var bufs = [], blksz = has_buf ? 256 : 2048;
  12218. var newblk = function ba_newblk(sz) {
  12219. var o = (new_buf(sz));
  12220. prep_blob(o, 0);
  12221. return o;
  12222. };
  12223. var curbuf = newblk(blksz);
  12224. var endbuf = function ba_endbuf() {
  12225. if(!curbuf) return;
  12226. if(curbuf.length > curbuf.l) { curbuf = curbuf.slice(0, curbuf.l); curbuf.l = curbuf.length; }
  12227. if(curbuf.length > 0) bufs.push(curbuf);
  12228. curbuf = null;
  12229. };
  12230. var next = function ba_next(sz) {
  12231. if(curbuf && (sz < (curbuf.length - curbuf.l))) return curbuf;
  12232. endbuf();
  12233. return (curbuf = newblk(Math.max(sz+1, blksz)));
  12234. };
  12235. var end = function ba_end() {
  12236. endbuf();
  12237. return __toBuffer([bufs]);
  12238. };
  12239. var push = function ba_push(buf) { endbuf(); curbuf = buf; if(curbuf.l == null) curbuf.l = curbuf.length; next(blksz); };
  12240. return ({ next:next, push:push, end:end, _bufs:bufs });
  12241. }
  12242. function write_record(ba, type, payload, length) {
  12243. var t = +XLSBRE[type], l;
  12244. if(isNaN(t)) return; // TODO: throw something here?
  12245. if(!length) length = XLSBRecordEnum[t].p || (payload||[]).length || 0;
  12246. l = 1 + (t >= 0x80 ? 1 : 0) + 1/* + length*/;
  12247. if(length >= 0x80) ++l; if(length >= 0x4000) ++l; if(length >= 0x200000) ++l;
  12248. var o = ba.next(l);
  12249. if(t <= 0x7F) o.write_shift(1, t);
  12250. else {
  12251. o.write_shift(1, (t & 0x7F) + 0x80);
  12252. o.write_shift(1, (t >> 7));
  12253. }
  12254. for(var i = 0; i != 4; ++i) {
  12255. if(length >= 0x80) { o.write_shift(1, (length & 0x7F)+0x80); length >>= 7; }
  12256. else { o.write_shift(1, length); break; }
  12257. }
  12258. if(length > 0 && is_buf(payload)) ba.push(payload);
  12259. }
  12260. /* XLS ranges enforced */
  12261. function shift_cell_xls(cell, tgt, opts) {
  12262. var out = dup(cell);
  12263. if(tgt.s) {
  12264. if(out.cRel) out.c += tgt.s.c;
  12265. if(out.rRel) out.r += tgt.s.r;
  12266. } else {
  12267. if(out.cRel) out.c += tgt.c;
  12268. if(out.rRel) out.r += tgt.r;
  12269. }
  12270. if(!opts || opts.biff < 12) {
  12271. while(out.c >= 0x100) out.c -= 0x100;
  12272. while(out.r >= 0x10000) out.r -= 0x10000;
  12273. }
  12274. return out;
  12275. }
  12276. function shift_range_xls(cell, range, opts) {
  12277. var out = dup(cell);
  12278. out.s = shift_cell_xls(out.s, range.s, opts);
  12279. out.e = shift_cell_xls(out.e, range.s, opts);
  12280. return out;
  12281. }
  12282. function encode_cell_xls(c, biff) {
  12283. if(c.cRel && c.c < 0) { c = dup(c); c.c += (biff > 8) ? 0x4000 : 0x100; }
  12284. if(c.rRel && c.r < 0) { c = dup(c); c.r += (biff > 8) ? 0x100000 : ((biff > 5) ? 0x10000 : 0x4000); }
  12285. var s = encode_cell(c);
  12286. if(c.cRel === 0) s = fix_col(s);
  12287. if(c.rRel === 0) s = fix_row(s);
  12288. return s;
  12289. }
  12290. function encode_range_xls(r, opts) {
  12291. if(r.s.r == 0 && !r.s.rRel) {
  12292. if(r.e.r == (opts.biff >= 12 ? 0xFFFFF : (opts.biff >= 8 ? 0x10000 : 0x4000)) && !r.e.rRel) {
  12293. return (r.s.cRel ? "" : "$") + encode_col(r.s.c) + ":" + (r.e.cRel ? "" : "$") + encode_col(r.e.c);
  12294. }
  12295. }
  12296. if(r.s.c == 0 && !r.s.cRel) {
  12297. if(r.e.c == (opts.biff >= 12 ? 0xFFFF : 0xFF) && !r.e.cRel) {
  12298. return (r.s.rRel ? "" : "$") + encode_row(r.s.r) + ":" + (r.e.rRel ? "" : "$") + encode_row(r.e.r);
  12299. }
  12300. }
  12301. return encode_cell_xls(r.s, opts.biff) + ":" + encode_cell_xls(r.e, opts.biff);
  12302. }
  12303. var OFFCRYPTO = {};
  12304. var make_offcrypto = function(O, _crypto) {
  12305. var crypto;
  12306. if(typeof _crypto !== 'undefined') crypto = _crypto;
  12307. else if(typeof require !== 'undefined') {
  12308. try { crypto = undefined; }
  12309. catch(e) { crypto = null; }
  12310. }
  12311. O.rc4 = function(key, data) {
  12312. var S = new Array(256);
  12313. var c = 0, i = 0, j = 0, t = 0;
  12314. for(i = 0; i != 256; ++i) S[i] = i;
  12315. for(i = 0; i != 256; ++i) {
  12316. j = (j + S[i] + (key[i%key.length]).charCodeAt(0))&255;
  12317. t = S[i]; S[i] = S[j]; S[j] = t;
  12318. }
  12319. // $FlowIgnore
  12320. i = j = 0; var out = Buffer(data.length);
  12321. for(c = 0; c != data.length; ++c) {
  12322. i = (i + 1)&255;
  12323. j = (j + S[i])%256;
  12324. t = S[i]; S[i] = S[j]; S[j] = t;
  12325. out[c] = (data[c] ^ S[(S[i]+S[j])&255]);
  12326. }
  12327. return out;
  12328. };
  12329. O.md5 = function(hex) {
  12330. if(!crypto) throw new Error("Unsupported crypto");
  12331. return crypto.createHash('md5').update(hex).digest('hex');
  12332. };
  12333. };
  12334. /*global crypto:true */
  12335. make_offcrypto(OFFCRYPTO, typeof crypto !== "undefined" ? crypto : undefined);
  12336. function decode_row(rowstr) { return parseInt(unfix_row(rowstr),10) - 1; }
  12337. function encode_row(row) { return "" + (row + 1); }
  12338. function fix_row(cstr) { return cstr.replace(/([A-Z]|^)(\d+)$/,"$1$$$2"); }
  12339. function unfix_row(cstr) { return cstr.replace(/\$(\d+)$/,"$1"); }
  12340. function decode_col(colstr) { var c = unfix_col(colstr), d = 0, i = 0; for(; i !== c.length; ++i) d = 26*d + c.charCodeAt(i) - 64; return d - 1; }
  12341. function encode_col(col) { var s=""; for(++col; col; col=Math.floor((col-1)/26)) s = String.fromCharCode(((col-1)%26) + 65) + s; return s; }
  12342. function fix_col(cstr) { return cstr.replace(/^([A-Z])/,"$$$1"); }
  12343. function unfix_col(cstr) { return cstr.replace(/^\$([A-Z])/,"$1"); }
  12344. function split_cell(cstr) { return cstr.replace(/(\$?[A-Z]*)(\$?\d*)/,"$1,$2").split(","); }
  12345. function decode_cell(cstr) { var splt = split_cell(cstr); return { c:decode_col(splt[0]), r:decode_row(splt[1]) }; }
  12346. function encode_cell(cell) { return encode_col(cell.c) + encode_row(cell.r); }
  12347. function decode_range(range) { var x =range.split(":").map(decode_cell); return {s:x[0],e:x[x.length-1]}; }
  12348. function encode_range(cs,ce) {
  12349. if(typeof ce === 'undefined' || typeof ce === 'number') {
  12350. return encode_range(cs.s, cs.e);
  12351. }
  12352. if(typeof cs !== 'string') cs = encode_cell((cs));
  12353. if(typeof ce !== 'string') ce = encode_cell((ce));
  12354. return cs == ce ? cs : cs + ":" + ce;
  12355. }
  12356. function safe_decode_range(range) {
  12357. var o = {s:{c:0,r:0},e:{c:0,r:0}};
  12358. var idx = 0, i = 0, cc = 0;
  12359. var len = range.length;
  12360. for(idx = 0; i < len; ++i) {
  12361. if((cc=range.charCodeAt(i)-64) < 1 || cc > 26) break;
  12362. idx = 26*idx + cc;
  12363. }
  12364. o.s.c = --idx;
  12365. for(idx = 0; i < len; ++i) {
  12366. if((cc=range.charCodeAt(i)-48) < 0 || cc > 9) break;
  12367. idx = 10*idx + cc;
  12368. }
  12369. o.s.r = --idx;
  12370. if(i === len || range.charCodeAt(++i) === 58) { o.e.c=o.s.c; o.e.r=o.s.r; return o; }
  12371. for(idx = 0; i != len; ++i) {
  12372. if((cc=range.charCodeAt(i)-64) < 1 || cc > 26) break;
  12373. idx = 26*idx + cc;
  12374. }
  12375. o.e.c = --idx;
  12376. for(idx = 0; i != len; ++i) {
  12377. if((cc=range.charCodeAt(i)-48) < 0 || cc > 9) break;
  12378. idx = 10*idx + cc;
  12379. }
  12380. o.e.r = --idx;
  12381. return o;
  12382. }
  12383. function safe_format_cell(cell, v) {
  12384. var q = (cell.t == 'd' && v instanceof Date);
  12385. if(cell.z != null) try { return (cell.w = SSF.format(cell.z, q ? datenum(v) : v)); } catch(e) { }
  12386. try { return (cell.w = SSF.format((cell.XF||{}).numFmtId||(q ? 14 : 0), q ? datenum(v) : v)); } catch(e) { return ''+v; }
  12387. }
  12388. function format_cell(cell, v, o) {
  12389. if(cell == null || cell.t == null || cell.t == 'z') return "";
  12390. if(cell.w !== undefined) return cell.w;
  12391. if(cell.t == 'd' && !cell.z && o && o.dateNF) cell.z = o.dateNF;
  12392. if(v == undefined) return safe_format_cell(cell, cell.v);
  12393. return safe_format_cell(cell, v);
  12394. }
  12395. function sheet_to_workbook(sheet, opts) {
  12396. var n = opts && opts.sheet ? opts.sheet : "Sheet1";
  12397. var sheets = {}; sheets[n] = sheet;
  12398. return { SheetNames: [n], Sheets: sheets };
  12399. }
  12400. function sheet_add_aoa(_ws, data, opts) {
  12401. var o = opts || {};
  12402. var dense = _ws ? Array.isArray(_ws) : o.dense;
  12403. if(DENSE != null && dense == null) dense = DENSE;
  12404. var ws = _ws || (dense ? ([]) : ({}));
  12405. var _R = 0, _C = 0;
  12406. if(ws && o.origin != null) {
  12407. if(typeof o.origin == 'number') _R = o.origin;
  12408. else {
  12409. var _origin = typeof o.origin == "string" ? decode_cell(o.origin) : o.origin;
  12410. _R = _origin.r; _C = _origin.c;
  12411. }
  12412. }
  12413. var range = ({s: {c:10000000, r:10000000}, e: {c:0, r:0}});
  12414. if(ws['!ref']) {
  12415. var _range = safe_decode_range(ws['!ref']);
  12416. range.s.c = _range.s.c;
  12417. range.s.r = _range.s.r;
  12418. range.e.c = Math.max(range.e.c, _range.e.c);
  12419. range.e.r = Math.max(range.e.r, _range.e.r);
  12420. if(_R == -1) range.e.r = _R = _range.e.r + 1;
  12421. }
  12422. for(var R = 0; R != data.length; ++R) {
  12423. if(!data[R]) continue;
  12424. if(!Array.isArray(data[R])) throw new Error("aoa_to_sheet expects an array of arrays");
  12425. for(var C = 0; C != data[R].length; ++C) {
  12426. if(typeof data[R][C] === 'undefined') continue;
  12427. var cell = ({v: data[R][C] });
  12428. var __R = _R + R, __C = _C + C;
  12429. if(range.s.r > __R) range.s.r = __R;
  12430. if(range.s.c > __C) range.s.c = __C;
  12431. if(range.e.r < __R) range.e.r = __R;
  12432. if(range.e.c < __C) range.e.c = __C;
  12433. if(data[R][C] && typeof data[R][C] === 'object' && !Array.isArray(data[R][C]) && !(data[R][C] instanceof Date)) cell = data[R][C];
  12434. else {
  12435. if(Array.isArray(cell.v)) { cell.f = data[R][C][1]; cell.v = cell.v[0]; }
  12436. if(cell.v === null) { if(cell.f) cell.t = 'n'; else if(!o.sheetStubs) continue; else cell.t = 'z'; }
  12437. else if(typeof cell.v === 'number') cell.t = 'n';
  12438. else if(typeof cell.v === 'boolean') cell.t = 'b';
  12439. else if(cell.v instanceof Date) {
  12440. cell.z = o.dateNF || SSF._table[14];
  12441. if(o.cellDates) { cell.t = 'd'; cell.w = SSF.format(cell.z, datenum(cell.v)); }
  12442. else { cell.t = 'n'; cell.v = datenum(cell.v); cell.w = SSF.format(cell.z, cell.v); }
  12443. }
  12444. else cell.t = 's';
  12445. }
  12446. if(dense) {
  12447. if(!ws[__R]) ws[__R] = [];
  12448. ws[__R][__C] = cell;
  12449. } else {
  12450. var cell_ref = encode_cell(({c:__C,r:__R}));
  12451. ws[cell_ref] = cell;
  12452. }
  12453. }
  12454. }
  12455. if(range.s.c < 10000000) ws['!ref'] = encode_range(range);
  12456. return ws;
  12457. }
  12458. function aoa_to_sheet(data, opts) { return sheet_add_aoa(null, data, opts); }
  12459. function write_UInt32LE(x, o) {
  12460. if(!o) o = new_buf(4);
  12461. o.write_shift(4, x);
  12462. return o;
  12463. }
  12464. /* [MS-XLSB] 2.5.168 */
  12465. function parse_XLWideString(data) {
  12466. var cchCharacters = data.read_shift(4);
  12467. return cchCharacters === 0 ? "" : data.read_shift(cchCharacters, 'dbcs');
  12468. }
  12469. function write_XLWideString(data, o) {
  12470. var _null = false; if(o == null) { _null = true; o = new_buf(4+2*data.length); }
  12471. o.write_shift(4, data.length);
  12472. if(data.length > 0) o.write_shift(0, data, 'dbcs');
  12473. return _null ? o.slice(0, o.l) : o;
  12474. }
  12475. /* [MS-XLSB] 2.5.143 */
  12476. function parse_StrRun(data) {
  12477. return { ich: data.read_shift(2), ifnt: data.read_shift(2) };
  12478. }
  12479. function write_StrRun(run, o) {
  12480. if(!o) o = new_buf(4);
  12481. o.write_shift(2, run.ich || 0);
  12482. o.write_shift(2, run.ifnt || 0);
  12483. return o;
  12484. }
  12485. /* [MS-XLSB] 2.5.121 */
  12486. function parse_RichStr(data, length) {
  12487. var start = data.l;
  12488. var flags = data.read_shift(1);
  12489. var str = parse_XLWideString(data);
  12490. var rgsStrRun = [];
  12491. var z = ({ t: str, h: str });
  12492. if((flags & 1) !== 0) { /* fRichStr */
  12493. /* TODO: formatted string */
  12494. var dwSizeStrRun = data.read_shift(4);
  12495. for(var i = 0; i != dwSizeStrRun; ++i) rgsStrRun.push(parse_StrRun(data));
  12496. z.r = rgsStrRun;
  12497. }
  12498. else z.r = [{ich:0, ifnt:0}];
  12499. //if((flags & 2) !== 0) { /* fExtStr */
  12500. // /* TODO: phonetic string */
  12501. //}
  12502. data.l = start + length;
  12503. return z;
  12504. }
  12505. function write_RichStr(str, o) {
  12506. /* TODO: formatted string */
  12507. var _null = false; if(o == null) { _null = true; o = new_buf(15+4*str.t.length); }
  12508. o.write_shift(1,0);
  12509. write_XLWideString(str.t, o);
  12510. return _null ? o.slice(0, o.l) : o;
  12511. }
  12512. /* [MS-XLSB] 2.4.328 BrtCommentText (RichStr w/1 run) */
  12513. var parse_BrtCommentText = parse_RichStr;
  12514. function write_BrtCommentText(str, o) {
  12515. /* TODO: formatted string */
  12516. var _null = false; if(o == null) { _null = true; o = new_buf(23+4*str.t.length); }
  12517. o.write_shift(1,1);
  12518. write_XLWideString(str.t, o);
  12519. o.write_shift(4,1);
  12520. write_StrRun({ich:0,ifnt:0}, o);
  12521. return _null ? o.slice(0, o.l) : o;
  12522. }
  12523. /* [MS-XLSB] 2.5.9 */
  12524. function parse_XLSBCell(data) {
  12525. var col = data.read_shift(4);
  12526. var iStyleRef = data.read_shift(2);
  12527. iStyleRef += data.read_shift(1) <<16;
  12528. data.l++; //var fPhShow = data.read_shift(1);
  12529. return { c:col, iStyleRef: iStyleRef };
  12530. }
  12531. function write_XLSBCell(cell, o) {
  12532. if(o == null) o = new_buf(8);
  12533. o.write_shift(-4, cell.c);
  12534. o.write_shift(3, cell.iStyleRef || cell.s);
  12535. o.write_shift(1, 0); /* fPhShow */
  12536. return o;
  12537. }
  12538. /* [MS-XLSB] 2.5.21 */
  12539. var parse_XLSBCodeName = parse_XLWideString;
  12540. var write_XLSBCodeName = write_XLWideString;
  12541. /* [MS-XLSB] 2.5.166 */
  12542. function parse_XLNullableWideString(data) {
  12543. var cchCharacters = data.read_shift(4);
  12544. return cchCharacters === 0 || cchCharacters === 0xFFFFFFFF ? "" : data.read_shift(cchCharacters, 'dbcs');
  12545. }
  12546. function write_XLNullableWideString(data, o) {
  12547. var _null = false; if(o == null) { _null = true; o = new_buf(127); }
  12548. o.write_shift(4, data.length > 0 ? data.length : 0xFFFFFFFF);
  12549. if(data.length > 0) o.write_shift(0, data, 'dbcs');
  12550. return _null ? o.slice(0, o.l) : o;
  12551. }
  12552. /* [MS-XLSB] 2.5.165 */
  12553. var parse_XLNameWideString = parse_XLWideString;
  12554. //var write_XLNameWideString = write_XLWideString;
  12555. /* [MS-XLSB] 2.5.114 */
  12556. var parse_RelID = parse_XLNullableWideString;
  12557. var write_RelID = write_XLNullableWideString;
  12558. /* [MS-XLS] 2.5.217 ; [MS-XLSB] 2.5.122 */
  12559. function parse_RkNumber(data) {
  12560. var b = data.slice(data.l, data.l+4);
  12561. var fX100 = (b[0] & 1), fInt = (b[0] & 2);
  12562. data.l+=4;
  12563. b[0] &= 0xFC; // b[0] &= ~3;
  12564. var RK = fInt === 0 ? __double([0,0,0,0,b[0],b[1],b[2],b[3]],0) : __readInt32LE(b,0)>>2;
  12565. return fX100 ? (RK/100) : RK;
  12566. }
  12567. function write_RkNumber(data, o) {
  12568. if(o == null) o = new_buf(4);
  12569. var fX100 = 0, fInt = 0, d100 = data * 100;
  12570. if((data == (data | 0)) && (data >= -(1<<29)) && (data < (1 << 29))) { fInt = 1; }
  12571. else if((d100 == (d100 | 0)) && (d100 >= -(1<<29)) && (d100 < (1 << 29))) { fInt = 1; fX100 = 1; }
  12572. if(fInt) o.write_shift(-4, ((fX100 ? d100 : data) << 2) + (fX100 + 2));
  12573. else throw new Error("unsupported RkNumber " + data); // TODO
  12574. }
  12575. /* [MS-XLSB] 2.5.117 RfX */
  12576. function parse_RfX(data ) {
  12577. var cell = ({s: {}, e: {}});
  12578. cell.s.r = data.read_shift(4);
  12579. cell.e.r = data.read_shift(4);
  12580. cell.s.c = data.read_shift(4);
  12581. cell.e.c = data.read_shift(4);
  12582. return cell;
  12583. }
  12584. function write_RfX(r, o) {
  12585. if(!o) o = new_buf(16);
  12586. o.write_shift(4, r.s.r);
  12587. o.write_shift(4, r.e.r);
  12588. o.write_shift(4, r.s.c);
  12589. o.write_shift(4, r.e.c);
  12590. return o;
  12591. }
  12592. /* [MS-XLSB] 2.5.153 UncheckedRfX */
  12593. var parse_UncheckedRfX = parse_RfX;
  12594. var write_UncheckedRfX = write_RfX;
  12595. /* [MS-XLS] 2.5.342 ; [MS-XLSB] 2.5.171 */
  12596. /* TODO: error checking, NaN and Infinity values are not valid Xnum */
  12597. function parse_Xnum(data) { return data.read_shift(8, 'f'); }
  12598. function write_Xnum(data, o) { return (o || new_buf(8)).write_shift(8, data, 'f'); }
  12599. /* [MS-XLSB] 2.5.97.2 */
  12600. var BErr = {
  12601. 0x00: "#NULL!",
  12602. 0x07: "#DIV/0!",
  12603. 0x0F: "#VALUE!",
  12604. 0x17: "#REF!",
  12605. 0x1D: "#NAME?",
  12606. 0x24: "#NUM!",
  12607. 0x2A: "#N/A",
  12608. 0x2B: "#GETTING_DATA",
  12609. 0xFF: "#WTF?"
  12610. };
  12611. var RBErr = evert_num(BErr);
  12612. /* [MS-XLSB] 2.4.324 BrtColor */
  12613. function parse_BrtColor(data) {
  12614. var out = {};
  12615. var d = data.read_shift(1);
  12616. //var fValidRGB = d & 1;
  12617. var xColorType = d >>> 1;
  12618. var index = data.read_shift(1);
  12619. var nTS = data.read_shift(2, 'i');
  12620. var bR = data.read_shift(1);
  12621. var bG = data.read_shift(1);
  12622. var bB = data.read_shift(1);
  12623. data.l++; //var bAlpha = data.read_shift(1);
  12624. switch(xColorType) {
  12625. case 0: out.auto = 1; break;
  12626. case 1:
  12627. out.index = index;
  12628. var icv = XLSIcv[index];
  12629. /* automatic pseudo index 81 */
  12630. if(icv) out.rgb = rgb2Hex(icv);
  12631. break;
  12632. case 2:
  12633. /* if(!fValidRGB) throw new error("invalid"); */
  12634. out.rgb = rgb2Hex([bR, bG, bB]);
  12635. break;
  12636. case 3: out.theme = index; break;
  12637. }
  12638. if(nTS != 0) out.tint = nTS > 0 ? nTS / 32767 : nTS / 32768;
  12639. return out;
  12640. }
  12641. function write_BrtColor(color, o) {
  12642. if(!o) o = new_buf(8);
  12643. if(!color||color.auto) { o.write_shift(4, 0); o.write_shift(4, 0); return o; }
  12644. if(color.index) {
  12645. o.write_shift(1, 0x02);
  12646. o.write_shift(1, color.index);
  12647. } else if(color.theme) {
  12648. o.write_shift(1, 0x06);
  12649. o.write_shift(1, color.theme);
  12650. } else {
  12651. o.write_shift(1, 0x05);
  12652. o.write_shift(1, 0);
  12653. }
  12654. var nTS = color.tint || 0;
  12655. if(nTS > 0) nTS *= 32767;
  12656. else if(nTS < 0) nTS *= 32768;
  12657. o.write_shift(2, nTS);
  12658. if(!color.rgb) {
  12659. o.write_shift(2, 0);
  12660. o.write_shift(1, 0);
  12661. o.write_shift(1, 0);
  12662. } else {
  12663. var rgb = (color.rgb || 'FFFFFF');
  12664. o.write_shift(1, parseInt(rgb.slice(0,2),16));
  12665. o.write_shift(1, parseInt(rgb.slice(2,4),16));
  12666. o.write_shift(1, parseInt(rgb.slice(4,6),16));
  12667. o.write_shift(1, 0xFF);
  12668. }
  12669. return o;
  12670. }
  12671. /* [MS-XLSB] 2.5.52 */
  12672. function parse_FontFlags(data) {
  12673. var d = data.read_shift(1);
  12674. data.l++;
  12675. var out = {
  12676. /* fBold: d & 0x01 */
  12677. fItalic: d & 0x02,
  12678. /* fUnderline: d & 0x04 */
  12679. fStrikeout: d & 0x08,
  12680. fOutline: d & 0x10,
  12681. fShadow: d & 0x20,
  12682. fCondense: d & 0x40,
  12683. fExtend: d & 0x80
  12684. };
  12685. return out;
  12686. }
  12687. function write_FontFlags(font, o) {
  12688. if(!o) o = new_buf(2);
  12689. var grbit =
  12690. (font.italic ? 0x02 : 0) |
  12691. (font.strike ? 0x08 : 0) |
  12692. (font.outline ? 0x10 : 0) |
  12693. (font.shadow ? 0x20 : 0) |
  12694. (font.condense ? 0x40 : 0) |
  12695. (font.extend ? 0x80 : 0);
  12696. o.write_shift(1, grbit);
  12697. o.write_shift(1, 0);
  12698. return o;
  12699. }
  12700. /* [MS-OLEDS] 2.3.1 and 2.3.2 */
  12701. function parse_ClipboardFormatOrString(o, w) {
  12702. // $FlowIgnore
  12703. var ClipFmt = {2:"BITMAP",3:"METAFILEPICT",8:"DIB",14:"ENHMETAFILE"};
  12704. var m = o.read_shift(4);
  12705. switch(m) {
  12706. case 0x00000000: return "";
  12707. case 0xffffffff: case 0xfffffffe: return ClipFmt[o.read_shift(4)]||"";
  12708. }
  12709. if(m > 0x190) throw new Error("Unsupported Clipboard: " + m.toString(16));
  12710. o.l -= 4;
  12711. return o.read_shift(0, w == 1 ? "lpstr" : "lpwstr");
  12712. }
  12713. function parse_ClipboardFormatOrAnsiString(o) { return parse_ClipboardFormatOrString(o, 1); }
  12714. function parse_ClipboardFormatOrUnicodeString(o) { return parse_ClipboardFormatOrString(o, 2); }
  12715. /* [MS-OLEPS] 2.2 PropertyType */
  12716. //var VT_EMPTY = 0x0000;
  12717. //var VT_NULL = 0x0001;
  12718. var VT_I2 = 0x0002;
  12719. var VT_I4 = 0x0003;
  12720. //var VT_R4 = 0x0004;
  12721. //var VT_R8 = 0x0005;
  12722. //var VT_CY = 0x0006;
  12723. //var VT_DATE = 0x0007;
  12724. //var VT_BSTR = 0x0008;
  12725. //var VT_ERROR = 0x000A;
  12726. var VT_BOOL = 0x000B;
  12727. var VT_VARIANT = 0x000C;
  12728. //var VT_DECIMAL = 0x000E;
  12729. //var VT_I1 = 0x0010;
  12730. //var VT_UI1 = 0x0011;
  12731. //var VT_UI2 = 0x0012;
  12732. var VT_UI4 = 0x0013;
  12733. //var VT_I8 = 0x0014;
  12734. //var VT_UI8 = 0x0015;
  12735. //var VT_INT = 0x0016;
  12736. //var VT_UINT = 0x0017;
  12737. var VT_LPSTR = 0x001E;
  12738. //var VT_LPWSTR = 0x001F;
  12739. var VT_FILETIME = 0x0040;
  12740. var VT_BLOB = 0x0041;
  12741. //var VT_STREAM = 0x0042;
  12742. //var VT_STORAGE = 0x0043;
  12743. //var VT_STREAMED_Object = 0x0044;
  12744. //var VT_STORED_Object = 0x0045;
  12745. //var VT_BLOB_Object = 0x0046;
  12746. var VT_CF = 0x0047;
  12747. //var VT_CLSID = 0x0048;
  12748. //var VT_VERSIONED_STREAM = 0x0049;
  12749. var VT_VECTOR = 0x1000;
  12750. //var VT_ARRAY = 0x2000;
  12751. var VT_STRING = 0x0050; // 2.3.3.1.11 VtString
  12752. var VT_USTR = 0x0051; // 2.3.3.1.12 VtUnalignedString
  12753. var VT_CUSTOM = [VT_STRING, VT_USTR];
  12754. /* [MS-OSHARED] 2.3.3.2.2.1 Document Summary Information PIDDSI */
  12755. var DocSummaryPIDDSI = {
  12756. 0x01: { n: 'CodePage', t: VT_I2 },
  12757. 0x02: { n: 'Category', t: VT_STRING },
  12758. 0x03: { n: 'PresentationFormat', t: VT_STRING },
  12759. 0x04: { n: 'ByteCount', t: VT_I4 },
  12760. 0x05: { n: 'LineCount', t: VT_I4 },
  12761. 0x06: { n: 'ParagraphCount', t: VT_I4 },
  12762. 0x07: { n: 'SlideCount', t: VT_I4 },
  12763. 0x08: { n: 'NoteCount', t: VT_I4 },
  12764. 0x09: { n: 'HiddenCount', t: VT_I4 },
  12765. 0x0a: { n: 'MultimediaClipCount', t: VT_I4 },
  12766. 0x0b: { n: 'ScaleCrop', t: VT_BOOL },
  12767. 0x0c: { n: 'HeadingPairs', t: VT_VECTOR | VT_VARIANT },
  12768. 0x0d: { n: 'TitlesOfParts', t: VT_VECTOR | VT_LPSTR },
  12769. 0x0e: { n: 'Manager', t: VT_STRING },
  12770. 0x0f: { n: 'Company', t: VT_STRING },
  12771. 0x10: { n: 'LinksUpToDate', t: VT_BOOL },
  12772. 0x11: { n: 'CharacterCount', t: VT_I4 },
  12773. 0x13: { n: 'SharedDoc', t: VT_BOOL },
  12774. 0x16: { n: 'HyperlinksChanged', t: VT_BOOL },
  12775. 0x17: { n: 'AppVersion', t: VT_I4, p: 'version' },
  12776. 0x18: { n: 'DigSig', t: VT_BLOB },
  12777. 0x1A: { n: 'ContentType', t: VT_STRING },
  12778. 0x1B: { n: 'ContentStatus', t: VT_STRING },
  12779. 0x1C: { n: 'Language', t: VT_STRING },
  12780. 0x1D: { n: 'Version', t: VT_STRING },
  12781. 0xFF: {}
  12782. };
  12783. /* [MS-OSHARED] 2.3.3.2.1.1 Summary Information Property Set PIDSI */
  12784. var SummaryPIDSI = {
  12785. 0x01: { n: 'CodePage', t: VT_I2 },
  12786. 0x02: { n: 'Title', t: VT_STRING },
  12787. 0x03: { n: 'Subject', t: VT_STRING },
  12788. 0x04: { n: 'Author', t: VT_STRING },
  12789. 0x05: { n: 'Keywords', t: VT_STRING },
  12790. 0x06: { n: 'Comments', t: VT_STRING },
  12791. 0x07: { n: 'Template', t: VT_STRING },
  12792. 0x08: { n: 'LastAuthor', t: VT_STRING },
  12793. 0x09: { n: 'RevNumber', t: VT_STRING },
  12794. 0x0A: { n: 'EditTime', t: VT_FILETIME },
  12795. 0x0B: { n: 'LastPrinted', t: VT_FILETIME },
  12796. 0x0C: { n: 'CreatedDate', t: VT_FILETIME },
  12797. 0x0D: { n: 'ModifiedDate', t: VT_FILETIME },
  12798. 0x0E: { n: 'PageCount', t: VT_I4 },
  12799. 0x0F: { n: 'WordCount', t: VT_I4 },
  12800. 0x10: { n: 'CharCount', t: VT_I4 },
  12801. 0x11: { n: 'Thumbnail', t: VT_CF },
  12802. 0x12: { n: 'Application', t: VT_STRING },
  12803. 0x13: { n: 'DocSecurity', t: VT_I4 },
  12804. 0xFF: {}
  12805. };
  12806. /* [MS-OLEPS] 2.18 */
  12807. var SpecialProperties = {
  12808. 0x80000000: { n: 'Locale', t: VT_UI4 },
  12809. 0x80000003: { n: 'Behavior', t: VT_UI4 },
  12810. 0x72627262: {}
  12811. };
  12812. (function() {
  12813. for(var y in SpecialProperties) if(SpecialProperties.hasOwnProperty(y))
  12814. DocSummaryPIDDSI[y] = SummaryPIDSI[y] = SpecialProperties[y];
  12815. })();
  12816. var DocSummaryRE = evert_key(DocSummaryPIDDSI, "n");
  12817. var SummaryRE = evert_key(SummaryPIDSI, "n");
  12818. /* [MS-XLS] 2.4.63 Country/Region codes */
  12819. var CountryEnum = {
  12820. 0x0001: "US", // United States
  12821. 0x0002: "CA", // Canada
  12822. 0x0003: "", // Latin America (except Brazil)
  12823. 0x0007: "RU", // Russia
  12824. 0x0014: "EG", // Egypt
  12825. 0x001E: "GR", // Greece
  12826. 0x001F: "NL", // Netherlands
  12827. 0x0020: "BE", // Belgium
  12828. 0x0021: "FR", // France
  12829. 0x0022: "ES", // Spain
  12830. 0x0024: "HU", // Hungary
  12831. 0x0027: "IT", // Italy
  12832. 0x0029: "CH", // Switzerland
  12833. 0x002B: "AT", // Austria
  12834. 0x002C: "GB", // United Kingdom
  12835. 0x002D: "DK", // Denmark
  12836. 0x002E: "SE", // Sweden
  12837. 0x002F: "NO", // Norway
  12838. 0x0030: "PL", // Poland
  12839. 0x0031: "DE", // Germany
  12840. 0x0034: "MX", // Mexico
  12841. 0x0037: "BR", // Brazil
  12842. 0x003d: "AU", // Australia
  12843. 0x0040: "NZ", // New Zealand
  12844. 0x0042: "TH", // Thailand
  12845. 0x0051: "JP", // Japan
  12846. 0x0052: "KR", // Korea
  12847. 0x0054: "VN", // Viet Nam
  12848. 0x0056: "CN", // China
  12849. 0x005A: "TR", // Turkey
  12850. 0x0069: "JS", // Ramastan
  12851. 0x00D5: "DZ", // Algeria
  12852. 0x00D8: "MA", // Morocco
  12853. 0x00DA: "LY", // Libya
  12854. 0x015F: "PT", // Portugal
  12855. 0x0162: "IS", // Iceland
  12856. 0x0166: "FI", // Finland
  12857. 0x01A4: "CZ", // Czech Republic
  12858. 0x0376: "TW", // Taiwan
  12859. 0x03C1: "LB", // Lebanon
  12860. 0x03C2: "JO", // Jordan
  12861. 0x03C3: "SY", // Syria
  12862. 0x03C4: "IQ", // Iraq
  12863. 0x03C5: "KW", // Kuwait
  12864. 0x03C6: "SA", // Saudi Arabia
  12865. 0x03CB: "AE", // United Arab Emirates
  12866. 0x03CC: "IL", // Israel
  12867. 0x03CE: "QA", // Qatar
  12868. 0x03D5: "IR", // Iran
  12869. 0xFFFF: "US" // United States
  12870. };
  12871. /* [MS-XLS] 2.5.127 */
  12872. var XLSFillPattern = [
  12873. null,
  12874. 'solid',
  12875. 'mediumGray',
  12876. 'darkGray',
  12877. 'lightGray',
  12878. 'darkHorizontal',
  12879. 'darkVertical',
  12880. 'darkDown',
  12881. 'darkUp',
  12882. 'darkGrid',
  12883. 'darkTrellis',
  12884. 'lightHorizontal',
  12885. 'lightVertical',
  12886. 'lightDown',
  12887. 'lightUp',
  12888. 'lightGrid',
  12889. 'lightTrellis',
  12890. 'gray125',
  12891. 'gray0625'
  12892. ];
  12893. function rgbify(arr) { return arr.map(function(x) { return [(x>>16)&255,(x>>8)&255,x&255]; }); }
  12894. /* [MS-XLS] 2.5.161 */
  12895. /* [MS-XLSB] 2.5.75 Icv */
  12896. var XLSIcv = rgbify([
  12897. /* Color Constants */
  12898. 0x000000,
  12899. 0xFFFFFF,
  12900. 0xFF0000,
  12901. 0x00FF00,
  12902. 0x0000FF,
  12903. 0xFFFF00,
  12904. 0xFF00FF,
  12905. 0x00FFFF,
  12906. /* Overridable Defaults */
  12907. 0x000000,
  12908. 0xFFFFFF,
  12909. 0xFF0000,
  12910. 0x00FF00,
  12911. 0x0000FF,
  12912. 0xFFFF00,
  12913. 0xFF00FF,
  12914. 0x00FFFF,
  12915. 0x800000,
  12916. 0x008000,
  12917. 0x000080,
  12918. 0x808000,
  12919. 0x800080,
  12920. 0x008080,
  12921. 0xC0C0C0,
  12922. 0x808080,
  12923. 0x9999FF,
  12924. 0x993366,
  12925. 0xFFFFCC,
  12926. 0xCCFFFF,
  12927. 0x660066,
  12928. 0xFF8080,
  12929. 0x0066CC,
  12930. 0xCCCCFF,
  12931. 0x000080,
  12932. 0xFF00FF,
  12933. 0xFFFF00,
  12934. 0x00FFFF,
  12935. 0x800080,
  12936. 0x800000,
  12937. 0x008080,
  12938. 0x0000FF,
  12939. 0x00CCFF,
  12940. 0xCCFFFF,
  12941. 0xCCFFCC,
  12942. 0xFFFF99,
  12943. 0x99CCFF,
  12944. 0xFF99CC,
  12945. 0xCC99FF,
  12946. 0xFFCC99,
  12947. 0x3366FF,
  12948. 0x33CCCC,
  12949. 0x99CC00,
  12950. 0xFFCC00,
  12951. 0xFF9900,
  12952. 0xFF6600,
  12953. 0x666699,
  12954. 0x969696,
  12955. 0x003366,
  12956. 0x339966,
  12957. 0x003300,
  12958. 0x333300,
  12959. 0x993300,
  12960. 0x993366,
  12961. 0x333399,
  12962. 0x333333,
  12963. /* Other entries to appease BIFF8/12 */
  12964. 0xFFFFFF, /* 0x40 icvForeground ?? */
  12965. 0x000000, /* 0x41 icvBackground ?? */
  12966. 0x000000, /* 0x42 icvFrame ?? */
  12967. 0x000000, /* 0x43 icv3D ?? */
  12968. 0x000000, /* 0x44 icv3DText ?? */
  12969. 0x000000, /* 0x45 icv3DHilite ?? */
  12970. 0x000000, /* 0x46 icv3DShadow ?? */
  12971. 0x000000, /* 0x47 icvHilite ?? */
  12972. 0x000000, /* 0x48 icvCtlText ?? */
  12973. 0x000000, /* 0x49 icvCtlScrl ?? */
  12974. 0x000000, /* 0x4A icvCtlInv ?? */
  12975. 0x000000, /* 0x4B icvCtlBody ?? */
  12976. 0x000000, /* 0x4C icvCtlFrame ?? */
  12977. 0x000000, /* 0x4D icvCtlFore ?? */
  12978. 0x000000, /* 0x4E icvCtlBack ?? */
  12979. 0x000000, /* 0x4F icvCtlNeutral */
  12980. 0x000000, /* 0x50 icvInfoBk ?? */
  12981. 0x000000 /* 0x51 icvInfoText ?? */
  12982. ]);
  12983. /* Parts enumerated in OPC spec, MS-XLSB and MS-XLSX */
  12984. /* 12.3 Part Summary <SpreadsheetML> */
  12985. /* 14.2 Part Summary <DrawingML> */
  12986. /* [MS-XLSX] 2.1 Part Enumerations ; [MS-XLSB] 2.1.7 Part Enumeration */
  12987. var ct2type/*{[string]:string}*/ = ({
  12988. /* Workbook */
  12989. "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml": "workbooks",
  12990. /* Worksheet */
  12991. "application/vnd.ms-excel.binIndexWs": "TODO", /* Binary Index */
  12992. /* Macrosheet */
  12993. "application/vnd.ms-excel.intlmacrosheet": "TODO",
  12994. "application/vnd.ms-excel.binIndexMs": "TODO", /* Binary Index */
  12995. /* File Properties */
  12996. "application/vnd.openxmlformats-package.core-properties+xml": "coreprops",
  12997. "application/vnd.openxmlformats-officedocument.custom-properties+xml": "custprops",
  12998. "application/vnd.openxmlformats-officedocument.extended-properties+xml": "extprops",
  12999. /* Custom Data Properties */
  13000. "application/vnd.openxmlformats-officedocument.customXmlProperties+xml": "TODO",
  13001. "application/vnd.openxmlformats-officedocument.spreadsheetml.customProperty": "TODO",
  13002. /* PivotTable */
  13003. "application/vnd.ms-excel.pivotTable": "TODO",
  13004. "application/vnd.openxmlformats-officedocument.spreadsheetml.pivotTable+xml": "TODO",
  13005. /* Chart Colors */
  13006. "application/vnd.ms-office.chartcolorstyle+xml": "TODO",
  13007. /* Chart Style */
  13008. "application/vnd.ms-office.chartstyle+xml": "TODO",
  13009. /* Calculation Chain */
  13010. "application/vnd.ms-excel.calcChain": "calcchains",
  13011. "application/vnd.openxmlformats-officedocument.spreadsheetml.calcChain+xml": "calcchains",
  13012. /* Printer Settings */
  13013. "application/vnd.openxmlformats-officedocument.spreadsheetml.printerSettings": "TODO",
  13014. /* ActiveX */
  13015. "application/vnd.ms-office.activeX": "TODO",
  13016. "application/vnd.ms-office.activeX+xml": "TODO",
  13017. /* Custom Toolbars */
  13018. "application/vnd.ms-excel.attachedToolbars": "TODO",
  13019. /* External Data Connections */
  13020. "application/vnd.ms-excel.connections": "TODO",
  13021. "application/vnd.openxmlformats-officedocument.spreadsheetml.connections+xml": "TODO",
  13022. /* External Links */
  13023. "application/vnd.ms-excel.externalLink": "links",
  13024. "application/vnd.openxmlformats-officedocument.spreadsheetml.externalLink+xml": "links",
  13025. /* Metadata */
  13026. "application/vnd.ms-excel.sheetMetadata": "TODO",
  13027. "application/vnd.openxmlformats-officedocument.spreadsheetml.sheetMetadata+xml": "TODO",
  13028. /* PivotCache */
  13029. "application/vnd.ms-excel.pivotCacheDefinition": "TODO",
  13030. "application/vnd.ms-excel.pivotCacheRecords": "TODO",
  13031. "application/vnd.openxmlformats-officedocument.spreadsheetml.pivotCacheDefinition+xml": "TODO",
  13032. "application/vnd.openxmlformats-officedocument.spreadsheetml.pivotCacheRecords+xml": "TODO",
  13033. /* Query Table */
  13034. "application/vnd.ms-excel.queryTable": "TODO",
  13035. "application/vnd.openxmlformats-officedocument.spreadsheetml.queryTable+xml": "TODO",
  13036. /* Shared Workbook */
  13037. "application/vnd.ms-excel.userNames": "TODO",
  13038. "application/vnd.ms-excel.revisionHeaders": "TODO",
  13039. "application/vnd.ms-excel.revisionLog": "TODO",
  13040. "application/vnd.openxmlformats-officedocument.spreadsheetml.revisionHeaders+xml": "TODO",
  13041. "application/vnd.openxmlformats-officedocument.spreadsheetml.revisionLog+xml": "TODO",
  13042. "application/vnd.openxmlformats-officedocument.spreadsheetml.userNames+xml": "TODO",
  13043. /* Single Cell Table */
  13044. "application/vnd.ms-excel.tableSingleCells": "TODO",
  13045. "application/vnd.openxmlformats-officedocument.spreadsheetml.tableSingleCells+xml": "TODO",
  13046. /* Slicer */
  13047. "application/vnd.ms-excel.slicer": "TODO",
  13048. "application/vnd.ms-excel.slicerCache": "TODO",
  13049. "application/vnd.ms-excel.slicer+xml": "TODO",
  13050. "application/vnd.ms-excel.slicerCache+xml": "TODO",
  13051. /* Sort Map */
  13052. "application/vnd.ms-excel.wsSortMap": "TODO",
  13053. /* Table */
  13054. "application/vnd.ms-excel.table": "TODO",
  13055. "application/vnd.openxmlformats-officedocument.spreadsheetml.table+xml": "TODO",
  13056. /* Themes */
  13057. "application/vnd.openxmlformats-officedocument.theme+xml": "themes",
  13058. /* Theme Override */
  13059. "application/vnd.openxmlformats-officedocument.themeOverride+xml": "TODO",
  13060. /* Timeline */
  13061. "application/vnd.ms-excel.Timeline+xml": "TODO", /* verify */
  13062. "application/vnd.ms-excel.TimelineCache+xml": "TODO", /* verify */
  13063. /* VBA */
  13064. "application/vnd.ms-office.vbaProject": "vba",
  13065. "application/vnd.ms-office.vbaProjectSignature": "vba",
  13066. /* Volatile Dependencies */
  13067. "application/vnd.ms-office.volatileDependencies": "TODO",
  13068. "application/vnd.openxmlformats-officedocument.spreadsheetml.volatileDependencies+xml": "TODO",
  13069. /* Control Properties */
  13070. "application/vnd.ms-excel.controlproperties+xml": "TODO",
  13071. /* Data Model */
  13072. "application/vnd.openxmlformats-officedocument.model+data": "TODO",
  13073. /* Survey */
  13074. "application/vnd.ms-excel.Survey+xml": "TODO",
  13075. /* Drawing */
  13076. "application/vnd.openxmlformats-officedocument.drawing+xml": "drawings",
  13077. "application/vnd.openxmlformats-officedocument.drawingml.chart+xml": "TODO",
  13078. "application/vnd.openxmlformats-officedocument.drawingml.chartshapes+xml": "TODO",
  13079. "application/vnd.openxmlformats-officedocument.drawingml.diagramColors+xml": "TODO",
  13080. "application/vnd.openxmlformats-officedocument.drawingml.diagramData+xml": "TODO",
  13081. "application/vnd.openxmlformats-officedocument.drawingml.diagramLayout+xml": "TODO",
  13082. "application/vnd.openxmlformats-officedocument.drawingml.diagramStyle+xml": "TODO",
  13083. /* VML */
  13084. "application/vnd.openxmlformats-officedocument.vmlDrawing": "TODO",
  13085. "application/vnd.openxmlformats-package.relationships+xml": "rels",
  13086. "application/vnd.openxmlformats-officedocument.oleObject": "TODO",
  13087. /* Image */
  13088. "image/png": "TODO",
  13089. "sheet": "js"
  13090. });
  13091. var CT_LIST = (function(){
  13092. var o = {
  13093. workbooks: {
  13094. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml",
  13095. xlsm: "application/vnd.ms-excel.sheet.macroEnabled.main+xml",
  13096. xlsb: "application/vnd.ms-excel.sheet.binary.macroEnabled.main",
  13097. xlam: "application/vnd.ms-excel.addin.macroEnabled.main+xml",
  13098. xltx: "application/vnd.openxmlformats-officedocument.spreadsheetml.template.main+xml"
  13099. },
  13100. strs: { /* Shared Strings */
  13101. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.sharedStrings+xml",
  13102. xlsb: "application/vnd.ms-excel.sharedStrings"
  13103. },
  13104. comments: { /* Comments */
  13105. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.comments+xml",
  13106. xlsb: "application/vnd.ms-excel.comments"
  13107. },
  13108. sheets: { /* Worksheet */
  13109. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.worksheet+xml",
  13110. xlsb: "application/vnd.ms-excel.worksheet"
  13111. },
  13112. charts: { /* Chartsheet */
  13113. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.chartsheet+xml",
  13114. xlsb: "application/vnd.ms-excel.chartsheet"
  13115. },
  13116. dialogs: { /* Dialogsheet */
  13117. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.dialogsheet+xml",
  13118. xlsb: "application/vnd.ms-excel.dialogsheet"
  13119. },
  13120. macros: { /* Macrosheet (Excel 4.0 Macros) */
  13121. xlsx: "application/vnd.ms-excel.macrosheet+xml",
  13122. xlsb: "application/vnd.ms-excel.macrosheet"
  13123. },
  13124. styles: { /* Styles */
  13125. xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.styles+xml",
  13126. xlsb: "application/vnd.ms-excel.styles"
  13127. }
  13128. };
  13129. keys(o).forEach(function(k) { ["xlsm", "xlam"].forEach(function(v) { if(!o[k][v]) o[k][v] = o[k].xlsx; }); });
  13130. keys(o).forEach(function(k){ keys(o[k]).forEach(function(v) { ct2type[o[k][v]] = k; }); });
  13131. return o;
  13132. })();
  13133. var type2ct/*{[string]:Array<string>}*/ = evert_arr(ct2type);
  13134. XMLNS.CT = 'http://schemas.openxmlformats.org/package/2006/content-types';
  13135. function new_ct() {
  13136. return ({
  13137. workbooks:[], sheets:[], charts:[], dialogs:[], macros:[],
  13138. rels:[], strs:[], comments:[], links:[],
  13139. coreprops:[], extprops:[], custprops:[], themes:[], styles:[],
  13140. calcchains:[], vba: [], drawings: [],
  13141. TODO:[], xmlns: "" });
  13142. }
  13143. function parse_ct(data) {
  13144. var ct = new_ct();
  13145. if(!data || !data.match) return ct;
  13146. var ctext = {};
  13147. (data.match(tagregex)||[]).forEach(function(x) {
  13148. var y = parsexmltag(x);
  13149. switch(y[0].replace(nsregex,"<")) {
  13150. case '<?xml': break;
  13151. case '<Types': ct.xmlns = y['xmlns' + (y[0].match(/<(\w+):/)||["",""])[1] ]; break;
  13152. case '<Default': ctext[y.Extension] = y.ContentType; break;
  13153. case '<Override':
  13154. if(ct[ct2type[y.ContentType]] !== undefined) ct[ct2type[y.ContentType]].push(y.PartName);
  13155. break;
  13156. }
  13157. });
  13158. if(ct.xmlns !== XMLNS.CT) throw new Error("Unknown Namespace: " + ct.xmlns);
  13159. ct.calcchain = ct.calcchains.length > 0 ? ct.calcchains[0] : "";
  13160. ct.sst = ct.strs.length > 0 ? ct.strs[0] : "";
  13161. ct.style = ct.styles.length > 0 ? ct.styles[0] : "";
  13162. ct.defaults = ctext;
  13163. delete ct.calcchains;
  13164. return ct;
  13165. }
  13166. var CTYPE_XML_ROOT = writextag('Types', null, {
  13167. 'xmlns': XMLNS.CT,
  13168. 'xmlns:xsd': XMLNS.xsd,
  13169. 'xmlns:xsi': XMLNS.xsi
  13170. });
  13171. var CTYPE_DEFAULTS = [
  13172. ['xml', 'application/xml'],
  13173. ['bin', 'application/vnd.ms-excel.sheet.binary.macroEnabled.main'],
  13174. ['vml', 'application/vnd.openxmlformats-officedocument.vmlDrawing'],
  13175. /* from test files */
  13176. ['bmp', 'image/bmp'],
  13177. ['png', 'image/png'],
  13178. ['gif', 'image/gif'],
  13179. ['emf', 'image/x-emf'],
  13180. ['wmf', 'image/x-wmf'],
  13181. ['jpg', 'image/jpeg'], ['jpeg', 'image/jpeg'],
  13182. ['tif', 'image/tiff'], ['tiff', 'image/tiff'],
  13183. ['pdf', 'application/pdf'],
  13184. ['rels', type2ct.rels[0]]
  13185. ].map(function(x) {
  13186. return writextag('Default', null, {'Extension':x[0], 'ContentType': x[1]});
  13187. });
  13188. function write_ct(ct, opts) {
  13189. var o = [], v;
  13190. o[o.length] = (XML_HEADER);
  13191. o[o.length] = (CTYPE_XML_ROOT);
  13192. o = o.concat(CTYPE_DEFAULTS);
  13193. var f1 = function(w) {
  13194. if(ct[w] && ct[w].length > 0) {
  13195. v = ct[w][0];
  13196. o[o.length] = (writextag('Override', null, {
  13197. 'PartName': (v[0] == '/' ? "":"/") + v,
  13198. 'ContentType': CT_LIST[w][opts.bookType || 'xlsx']
  13199. }));
  13200. }
  13201. };
  13202. var f2 = function(w) {
  13203. (ct[w]||[]).forEach(function(v) {
  13204. o[o.length] = (writextag('Override', null, {
  13205. 'PartName': (v[0] == '/' ? "":"/") + v,
  13206. 'ContentType': CT_LIST[w][opts.bookType || 'xlsx']
  13207. }));
  13208. });
  13209. };
  13210. var f3 = function(t) {
  13211. (ct[t]||[]).forEach(function(v) {
  13212. o[o.length] = (writextag('Override', null, {
  13213. 'PartName': (v[0] == '/' ? "":"/") + v,
  13214. 'ContentType': type2ct[t][0]
  13215. }));
  13216. });
  13217. };
  13218. f1('workbooks');
  13219. f2('sheets');
  13220. f2('charts');
  13221. f3('themes');
  13222. ['strs', 'styles'].forEach(f1);
  13223. ['coreprops', 'extprops', 'custprops'].forEach(f3);
  13224. f3('vba');
  13225. f3('comments');
  13226. f3('drawings');
  13227. if(o.length>2){ o[o.length] = ('</Types>'); o[1]=o[1].replace("/>",">"); }
  13228. return o.join("");
  13229. }
  13230. /* 9.3 Relationships */
  13231. var RELS = ({
  13232. WB: "http://schemas.openxmlformats.org/officeDocument/2006/relationships/officeDocument",
  13233. SHEET: "http://sheetjs.openxmlformats.org/officeDocument/2006/relationships/officeDocument",
  13234. HLINK: "http://schemas.openxmlformats.org/officeDocument/2006/relationships/hyperlink",
  13235. VML: "http://schemas.openxmlformats.org/officeDocument/2006/relationships/vmlDrawing",
  13236. VBA: "http://schemas.microsoft.com/office/2006/relationships/vbaProject"
  13237. });
  13238. /* 9.3.3 Representing Relationships */
  13239. function get_rels_path(file) {
  13240. var n = file.lastIndexOf("/");
  13241. return file.slice(0,n+1) + '_rels/' + file.slice(n+1) + ".rels";
  13242. }
  13243. function parse_rels(data, currentFilePath) {
  13244. if (!data) return data;
  13245. if (currentFilePath.charAt(0) !== '/') {
  13246. currentFilePath = '/'+currentFilePath;
  13247. }
  13248. var rels = {};
  13249. var hash = {};
  13250. (data.match(tagregex)||[]).forEach(function(x) {
  13251. var y = parsexmltag(x);
  13252. /* 9.3.2.2 OPC_Relationships */
  13253. if (y[0] === '<Relationship') {
  13254. var rel = {}; rel.Type = y.Type; rel.Target = y.Target; rel.Id = y.Id; rel.TargetMode = y.TargetMode;
  13255. var canonictarget = y.TargetMode === 'External' ? y.Target : resolve_path(y.Target, currentFilePath);
  13256. rels[canonictarget] = rel;
  13257. hash[y.Id] = rel;
  13258. }
  13259. });
  13260. rels["!id"] = hash;
  13261. return rels;
  13262. }
  13263. XMLNS.RELS = 'http://schemas.openxmlformats.org/package/2006/relationships';
  13264. var RELS_ROOT = writextag('Relationships', null, {
  13265. //'xmlns:ns0': XMLNS.RELS,
  13266. 'xmlns': XMLNS.RELS
  13267. });
  13268. /* TODO */
  13269. function write_rels(rels) {
  13270. var o = [XML_HEADER, RELS_ROOT];
  13271. keys(rels['!id']).forEach(function(rid) {
  13272. o[o.length] = (writextag('Relationship', null, rels['!id'][rid]));
  13273. });
  13274. if(o.length>2){ o[o.length] = ('</Relationships>'); o[1]=o[1].replace("/>",">"); }
  13275. return o.join("");
  13276. }
  13277. function add_rels(rels, rId, f, type, relobj) {
  13278. if(!relobj) relobj = {};
  13279. if(!rels['!id']) rels['!id'] = {};
  13280. if(rId < 0) for(rId = 1; rels['!id']['rId' + rId]; ++rId){/* empty */}
  13281. relobj.Id = 'rId' + rId;
  13282. relobj.Type = type;
  13283. relobj.Target = f;
  13284. if(relobj.Type == RELS.HLINK) relobj.TargetMode = "External";
  13285. if(rels['!id'][relobj.Id]) throw new Error("Cannot rewrite rId " + rId);
  13286. rels['!id'][relobj.Id] = relobj;
  13287. rels[('/' + relobj.Target).replace("//","/")] = relobj;
  13288. return rId;
  13289. }
  13290. /* Open Document Format for Office Applications (OpenDocument) Version 1.2 */
  13291. /* Part 3 Section 4 Manifest File */
  13292. var CT_ODS = "application/vnd.oasis.opendocument.spreadsheet";
  13293. function parse_manifest(d, opts) {
  13294. var str = xlml_normalize(d);
  13295. var Rn;
  13296. var FEtag;
  13297. while((Rn = xlmlregex.exec(str))) switch(Rn[3]) {
  13298. case 'manifest': break; // 4.2 <manifest:manifest>
  13299. case 'file-entry': // 4.3 <manifest:file-entry>
  13300. FEtag = parsexmltag(Rn[0], false);
  13301. if(FEtag.path == '/' && FEtag.type !== CT_ODS) throw new Error("This OpenDocument is not a spreadsheet");
  13302. break;
  13303. case 'encryption-data': // 4.4 <manifest:encryption-data>
  13304. case 'algorithm': // 4.5 <manifest:algorithm>
  13305. case 'start-key-generation': // 4.6 <manifest:start-key-generation>
  13306. case 'key-derivation': // 4.7 <manifest:key-derivation>
  13307. throw new Error("Unsupported ODS Encryption");
  13308. default: if(opts && opts.WTF) throw Rn;
  13309. }
  13310. }
  13311. function write_manifest(manifest) {
  13312. var o = [XML_HEADER];
  13313. o.push('<manifest:manifest xmlns:manifest="urn:oasis:names:tc:opendocument:xmlns:manifest:1.0" manifest:version="1.2">\n');
  13314. o.push(' <manifest:file-entry manifest:full-path="/" manifest:version="1.2" manifest:media-type="application/vnd.oasis.opendocument.spreadsheet"/>\n');
  13315. for(var i = 0; i < manifest.length; ++i) o.push(' <manifest:file-entry manifest:full-path="' + manifest[i][0] + '" manifest:media-type="' + manifest[i][1] + '"/>\n');
  13316. o.push('</manifest:manifest>');
  13317. return o.join("");
  13318. }
  13319. /* Part 3 Section 6 Metadata Manifest File */
  13320. function write_rdf_type(file, res, tag) {
  13321. return [
  13322. ' <rdf:Description rdf:about="' + file + '">\n',
  13323. ' <rdf:type rdf:resource="http://docs.oasis-open.org/ns/office/1.2/meta/' + (tag || "odf") + '#' + res + '"/>\n',
  13324. ' </rdf:Description>\n'
  13325. ].join("");
  13326. }
  13327. function write_rdf_has(base, file) {
  13328. return [
  13329. ' <rdf:Description rdf:about="' + base + '">\n',
  13330. ' <ns0:hasPart xmlns:ns0="http://docs.oasis-open.org/ns/office/1.2/meta/pkg#" rdf:resource="' + file + '"/>\n',
  13331. ' </rdf:Description>\n'
  13332. ].join("");
  13333. }
  13334. function write_rdf(rdf) {
  13335. var o = [XML_HEADER];
  13336. o.push('<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">\n');
  13337. for(var i = 0; i != rdf.length; ++i) {
  13338. o.push(write_rdf_type(rdf[i][0], rdf[i][1]));
  13339. o.push(write_rdf_has("",rdf[i][0]));
  13340. }
  13341. o.push(write_rdf_type("","Document", "pkg"));
  13342. o.push('</rdf:RDF>');
  13343. return o.join("");
  13344. }
  13345. /* TODO: pull properties */
  13346. var write_meta_ods = (function() {
  13347. var payload = '<?xml version="1.0" encoding="UTF-8" standalone="yes"?><office:document-meta xmlns:office="urn:oasis:names:tc:opendocument:xmlns:office:1.0" xmlns:meta="urn:oasis:names:tc:opendocument:xmlns:meta:1.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:xlink="http://www.w3.org/1999/xlink" office:version="1.2"><office:meta><meta:generator>Sheet' + 'JS ' + XLSX.version + '</meta:generator></office:meta></office:document-meta>';
  13348. return function wmo() {
  13349. return payload;
  13350. };
  13351. })();
  13352. /* ECMA-376 Part II 11.1 Core Properties Part */
  13353. /* [MS-OSHARED] 2.3.3.2.[1-2].1 (PIDSI/PIDDSI) */
  13354. var CORE_PROPS = [
  13355. ["cp:category", "Category"],
  13356. ["cp:contentStatus", "ContentStatus"],
  13357. ["cp:keywords", "Keywords"],
  13358. ["cp:lastModifiedBy", "LastAuthor"],
  13359. ["cp:lastPrinted", "LastPrinted"],
  13360. ["cp:revision", "RevNumber"],
  13361. ["cp:version", "Version"],
  13362. ["dc:creator", "Author"],
  13363. ["dc:description", "Comments"],
  13364. ["dc:identifier", "Identifier"],
  13365. ["dc:language", "Language"],
  13366. ["dc:subject", "Subject"],
  13367. ["dc:title", "Title"],
  13368. ["dcterms:created", "CreatedDate", 'date'],
  13369. ["dcterms:modified", "ModifiedDate", 'date']
  13370. ];
  13371. XMLNS.CORE_PROPS = "http://schemas.openxmlformats.org/package/2006/metadata/core-properties";
  13372. RELS.CORE_PROPS = 'http://schemas.openxmlformats.org/package/2006/relationships/metadata/core-properties';
  13373. var CORE_PROPS_REGEX = (function() {
  13374. var r = new Array(CORE_PROPS.length);
  13375. for(var i = 0; i < CORE_PROPS.length; ++i) {
  13376. var f = CORE_PROPS[i];
  13377. var g = "(?:"+ f[0].slice(0,f[0].indexOf(":")) +":)"+ f[0].slice(f[0].indexOf(":")+1);
  13378. r[i] = new RegExp("<" + g + "[^>]*>([\\s\\S]*?)<\/" + g + ">");
  13379. }
  13380. return r;
  13381. })();
  13382. function parse_core_props(data) {
  13383. var p = {};
  13384. data = utf8read(data);
  13385. for(var i = 0; i < CORE_PROPS.length; ++i) {
  13386. var f = CORE_PROPS[i], cur = data.match(CORE_PROPS_REGEX[i]);
  13387. if(cur != null && cur.length > 0) p[f[1]] = cur[1];
  13388. if(f[2] === 'date' && p[f[1]]) p[f[1]] = parseDate(p[f[1]]);
  13389. }
  13390. return p;
  13391. }
  13392. var CORE_PROPS_XML_ROOT = writextag('cp:coreProperties', null, {
  13393. //'xmlns': XMLNS.CORE_PROPS,
  13394. 'xmlns:cp': XMLNS.CORE_PROPS,
  13395. 'xmlns:dc': XMLNS.dc,
  13396. 'xmlns:dcterms': XMLNS.dcterms,
  13397. 'xmlns:dcmitype': XMLNS.dcmitype,
  13398. 'xmlns:xsi': XMLNS.xsi
  13399. });
  13400. function cp_doit(f, g, h, o, p) {
  13401. if(p[f] != null || g == null || g === "") return;
  13402. p[f] = g;
  13403. o[o.length] = (h ? writextag(f,g,h) : writetag(f,g));
  13404. }
  13405. function write_core_props(cp, _opts) {
  13406. var opts = _opts || {};
  13407. var o = [XML_HEADER, CORE_PROPS_XML_ROOT], p = {};
  13408. if(!cp && !opts.Props) return o.join("");
  13409. if(cp) {
  13410. if(cp.CreatedDate != null) cp_doit("dcterms:created", typeof cp.CreatedDate === "string" ? cp.CreatedDate : write_w3cdtf(cp.CreatedDate, opts.WTF), {"xsi:type":"dcterms:W3CDTF"}, o, p);
  13411. if(cp.ModifiedDate != null) cp_doit("dcterms:modified", typeof cp.ModifiedDate === "string" ? cp.ModifiedDate : write_w3cdtf(cp.ModifiedDate, opts.WTF), {"xsi:type":"dcterms:W3CDTF"}, o, p);
  13412. }
  13413. for(var i = 0; i != CORE_PROPS.length; ++i) {
  13414. var f = CORE_PROPS[i];
  13415. var v = opts.Props && opts.Props[f[1]] != null ? opts.Props[f[1]] : cp ? cp[f[1]] : null;
  13416. if(v === true) v = "1";
  13417. else if(v === false) v = "0";
  13418. else if(typeof v == "number") v = String(v);
  13419. if(v != null) cp_doit(f[0], v, null, o, p);
  13420. }
  13421. if(o.length>2){ o[o.length] = ('</cp:coreProperties>'); o[1]=o[1].replace("/>",">"); }
  13422. return o.join("");
  13423. }
  13424. /* 15.2.12.3 Extended File Properties Part */
  13425. /* [MS-OSHARED] 2.3.3.2.[1-2].1 (PIDSI/PIDDSI) */
  13426. var EXT_PROPS = [
  13427. ["Application", "Application", "string"],
  13428. ["AppVersion", "AppVersion", "string"],
  13429. ["Company", "Company", "string"],
  13430. ["DocSecurity", "DocSecurity", "string"],
  13431. ["Manager", "Manager", "string"],
  13432. ["HyperlinksChanged", "HyperlinksChanged", "bool"],
  13433. ["SharedDoc", "SharedDoc", "bool"],
  13434. ["LinksUpToDate", "LinksUpToDate", "bool"],
  13435. ["ScaleCrop", "ScaleCrop", "bool"],
  13436. ["HeadingPairs", "HeadingPairs", "raw"],
  13437. ["TitlesOfParts", "TitlesOfParts", "raw"]
  13438. ];
  13439. XMLNS.EXT_PROPS = "http://schemas.openxmlformats.org/officeDocument/2006/extended-properties";
  13440. RELS.EXT_PROPS = 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/extended-properties';
  13441. var PseudoPropsPairs = [
  13442. "Worksheets", "SheetNames",
  13443. "NamedRanges", "DefinedNames",
  13444. "Chartsheets", "ChartNames"
  13445. ];
  13446. function load_props_pairs(HP, TOP, props, opts) {
  13447. var v = [];
  13448. if(typeof HP == "string") v = parseVector(HP, opts);
  13449. else for(var j = 0; j < HP.length; ++j) v = v.concat(HP[j].map(function(hp) { return {v:hp}; }));
  13450. var parts = (typeof TOP == "string") ? parseVector(TOP, opts).map(function (x) { return x.v; }) : TOP;
  13451. var idx = 0, len = 0;
  13452. if(parts.length > 0) for(var i = 0; i !== v.length; i += 2) {
  13453. len = +(v[i+1].v);
  13454. switch(v[i].v) {
  13455. case "Worksheets":
  13456. case "工作表":
  13457. case "Листы":
  13458. case "أوراق العمل":
  13459. case "ワークシート":
  13460. case "גליונות עבודה":
  13461. case "Arbeitsblätter":
  13462. case "Çalışma Sayfaları":
  13463. case "Feuilles de calcul":
  13464. case "Fogli di lavoro":
  13465. case "Folhas de cálculo":
  13466. case "Planilhas":
  13467. case "Regneark":
  13468. case "Werkbladen":
  13469. props.Worksheets = len;
  13470. props.SheetNames = parts.slice(idx, idx + len);
  13471. break;
  13472. case "Named Ranges":
  13473. case "名前付き一覧":
  13474. case "Benannte Bereiche":
  13475. case "Navngivne områder":
  13476. props.NamedRanges = len;
  13477. props.DefinedNames = parts.slice(idx, idx + len);
  13478. break;
  13479. case "Charts":
  13480. case "Diagramme":
  13481. props.Chartsheets = len;
  13482. props.ChartNames = parts.slice(idx, idx + len);
  13483. break;
  13484. }
  13485. idx += len;
  13486. }
  13487. }
  13488. function parse_ext_props(data, p, opts) {
  13489. var q = {}; if(!p) p = {};
  13490. data = utf8read(data);
  13491. EXT_PROPS.forEach(function(f) {
  13492. switch(f[2]) {
  13493. case "string": p[f[1]] = (data.match(matchtag(f[0]))||[])[1]; break;
  13494. case "bool": p[f[1]] = (data.match(matchtag(f[0]))||[])[1] === "true"; break;
  13495. case "raw":
  13496. var cur = data.match(new RegExp("<" + f[0] + "[^>]*>([\\s\\S]*?)<\/" + f[0] + ">"));
  13497. if(cur && cur.length > 0) q[f[1]] = cur[1];
  13498. break;
  13499. }
  13500. });
  13501. if(q.HeadingPairs && q.TitlesOfParts) load_props_pairs(q.HeadingPairs, q.TitlesOfParts, p, opts);
  13502. return p;
  13503. }
  13504. var EXT_PROPS_XML_ROOT = writextag('Properties', null, {
  13505. 'xmlns': XMLNS.EXT_PROPS,
  13506. 'xmlns:vt': XMLNS.vt
  13507. });
  13508. function write_ext_props(cp) {
  13509. var o = [], W = writextag;
  13510. if(!cp) cp = {};
  13511. cp.Application = "SheetJS";
  13512. o[o.length] = (XML_HEADER);
  13513. o[o.length] = (EXT_PROPS_XML_ROOT);
  13514. EXT_PROPS.forEach(function(f) {
  13515. if(cp[f[1]] === undefined) return;
  13516. var v;
  13517. switch(f[2]) {
  13518. case 'string': v = String(cp[f[1]]); break;
  13519. case 'bool': v = cp[f[1]] ? 'true' : 'false'; break;
  13520. }
  13521. if(v !== undefined) o[o.length] = (W(f[0], v));
  13522. });
  13523. /* TODO: HeadingPairs, TitlesOfParts */
  13524. o[o.length] = (W('HeadingPairs', W('vt:vector', W('vt:variant', '<vt:lpstr>Worksheets</vt:lpstr>')+W('vt:variant', W('vt:i4', String(cp.Worksheets))), {size:2, baseType:"variant"})));
  13525. o[o.length] = (W('TitlesOfParts', W('vt:vector', cp.SheetNames.map(function(s) { return "<vt:lpstr>" + escapexml(s) + "</vt:lpstr>"; }).join(""), {size: cp.Worksheets, baseType:"lpstr"})));
  13526. if(o.length>2){ o[o.length] = ('</Properties>'); o[1]=o[1].replace("/>",">"); }
  13527. return o.join("");
  13528. }
  13529. /* 15.2.12.2 Custom File Properties Part */
  13530. XMLNS.CUST_PROPS = "http://schemas.openxmlformats.org/officeDocument/2006/custom-properties";
  13531. RELS.CUST_PROPS = 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/custom-properties';
  13532. var custregex = /<[^>]+>[^<]*/g;
  13533. function parse_cust_props(data, opts) {
  13534. var p = {}, name = "";
  13535. var m = data.match(custregex);
  13536. if(m) for(var i = 0; i != m.length; ++i) {
  13537. var x = m[i], y = parsexmltag(x);
  13538. switch(y[0]) {
  13539. case '<?xml': break;
  13540. case '<Properties': break;
  13541. case '<property': name = y.name; break;
  13542. case '</property>': name = null; break;
  13543. default: if (x.indexOf('<vt:') === 0) {
  13544. var toks = x.split('>');
  13545. var type = toks[0].slice(4), text = toks[1];
  13546. /* 22.4.2.32 (CT_Variant). Omit the binary types from 22.4 (Variant Types) */
  13547. switch(type) {
  13548. case 'lpstr': case 'bstr': case 'lpwstr':
  13549. p[name] = unescapexml(text);
  13550. break;
  13551. case 'bool':
  13552. p[name] = parsexmlbool(text);
  13553. break;
  13554. case 'i1': case 'i2': case 'i4': case 'i8': case 'int': case 'uint':
  13555. p[name] = parseInt(text, 10);
  13556. break;
  13557. case 'r4': case 'r8': case 'decimal':
  13558. p[name] = parseFloat(text);
  13559. break;
  13560. case 'filetime': case 'date':
  13561. p[name] = parseDate(text);
  13562. break;
  13563. case 'cy': case 'error':
  13564. p[name] = unescapexml(text);
  13565. break;
  13566. default:
  13567. if(type.slice(-1) == '/') break;
  13568. if(opts.WTF && typeof console !== 'undefined') console.warn('Unexpected', x, type, toks);
  13569. }
  13570. } else if(x.slice(0,2) === "</") {/* empty */
  13571. } else if(opts.WTF) throw new Error(x);
  13572. }
  13573. }
  13574. return p;
  13575. }
  13576. var CUST_PROPS_XML_ROOT = writextag('Properties', null, {
  13577. 'xmlns': XMLNS.CUST_PROPS,
  13578. 'xmlns:vt': XMLNS.vt
  13579. });
  13580. function write_cust_props(cp) {
  13581. var o = [XML_HEADER, CUST_PROPS_XML_ROOT];
  13582. if(!cp) return o.join("");
  13583. var pid = 1;
  13584. keys(cp).forEach(function custprop(k) { ++pid;
  13585. o[o.length] = (writextag('property', write_vt(cp[k]), {
  13586. 'fmtid': '{D5CDD505-2E9C-101B-9397-08002B2CF9AE}',
  13587. 'pid': pid,
  13588. 'name': k
  13589. }));
  13590. });
  13591. if(o.length>2){ o[o.length] = '</Properties>'; o[1]=o[1].replace("/>",">"); }
  13592. return o.join("");
  13593. }
  13594. /* Common Name -> XLML Name */
  13595. var XLMLDocPropsMap = {
  13596. Title: 'Title',
  13597. Subject: 'Subject',
  13598. Author: 'Author',
  13599. Keywords: 'Keywords',
  13600. Comments: 'Description',
  13601. LastAuthor: 'LastAuthor',
  13602. RevNumber: 'Revision',
  13603. Application: 'AppName',
  13604. /* TotalTime: 'TotalTime', */
  13605. LastPrinted: 'LastPrinted',
  13606. CreatedDate: 'Created',
  13607. ModifiedDate: 'LastSaved',
  13608. /* Pages */
  13609. /* Words */
  13610. /* Characters */
  13611. Category: 'Category',
  13612. /* PresentationFormat */
  13613. Manager: 'Manager',
  13614. Company: 'Company',
  13615. /* Guid */
  13616. /* HyperlinkBase */
  13617. /* Bytes */
  13618. /* Lines */
  13619. /* Paragraphs */
  13620. /* CharactersWithSpaces */
  13621. AppVersion: 'Version',
  13622. ContentStatus: 'ContentStatus', /* NOTE: missing from schema */
  13623. Identifier: 'Identifier', /* NOTE: missing from schema */
  13624. Language: 'Language' /* NOTE: missing from schema */
  13625. };
  13626. var evert_XLMLDPM = evert(XLMLDocPropsMap);
  13627. function xlml_set_prop(Props, tag, val) {
  13628. tag = evert_XLMLDPM[tag] || tag;
  13629. Props[tag] = val;
  13630. }
  13631. function xlml_write_docprops(Props, opts) {
  13632. var o = [];
  13633. keys(XLMLDocPropsMap).map(function(m) {
  13634. for(var i = 0; i < CORE_PROPS.length; ++i) if(CORE_PROPS[i][1] == m) return CORE_PROPS[i];
  13635. for(i = 0; i < EXT_PROPS.length; ++i) if(EXT_PROPS[i][1] == m) return EXT_PROPS[i];
  13636. throw m;
  13637. }).forEach(function(p) {
  13638. if(Props[p[1]] == null) return;
  13639. var m = opts && opts.Props && opts.Props[p[1]] != null ? opts.Props[p[1]] : Props[p[1]];
  13640. switch(p[2]) {
  13641. case 'date': m = new Date(m).toISOString().replace(/\.\d*Z/,"Z"); break;
  13642. }
  13643. if(typeof m == 'number') m = String(m);
  13644. else if(m === true || m === false) { m = m ? "1" : "0"; }
  13645. else if(m instanceof Date) m = new Date(m).toISOString().replace(/\.\d*Z/,"");
  13646. o.push(writetag(XLMLDocPropsMap[p[1]] || p[1], m));
  13647. });
  13648. return writextag('DocumentProperties', o.join(""), {xmlns:XLMLNS.o });
  13649. }
  13650. function xlml_write_custprops(Props, Custprops) {
  13651. var BLACKLIST = ["Worksheets","SheetNames"];
  13652. var T = 'CustomDocumentProperties';
  13653. var o = [];
  13654. if(Props) keys(Props).forEach(function(k) {
  13655. if(!Props.hasOwnProperty(k)) return;
  13656. for(var i = 0; i < CORE_PROPS.length; ++i) if(k == CORE_PROPS[i][1]) return;
  13657. for(i = 0; i < EXT_PROPS.length; ++i) if(k == EXT_PROPS[i][1]) return;
  13658. for(i = 0; i < BLACKLIST.length; ++i) if(k == BLACKLIST[i]) return;
  13659. var m = Props[k];
  13660. var t = "string";
  13661. if(typeof m == 'number') { t = "float"; m = String(m); }
  13662. else if(m === true || m === false) { t = "boolean"; m = m ? "1" : "0"; }
  13663. else m = String(m);
  13664. o.push(writextag(escapexmltag(k), m, {"dt:dt":t}));
  13665. });
  13666. if(Custprops) keys(Custprops).forEach(function(k) {
  13667. if(!Custprops.hasOwnProperty(k)) return;
  13668. if(Props && Props.hasOwnProperty(k)) return;
  13669. var m = Custprops[k];
  13670. var t = "string";
  13671. if(typeof m == 'number') { t = "float"; m = String(m); }
  13672. else if(m === true || m === false) { t = "boolean"; m = m ? "1" : "0"; }
  13673. else if(m instanceof Date) { t = "dateTime.tz"; m = m.toISOString(); }
  13674. else m = String(m);
  13675. o.push(writextag(escapexmltag(k), m, {"dt:dt":t}));
  13676. });
  13677. return '<' + T + ' xmlns="' + XLMLNS.o + '">' + o.join("") + '</' + T + '>';
  13678. }
  13679. /* [MS-DTYP] 2.3.3 FILETIME */
  13680. /* [MS-OLEDS] 2.1.3 FILETIME (Packet Version) */
  13681. /* [MS-OLEPS] 2.8 FILETIME (Packet Version) */
  13682. function parse_FILETIME(blob) {
  13683. var dwLowDateTime = blob.read_shift(4), dwHighDateTime = blob.read_shift(4);
  13684. return new Date(((dwHighDateTime/1e7*Math.pow(2,32) + dwLowDateTime/1e7) - 11644473600)*1000).toISOString().replace(/\.000/,"");
  13685. }
  13686. function write_FILETIME(time) {
  13687. var date = (typeof time == "string") ? new Date(Date.parse(time)) : time;
  13688. var t = date.getTime() / 1000 + 11644473600;
  13689. var l = t % Math.pow(2,32), h = (t - l) / Math.pow(2,32);
  13690. l *= 1e7; h *= 1e7;
  13691. var w = (l / Math.pow(2,32)) | 0;
  13692. if(w > 0) { l = l % Math.pow(2,32); h += w; }
  13693. var o = new_buf(8); o.write_shift(4, l); o.write_shift(4, h); return o;
  13694. }
  13695. /* [MS-OSHARED] 2.3.3.1.4 Lpstr */
  13696. function parse_lpstr(blob, type, pad) {
  13697. var start = blob.l;
  13698. var str = blob.read_shift(0, 'lpstr-cp');
  13699. if(pad) while((blob.l - start) & 3) ++blob.l;
  13700. return str;
  13701. }
  13702. /* [MS-OSHARED] 2.3.3.1.6 Lpwstr */
  13703. function parse_lpwstr(blob, type, pad) {
  13704. var str = blob.read_shift(0, 'lpwstr');
  13705. if(pad) blob.l += (4 - ((str.length+1) & 3)) & 3;
  13706. return str;
  13707. }
  13708. /* [MS-OSHARED] 2.3.3.1.11 VtString */
  13709. /* [MS-OSHARED] 2.3.3.1.12 VtUnalignedString */
  13710. function parse_VtStringBase(blob, stringType, pad) {
  13711. if(stringType === 0x1F /*VT_LPWSTR*/) return parse_lpwstr(blob);
  13712. return parse_lpstr(blob, stringType, pad);
  13713. }
  13714. function parse_VtString(blob, t, pad) { return parse_VtStringBase(blob, t, pad === false ? 0: 4); }
  13715. function parse_VtUnalignedString(blob, t) { if(!t) throw new Error("VtUnalignedString must have positive length"); return parse_VtStringBase(blob, t, 0); }
  13716. /* [MS-OSHARED] 2.3.3.1.9 VtVecUnalignedLpstrValue */
  13717. function parse_VtVecUnalignedLpstrValue(blob) {
  13718. var length = blob.read_shift(4);
  13719. var ret = [];
  13720. for(var i = 0; i != length; ++i) ret[i] = blob.read_shift(0, 'lpstr-cp').replace(chr0,'');
  13721. return ret;
  13722. }
  13723. /* [MS-OSHARED] 2.3.3.1.10 VtVecUnalignedLpstr */
  13724. function parse_VtVecUnalignedLpstr(blob) {
  13725. return parse_VtVecUnalignedLpstrValue(blob);
  13726. }
  13727. /* [MS-OSHARED] 2.3.3.1.13 VtHeadingPair */
  13728. function parse_VtHeadingPair(blob) {
  13729. var headingString = parse_TypedPropertyValue(blob, VT_USTR);
  13730. var headerParts = parse_TypedPropertyValue(blob, VT_I4);
  13731. return [headingString, headerParts];
  13732. }
  13733. /* [MS-OSHARED] 2.3.3.1.14 VtVecHeadingPairValue */
  13734. function parse_VtVecHeadingPairValue(blob) {
  13735. var cElements = blob.read_shift(4);
  13736. var out = [];
  13737. for(var i = 0; i != cElements / 2; ++i) out.push(parse_VtHeadingPair(blob));
  13738. return out;
  13739. }
  13740. /* [MS-OSHARED] 2.3.3.1.15 VtVecHeadingPair */
  13741. function parse_VtVecHeadingPair(blob) {
  13742. // NOTE: When invoked, wType & padding were already consumed
  13743. return parse_VtVecHeadingPairValue(blob);
  13744. }
  13745. /* [MS-OLEPS] 2.18.1 Dictionary (uses 2.17, 2.16) */
  13746. function parse_dictionary(blob,CodePage) {
  13747. var cnt = blob.read_shift(4);
  13748. var dict = ({});
  13749. for(var j = 0; j != cnt; ++j) {
  13750. var pid = blob.read_shift(4);
  13751. var len = blob.read_shift(4);
  13752. dict[pid] = blob.read_shift(len, (CodePage === 0x4B0 ?'utf16le':'utf8')).replace(chr0,'').replace(chr1,'!');
  13753. if(CodePage === 0x4B0 && (len % 2)) blob.l += 2;
  13754. }
  13755. if(blob.l & 3) blob.l = (blob.l>>2+1)<<2;
  13756. return dict;
  13757. }
  13758. /* [MS-OLEPS] 2.9 BLOB */
  13759. function parse_BLOB(blob) {
  13760. var size = blob.read_shift(4);
  13761. var bytes = blob.slice(blob.l,blob.l+size);
  13762. blob.l += size;
  13763. if((size & 3) > 0) blob.l += (4 - (size & 3)) & 3;
  13764. return bytes;
  13765. }
  13766. /* [MS-OLEPS] 2.11 ClipboardData */
  13767. function parse_ClipboardData(blob) {
  13768. // TODO
  13769. var o = {};
  13770. o.Size = blob.read_shift(4);
  13771. //o.Format = blob.read_shift(4);
  13772. blob.l += o.Size + 3 - (o.Size - 1) % 4;
  13773. return o;
  13774. }
  13775. /* [MS-OLEPS] 2.15 TypedPropertyValue */
  13776. function parse_TypedPropertyValue(blob, type, _opts) {
  13777. var t = blob.read_shift(2), ret, opts = _opts||{};
  13778. blob.l += 2;
  13779. if(type !== VT_VARIANT)
  13780. if(t !== type && VT_CUSTOM.indexOf(type)===-1) throw new Error('Expected type ' + type + ' saw ' + t);
  13781. switch(type === VT_VARIANT ? t : type) {
  13782. case 0x02 /*VT_I2*/: ret = blob.read_shift(2, 'i'); if(!opts.raw) blob.l += 2; return ret;
  13783. case 0x03 /*VT_I4*/: ret = blob.read_shift(4, 'i'); return ret;
  13784. case 0x0B /*VT_BOOL*/: return blob.read_shift(4) !== 0x0;
  13785. case 0x13 /*VT_UI4*/: ret = blob.read_shift(4); return ret;
  13786. case 0x1E /*VT_LPSTR*/: return parse_lpstr(blob, t, 4).replace(chr0,'');
  13787. case 0x1F /*VT_LPWSTR*/: return parse_lpwstr(blob);
  13788. case 0x40 /*VT_FILETIME*/: return parse_FILETIME(blob);
  13789. case 0x41 /*VT_BLOB*/: return parse_BLOB(blob);
  13790. case 0x47 /*VT_CF*/: return parse_ClipboardData(blob);
  13791. case 0x50 /*VT_STRING*/: return parse_VtString(blob, t, !opts.raw).replace(chr0,'');
  13792. case 0x51 /*VT_USTR*/: return parse_VtUnalignedString(blob, t/*, 4*/).replace(chr0,'');
  13793. case 0x100C /*VT_VECTOR|VT_VARIANT*/: return parse_VtVecHeadingPair(blob);
  13794. case 0x101E /*VT_LPSTR*/: return parse_VtVecUnalignedLpstr(blob);
  13795. default: throw new Error("TypedPropertyValue unrecognized type " + type + " " + t);
  13796. }
  13797. }
  13798. function write_TypedPropertyValue(type, value) {
  13799. var o = new_buf(4), p = new_buf(4);
  13800. o.write_shift(4, type == 0x50 ? 0x1F : type);
  13801. switch(type) {
  13802. case 0x03 /*VT_I4*/: p.write_shift(-4, value); break;
  13803. case 0x05 /*VT_I4*/: p = new_buf(8); p.write_shift(8, value, 'f'); break;
  13804. case 0x0B /*VT_BOOL*/: p.write_shift(4, value ? 0x01 : 0x00); break;
  13805. case 0x40 /*VT_FILETIME*/: p = write_FILETIME(value); break;
  13806. case 0x1F /*VT_LPWSTR*/:
  13807. case 0x50 /*VT_STRING*/:
  13808. p = new_buf(4 + 2 * (value.length + 1) + (value.length % 2 ? 0 : 2));
  13809. p.write_shift(4, value.length + 1);
  13810. p.write_shift(0, value, "dbcs");
  13811. while(p.l != p.length) p.write_shift(1, 0);
  13812. break;
  13813. default: throw new Error("TypedPropertyValue unrecognized type " + type + " " + value);
  13814. }
  13815. return bconcat([o, p]);
  13816. }
  13817. /* [MS-OLEPS] 2.20 PropertySet */
  13818. function parse_PropertySet(blob, PIDSI) {
  13819. var start_addr = blob.l;
  13820. var size = blob.read_shift(4);
  13821. var NumProps = blob.read_shift(4);
  13822. var Props = [], i = 0;
  13823. var CodePage = 0;
  13824. var Dictionary = -1, DictObj = ({});
  13825. for(i = 0; i != NumProps; ++i) {
  13826. var PropID = blob.read_shift(4);
  13827. var Offset = blob.read_shift(4);
  13828. Props[i] = [PropID, Offset + start_addr];
  13829. }
  13830. Props.sort(function(x,y) { return x[1] - y[1]; });
  13831. var PropH = {};
  13832. for(i = 0; i != NumProps; ++i) {
  13833. if(blob.l !== Props[i][1]) {
  13834. var fail = true;
  13835. if(i>0 && PIDSI) switch(PIDSI[Props[i-1][0]].t) {
  13836. case 0x02 /*VT_I2*/: if(blob.l+2 === Props[i][1]) { blob.l+=2; fail = false; } break;
  13837. case 0x50 /*VT_STRING*/: if(blob.l <= Props[i][1]) { blob.l=Props[i][1]; fail = false; } break;
  13838. case 0x100C /*VT_VECTOR|VT_VARIANT*/: if(blob.l <= Props[i][1]) { blob.l=Props[i][1]; fail = false; } break;
  13839. }
  13840. if((!PIDSI||i==0) && blob.l <= Props[i][1]) { fail=false; blob.l = Props[i][1]; }
  13841. if(fail) throw new Error("Read error: Expected address " + Props[i][1] + ' at ' + blob.l + ' :' + i);
  13842. }
  13843. if(PIDSI) {
  13844. var piddsi = PIDSI[Props[i][0]];
  13845. PropH[piddsi.n] = parse_TypedPropertyValue(blob, piddsi.t, {raw:true});
  13846. if(piddsi.p === 'version') PropH[piddsi.n] = String(PropH[piddsi.n] >> 16) + "." + ("0000" + String(PropH[piddsi.n] & 0xFFFF)).slice(-4);
  13847. if(piddsi.n == "CodePage") switch(PropH[piddsi.n]) {
  13848. case 0: PropH[piddsi.n] = 1252;
  13849. /* falls through */
  13850. case 874:
  13851. case 932:
  13852. case 936:
  13853. case 949:
  13854. case 950:
  13855. case 1250:
  13856. case 1251:
  13857. case 1253:
  13858. case 1254:
  13859. case 1255:
  13860. case 1256:
  13861. case 1257:
  13862. case 1258:
  13863. case 10000:
  13864. case 1200:
  13865. case 1201:
  13866. case 1252:
  13867. case 65000: case -536:
  13868. case 65001: case -535:
  13869. set_cp(CodePage = (PropH[piddsi.n]>>>0) & 0xFFFF); break;
  13870. default: throw new Error("Unsupported CodePage: " + PropH[piddsi.n]);
  13871. }
  13872. } else {
  13873. if(Props[i][0] === 0x1) {
  13874. CodePage = PropH.CodePage = (parse_TypedPropertyValue(blob, VT_I2));
  13875. set_cp(CodePage);
  13876. if(Dictionary !== -1) {
  13877. var oldpos = blob.l;
  13878. blob.l = Props[Dictionary][1];
  13879. DictObj = parse_dictionary(blob,CodePage);
  13880. blob.l = oldpos;
  13881. }
  13882. } else if(Props[i][0] === 0) {
  13883. if(CodePage === 0) { Dictionary = i; blob.l = Props[i+1][1]; continue; }
  13884. DictObj = parse_dictionary(blob,CodePage);
  13885. } else {
  13886. var name = DictObj[Props[i][0]];
  13887. var val;
  13888. /* [MS-OSHARED] 2.3.3.2.3.1.2 + PROPVARIANT */
  13889. switch(blob[blob.l]) {
  13890. case 0x41 /*VT_BLOB*/: blob.l += 4; val = parse_BLOB(blob); break;
  13891. case 0x1E /*VT_LPSTR*/: blob.l += 4; val = parse_VtString(blob, blob[blob.l-4]).replace(/\u0000+$/,""); break;
  13892. case 0x1F /*VT_LPWSTR*/: blob.l += 4; val = parse_VtString(blob, blob[blob.l-4]).replace(/\u0000+$/,""); break;
  13893. case 0x03 /*VT_I4*/: blob.l += 4; val = blob.read_shift(4, 'i'); break;
  13894. case 0x13 /*VT_UI4*/: blob.l += 4; val = blob.read_shift(4); break;
  13895. case 0x05 /*VT_R8*/: blob.l += 4; val = blob.read_shift(8, 'f'); break;
  13896. case 0x0B /*VT_BOOL*/: blob.l += 4; val = parsebool(blob, 4); break;
  13897. case 0x40 /*VT_FILETIME*/: blob.l += 4; val = parseDate(parse_FILETIME(blob)); break;
  13898. default: throw new Error("unparsed value: " + blob[blob.l]);
  13899. }
  13900. PropH[name] = val;
  13901. }
  13902. }
  13903. }
  13904. blob.l = start_addr + size; /* step ahead to skip padding */
  13905. return PropH;
  13906. }
  13907. var XLSPSSkip = [ "CodePage", "Thumbnail", "_PID_LINKBASE", "_PID_HLINKS", "SystemIdentifier", "FMTID" ].concat(PseudoPropsPairs);
  13908. function guess_property_type(val) {
  13909. switch(typeof val) {
  13910. case "boolean": return 0x0B;
  13911. case "number": return ((val|0)==val) ? 0x03 : 0x05;
  13912. case "string": return 0x1F;
  13913. case "object": if(val instanceof Date) return 0x40; break;
  13914. }
  13915. return -1;
  13916. }
  13917. function write_PropertySet(entries, RE, PIDSI) {
  13918. var hdr = new_buf(8), piao = [], prop = [];
  13919. var sz = 8, i = 0;
  13920. var pr = new_buf(8), pio = new_buf(8);
  13921. pr.write_shift(4, 0x0002);
  13922. pr.write_shift(4, 0x04B0);
  13923. pio.write_shift(4, 0x0001);
  13924. prop.push(pr); piao.push(pio);
  13925. sz += 8 + pr.length;
  13926. if(!RE) {
  13927. pio = new_buf(8);
  13928. pio.write_shift(4, 0);
  13929. piao.unshift(pio);
  13930. var bufs = [new_buf(4)];
  13931. bufs[0].write_shift(4, entries.length);
  13932. for(i = 0; i < entries.length; ++i) {
  13933. var value = entries[i][0];
  13934. pr = new_buf(4 + 4 + 2 * (value.length + 1) + (value.length % 2 ? 0 : 2));
  13935. pr.write_shift(4, i+2);
  13936. pr.write_shift(4, value.length + 1);
  13937. pr.write_shift(0, value, "dbcs");
  13938. while(pr.l != pr.length) pr.write_shift(1, 0);
  13939. bufs.push(pr);
  13940. }
  13941. pr = bconcat(bufs);
  13942. prop.unshift(pr);
  13943. sz += 8 + pr.length;
  13944. }
  13945. for(i = 0; i < entries.length; ++i) {
  13946. if(RE && !RE[entries[i][0]]) continue;
  13947. if(XLSPSSkip.indexOf(entries[i][0]) > -1) continue;
  13948. if(entries[i][1] == null) continue;
  13949. var val = entries[i][1], idx = 0;
  13950. if(RE) {
  13951. idx = +RE[entries[i][0]];
  13952. var pinfo = (PIDSI)[idx];
  13953. if(pinfo.p == "version" && typeof val == "string") {
  13954. var arr = val.split(".");
  13955. val = ((+arr[0])<<16) + ((+arr[1])||0);
  13956. }
  13957. pr = write_TypedPropertyValue(pinfo.t, val);
  13958. } else {
  13959. var T = guess_property_type(val);
  13960. if(T == -1) { T = 0x1F; val = String(val); }
  13961. pr = write_TypedPropertyValue(T, val);
  13962. }
  13963. prop.push(pr);
  13964. pio = new_buf(8);
  13965. pio.write_shift(4, !RE ? 2+i : idx);
  13966. piao.push(pio);
  13967. sz += 8 + pr.length;
  13968. }
  13969. var w = 8 * (prop.length + 1);
  13970. for(i = 0; i < prop.length; ++i) { piao[i].write_shift(4, w); w += prop[i].length; }
  13971. hdr.write_shift(4, sz);
  13972. hdr.write_shift(4, prop.length);
  13973. return bconcat([hdr].concat(piao).concat(prop));
  13974. }
  13975. /* [MS-OLEPS] 2.21 PropertySetStream */
  13976. function parse_PropertySetStream(file, PIDSI, clsid) {
  13977. var blob = file.content;
  13978. if(!blob) return ({});
  13979. prep_blob(blob, 0);
  13980. var NumSets, FMTID0, FMTID1, Offset0, Offset1 = 0;
  13981. blob.chk('feff', 'Byte Orders: ');
  13982. /*var vers = */blob.read_shift(2); // TODO: check version
  13983. var SystemIdentifier = blob.read_shift(4);
  13984. var CLSID = blob.read_shift(16);
  13985. if(CLSID !== CFB.utils.consts.HEADER_CLSID && CLSID !== clsid) throw new Error("Bad PropertySet CLSID " + CLSID);
  13986. NumSets = blob.read_shift(4);
  13987. if(NumSets !== 1 && NumSets !== 2) throw new Error("Unrecognized #Sets: " + NumSets);
  13988. FMTID0 = blob.read_shift(16); Offset0 = blob.read_shift(4);
  13989. if(NumSets === 1 && Offset0 !== blob.l) throw new Error("Length mismatch: " + Offset0 + " !== " + blob.l);
  13990. else if(NumSets === 2) { FMTID1 = blob.read_shift(16); Offset1 = blob.read_shift(4); }
  13991. var PSet0 = parse_PropertySet(blob, PIDSI);
  13992. var rval = ({ SystemIdentifier: SystemIdentifier });
  13993. for(var y in PSet0) rval[y] = PSet0[y];
  13994. //rval.blob = blob;
  13995. rval.FMTID = FMTID0;
  13996. //rval.PSet0 = PSet0;
  13997. if(NumSets === 1) return rval;
  13998. if(Offset1 - blob.l == 2) blob.l += 2;
  13999. if(blob.l !== Offset1) throw new Error("Length mismatch 2: " + blob.l + " !== " + Offset1);
  14000. var PSet1;
  14001. try { PSet1 = parse_PropertySet(blob, null); } catch(e) {/* empty */}
  14002. for(y in PSet1) rval[y] = PSet1[y];
  14003. rval.FMTID = [FMTID0, FMTID1]; // TODO: verify FMTID0/1
  14004. return rval;
  14005. }
  14006. function write_PropertySetStream(entries, clsid, RE, PIDSI, entries2, clsid2) {
  14007. var hdr = new_buf(entries2 ? 68 : 48);
  14008. var bufs = [hdr];
  14009. hdr.write_shift(2, 0xFFFE);
  14010. hdr.write_shift(2, 0x0000); /* TODO: type 1 props */
  14011. hdr.write_shift(4, 0x32363237);
  14012. hdr.write_shift(16, CFB.utils.consts.HEADER_CLSID, "hex");
  14013. hdr.write_shift(4, (entries2 ? 2 : 1));
  14014. hdr.write_shift(16, clsid, "hex");
  14015. hdr.write_shift(4, (entries2 ? 68 : 48));
  14016. var ps0 = write_PropertySet(entries, RE, PIDSI);
  14017. bufs.push(ps0);
  14018. if(entries2) {
  14019. var ps1 = write_PropertySet(entries2, null, null);
  14020. hdr.write_shift(16, clsid2, "hex");
  14021. hdr.write_shift(4, 68 + ps0.length);
  14022. bufs.push(ps1);
  14023. }
  14024. return bconcat(bufs);
  14025. }
  14026. function parsenoop2(blob, length) { blob.read_shift(length); return null; }
  14027. function writezeroes(n, o) { if(!o) o=new_buf(n); for(var j=0; j<n; ++j) o.write_shift(1, 0); return o; }
  14028. function parslurp(blob, length, cb) {
  14029. var arr = [], target = blob.l + length;
  14030. while(blob.l < target) arr.push(cb(blob, target - blob.l));
  14031. if(target !== blob.l) throw new Error("Slurp error");
  14032. return arr;
  14033. }
  14034. function parsebool(blob, length) { return blob.read_shift(length) === 0x1; }
  14035. function writebool(v, o) { if(!o) o=new_buf(2); o.write_shift(2, +!!v); return o; }
  14036. function parseuint16(blob) { return blob.read_shift(2, 'u'); }
  14037. function writeuint16(v, o) { if(!o) o=new_buf(2); o.write_shift(2, v); return o; }
  14038. function parseuint16a(blob, length) { return parslurp(blob,length,parseuint16);}
  14039. /* --- 2.5 Structures --- */
  14040. /* [MS-XLS] 2.5.10 Bes (boolean or error) */
  14041. function parse_Bes(blob) {
  14042. var v = blob.read_shift(1), t = blob.read_shift(1);
  14043. return t === 0x01 ? v : v === 0x01;
  14044. }
  14045. function write_Bes(v, t, o) {
  14046. if(!o) o = new_buf(2);
  14047. o.write_shift(1, +v);
  14048. o.write_shift(1, ((t == 'e') ? 1 : 0));
  14049. return o;
  14050. }
  14051. /* [MS-XLS] 2.5.240 ShortXLUnicodeString */
  14052. function parse_ShortXLUnicodeString(blob, length, opts) {
  14053. var cch = blob.read_shift(opts && opts.biff >= 12 ? 2 : 1);
  14054. var encoding = 'sbcs-cont';
  14055. var cp = current_codepage;
  14056. if(opts && opts.biff >= 8) current_codepage = 1200;
  14057. if(!opts || opts.biff == 8 ) {
  14058. var fHighByte = blob.read_shift(1);
  14059. if(fHighByte) { encoding = 'dbcs-cont'; }
  14060. } else if(opts.biff == 12) {
  14061. encoding = 'wstr';
  14062. }
  14063. if(opts.biff >= 2 && opts.biff <= 5) encoding = 'cpstr';
  14064. var o = cch ? blob.read_shift(cch, encoding) : "";
  14065. current_codepage = cp;
  14066. return o;
  14067. }
  14068. /* 2.5.293 XLUnicodeRichExtendedString */
  14069. function parse_XLUnicodeRichExtendedString(blob) {
  14070. var cp = current_codepage;
  14071. current_codepage = 1200;
  14072. var cch = blob.read_shift(2), flags = blob.read_shift(1);
  14073. var /*fHighByte = flags & 0x1,*/ fExtSt = flags & 0x4, fRichSt = flags & 0x8;
  14074. var width = 1 + (flags & 0x1); // 0x0 -> utf8, 0x1 -> dbcs
  14075. var cRun = 0, cbExtRst;
  14076. var z = {};
  14077. if(fRichSt) cRun = blob.read_shift(2);
  14078. if(fExtSt) cbExtRst = blob.read_shift(4);
  14079. var encoding = width == 2 ? 'dbcs-cont' : 'sbcs-cont';
  14080. var msg = cch === 0 ? "" : blob.read_shift(cch, encoding);
  14081. if(fRichSt) blob.l += 4 * cRun; //TODO: parse this
  14082. if(fExtSt) blob.l += cbExtRst; //TODO: parse this
  14083. z.t = msg;
  14084. if(!fRichSt) { z.raw = "<t>" + z.t + "</t>"; z.r = z.t; }
  14085. current_codepage = cp;
  14086. return z;
  14087. }
  14088. /* 2.5.296 XLUnicodeStringNoCch */
  14089. function parse_XLUnicodeStringNoCch(blob, cch, opts) {
  14090. var retval;
  14091. if(opts) {
  14092. if(opts.biff >= 2 && opts.biff <= 5) return blob.read_shift(cch, 'cpstr');
  14093. if(opts.biff >= 12) return blob.read_shift(cch, 'dbcs-cont');
  14094. }
  14095. var fHighByte = blob.read_shift(1);
  14096. if(fHighByte===0) { retval = blob.read_shift(cch, 'sbcs-cont'); }
  14097. else { retval = blob.read_shift(cch, 'dbcs-cont'); }
  14098. return retval;
  14099. }
  14100. /* 2.5.294 XLUnicodeString */
  14101. function parse_XLUnicodeString(blob, length, opts) {
  14102. var cch = blob.read_shift(opts && opts.biff == 2 ? 1 : 2);
  14103. if(cch === 0) { blob.l++; return ""; }
  14104. return parse_XLUnicodeStringNoCch(blob, cch, opts);
  14105. }
  14106. /* BIFF5 override */
  14107. function parse_XLUnicodeString2(blob, length, opts) {
  14108. if(opts.biff > 5) return parse_XLUnicodeString(blob, length, opts);
  14109. var cch = blob.read_shift(1);
  14110. if(cch === 0) { blob.l++; return ""; }
  14111. return blob.read_shift(cch, (opts.biff <= 4 || !blob.lens ) ? 'cpstr' : 'sbcs-cont');
  14112. }
  14113. /* TODO: BIFF5 and lower, codepage awareness */
  14114. function write_XLUnicodeString(str, opts, o) {
  14115. if(!o) o = new_buf(3 + 2 * str.length);
  14116. o.write_shift(2, str.length);
  14117. o.write_shift(1, 1);
  14118. o.write_shift(31, str, 'utf16le');
  14119. return o;
  14120. }
  14121. /* [MS-XLS] 2.5.61 ControlInfo */
  14122. function parse_ControlInfo(blob) {
  14123. var flags = blob.read_shift(1);
  14124. blob.l++;
  14125. var accel = blob.read_shift(2);
  14126. blob.l += 2;
  14127. return [flags, accel];
  14128. }
  14129. /* [MS-OSHARED] 2.3.7.6 URLMoniker TODO: flags */
  14130. function parse_URLMoniker(blob) {
  14131. var len = blob.read_shift(4), start = blob.l;
  14132. var extra = false;
  14133. if(len > 24) {
  14134. /* look ahead */
  14135. blob.l += len - 24;
  14136. if(blob.read_shift(16) === "795881f43b1d7f48af2c825dc4852763") extra = true;
  14137. blob.l = start;
  14138. }
  14139. var url = blob.read_shift((extra?len-24:len)>>1, 'utf16le').replace(chr0,"");
  14140. if(extra) blob.l += 24;
  14141. return url;
  14142. }
  14143. /* [MS-OSHARED] 2.3.7.8 FileMoniker TODO: all fields */
  14144. function parse_FileMoniker(blob) {
  14145. blob.l += 2; //var cAnti = blob.read_shift(2);
  14146. var ansiPath = blob.read_shift(0, 'lpstr-ansi');
  14147. blob.l += 2; //var endServer = blob.read_shift(2);
  14148. if(blob.read_shift(2) != 0xDEAD) throw new Error("Bad FileMoniker");
  14149. var sz = blob.read_shift(4);
  14150. if(sz === 0) return ansiPath.replace(/\\/g,"/");
  14151. var bytes = blob.read_shift(4);
  14152. if(blob.read_shift(2) != 3) throw new Error("Bad FileMoniker");
  14153. var unicodePath = blob.read_shift(bytes>>1, 'utf16le').replace(chr0,"");
  14154. return unicodePath;
  14155. }
  14156. /* [MS-OSHARED] 2.3.7.2 HyperlinkMoniker TODO: all the monikers */
  14157. function parse_HyperlinkMoniker(blob, length) {
  14158. var clsid = blob.read_shift(16); length -= 16;
  14159. switch(clsid) {
  14160. case "e0c9ea79f9bace118c8200aa004ba90b": return parse_URLMoniker(blob, length);
  14161. case "0303000000000000c000000000000046": return parse_FileMoniker(blob, length);
  14162. default: throw new Error("Unsupported Moniker " + clsid);
  14163. }
  14164. }
  14165. /* [MS-OSHARED] 2.3.7.9 HyperlinkString */
  14166. function parse_HyperlinkString(blob) {
  14167. var len = blob.read_shift(4);
  14168. var o = len > 0 ? blob.read_shift(len, 'utf16le').replace(chr0, "") : "";
  14169. return o;
  14170. }
  14171. /* [MS-OSHARED] 2.3.7.1 Hyperlink Object */
  14172. function parse_Hyperlink(blob, length) {
  14173. var end = blob.l + length;
  14174. var sVer = blob.read_shift(4);
  14175. if(sVer !== 2) throw new Error("Unrecognized streamVersion: " + sVer);
  14176. var flags = blob.read_shift(2);
  14177. blob.l += 2;
  14178. var displayName, targetFrameName, moniker, oleMoniker, Loc="", guid, fileTime;
  14179. if(flags & 0x0010) displayName = parse_HyperlinkString(blob, end - blob.l);
  14180. if(flags & 0x0080) targetFrameName = parse_HyperlinkString(blob, end - blob.l);
  14181. if((flags & 0x0101) === 0x0101) moniker = parse_HyperlinkString(blob, end - blob.l);
  14182. if((flags & 0x0101) === 0x0001) oleMoniker = parse_HyperlinkMoniker(blob, end - blob.l);
  14183. if(flags & 0x0008) Loc = parse_HyperlinkString(blob, end - blob.l);
  14184. if(flags & 0x0020) guid = blob.read_shift(16);
  14185. if(flags & 0x0040) fileTime = parse_FILETIME(blob/*, 8*/);
  14186. blob.l = end;
  14187. var target = targetFrameName||moniker||oleMoniker||"";
  14188. if(target && Loc) target+="#"+Loc;
  14189. if(!target) target = "#" + Loc;
  14190. var out = ({Target:target});
  14191. if(guid) out.guid = guid;
  14192. if(fileTime) out.time = fileTime;
  14193. if(displayName) out.Tooltip = displayName;
  14194. return out;
  14195. }
  14196. function write_Hyperlink(hl) {
  14197. var out = new_buf(512), i = 0;
  14198. var Target = hl.Target;
  14199. var F = Target.indexOf("#") > -1 ? 0x1f : 0x17;
  14200. switch(Target.charAt(0)) { case "#": F=0x1c; break; case ".": F&=~2; break; }
  14201. out.write_shift(4,2); out.write_shift(4, F);
  14202. var data = [8,6815827,6619237,4849780,83]; for(i = 0; i < data.length; ++i) out.write_shift(4, data[i]);
  14203. if(F == 0x1C) {
  14204. Target = Target.slice(1);
  14205. out.write_shift(4, Target.length + 1);
  14206. for(i = 0; i < Target.length; ++i) out.write_shift(2, Target.charCodeAt(i));
  14207. out.write_shift(2, 0);
  14208. } else if(F & 0x02) {
  14209. data = "e0 c9 ea 79 f9 ba ce 11 8c 82 00 aa 00 4b a9 0b".split(" ");
  14210. for(i = 0; i < data.length; ++i) out.write_shift(1, parseInt(data[i], 16));
  14211. out.write_shift(4, 2*(Target.length + 1));
  14212. for(i = 0; i < Target.length; ++i) out.write_shift(2, Target.charCodeAt(i));
  14213. out.write_shift(2, 0);
  14214. } else {
  14215. data = "03 03 00 00 00 00 00 00 c0 00 00 00 00 00 00 46".split(" ");
  14216. for(i = 0; i < data.length; ++i) out.write_shift(1, parseInt(data[i], 16));
  14217. var P = 0;
  14218. while(Target.slice(P*3,P*3+3)=="../"||Target.slice(P*3,P*3+3)=="..\\") ++P;
  14219. out.write_shift(2, P);
  14220. out.write_shift(4, Target.length + 1);
  14221. for(i = 0; i < Target.length; ++i) out.write_shift(1, Target.charCodeAt(i) & 0xFF);
  14222. out.write_shift(1, 0);
  14223. out.write_shift(2, 0xFFFF);
  14224. out.write_shift(2, 0xDEAD);
  14225. for(i = 0; i < 6; ++i) out.write_shift(4, 0);
  14226. }
  14227. return out.slice(0, out.l);
  14228. }
  14229. /* 2.5.178 LongRGBA */
  14230. function parse_LongRGBA(blob) { var r = blob.read_shift(1), g = blob.read_shift(1), b = blob.read_shift(1), a = blob.read_shift(1); return [r,g,b,a]; }
  14231. /* 2.5.177 LongRGB */
  14232. function parse_LongRGB(blob, length) { var x = parse_LongRGBA(blob, length); x[3] = 0; return x; }
  14233. /* [MS-XLS] 2.5.19 */
  14234. function parse_XLSCell(blob) {
  14235. var rw = blob.read_shift(2); // 0-indexed
  14236. var col = blob.read_shift(2);
  14237. var ixfe = blob.read_shift(2);
  14238. return ({r:rw, c:col, ixfe:ixfe});
  14239. }
  14240. function write_XLSCell(R, C, ixfe, o) {
  14241. if(!o) o = new_buf(6);
  14242. o.write_shift(2, R);
  14243. o.write_shift(2, C);
  14244. o.write_shift(2, ixfe||0);
  14245. return o;
  14246. }
  14247. /* [MS-XLS] 2.5.134 */
  14248. function parse_frtHeader(blob) {
  14249. var rt = blob.read_shift(2);
  14250. var flags = blob.read_shift(2); // TODO: parse these flags
  14251. blob.l += 8;
  14252. return {type: rt, flags: flags};
  14253. }
  14254. function parse_OptXLUnicodeString(blob, length, opts) { return length === 0 ? "" : parse_XLUnicodeString2(blob, length, opts); }
  14255. /* [MS-XLS] 2.5.344 */
  14256. function parse_XTI(blob, length, opts) {
  14257. var w = opts.biff > 8 ? 4 : 2;
  14258. var iSupBook = blob.read_shift(w), itabFirst = blob.read_shift(w,'i'), itabLast = blob.read_shift(w,'i');
  14259. return [iSupBook, itabFirst, itabLast];
  14260. }
  14261. /* [MS-XLS] 2.5.218 */
  14262. function parse_RkRec(blob) {
  14263. var ixfe = blob.read_shift(2);
  14264. var RK = parse_RkNumber(blob);
  14265. return [ixfe, RK];
  14266. }
  14267. /* [MS-XLS] 2.5.1 */
  14268. function parse_AddinUdf(blob, length, opts) {
  14269. blob.l += 4; length -= 4;
  14270. var l = blob.l + length;
  14271. var udfName = parse_ShortXLUnicodeString(blob, length, opts);
  14272. var cb = blob.read_shift(2);
  14273. l -= blob.l;
  14274. if(cb !== l) throw new Error("Malformed AddinUdf: padding = " + l + " != " + cb);
  14275. blob.l += cb;
  14276. return udfName;
  14277. }
  14278. /* [MS-XLS] 2.5.209 TODO: Check sizes */
  14279. function parse_Ref8U(blob) {
  14280. var rwFirst = blob.read_shift(2);
  14281. var rwLast = blob.read_shift(2);
  14282. var colFirst = blob.read_shift(2);
  14283. var colLast = blob.read_shift(2);
  14284. return {s:{c:colFirst, r:rwFirst}, e:{c:colLast,r:rwLast}};
  14285. }
  14286. function write_Ref8U(r, o) {
  14287. if(!o) o = new_buf(8);
  14288. o.write_shift(2, r.s.r);
  14289. o.write_shift(2, r.e.r);
  14290. o.write_shift(2, r.s.c);
  14291. o.write_shift(2, r.e.c);
  14292. return o;
  14293. }
  14294. /* [MS-XLS] 2.5.211 */
  14295. function parse_RefU(blob) {
  14296. var rwFirst = blob.read_shift(2);
  14297. var rwLast = blob.read_shift(2);
  14298. var colFirst = blob.read_shift(1);
  14299. var colLast = blob.read_shift(1);
  14300. return {s:{c:colFirst, r:rwFirst}, e:{c:colLast,r:rwLast}};
  14301. }
  14302. /* [MS-XLS] 2.5.207 */
  14303. var parse_Ref = parse_RefU;
  14304. /* [MS-XLS] 2.5.143 */
  14305. function parse_FtCmo(blob) {
  14306. blob.l += 4;
  14307. var ot = blob.read_shift(2);
  14308. var id = blob.read_shift(2);
  14309. var flags = blob.read_shift(2);
  14310. blob.l+=12;
  14311. return [id, ot, flags];
  14312. }
  14313. /* [MS-XLS] 2.5.149 */
  14314. function parse_FtNts(blob) {
  14315. var out = {};
  14316. blob.l += 4;
  14317. blob.l += 16; // GUID TODO
  14318. out.fSharedNote = blob.read_shift(2);
  14319. blob.l += 4;
  14320. return out;
  14321. }
  14322. /* [MS-XLS] 2.5.142 */
  14323. function parse_FtCf(blob) {
  14324. var out = {};
  14325. blob.l += 4;
  14326. blob.cf = blob.read_shift(2);
  14327. return out;
  14328. }
  14329. /* [MS-XLS] 2.5.140 - 2.5.154 and friends */
  14330. function parse_FtSkip(blob) { blob.l += 2; blob.l += blob.read_shift(2); }
  14331. var FtTab = {
  14332. 0x00: parse_FtSkip, /* FtEnd */
  14333. 0x04: parse_FtSkip, /* FtMacro */
  14334. 0x05: parse_FtSkip, /* FtButton */
  14335. 0x06: parse_FtSkip, /* FtGmo */
  14336. 0x07: parse_FtCf, /* FtCf */
  14337. 0x08: parse_FtSkip, /* FtPioGrbit */
  14338. 0x09: parse_FtSkip, /* FtPictFmla */
  14339. 0x0A: parse_FtSkip, /* FtCbls */
  14340. 0x0B: parse_FtSkip, /* FtRbo */
  14341. 0x0C: parse_FtSkip, /* FtSbs */
  14342. 0x0D: parse_FtNts, /* FtNts */
  14343. 0x0E: parse_FtSkip, /* FtSbsFmla */
  14344. 0x0F: parse_FtSkip, /* FtGboData */
  14345. 0x10: parse_FtSkip, /* FtEdoData */
  14346. 0x11: parse_FtSkip, /* FtRboData */
  14347. 0x12: parse_FtSkip, /* FtCblsData */
  14348. 0x13: parse_FtSkip, /* FtLbsData */
  14349. 0x14: parse_FtSkip, /* FtCblsFmla */
  14350. 0x15: parse_FtCmo
  14351. };
  14352. function parse_FtArray(blob, length) {
  14353. var tgt = blob.l + length;
  14354. var fts = [];
  14355. while(blob.l < tgt) {
  14356. var ft = blob.read_shift(2);
  14357. blob.l-=2;
  14358. try {
  14359. fts.push(FtTab[ft](blob, tgt - blob.l));
  14360. } catch(e) { blob.l = tgt; return fts; }
  14361. }
  14362. if(blob.l != tgt) blob.l = tgt; //throw new error("bad Object Ft-sequence");
  14363. return fts;
  14364. }
  14365. /* --- 2.4 Records --- */
  14366. /* [MS-XLS] 2.4.21 */
  14367. function parse_BOF(blob, length) {
  14368. var o = {BIFFVer:0, dt:0};
  14369. o.BIFFVer = blob.read_shift(2); length -= 2;
  14370. if(length >= 2) { o.dt = blob.read_shift(2); blob.l -= 2; }
  14371. switch(o.BIFFVer) {
  14372. case 0x0600: /* BIFF8 */
  14373. case 0x0500: /* BIFF5 */
  14374. case 0x0400: /* BIFF4 */
  14375. case 0x0300: /* BIFF3 */
  14376. case 0x0200: /* BIFF2 */
  14377. case 0x0002: case 0x0007: /* BIFF2 */
  14378. break;
  14379. default: if(length > 6) throw new Error("Unexpected BIFF Ver " + o.BIFFVer);
  14380. }
  14381. blob.read_shift(length);
  14382. return o;
  14383. }
  14384. function write_BOF(wb, t, o) {
  14385. var h = 0x0600, w = 16;
  14386. switch(o.bookType) {
  14387. case 'biff8': break;
  14388. case 'biff5': h = 0x0500; w = 8; break;
  14389. case 'biff4': h = 0x0004; w = 6; break;
  14390. case 'biff3': h = 0x0003; w = 6; break;
  14391. case 'biff2': h = 0x0002; w = 4; break;
  14392. case 'xla': break;
  14393. default: throw new Error("unsupported BIFF version");
  14394. }
  14395. var out = new_buf(w);
  14396. out.write_shift(2, h);
  14397. out.write_shift(2, t);
  14398. if(w > 4) out.write_shift(2, 0x7262);
  14399. if(w > 6) out.write_shift(2, 0x07CD);
  14400. if(w > 8) {
  14401. out.write_shift(2, 0xC009);
  14402. out.write_shift(2, 0x0001);
  14403. out.write_shift(2, 0x0706);
  14404. out.write_shift(2, 0x0000);
  14405. }
  14406. return out;
  14407. }
  14408. /* [MS-XLS] 2.4.146 */
  14409. function parse_InterfaceHdr(blob, length) {
  14410. if(length === 0) return 0x04b0;
  14411. if((blob.read_shift(2))!==0x04b0){/* empty */}
  14412. return 0x04b0;
  14413. }
  14414. /* [MS-XLS] 2.4.349 */
  14415. function parse_WriteAccess(blob, length, opts) {
  14416. if(opts.enc) { blob.l += length; return ""; }
  14417. var l = blob.l;
  14418. // TODO: make sure XLUnicodeString doesnt overrun
  14419. var UserName = parse_XLUnicodeString2(blob, 0, opts);
  14420. blob.read_shift(length + l - blob.l);
  14421. return UserName;
  14422. }
  14423. function write_WriteAccess(s, opts) {
  14424. var b8 = !opts || opts.biff == 8;
  14425. var o = new_buf(b8 ? 112 : 54);
  14426. o.write_shift(opts.biff == 8 ? 2 : 1, 7);
  14427. if(b8) o.write_shift(1, 0);
  14428. o.write_shift(4, 0x33336853);
  14429. o.write_shift(4, (0x00534A74 | (b8 ? 0 : 0x20000000)));
  14430. while(o.l < o.length) o.write_shift(1, (b8 ? 0 : 32));
  14431. return o;
  14432. }
  14433. /* [MS-XLS] 2.4.351 */
  14434. function parse_WsBool(blob, length, opts) {
  14435. var flags = opts && opts.biff == 8 || length == 2 ? blob.read_shift(2) : (blob.l += length, 0);
  14436. return { fDialog: flags & 0x10 };
  14437. }
  14438. /* [MS-XLS] 2.4.28 */
  14439. function parse_BoundSheet8(blob, length, opts) {
  14440. var pos = blob.read_shift(4);
  14441. var hidden = blob.read_shift(1) & 0x03;
  14442. var dt = blob.read_shift(1);
  14443. switch(dt) {
  14444. case 0: dt = 'Worksheet'; break;
  14445. case 1: dt = 'Macrosheet'; break;
  14446. case 2: dt = 'Chartsheet'; break;
  14447. case 6: dt = 'VBAModule'; break;
  14448. }
  14449. var name = parse_ShortXLUnicodeString(blob, 0, opts);
  14450. if(name.length === 0) name = "Sheet1";
  14451. return { pos:pos, hs:hidden, dt:dt, name:name };
  14452. }
  14453. function write_BoundSheet8(data, opts) {
  14454. var w = (!opts || opts.biff >= 8 ? 2 : 1);
  14455. var o = new_buf(8 + w * data.name.length);
  14456. o.write_shift(4, data.pos);
  14457. o.write_shift(1, data.hs || 0);
  14458. o.write_shift(1, data.dt);
  14459. o.write_shift(1, data.name.length);
  14460. if(opts.biff >= 8) o.write_shift(1, 1);
  14461. o.write_shift(w * data.name.length, data.name, opts.biff < 8 ? 'sbcs' : 'utf16le');
  14462. var out = o.slice(0, o.l);
  14463. out.l = o.l; return out;
  14464. }
  14465. /* [MS-XLS] 2.4.265 TODO */
  14466. function parse_SST(blob, length) {
  14467. var end = blob.l + length;
  14468. var cnt = blob.read_shift(4);
  14469. var ucnt = blob.read_shift(4);
  14470. var strs = ([]);
  14471. for(var i = 0; i != ucnt && blob.l < end; ++i) {
  14472. strs.push(parse_XLUnicodeRichExtendedString(blob));
  14473. }
  14474. strs.Count = cnt; strs.Unique = ucnt;
  14475. return strs;
  14476. }
  14477. /* [MS-XLS] 2.4.107 */
  14478. function parse_ExtSST(blob, length) {
  14479. var extsst = {};
  14480. extsst.dsst = blob.read_shift(2);
  14481. blob.l += length-2;
  14482. return extsst;
  14483. }
  14484. /* [MS-XLS] 2.4.221 TODO: check BIFF2-4 */
  14485. function parse_Row(blob) {
  14486. var z = ({});
  14487. z.r = blob.read_shift(2);
  14488. z.c = blob.read_shift(2);
  14489. z.cnt = blob.read_shift(2) - z.c;
  14490. var miyRw = blob.read_shift(2);
  14491. blob.l += 4; // reserved(2), unused(2)
  14492. var flags = blob.read_shift(1); // various flags
  14493. blob.l += 3; // reserved(8), ixfe(12), flags(4)
  14494. if(flags & 0x07) z.level = flags & 0x07;
  14495. // collapsed: flags & 0x10
  14496. if(flags & 0x20) z.hidden = true;
  14497. if(flags & 0x40) z.hpt = miyRw / 20;
  14498. return z;
  14499. }
  14500. /* [MS-XLS] 2.4.125 */
  14501. function parse_ForceFullCalculation(blob) {
  14502. var header = parse_frtHeader(blob);
  14503. if(header.type != 0x08A3) throw new Error("Invalid Future Record " + header.type);
  14504. var fullcalc = blob.read_shift(4);
  14505. return fullcalc !== 0x0;
  14506. }
  14507. /* [MS-XLS] 2.4.215 rt */
  14508. function parse_RecalcId(blob) {
  14509. blob.read_shift(2);
  14510. return blob.read_shift(4);
  14511. }
  14512. /* [MS-XLS] 2.4.87 */
  14513. function parse_DefaultRowHeight(blob, length, opts) {
  14514. var f = 0;
  14515. if(!(opts && opts.biff == 2)) {
  14516. f = blob.read_shift(2);
  14517. }
  14518. var miyRw = blob.read_shift(2);
  14519. if((opts && opts.biff == 2)) {
  14520. f = 1 - (miyRw >> 15); miyRw &= 0x7fff;
  14521. }
  14522. var fl = {Unsynced:f&1,DyZero:(f&2)>>1,ExAsc:(f&4)>>2,ExDsc:(f&8)>>3};
  14523. return [fl, miyRw];
  14524. }
  14525. /* [MS-XLS] 2.4.345 TODO */
  14526. function parse_Window1(blob) {
  14527. var xWn = blob.read_shift(2), yWn = blob.read_shift(2), dxWn = blob.read_shift(2), dyWn = blob.read_shift(2);
  14528. var flags = blob.read_shift(2), iTabCur = blob.read_shift(2), iTabFirst = blob.read_shift(2);
  14529. var ctabSel = blob.read_shift(2), wTabRatio = blob.read_shift(2);
  14530. return { Pos: [xWn, yWn], Dim: [dxWn, dyWn], Flags: flags, CurTab: iTabCur,
  14531. FirstTab: iTabFirst, Selected: ctabSel, TabRatio: wTabRatio };
  14532. }
  14533. function write_Window1() {
  14534. var o = new_buf(18);
  14535. o.write_shift(2, 0);
  14536. o.write_shift(2, 0);
  14537. o.write_shift(2, 0x7260);
  14538. o.write_shift(2, 0x44c0);
  14539. o.write_shift(2, 0x38);
  14540. o.write_shift(2, 0);
  14541. o.write_shift(2, 0);
  14542. o.write_shift(2, 1);
  14543. o.write_shift(2, 0x01f4);
  14544. return o;
  14545. }
  14546. /* [MS-XLS] 2.4.346 TODO */
  14547. function parse_Window2(blob, length, opts) {
  14548. if(opts && opts.biff >= 2 && opts.biff < 8) return {};
  14549. var f = blob.read_shift(2);
  14550. return { RTL: f & 0x40 };
  14551. }
  14552. function write_Window2(view) {
  14553. var o = new_buf(18), f = 0x6b6;
  14554. if(view && view.RTL) f |= 0x40;
  14555. o.write_shift(2, f);
  14556. o.write_shift(4, 0);
  14557. o.write_shift(4, 64);
  14558. o.write_shift(4, 0);
  14559. o.write_shift(4, 0);
  14560. return o;
  14561. }
  14562. /* [MS-XLS] 2.4.122 TODO */
  14563. function parse_Font(blob, length, opts) {
  14564. var o = {
  14565. dyHeight: blob.read_shift(2),
  14566. fl: blob.read_shift(2)
  14567. };
  14568. switch((opts && opts.biff) || 8) {
  14569. case 2: break;
  14570. case 3: case 4: blob.l += 2; break;
  14571. default: blob.l += 10; break;
  14572. }
  14573. o.name = parse_ShortXLUnicodeString(blob, 0, opts);
  14574. return o;
  14575. }
  14576. function write_Font(data, opts) {
  14577. var name = data.name || "Arial";
  14578. var b5 = (opts && (opts.biff == 5)), w = (b5 ? (15 + name.length) : (16 + 2 * name.length));
  14579. var o = new_buf(w);
  14580. o.write_shift(2, (data.sz || 12) * 20);
  14581. o.write_shift(4, 0);
  14582. o.write_shift(2, 400);
  14583. o.write_shift(4, 0);
  14584. o.write_shift(2, 0);
  14585. o.write_shift(1, name.length);
  14586. if(!b5) o.write_shift(1, 1);
  14587. o.write_shift((b5 ? 1 : 2) * name.length, name, (b5 ? "sbcs" : "utf16le"));
  14588. return o;
  14589. }
  14590. /* [MS-XLS] 2.4.149 */
  14591. function parse_LabelSst(blob) {
  14592. var cell = parse_XLSCell(blob);
  14593. cell.isst = blob.read_shift(4);
  14594. return cell;
  14595. }
  14596. /* [MS-XLS] 2.4.148 */
  14597. function parse_Label(blob, length, opts) {
  14598. var target = blob.l + length;
  14599. var cell = parse_XLSCell(blob, 6);
  14600. if(opts.biff == 2) blob.l++;
  14601. var str = parse_XLUnicodeString(blob, target - blob.l, opts);
  14602. cell.val = str;
  14603. return cell;
  14604. }
  14605. function write_Label(R, C, v, os, opts) {
  14606. var b8 = !opts || opts.biff == 8;
  14607. var o = new_buf(6 + 2 + (+b8) + (1 + b8) * v.length);
  14608. write_XLSCell(R, C, os, o);
  14609. o.write_shift(2, v.length);
  14610. if(b8) o.write_shift(1, 1);
  14611. o.write_shift((1 + b8) * v.length, v, b8 ? 'utf16le' : 'sbcs');
  14612. return o;
  14613. }
  14614. /* [MS-XLS] 2.4.126 Number Formats */
  14615. function parse_Format(blob, length, opts) {
  14616. var numFmtId = blob.read_shift(2);
  14617. var fmtstr = parse_XLUnicodeString2(blob, 0, opts);
  14618. return [numFmtId, fmtstr];
  14619. }
  14620. function write_Format(i, f, opts, o) {
  14621. var b5 = (opts && (opts.biff == 5));
  14622. if(!o) o = new_buf(b5 ? (3 + f.length) : (5 + 2 * f.length));
  14623. o.write_shift(2, i);
  14624. o.write_shift((b5 ? 1 : 2), f.length);
  14625. if(!b5) o.write_shift(1, 1);
  14626. o.write_shift((b5 ? 1 : 2) * f.length, f, (b5 ? 'sbcs' : 'utf16le'));
  14627. var out = (o.length > o.l) ? o.slice(0, o.l) : o;
  14628. if(out.l == null) out.l = out.length;
  14629. return out;
  14630. }
  14631. var parse_BIFF2Format = parse_XLUnicodeString2;
  14632. /* [MS-XLS] 2.4.90 */
  14633. function parse_Dimensions(blob, length, opts) {
  14634. var end = blob.l + length;
  14635. var w = opts.biff == 8 || !opts.biff ? 4 : 2;
  14636. var r = blob.read_shift(w), R = blob.read_shift(w);
  14637. var c = blob.read_shift(2), C = blob.read_shift(2);
  14638. blob.l = end;
  14639. return {s: {r:r, c:c}, e: {r:R, c:C}};
  14640. }
  14641. function write_Dimensions(range, opts) {
  14642. var w = opts.biff == 8 || !opts.biff ? 4 : 2;
  14643. var o = new_buf(2*w + 6);
  14644. o.write_shift(w, range.s.r);
  14645. o.write_shift(w, range.e.r + 1);
  14646. o.write_shift(2, range.s.c);
  14647. o.write_shift(2, range.e.c + 1);
  14648. o.write_shift(2, 0);
  14649. return o;
  14650. }
  14651. /* [MS-XLS] 2.4.220 */
  14652. function parse_RK(blob) {
  14653. var rw = blob.read_shift(2), col = blob.read_shift(2);
  14654. var rkrec = parse_RkRec(blob);
  14655. return {r:rw, c:col, ixfe:rkrec[0], rknum:rkrec[1]};
  14656. }
  14657. /* [MS-XLS] 2.4.175 */
  14658. function parse_MulRk(blob, length) {
  14659. var target = blob.l + length - 2;
  14660. var rw = blob.read_shift(2), col = blob.read_shift(2);
  14661. var rkrecs = [];
  14662. while(blob.l < target) rkrecs.push(parse_RkRec(blob));
  14663. if(blob.l !== target) throw new Error("MulRK read error");
  14664. var lastcol = blob.read_shift(2);
  14665. if(rkrecs.length != lastcol - col + 1) throw new Error("MulRK length mismatch");
  14666. return {r:rw, c:col, C:lastcol, rkrec:rkrecs};
  14667. }
  14668. /* [MS-XLS] 2.4.174 */
  14669. function parse_MulBlank(blob, length) {
  14670. var target = blob.l + length - 2;
  14671. var rw = blob.read_shift(2), col = blob.read_shift(2);
  14672. var ixfes = [];
  14673. while(blob.l < target) ixfes.push(blob.read_shift(2));
  14674. if(blob.l !== target) throw new Error("MulBlank read error");
  14675. var lastcol = blob.read_shift(2);
  14676. if(ixfes.length != lastcol - col + 1) throw new Error("MulBlank length mismatch");
  14677. return {r:rw, c:col, C:lastcol, ixfe:ixfes};
  14678. }
  14679. /* [MS-XLS] 2.5.20 2.5.249 TODO: interpret values here */
  14680. function parse_CellStyleXF(blob, length, style, opts) {
  14681. var o = {};
  14682. var a = blob.read_shift(4), b = blob.read_shift(4);
  14683. var c = blob.read_shift(4), d = blob.read_shift(2);
  14684. o.patternType = XLSFillPattern[c >> 26];
  14685. if(!opts.cellStyles) return o;
  14686. o.alc = a & 0x07;
  14687. o.fWrap = (a >> 3) & 0x01;
  14688. o.alcV = (a >> 4) & 0x07;
  14689. o.fJustLast = (a >> 7) & 0x01;
  14690. o.trot = (a >> 8) & 0xFF;
  14691. o.cIndent = (a >> 16) & 0x0F;
  14692. o.fShrinkToFit = (a >> 20) & 0x01;
  14693. o.iReadOrder = (a >> 22) & 0x02;
  14694. o.fAtrNum = (a >> 26) & 0x01;
  14695. o.fAtrFnt = (a >> 27) & 0x01;
  14696. o.fAtrAlc = (a >> 28) & 0x01;
  14697. o.fAtrBdr = (a >> 29) & 0x01;
  14698. o.fAtrPat = (a >> 30) & 0x01;
  14699. o.fAtrProt = (a >> 31) & 0x01;
  14700. o.dgLeft = b & 0x0F;
  14701. o.dgRight = (b >> 4) & 0x0F;
  14702. o.dgTop = (b >> 8) & 0x0F;
  14703. o.dgBottom = (b >> 12) & 0x0F;
  14704. o.icvLeft = (b >> 16) & 0x7F;
  14705. o.icvRight = (b >> 23) & 0x7F;
  14706. o.grbitDiag = (b >> 30) & 0x03;
  14707. o.icvTop = c & 0x7F;
  14708. o.icvBottom = (c >> 7) & 0x7F;
  14709. o.icvDiag = (c >> 14) & 0x7F;
  14710. o.dgDiag = (c >> 21) & 0x0F;
  14711. o.icvFore = d & 0x7F;
  14712. o.icvBack = (d >> 7) & 0x7F;
  14713. o.fsxButton = (d >> 14) & 0x01;
  14714. return o;
  14715. }
  14716. //function parse_CellXF(blob, length, opts) {return parse_CellStyleXF(blob,length,0, opts);}
  14717. //function parse_StyleXF(blob, length, opts) {return parse_CellStyleXF(blob,length,1, opts);}
  14718. /* [MS-XLS] 2.4.353 TODO: actually do this right */
  14719. function parse_XF(blob, length, opts) {
  14720. var o = {};
  14721. o.ifnt = blob.read_shift(2); o.numFmtId = blob.read_shift(2); o.flags = blob.read_shift(2);
  14722. o.fStyle = (o.flags >> 2) & 0x01;
  14723. length -= 6;
  14724. o.data = parse_CellStyleXF(blob, length, o.fStyle, opts);
  14725. return o;
  14726. }
  14727. function write_XF(data, ixfeP, opts, o) {
  14728. var b5 = (opts && (opts.biff == 5));
  14729. if(!o) o = new_buf(b5 ? 16 : 20);
  14730. o.write_shift(2, 0);
  14731. if(data.style) {
  14732. o.write_shift(2, (data.numFmtId||0));
  14733. o.write_shift(2, 0xFFF4);
  14734. } else {
  14735. o.write_shift(2, (data.numFmtId||0));
  14736. o.write_shift(2, (ixfeP<<4));
  14737. }
  14738. o.write_shift(4, 0);
  14739. o.write_shift(4, 0);
  14740. if(!b5) o.write_shift(4, 0);
  14741. o.write_shift(2, 0);
  14742. return o;
  14743. }
  14744. /* [MS-XLS] 2.4.134 */
  14745. function parse_Guts(blob) {
  14746. blob.l += 4;
  14747. var out = [blob.read_shift(2), blob.read_shift(2)];
  14748. if(out[0] !== 0) out[0]--;
  14749. if(out[1] !== 0) out[1]--;
  14750. if(out[0] > 7 || out[1] > 7) throw new Error("Bad Gutters: " + out.join("|"));
  14751. return out;
  14752. }
  14753. function write_Guts(guts) {
  14754. var o = new_buf(8);
  14755. o.write_shift(4, 0);
  14756. o.write_shift(2, guts[0] ? guts[0] + 1 : 0);
  14757. o.write_shift(2, guts[1] ? guts[1] + 1 : 0);
  14758. return o;
  14759. }
  14760. /* [MS-XLS] 2.4.24 */
  14761. function parse_BoolErr(blob, length, opts) {
  14762. var cell = parse_XLSCell(blob, 6);
  14763. if(opts.biff == 2) ++blob.l;
  14764. var val = parse_Bes(blob, 2);
  14765. cell.val = val;
  14766. cell.t = (val === true || val === false) ? 'b' : 'e';
  14767. return cell;
  14768. }
  14769. function write_BoolErr(R, C, v, os, opts, t) {
  14770. var o = new_buf(8);
  14771. write_XLSCell(R, C, os, o);
  14772. write_Bes(v, t, o);
  14773. return o;
  14774. }
  14775. /* [MS-XLS] 2.4.180 Number */
  14776. function parse_Number(blob) {
  14777. var cell = parse_XLSCell(blob, 6);
  14778. var xnum = parse_Xnum(blob, 8);
  14779. cell.val = xnum;
  14780. return cell;
  14781. }
  14782. function write_Number(R, C, v, os) {
  14783. var o = new_buf(14);
  14784. write_XLSCell(R, C, os, o);
  14785. write_Xnum(v, o);
  14786. return o;
  14787. }
  14788. var parse_XLHeaderFooter = parse_OptXLUnicodeString; // TODO: parse 2.4.136
  14789. /* [MS-XLS] 2.4.271 */
  14790. function parse_SupBook(blob, length, opts) {
  14791. var end = blob.l + length;
  14792. var ctab = blob.read_shift(2);
  14793. var cch = blob.read_shift(2);
  14794. opts.sbcch = cch;
  14795. if(cch == 0x0401 || cch == 0x3A01) return [cch, ctab];
  14796. if(cch < 0x01 || cch >0xff) throw new Error("Unexpected SupBook type: "+cch);
  14797. var virtPath = parse_XLUnicodeStringNoCch(blob, cch);
  14798. /* TODO: 2.5.277 Virtual Path */
  14799. var rgst = [];
  14800. while(end > blob.l) rgst.push(parse_XLUnicodeString(blob));
  14801. return [cch, ctab, virtPath, rgst];
  14802. }
  14803. /* [MS-XLS] 2.4.105 TODO */
  14804. function parse_ExternName(blob, length, opts) {
  14805. var flags = blob.read_shift(2);
  14806. var body;
  14807. var o = ({
  14808. fBuiltIn: flags & 0x01,
  14809. fWantAdvise: (flags >>> 1) & 0x01,
  14810. fWantPict: (flags >>> 2) & 0x01,
  14811. fOle: (flags >>> 3) & 0x01,
  14812. fOleLink: (flags >>> 4) & 0x01,
  14813. cf: (flags >>> 5) & 0x3FF,
  14814. fIcon: flags >>> 15 & 0x01
  14815. });
  14816. if(opts.sbcch === 0x3A01) body = parse_AddinUdf(blob, length-2, opts);
  14817. //else throw new error("unsupported SupBook cch: " + opts.sbcch);
  14818. o.body = body || blob.read_shift(length-2);
  14819. if(typeof body === "string") o.Name = body;
  14820. return o;
  14821. }
  14822. /* [MS-XLS] 2.4.150 TODO */
  14823. var XLSLblBuiltIn = [
  14824. "_xlnm.Consolidate_Area",
  14825. "_xlnm.Auto_Open",
  14826. "_xlnm.Auto_Close",
  14827. "_xlnm.Extract",
  14828. "_xlnm.Database",
  14829. "_xlnm.Criteria",
  14830. "_xlnm.Print_Area",
  14831. "_xlnm.Print_Titles",
  14832. "_xlnm.Recorder",
  14833. "_xlnm.Data_Form",
  14834. "_xlnm.Auto_Activate",
  14835. "_xlnm.Auto_Deactivate",
  14836. "_xlnm.Sheet_Title",
  14837. "_xlnm._FilterDatabase"
  14838. ];
  14839. function parse_Lbl(blob, length, opts) {
  14840. var target = blob.l + length;
  14841. var flags = blob.read_shift(2);
  14842. var chKey = blob.read_shift(1);
  14843. var cch = blob.read_shift(1);
  14844. var cce = blob.read_shift(opts && opts.biff == 2 ? 1 : 2);
  14845. var itab = 0;
  14846. if(!opts || opts.biff >= 5) {
  14847. if(opts.biff != 5) blob.l += 2;
  14848. itab = blob.read_shift(2);
  14849. if(opts.biff == 5) blob.l += 2;
  14850. blob.l += 4;
  14851. }
  14852. var name = parse_XLUnicodeStringNoCch(blob, cch, opts);
  14853. if(flags & 0x20) name = XLSLblBuiltIn[name.charCodeAt(0)];
  14854. var npflen = target - blob.l; if(opts && opts.biff == 2) --npflen;
  14855. var rgce = target == blob.l || cce === 0 ? [] : parse_NameParsedFormula(blob, npflen, opts, cce);
  14856. return {
  14857. chKey: chKey,
  14858. Name: name,
  14859. itab: itab,
  14860. rgce: rgce
  14861. };
  14862. }
  14863. /* [MS-XLS] 2.4.106 TODO: verify filename encoding */
  14864. function parse_ExternSheet(blob, length, opts) {
  14865. if(opts.biff < 8) return parse_BIFF5ExternSheet(blob, length, opts);
  14866. var o = [], target = blob.l + length, len = blob.read_shift(opts.biff > 8 ? 4 : 2);
  14867. while(len-- !== 0) o.push(parse_XTI(blob, opts.biff > 8 ? 12 : 6, opts));
  14868. // [iSupBook, itabFirst, itabLast];
  14869. if(blob.l != target) throw new Error("Bad ExternSheet: " + blob.l + " != " + target);
  14870. return o;
  14871. }
  14872. function parse_BIFF5ExternSheet(blob, length, opts) {
  14873. if(blob[blob.l + 1] == 0x03) blob[blob.l]++;
  14874. var o = parse_ShortXLUnicodeString(blob, length, opts);
  14875. return o.charCodeAt(0) == 0x03 ? o.slice(1) : o;
  14876. }
  14877. /* [MS-XLS] 2.4.176 TODO: check older biff */
  14878. function parse_NameCmt(blob, length, opts) {
  14879. if(opts.biff < 8) { blob.l += length; return; }
  14880. var cchName = blob.read_shift(2);
  14881. var cchComment = blob.read_shift(2);
  14882. var name = parse_XLUnicodeStringNoCch(blob, cchName, opts);
  14883. var comment = parse_XLUnicodeStringNoCch(blob, cchComment, opts);
  14884. return [name, comment];
  14885. }
  14886. /* [MS-XLS] 2.4.260 */
  14887. function parse_ShrFmla(blob, length, opts) {
  14888. var ref = parse_RefU(blob, 6);
  14889. blob.l++;
  14890. var cUse = blob.read_shift(1);
  14891. length -= 8;
  14892. return [parse_SharedParsedFormula(blob, length, opts), cUse, ref];
  14893. }
  14894. /* [MS-XLS] 2.4.4 TODO */
  14895. function parse_Array(blob, length, opts) {
  14896. var ref = parse_Ref(blob, 6);
  14897. /* TODO: fAlwaysCalc */
  14898. switch(opts.biff) {
  14899. case 2: blob.l ++; length -= 7; break;
  14900. case 3: case 4: blob.l += 2; length -= 8; break;
  14901. default: blob.l += 6; length -= 12;
  14902. }
  14903. return [ref, parse_ArrayParsedFormula(blob, length, opts, ref)];
  14904. }
  14905. /* [MS-XLS] 2.4.173 */
  14906. function parse_MTRSettings(blob) {
  14907. var fMTREnabled = blob.read_shift(4) !== 0x00;
  14908. var fUserSetThreadCount = blob.read_shift(4) !== 0x00;
  14909. var cUserThreadCount = blob.read_shift(4);
  14910. return [fMTREnabled, fUserSetThreadCount, cUserThreadCount];
  14911. }
  14912. /* [MS-XLS] 2.5.186 TODO: BIFF5 */
  14913. function parse_NoteSh(blob, length, opts) {
  14914. if(opts.biff < 8) return;
  14915. var row = blob.read_shift(2), col = blob.read_shift(2);
  14916. var flags = blob.read_shift(2), idObj = blob.read_shift(2);
  14917. var stAuthor = parse_XLUnicodeString2(blob, 0, opts);
  14918. if(opts.biff < 8) blob.read_shift(1);
  14919. return [{r:row,c:col}, stAuthor, idObj, flags];
  14920. }
  14921. /* [MS-XLS] 2.4.179 */
  14922. function parse_Note(blob, length, opts) {
  14923. /* TODO: Support revisions */
  14924. return parse_NoteSh(blob, length, opts);
  14925. }
  14926. /* [MS-XLS] 2.4.168 */
  14927. function parse_MergeCells(blob, length) {
  14928. var merges = [];
  14929. var cmcs = blob.read_shift(2);
  14930. while (cmcs--) merges.push(parse_Ref8U(blob,length));
  14931. return merges;
  14932. }
  14933. function write_MergeCells(merges) {
  14934. var o = new_buf(2 + merges.length * 8);
  14935. o.write_shift(2, merges.length);
  14936. for(var i = 0; i < merges.length; ++i) write_Ref8U(merges[i], o);
  14937. return o;
  14938. }
  14939. /* [MS-XLS] 2.4.181 TODO: parse all the things! */
  14940. function parse_Obj(blob, length, opts) {
  14941. if(opts && opts.biff < 8) return parse_BIFF5Obj(blob, length, opts);
  14942. var cmo = parse_FtCmo(blob, 22); // id, ot, flags
  14943. var fts = parse_FtArray(blob, length-22, cmo[1]);
  14944. return { cmo: cmo, ft:fts };
  14945. }
  14946. /* from older spec */
  14947. var parse_BIFF5OT = [];
  14948. parse_BIFF5OT[0x08] = function(blob, length) {
  14949. var tgt = blob.l + length;
  14950. blob.l += 10; // todo
  14951. var cf = blob.read_shift(2);
  14952. blob.l += 4;
  14953. blob.l += 2; //var cbPictFmla = blob.read_shift(2);
  14954. blob.l += 2;
  14955. blob.l += 2; //var grbit = blob.read_shift(2);
  14956. blob.l += 4;
  14957. var cchName = blob.read_shift(1);
  14958. blob.l += cchName; // TODO: stName
  14959. blob.l = tgt; // TODO: fmla
  14960. return { fmt:cf };
  14961. };
  14962. function parse_BIFF5Obj(blob, length, opts) {
  14963. blob.l += 4; //var cnt = blob.read_shift(4);
  14964. var ot = blob.read_shift(2);
  14965. var id = blob.read_shift(2);
  14966. var grbit = blob.read_shift(2);
  14967. blob.l += 2; //var colL = blob.read_shift(2);
  14968. blob.l += 2; //var dxL = blob.read_shift(2);
  14969. blob.l += 2; //var rwT = blob.read_shift(2);
  14970. blob.l += 2; //var dyT = blob.read_shift(2);
  14971. blob.l += 2; //var colR = blob.read_shift(2);
  14972. blob.l += 2; //var dxR = blob.read_shift(2);
  14973. blob.l += 2; //var rwB = blob.read_shift(2);
  14974. blob.l += 2; //var dyB = blob.read_shift(2);
  14975. blob.l += 2; //var cbMacro = blob.read_shift(2);
  14976. blob.l += 6;
  14977. length -= 36;
  14978. var fts = [];
  14979. fts.push((parse_BIFF5OT[ot]||parsenoop)(blob, length, opts));
  14980. return { cmo: [id, ot, grbit], ft:fts };
  14981. }
  14982. /* [MS-XLS] 2.4.329 TODO: parse properly */
  14983. function parse_TxO(blob, length, opts) {
  14984. var s = blob.l;
  14985. var texts = "";
  14986. try {
  14987. blob.l += 4;
  14988. var ot = (opts.lastobj||{cmo:[0,0]}).cmo[1];
  14989. var controlInfo; // eslint-disable-line no-unused-vars
  14990. if([0,5,7,11,12,14].indexOf(ot) == -1) blob.l += 6;
  14991. else controlInfo = parse_ControlInfo(blob, 6, opts);
  14992. var cchText = blob.read_shift(2);
  14993. /*var cbRuns = */blob.read_shift(2);
  14994. /*var ifntEmpty = */parseuint16(blob, 2);
  14995. var len = blob.read_shift(2);
  14996. blob.l += len;
  14997. //var fmla = parse_ObjFmla(blob, s + length - blob.l);
  14998. for(var i = 1; i < blob.lens.length-1; ++i) {
  14999. if(blob.l-s != blob.lens[i]) throw new Error("TxO: bad continue record");
  15000. var hdr = blob[blob.l];
  15001. var t = parse_XLUnicodeStringNoCch(blob, blob.lens[i+1]-blob.lens[i]-1);
  15002. texts += t;
  15003. if(texts.length >= (hdr ? cchText : 2*cchText)) break;
  15004. }
  15005. if(texts.length !== cchText && texts.length !== cchText*2) {
  15006. throw new Error("cchText: " + cchText + " != " + texts.length);
  15007. }
  15008. blob.l = s + length;
  15009. /* [MS-XLS] 2.5.272 TxORuns */
  15010. // var rgTxoRuns = [];
  15011. // for(var j = 0; j != cbRuns/8-1; ++j) blob.l += 8;
  15012. // var cchText2 = blob.read_shift(2);
  15013. // if(cchText2 !== cchText) throw new error("TxOLastRun mismatch: " + cchText2 + " " + cchText);
  15014. // blob.l += 6;
  15015. // if(s + length != blob.l) throw new error("TxO " + (s + length) + ", at " + blob.l);
  15016. return { t: texts };
  15017. } catch(e) { blob.l = s + length; return { t: texts }; }
  15018. }
  15019. /* [MS-XLS] 2.4.140 */
  15020. function parse_HLink(blob, length) {
  15021. var ref = parse_Ref8U(blob, 8);
  15022. blob.l += 16; /* CLSID */
  15023. var hlink = parse_Hyperlink(blob, length-24);
  15024. return [ref, hlink];
  15025. }
  15026. function write_HLink(hl) {
  15027. var O = new_buf(24);
  15028. var ref = decode_cell(hl[0]);
  15029. O.write_shift(2, ref.r); O.write_shift(2, ref.r);
  15030. O.write_shift(2, ref.c); O.write_shift(2, ref.c);
  15031. var clsid = "d0 c9 ea 79 f9 ba ce 11 8c 82 00 aa 00 4b a9 0b".split(" ");
  15032. for(var i = 0; i < 16; ++i) O.write_shift(1, parseInt(clsid[i], 16));
  15033. return bconcat([O, write_Hyperlink(hl[1])]);
  15034. }
  15035. /* [MS-XLS] 2.4.141 */
  15036. function parse_HLinkTooltip(blob, length) {
  15037. blob.read_shift(2);
  15038. var ref = parse_Ref8U(blob, 8);
  15039. var wzTooltip = blob.read_shift((length-10)/2, 'dbcs-cont');
  15040. wzTooltip = wzTooltip.replace(chr0,"");
  15041. return [ref, wzTooltip];
  15042. }
  15043. function write_HLinkTooltip(hl) {
  15044. var TT = hl[1].Tooltip;
  15045. var O = new_buf(10 + 2 * (TT.length + 1));
  15046. O.write_shift(2, 0x0800);
  15047. var ref = decode_cell(hl[0]);
  15048. O.write_shift(2, ref.r); O.write_shift(2, ref.r);
  15049. O.write_shift(2, ref.c); O.write_shift(2, ref.c);
  15050. for(var i = 0; i < TT.length; ++i) O.write_shift(2, TT.charCodeAt(i));
  15051. O.write_shift(2, 0);
  15052. return O;
  15053. }
  15054. /* [MS-XLS] 2.4.63 */
  15055. function parse_Country(blob) {
  15056. var o = [0,0], d;
  15057. d = blob.read_shift(2); o[0] = CountryEnum[d] || d;
  15058. d = blob.read_shift(2); o[1] = CountryEnum[d] || d;
  15059. return o;
  15060. }
  15061. function write_Country(o) {
  15062. if(!o) o = new_buf(4);
  15063. o.write_shift(2, 0x01);
  15064. o.write_shift(2, 0x01);
  15065. return o;
  15066. }
  15067. /* [MS-XLS] 2.4.50 ClrtClient */
  15068. function parse_ClrtClient(blob) {
  15069. var ccv = blob.read_shift(2);
  15070. var o = [];
  15071. while(ccv-->0) o.push(parse_LongRGB(blob, 8));
  15072. return o;
  15073. }
  15074. /* [MS-XLS] 2.4.188 */
  15075. function parse_Palette(blob) {
  15076. var ccv = blob.read_shift(2);
  15077. var o = [];
  15078. while(ccv-->0) o.push(parse_LongRGB(blob, 8));
  15079. return o;
  15080. }
  15081. /* [MS-XLS] 2.4.354 */
  15082. function parse_XFCRC(blob) {
  15083. blob.l += 2;
  15084. var o = {cxfs:0, crc:0};
  15085. o.cxfs = blob.read_shift(2);
  15086. o.crc = blob.read_shift(4);
  15087. return o;
  15088. }
  15089. /* [MS-XLS] 2.4.53 TODO: parse flags */
  15090. /* [MS-XLSB] 2.4.323 TODO: parse flags */
  15091. function parse_ColInfo(blob, length, opts) {
  15092. if(!opts.cellStyles) return parsenoop(blob, length);
  15093. var w = opts && opts.biff >= 12 ? 4 : 2;
  15094. var colFirst = blob.read_shift(w);
  15095. var colLast = blob.read_shift(w);
  15096. var coldx = blob.read_shift(w);
  15097. var ixfe = blob.read_shift(w);
  15098. var flags = blob.read_shift(2);
  15099. if(w == 2) blob.l += 2;
  15100. return {s:colFirst, e:colLast, w:coldx, ixfe:ixfe, flags:flags};
  15101. }
  15102. /* [MS-XLS] 2.4.257 */
  15103. function parse_Setup(blob, length) {
  15104. var o = {};
  15105. if(length < 32) return o;
  15106. blob.l += 16;
  15107. o.header = parse_Xnum(blob, 8);
  15108. o.footer = parse_Xnum(blob, 8);
  15109. blob.l += 2;
  15110. return o;
  15111. }
  15112. /* [MS-XLS] 2.4.261 */
  15113. function parse_ShtProps(blob, length, opts) {
  15114. var def = {area:false};
  15115. if(opts.biff != 5) { blob.l += length; return def; }
  15116. var d = blob.read_shift(1); blob.l += 3;
  15117. if((d & 0x10)) def.area = true;
  15118. return def;
  15119. }
  15120. /* [MS-XLS] 2.4.241 */
  15121. function write_RRTabId(n) {
  15122. var out = new_buf(2 * n);
  15123. for(var i = 0; i < n; ++i) out.write_shift(2, i+1);
  15124. return out;
  15125. }
  15126. var parse_Blank = parse_XLSCell; /* [MS-XLS] 2.4.20 Just the cell */
  15127. var parse_Scl = parseuint16a; /* [MS-XLS] 2.4.247 num, den */
  15128. var parse_String = parse_XLUnicodeString; /* [MS-XLS] 2.4.268 */
  15129. /* --- Specific to versions before BIFF8 --- */
  15130. function parse_ImData(blob) {
  15131. var cf = blob.read_shift(2);
  15132. var env = blob.read_shift(2);
  15133. var lcb = blob.read_shift(4);
  15134. var o = {fmt:cf, env:env, len:lcb, data:blob.slice(blob.l,blob.l+lcb)};
  15135. blob.l += lcb;
  15136. return o;
  15137. }
  15138. /* BIFF2_??? where ??? is the name from [XLS] */
  15139. function parse_BIFF2STR(blob, length, opts) {
  15140. var cell = parse_XLSCell(blob, 6);
  15141. ++blob.l;
  15142. var str = parse_XLUnicodeString2(blob, length-7, opts);
  15143. cell.t = 'str';
  15144. cell.val = str;
  15145. return cell;
  15146. }
  15147. function parse_BIFF2NUM(blob) {
  15148. var cell = parse_XLSCell(blob, 6);
  15149. ++blob.l;
  15150. var num = parse_Xnum(blob, 8);
  15151. cell.t = 'n';
  15152. cell.val = num;
  15153. return cell;
  15154. }
  15155. function write_BIFF2NUM(r, c, val) {
  15156. var out = new_buf(15);
  15157. write_BIFF2Cell(out, r, c);
  15158. out.write_shift(8, val, 'f');
  15159. return out;
  15160. }
  15161. function parse_BIFF2INT(blob) {
  15162. var cell = parse_XLSCell(blob, 6);
  15163. ++blob.l;
  15164. var num = blob.read_shift(2);
  15165. cell.t = 'n';
  15166. cell.val = num;
  15167. return cell;
  15168. }
  15169. function write_BIFF2INT(r, c, val) {
  15170. var out = new_buf(9);
  15171. write_BIFF2Cell(out, r, c);
  15172. out.write_shift(2, val);
  15173. return out;
  15174. }
  15175. function parse_BIFF2STRING(blob) {
  15176. var cch = blob.read_shift(1);
  15177. if(cch === 0) { blob.l++; return ""; }
  15178. return blob.read_shift(cch, 'sbcs-cont');
  15179. }
  15180. /* TODO: convert to BIFF8 font struct */
  15181. function parse_BIFF2FONTXTRA(blob, length) {
  15182. blob.l += 6; // unknown
  15183. blob.l += 2; // font weight "bls"
  15184. blob.l += 1; // charset
  15185. blob.l += 3; // unknown
  15186. blob.l += 1; // font family
  15187. blob.l += length - 13;
  15188. }
  15189. /* TODO: parse rich text runs */
  15190. function parse_RString(blob, length, opts) {
  15191. var end = blob.l + length;
  15192. var cell = parse_XLSCell(blob, 6);
  15193. var cch = blob.read_shift(2);
  15194. var str = parse_XLUnicodeStringNoCch(blob, cch, opts);
  15195. blob.l = end;
  15196. cell.t = 'str';
  15197. cell.val = str;
  15198. return cell;
  15199. }
  15200. /* from js-harb (C) 2014-present SheetJS */
  15201. var DBF = (function() {
  15202. var dbf_codepage_map = {
  15203. /* Code Pages Supported by Visual FoxPro */
  15204. 0x01: 437, 0x02: 850,
  15205. 0x03: 1252, 0x04: 10000,
  15206. 0x64: 852, 0x65: 866,
  15207. 0x66: 865, 0x67: 861,
  15208. 0x68: 895, 0x69: 620,
  15209. 0x6A: 737, 0x6B: 857,
  15210. 0x78: 950, 0x79: 949,
  15211. 0x7A: 936, 0x7B: 932,
  15212. 0x7C: 874, 0x7D: 1255,
  15213. 0x7E: 1256, 0x96: 10007,
  15214. 0x97: 10029, 0x98: 10006,
  15215. 0xC8: 1250, 0xC9: 1251,
  15216. 0xCA: 1254, 0xCB: 1253,
  15217. /* shapefile DBF extension */
  15218. 0x00: 20127, 0x08: 865,
  15219. 0x09: 437, 0x0A: 850,
  15220. 0x0B: 437, 0x0D: 437,
  15221. 0x0E: 850, 0x0F: 437,
  15222. 0x10: 850, 0x11: 437,
  15223. 0x12: 850, 0x13: 932,
  15224. 0x14: 850, 0x15: 437,
  15225. 0x16: 850, 0x17: 865,
  15226. 0x18: 437, 0x19: 437,
  15227. 0x1A: 850, 0x1B: 437,
  15228. 0x1C: 863, 0x1D: 850,
  15229. 0x1F: 852, 0x22: 852,
  15230. 0x23: 852, 0x24: 860,
  15231. 0x25: 850, 0x26: 866,
  15232. 0x37: 850, 0x40: 852,
  15233. 0x4D: 936, 0x4E: 949,
  15234. 0x4F: 950, 0x50: 874,
  15235. 0x57: 1252, 0x58: 1252,
  15236. 0x59: 1252,
  15237. 0xFF: 16969
  15238. };
  15239. /* TODO: find an actual specification */
  15240. function dbf_to_aoa(buf, opts) {
  15241. var out = [];
  15242. /* TODO: browser based */
  15243. var d = (new_raw_buf(1));
  15244. switch(opts.type) {
  15245. case 'base64': d = s2a(Base64.decode(buf)); break;
  15246. case 'binary': d = s2a(buf); break;
  15247. case 'buffer':
  15248. case 'array': d = buf; break;
  15249. }
  15250. prep_blob(d, 0);
  15251. /* header */
  15252. var ft = d.read_shift(1);
  15253. var memo = false;
  15254. var vfp = false, l7 = false;
  15255. switch(ft) {
  15256. case 0x02: case 0x03: break;
  15257. case 0x30: vfp = true; memo = true; break;
  15258. case 0x31: vfp = true; break;
  15259. case 0x83: memo = true; break;
  15260. case 0x8B: memo = true; break;
  15261. case 0x8C: memo = true; l7 = true; break;
  15262. case 0xF5: memo = true; break;
  15263. default: throw new Error("DBF Unsupported Version: " + ft.toString(16));
  15264. }
  15265. var /*filedate = new Date(),*/ nrow = 0, fpos = 0;
  15266. if(ft == 0x02) nrow = d.read_shift(2);
  15267. /*filedate = new Date(d.read_shift(1) + 1900, d.read_shift(1) - 1, d.read_shift(1));*/d.l += 3;
  15268. if(ft != 0x02) nrow = d.read_shift(4);
  15269. if(ft != 0x02) fpos = d.read_shift(2);
  15270. var rlen = d.read_shift(2);
  15271. var /*flags = 0,*/ current_cp = 1252;
  15272. if(ft != 0x02) {
  15273. d.l+=16;
  15274. /*flags = */d.read_shift(1);
  15275. //if(memo && ((flags & 0x02) === 0)) throw new error("DBF Flags " + flags.toString(16) + " ft " + ft.toString(16));
  15276. /* codepage present in FoxPro */
  15277. if(d[d.l] !== 0) current_cp = dbf_codepage_map[d[d.l]];
  15278. d.l+=1;
  15279. d.l+=2;
  15280. }
  15281. if(l7) d.l += 36;
  15282. var fields = [], field = ({});
  15283. var hend = fpos - 10 - (vfp ? 264 : 0), ww = l7 ? 32 : 11;
  15284. while(ft == 0x02 ? d.l < d.length && d[d.l] != 0x0d: d.l < hend) {
  15285. field = ({});
  15286. field.name = cptable.utils.decode(current_cp, d.slice(d.l, d.l+ww)).replace(/[\u0000\r\n].*$/g,"");
  15287. d.l += ww;
  15288. field.type = String.fromCharCode(d.read_shift(1));
  15289. if(ft != 0x02 && !l7) field.offset = d.read_shift(4);
  15290. field.len = d.read_shift(1);
  15291. if(ft == 0x02) field.offset = d.read_shift(2);
  15292. field.dec = d.read_shift(1);
  15293. if(field.name.length) fields.push(field);
  15294. if(ft != 0x02) d.l += l7 ? 13 : 14;
  15295. switch(field.type) {
  15296. case 'B': // VFP Double
  15297. if((!vfp || field.len != 8) && opts.WTF) console.log('Skipping ' + field.name + ':' + field.type);
  15298. break;
  15299. case 'G': // General
  15300. case 'P': // Picture
  15301. if(opts.WTF) console.log('Skipping ' + field.name + ':' + field.type);
  15302. break;
  15303. case 'C': // character
  15304. case 'D': // date
  15305. case 'F': // floating point
  15306. case 'I': // long
  15307. case 'L': // boolean
  15308. case 'M': // memo
  15309. case 'N': // number
  15310. case 'O': // double
  15311. case 'T': // datetime
  15312. case 'Y': // currency
  15313. case '0': // VFP _NullFlags
  15314. case '@': // timestamp
  15315. case '+': // autoincrement
  15316. break;
  15317. default: throw new Error('Unknown Field Type: ' + field.type);
  15318. }
  15319. }
  15320. if(d[d.l] !== 0x0D) d.l = fpos-1;
  15321. else if(ft == 0x02) d.l = 0x209;
  15322. if(ft != 0x02) {
  15323. if(d.read_shift(1) !== 0x0D) throw new Error("DBF Terminator not found " + d.l + " " + d[d.l]);
  15324. d.l = fpos;
  15325. }
  15326. /* data */
  15327. var R = 0, C = 0;
  15328. out[0] = [];
  15329. for(C = 0; C != fields.length; ++C) out[0][C] = fields[C].name;
  15330. while(nrow-- > 0) {
  15331. if(d[d.l] === 0x2A) { d.l+=rlen; continue; }
  15332. ++d.l;
  15333. out[++R] = []; C = 0;
  15334. for(C = 0; C != fields.length; ++C) {
  15335. var dd = d.slice(d.l, d.l+fields[C].len); d.l+=fields[C].len;
  15336. prep_blob(dd, 0);
  15337. var s = cptable.utils.decode(current_cp, dd);
  15338. switch(fields[C].type) {
  15339. case 'C':
  15340. out[R][C] = cptable.utils.decode(current_cp, dd);
  15341. out[R][C] = out[R][C].trim();
  15342. break;
  15343. case 'D':
  15344. if(s.length === 8) out[R][C] = new Date(+s.slice(0,4), +s.slice(4,6)-1, +s.slice(6,8));
  15345. else out[R][C] = s;
  15346. break;
  15347. case 'F': out[R][C] = parseFloat(s.trim()); break;
  15348. case '+': case 'I': out[R][C] = l7 ? dd.read_shift(-4, 'i') ^ 0x80000000 : dd.read_shift(4, 'i'); break;
  15349. case 'L': switch(s.toUpperCase()) {
  15350. case 'Y': case 'T': out[R][C] = true; break;
  15351. case 'N': case 'F': out[R][C] = false; break;
  15352. case ' ': case '?': out[R][C] = false; break; /* NOTE: technically uninitialized */
  15353. default: throw new Error("DBF Unrecognized L:|" + s + "|");
  15354. } break;
  15355. case 'M': /* TODO: handle memo files */
  15356. if(!memo) throw new Error("DBF Unexpected MEMO for type " + ft.toString(16));
  15357. out[R][C] = "##MEMO##" + (l7 ? parseInt(s.trim(), 10): dd.read_shift(4));
  15358. break;
  15359. case 'N': out[R][C] = +s.replace(/\u0000/g,"").trim(); break;
  15360. case '@': out[R][C] = new Date(dd.read_shift(-8, 'f') - 0x388317533400); break;
  15361. case 'T': out[R][C] = new Date((dd.read_shift(4) - 0x253D8C) * 0x5265C00 + dd.read_shift(4)); break;
  15362. case 'Y': out[R][C] = dd.read_shift(4,'i')/1e4; break;
  15363. case 'O': out[R][C] = -dd.read_shift(-8, 'f'); break;
  15364. case 'B': if(vfp && fields[C].len == 8) { out[R][C] = dd.read_shift(8,'f'); break; }
  15365. /* falls through */
  15366. case 'G': case 'P': dd.l += fields[C].len; break;
  15367. case '0':
  15368. if(fields[C].name === '_NullFlags') break;
  15369. /* falls through */
  15370. default: throw new Error("DBF Unsupported data type " + fields[C].type);
  15371. }
  15372. }
  15373. }
  15374. if(ft != 0x02) if(d.l < d.length && d[d.l++] != 0x1A) throw new Error("DBF EOF Marker missing " + (d.l-1) + " of " + d.length + " " + d[d.l-1].toString(16));
  15375. if(opts && opts.sheetRows) out = out.slice(0, opts.sheetRows);
  15376. return out;
  15377. }
  15378. function dbf_to_sheet(buf, opts) {
  15379. var o = opts || {};
  15380. if(!o.dateNF) o.dateNF = "yyyymmdd";
  15381. return aoa_to_sheet(dbf_to_aoa(buf, o), o);
  15382. }
  15383. function dbf_to_workbook(buf, opts) {
  15384. try { return sheet_to_workbook(dbf_to_sheet(buf, opts), opts); }
  15385. catch(e) { if(opts && opts.WTF) throw e; }
  15386. return ({SheetNames:[],Sheets:{}});
  15387. }
  15388. var _RLEN = { 'B': 8, 'C': 250, 'L': 1, 'D': 8, '?': 0, '': 0 };
  15389. function sheet_to_dbf(ws, opts) {
  15390. var o = opts || {};
  15391. if(o.type == "string") throw new Error("Cannot write DBF to JS string");
  15392. var ba = buf_array();
  15393. var aoa = sheet_to_json(ws, {header:1, cellDates:true});
  15394. var headers = aoa[0], data = aoa.slice(1);
  15395. var i = 0, j = 0, hcnt = 0, rlen = 1;
  15396. for(i = 0; i < headers.length; ++i) {
  15397. if(i == null) continue;
  15398. ++hcnt;
  15399. if(typeof headers[i] === 'number') headers[i] = headers[i].toString(10);
  15400. if(typeof headers[i] !== 'string') throw new Error("DBF Invalid column name " + headers[i] + " |" + (typeof headers[i]) + "|");
  15401. if(headers.indexOf(headers[i]) !== i) for(j=0; j<1024;++j)
  15402. if(headers.indexOf(headers[i] + "_" + j) == -1) { headers[i] += "_" + j; break; }
  15403. }
  15404. var range = safe_decode_range(ws['!ref']);
  15405. var coltypes = [];
  15406. for(i = 0; i <= range.e.c - range.s.c; ++i) {
  15407. var col = [];
  15408. for(j=0; j < data.length; ++j) {
  15409. if(data[j][i] != null) col.push(data[j][i]);
  15410. }
  15411. if(col.length == 0 || headers[i] == null) { coltypes[i] = '?'; continue; }
  15412. var guess = '', _guess = '';
  15413. for(j = 0; j < col.length; ++j) {
  15414. switch(typeof col[j]) {
  15415. /* TODO: check if L2 compat is desired */
  15416. case 'number': _guess = 'B'; break;
  15417. case 'string': _guess = 'C'; break;
  15418. case 'boolean': _guess = 'L'; break;
  15419. case 'object': _guess = col[j] instanceof Date ? 'D' : 'C'; break;
  15420. default: _guess = 'C';
  15421. }
  15422. guess = guess && guess != _guess ? 'C' : _guess;
  15423. if(guess == 'C') break;
  15424. }
  15425. rlen += _RLEN[guess] || 0;
  15426. coltypes[i] = guess;
  15427. }
  15428. var h = ba.next(32);
  15429. h.write_shift(4, 0x13021130);
  15430. h.write_shift(4, data.length);
  15431. h.write_shift(2, 296 + 32 * hcnt);
  15432. h.write_shift(2, rlen);
  15433. for(i=0; i < 4; ++i) h.write_shift(4, 0);
  15434. h.write_shift(4, 0x00000300); // TODO: CP
  15435. for(i = 0, j = 0; i < headers.length; ++i) {
  15436. if(headers[i] == null) continue;
  15437. var hf = ba.next(32);
  15438. var _f = (headers[i].slice(-10) + "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00").slice(0, 11);
  15439. hf.write_shift(1, _f, "sbcs");
  15440. hf.write_shift(1, coltypes[i] == '?' ? 'C' : coltypes[i], "sbcs");
  15441. hf.write_shift(4, j);
  15442. hf.write_shift(1, _RLEN[coltypes[i]] || 0);
  15443. hf.write_shift(1, 0);
  15444. hf.write_shift(1, 0x02);
  15445. hf.write_shift(4, 0);
  15446. hf.write_shift(1, 0);
  15447. hf.write_shift(4, 0);
  15448. hf.write_shift(4, 0);
  15449. j += _RLEN[coltypes[i]] || 0;
  15450. }
  15451. var hb = ba.next(264);
  15452. hb.write_shift(4, 0x0000000D);
  15453. for(i=0; i < 65;++i) hb.write_shift(4, 0x00000000);
  15454. for(i=0; i < data.length; ++i) {
  15455. var rout = ba.next(rlen);
  15456. rout.write_shift(1, 0);
  15457. for(j=0; j<headers.length; ++j) {
  15458. if(headers[j] == null) continue;
  15459. switch(coltypes[j]) {
  15460. case 'L': rout.write_shift(1, data[i][j] == null ? 0x3F : data[i][j] ? 0x54 : 0x46); break;
  15461. case 'B': rout.write_shift(8, data[i][j]||0, 'f'); break;
  15462. case 'D':
  15463. if(!data[i][j]) rout.write_shift(8, "00000000", "sbcs");
  15464. else {
  15465. rout.write_shift(4, ("0000"+data[i][j].getFullYear()).slice(-4), "sbcs");
  15466. rout.write_shift(2, ("00"+(data[i][j].getMonth()+1)).slice(-2), "sbcs");
  15467. rout.write_shift(2, ("00"+data[i][j].getDate()).slice(-2), "sbcs");
  15468. } break;
  15469. case 'C':
  15470. var _s = String(data[i][j]||"");
  15471. rout.write_shift(1, _s, "sbcs");
  15472. for(hcnt=0; hcnt < 250-_s.length; ++hcnt) rout.write_shift(1, 0x20); break;
  15473. }
  15474. }
  15475. // data
  15476. }
  15477. ba.next(1).write_shift(1, 0x1A);
  15478. return ba.end();
  15479. }
  15480. return {
  15481. to_workbook: dbf_to_workbook,
  15482. to_sheet: dbf_to_sheet,
  15483. from_sheet: sheet_to_dbf
  15484. };
  15485. })();
  15486. var SYLK = (function() {
  15487. /* TODO: find an actual specification */
  15488. function sylk_to_aoa(d, opts) {
  15489. switch(opts.type) {
  15490. case 'base64': return sylk_to_aoa_str(Base64.decode(d), opts);
  15491. case 'binary': return sylk_to_aoa_str(d, opts);
  15492. case 'buffer': return sylk_to_aoa_str(d.toString('binary'), opts);
  15493. case 'array': return sylk_to_aoa_str(cc2str(d), opts);
  15494. }
  15495. throw new Error("Unrecognized type " + opts.type);
  15496. }
  15497. function sylk_to_aoa_str(str, opts) {
  15498. var records = str.split(/[\n\r]+/), R = -1, C = -1, ri = 0, rj = 0, arr = [];
  15499. var formats = [];
  15500. var next_cell_format = null;
  15501. var sht = {}, rowinfo = [], colinfo = [], cw = [];
  15502. var Mval = 0, j;
  15503. for (; ri !== records.length; ++ri) {
  15504. Mval = 0;
  15505. var rstr=records[ri].trim();
  15506. var record=rstr.replace(/;;/g, "\u0001").split(";").map(function(x) { return x.replace(/\u0001/g, ";"); });
  15507. var RT=record[0], val;
  15508. if(rstr.length > 0) switch(RT) {
  15509. case 'ID': break; /* header */
  15510. case 'E': break; /* EOF */
  15511. case 'B': break; /* dimensions */
  15512. case 'O': break; /* options? */
  15513. case 'P':
  15514. if(record[1].charAt(0) == 'P')
  15515. formats.push(rstr.slice(3).replace(/;;/g, ";"));
  15516. break;
  15517. case 'C':
  15518. var C_seen_K = false, C_seen_X = false;
  15519. for(rj=1; rj<record.length; ++rj) switch(record[rj].charAt(0)) {
  15520. case 'X': C = parseInt(record[rj].slice(1))-1; C_seen_X = true; break;
  15521. case 'Y':
  15522. R = parseInt(record[rj].slice(1))-1; if(!C_seen_X) C = 0;
  15523. for(j = arr.length; j <= R; ++j) arr[j] = [];
  15524. break;
  15525. case 'K':
  15526. val = record[rj].slice(1);
  15527. if(val.charAt(0) === '"') val = val.slice(1,val.length - 1);
  15528. else if(val === 'TRUE') val = true;
  15529. else if(val === 'FALSE') val = false;
  15530. else if(!isNaN(fuzzynum(val))) {
  15531. val = fuzzynum(val);
  15532. if(next_cell_format !== null && SSF.is_date(next_cell_format)) val = numdate(val);
  15533. } else if(!isNaN(fuzzydate(val).getDate())) {
  15534. val = parseDate(val);
  15535. }
  15536. if(typeof cptable !== 'undefined' && typeof val == "string" && ((opts||{}).type != "string") && (opts||{}).codepage) val = cptable.utils.decode(opts.codepage, val);
  15537. C_seen_K = true;
  15538. break;
  15539. case 'E':
  15540. var formula = rc_to_a1(record[rj].slice(1), {r:R,c:C});
  15541. arr[R][C] = [arr[R][C], formula];
  15542. break;
  15543. default: if(opts && opts.WTF) throw new Error("SYLK bad record " + rstr);
  15544. }
  15545. if(C_seen_K) { arr[R][C] = val; next_cell_format = null; }
  15546. break;
  15547. case 'F':
  15548. var F_seen = 0;
  15549. for(rj=1; rj<record.length; ++rj) switch(record[rj].charAt(0)) {
  15550. case 'X': C = parseInt(record[rj].slice(1))-1; ++F_seen; break;
  15551. case 'Y':
  15552. R = parseInt(record[rj].slice(1))-1; /*C = 0;*/
  15553. for(j = arr.length; j <= R; ++j) arr[j] = [];
  15554. break;
  15555. case 'M': Mval = parseInt(record[rj].slice(1)) / 20; break;
  15556. case 'F': break; /* ??? */
  15557. case 'G': break; /* hide grid */
  15558. case 'P':
  15559. next_cell_format = formats[parseInt(record[rj].slice(1))];
  15560. break;
  15561. case 'S': break; /* cell style */
  15562. case 'D': break; /* column */
  15563. case 'N': break; /* font */
  15564. case 'W':
  15565. cw = record[rj].slice(1).split(" ");
  15566. for(j = parseInt(cw[0], 10); j <= parseInt(cw[1], 10); ++j) {
  15567. Mval = parseInt(cw[2], 10);
  15568. colinfo[j-1] = Mval === 0 ? {hidden:true}: {wch:Mval}; process_col(colinfo[j-1]);
  15569. } break;
  15570. case 'C': /* default column format */
  15571. C = parseInt(record[rj].slice(1))-1;
  15572. if(!colinfo[C]) colinfo[C] = {};
  15573. break;
  15574. case 'R': /* row properties */
  15575. R = parseInt(record[rj].slice(1))-1;
  15576. if(!rowinfo[R]) rowinfo[R] = {};
  15577. if(Mval > 0) { rowinfo[R].hpt = Mval; rowinfo[R].hpx = pt2px(Mval); }
  15578. else if(Mval === 0) rowinfo[R].hidden = true;
  15579. break;
  15580. default: if(opts && opts.WTF) throw new Error("SYLK bad record " + rstr);
  15581. }
  15582. if(F_seen < 1) next_cell_format = null; break;
  15583. default: if(opts && opts.WTF) throw new Error("SYLK bad record " + rstr);
  15584. }
  15585. }
  15586. if(rowinfo.length > 0) sht['!rows'] = rowinfo;
  15587. if(colinfo.length > 0) sht['!cols'] = colinfo;
  15588. if(opts && opts.sheetRows) arr = arr.slice(0, opts.sheetRows);
  15589. return [arr, sht];
  15590. }
  15591. function sylk_to_sheet(d, opts) {
  15592. var aoasht = sylk_to_aoa(d, opts);
  15593. var aoa = aoasht[0], ws = aoasht[1];
  15594. var o = aoa_to_sheet(aoa, opts);
  15595. keys(ws).forEach(function(k) { o[k] = ws[k]; });
  15596. return o;
  15597. }
  15598. function sylk_to_workbook(d, opts) { return sheet_to_workbook(sylk_to_sheet(d, opts), opts); }
  15599. function write_ws_cell_sylk(cell, ws, R, C) {
  15600. var o = "C;Y" + (R+1) + ";X" + (C+1) + ";K";
  15601. switch(cell.t) {
  15602. case 'n':
  15603. o += (cell.v||0);
  15604. if(cell.f && !cell.F) o += ";E" + a1_to_rc(cell.f, {r:R, c:C}); break;
  15605. case 'b': o += cell.v ? "TRUE" : "FALSE"; break;
  15606. case 'e': o += cell.w || cell.v; break;
  15607. case 'd': o += '"' + (cell.w || cell.v) + '"'; break;
  15608. case 's': o += '"' + cell.v.replace(/"/g,"") + '"'; break;
  15609. }
  15610. return o;
  15611. }
  15612. function write_ws_cols_sylk(out, cols) {
  15613. cols.forEach(function(col, i) {
  15614. var rec = "F;W" + (i+1) + " " + (i+1) + " ";
  15615. if(col.hidden) rec += "0";
  15616. else {
  15617. if(typeof col.width == 'number') col.wpx = width2px(col.width);
  15618. if(typeof col.wpx == 'number') col.wch = px2char(col.wpx);
  15619. if(typeof col.wch == 'number') rec += Math.round(col.wch);
  15620. }
  15621. if(rec.charAt(rec.length - 1) != " ") out.push(rec);
  15622. });
  15623. }
  15624. function write_ws_rows_sylk(out, rows) {
  15625. rows.forEach(function(row, i) {
  15626. var rec = "F;";
  15627. if(row.hidden) rec += "M0;";
  15628. else if(row.hpt) rec += "M" + 20 * row.hpt + ";";
  15629. else if(row.hpx) rec += "M" + 20 * px2pt(row.hpx) + ";";
  15630. if(rec.length > 2) out.push(rec + "R" + (i+1));
  15631. });
  15632. }
  15633. function sheet_to_sylk(ws, opts) {
  15634. var preamble = ["ID;PWXL;N;E"], o = [];
  15635. var r = safe_decode_range(ws['!ref']), cell;
  15636. var dense = Array.isArray(ws);
  15637. var RS = "\r\n";
  15638. preamble.push("P;PGeneral");
  15639. preamble.push("F;P0;DG0G8;M255");
  15640. if(ws['!cols']) write_ws_cols_sylk(preamble, ws['!cols']);
  15641. if(ws['!rows']) write_ws_rows_sylk(preamble, ws['!rows']);
  15642. preamble.push("B;Y" + (r.e.r - r.s.r + 1) + ";X" + (r.e.c - r.s.c + 1) + ";D" + [r.s.c,r.s.r,r.e.c,r.e.r].join(" "));
  15643. for(var R = r.s.r; R <= r.e.r; ++R) {
  15644. for(var C = r.s.c; C <= r.e.c; ++C) {
  15645. var coord = encode_cell({r:R,c:C});
  15646. cell = dense ? (ws[R]||[])[C]: ws[coord];
  15647. if(!cell || (cell.v == null && (!cell.f || cell.F))) continue;
  15648. o.push(write_ws_cell_sylk(cell, ws, R, C, opts));
  15649. }
  15650. }
  15651. return preamble.join(RS) + RS + o.join(RS) + RS + "E" + RS;
  15652. }
  15653. return {
  15654. to_workbook: sylk_to_workbook,
  15655. to_sheet: sylk_to_sheet,
  15656. from_sheet: sheet_to_sylk
  15657. };
  15658. })();
  15659. var DIF = (function() {
  15660. function dif_to_aoa(d, opts) {
  15661. switch(opts.type) {
  15662. case 'base64': return dif_to_aoa_str(Base64.decode(d), opts);
  15663. case 'binary': return dif_to_aoa_str(d, opts);
  15664. case 'buffer': return dif_to_aoa_str(d.toString('binary'), opts);
  15665. case 'array': return dif_to_aoa_str(cc2str(d), opts);
  15666. }
  15667. throw new Error("Unrecognized type " + opts.type);
  15668. }
  15669. function dif_to_aoa_str(str, opts) {
  15670. var records = str.split('\n'), R = -1, C = -1, ri = 0, arr = [];
  15671. for (; ri !== records.length; ++ri) {
  15672. if (records[ri].trim() === 'BOT') { arr[++R] = []; C = 0; continue; }
  15673. if (R < 0) continue;
  15674. var metadata = records[ri].trim().split(",");
  15675. var type = metadata[0], value = metadata[1];
  15676. ++ri;
  15677. var data = records[ri].trim();
  15678. switch (+type) {
  15679. case -1:
  15680. if (data === 'BOT') { arr[++R] = []; C = 0; continue; }
  15681. else if (data !== 'EOD') throw new Error("Unrecognized DIF special command " + data);
  15682. break;
  15683. case 0:
  15684. if(data === 'TRUE') arr[R][C] = true;
  15685. else if(data === 'FALSE') arr[R][C] = false;
  15686. else if(!isNaN(fuzzynum(value))) arr[R][C] = fuzzynum(value);
  15687. else if(!isNaN(fuzzydate(value).getDate())) arr[R][C] = parseDate(value);
  15688. else arr[R][C] = value;
  15689. ++C; break;
  15690. case 1:
  15691. data = data.slice(1,data.length-1);
  15692. arr[R][C++] = data !== '' ? data : null;
  15693. break;
  15694. }
  15695. if (data === 'EOD') break;
  15696. }
  15697. if(opts && opts.sheetRows) arr = arr.slice(0, opts.sheetRows);
  15698. return arr;
  15699. }
  15700. function dif_to_sheet(str, opts) { return aoa_to_sheet(dif_to_aoa(str, opts), opts); }
  15701. function dif_to_workbook(str, opts) { return sheet_to_workbook(dif_to_sheet(str, opts), opts); }
  15702. var sheet_to_dif = (function() {
  15703. var push_field = function pf(o, topic, v, n, s) {
  15704. o.push(topic);
  15705. o.push(v + "," + n);
  15706. o.push('"' + s.replace(/"/g,'""') + '"');
  15707. };
  15708. var push_value = function po(o, type, v, s) {
  15709. o.push(type + "," + v);
  15710. o.push(type == 1 ? '"' + s.replace(/"/g,'""') + '"' : s);
  15711. };
  15712. return function sheet_to_dif(ws) {
  15713. var o = [];
  15714. var r = safe_decode_range(ws['!ref']), cell;
  15715. var dense = Array.isArray(ws);
  15716. push_field(o, "TABLE", 0, 1, "sheetjs");
  15717. push_field(o, "VECTORS", 0, r.e.r - r.s.r + 1,"");
  15718. push_field(o, "TUPLES", 0, r.e.c - r.s.c + 1,"");
  15719. push_field(o, "DATA", 0, 0,"");
  15720. for(var R = r.s.r; R <= r.e.r; ++R) {
  15721. push_value(o, -1, 0, "BOT");
  15722. for(var C = r.s.c; C <= r.e.c; ++C) {
  15723. var coord = encode_cell({r:R,c:C});
  15724. cell = dense ? (ws[R]||[])[C] : ws[coord];
  15725. if(!cell) { push_value(o, 1, 0, ""); continue;}
  15726. switch(cell.t) {
  15727. case 'n':
  15728. var val = DIF_XL ? cell.w : cell.v;
  15729. if(!val && cell.v != null) val = cell.v;
  15730. if(val == null) {
  15731. if(DIF_XL && cell.f && !cell.F) push_value(o, 1, 0, "=" + cell.f);
  15732. else push_value(o, 1, 0, "");
  15733. }
  15734. else push_value(o, 0, val, "V");
  15735. break;
  15736. case 'b':
  15737. push_value(o, 0, cell.v ? 1 : 0, cell.v ? "TRUE" : "FALSE");
  15738. break;
  15739. case 's':
  15740. push_value(o, 1, 0, (!DIF_XL || isNaN(cell.v)) ? cell.v : '="' + cell.v + '"');
  15741. break;
  15742. case 'd':
  15743. if(!cell.w) cell.w = SSF.format(cell.z || SSF._table[14], datenum(parseDate(cell.v)));
  15744. if(DIF_XL) push_value(o, 0, cell.w, "V");
  15745. else push_value(o, 1, 0, cell.w);
  15746. break;
  15747. default: push_value(o, 1, 0, "");
  15748. }
  15749. }
  15750. }
  15751. push_value(o, -1, 0, "EOD");
  15752. var RS = "\r\n";
  15753. var oo = o.join(RS);
  15754. //while((oo.length & 0x7F) != 0) oo += "\0";
  15755. return oo;
  15756. };
  15757. })();
  15758. return {
  15759. to_workbook: dif_to_workbook,
  15760. to_sheet: dif_to_sheet,
  15761. from_sheet: sheet_to_dif
  15762. };
  15763. })();
  15764. var ETH = (function() {
  15765. function decode(s) { return s.replace(/\\b/g,"\\").replace(/\\c/g,":").replace(/\\n/g,"\n"); }
  15766. function encode(s) { return s.replace(/\\/g, "\\b").replace(/:/g, "\\c").replace(/\n/g,"\\n"); }
  15767. function eth_to_aoa(str, opts) {
  15768. var records = str.split('\n'), R = -1, C = -1, ri = 0, arr = [];
  15769. for (; ri !== records.length; ++ri) {
  15770. var record = records[ri].trim().split(":");
  15771. if(record[0] !== 'cell') continue;
  15772. var addr = decode_cell(record[1]);
  15773. if(arr.length <= addr.r) for(R = arr.length; R <= addr.r; ++R) if(!arr[R]) arr[R] = [];
  15774. R = addr.r; C = addr.c;
  15775. switch(record[2]) {
  15776. case 't': arr[R][C] = decode(record[3]); break;
  15777. case 'v': arr[R][C] = +record[3]; break;
  15778. case 'vtf': var _f = record[record.length - 1];
  15779. /* falls through */
  15780. case 'vtc':
  15781. switch(record[3]) {
  15782. case 'nl': arr[R][C] = +record[4] ? true : false; break;
  15783. default: arr[R][C] = +record[4]; break;
  15784. }
  15785. if(record[2] == 'vtf') arr[R][C] = [arr[R][C], _f];
  15786. }
  15787. }
  15788. if(opts && opts.sheetRows) arr = arr.slice(0, opts.sheetRows);
  15789. return arr;
  15790. }
  15791. function eth_to_sheet(d, opts) { return aoa_to_sheet(eth_to_aoa(d, opts), opts); }
  15792. function eth_to_workbook(d, opts) { return sheet_to_workbook(eth_to_sheet(d, opts), opts); }
  15793. var header = [
  15794. "socialcalc:version:1.5",
  15795. "MIME-Version: 1.0",
  15796. "Content-Type: multipart/mixed; boundary=SocialCalcSpreadsheetControlSave"
  15797. ].join("\n");
  15798. var sep = [
  15799. "--SocialCalcSpreadsheetControlSave",
  15800. "Content-type: text/plain; charset=UTF-8"
  15801. ].join("\n") + "\n";
  15802. /* TODO: the other parts */
  15803. var meta = [
  15804. "# SocialCalc Spreadsheet Control Save",
  15805. "part:sheet"
  15806. ].join("\n");
  15807. var end = "--SocialCalcSpreadsheetControlSave--";
  15808. function sheet_to_eth_data(ws) {
  15809. if(!ws || !ws['!ref']) return "";
  15810. var o = [], oo = [], cell, coord = "";
  15811. var r = decode_range(ws['!ref']);
  15812. var dense = Array.isArray(ws);
  15813. for(var R = r.s.r; R <= r.e.r; ++R) {
  15814. for(var C = r.s.c; C <= r.e.c; ++C) {
  15815. coord = encode_cell({r:R,c:C});
  15816. cell = dense ? (ws[R]||[])[C] : ws[coord];
  15817. if(!cell || cell.v == null || cell.t === 'z') continue;
  15818. oo = ["cell", coord, 't'];
  15819. switch(cell.t) {
  15820. case 's': case 'str': oo.push(encode(cell.v)); break;
  15821. case 'n':
  15822. if(!cell.f) { oo[2]='v'; oo[3]=cell.v; }
  15823. else { oo[2]='vtf'; oo[3]='n'; oo[4]=cell.v; oo[5]=encode(cell.f); }
  15824. break;
  15825. case 'b':
  15826. oo[2] = 'vt'+(cell.f?'f':'c'); oo[3]='nl'; oo[4]=cell.v?"1":"0";
  15827. oo[5] = encode(cell.f||(cell.v?'TRUE':'FALSE'));
  15828. break;
  15829. case 'd':
  15830. var t = datenum(parseDate(cell.v));
  15831. oo[2] = 'vtc'; oo[3] = 'nd'; oo[4] = ""+t;
  15832. oo[5] = cell.w || SSF.format(cell.z || SSF._table[14], t);
  15833. break;
  15834. case 'e': continue;
  15835. }
  15836. o.push(oo.join(":"));
  15837. }
  15838. }
  15839. o.push("sheet:c:" + (r.e.c-r.s.c+1) + ":r:" + (r.e.r-r.s.r+1) + ":tvf:1");
  15840. o.push("valueformat:1:text-wiki");
  15841. //o.push("copiedfrom:" + ws['!ref']); // clipboard only
  15842. return o.join("\n");
  15843. }
  15844. function sheet_to_eth(ws) {
  15845. return [header, sep, meta, sep, sheet_to_eth_data(ws), end].join("\n");
  15846. // return ["version:1.5", sheet_to_eth_data(ws)].join("\n"); // clipboard form
  15847. }
  15848. return {
  15849. to_workbook: eth_to_workbook,
  15850. to_sheet: eth_to_sheet,
  15851. from_sheet: sheet_to_eth
  15852. };
  15853. })();
  15854. var PRN = (function() {
  15855. function set_text_arr(data, arr, R, C, o) {
  15856. if(o.raw) arr[R][C] = data;
  15857. else if(data === 'TRUE') arr[R][C] = true;
  15858. else if(data === 'FALSE') arr[R][C] = false;
  15859. else if(data === ""){/* empty */}
  15860. else if(!isNaN(fuzzynum(data))) arr[R][C] = fuzzynum(data);
  15861. else if(!isNaN(fuzzydate(data).getDate())) arr[R][C] = parseDate(data);
  15862. else arr[R][C] = data;
  15863. }
  15864. function prn_to_aoa_str(f, opts) {
  15865. var o = opts || {};
  15866. var arr = ([]);
  15867. if(!f || f.length === 0) return arr;
  15868. var lines = f.split(/[\r\n]/);
  15869. var L = lines.length - 1;
  15870. while(L >= 0 && lines[L].length === 0) --L;
  15871. var start = 10, idx = 0;
  15872. var R = 0;
  15873. for(; R <= L; ++R) {
  15874. idx = lines[R].indexOf(" ");
  15875. if(idx == -1) idx = lines[R].length; else idx++;
  15876. start = Math.max(start, idx);
  15877. }
  15878. for(R = 0; R <= L; ++R) {
  15879. arr[R] = [];
  15880. /* TODO: confirm that widths are always 10 */
  15881. var C = 0;
  15882. set_text_arr(lines[R].slice(0, start).trim(), arr, R, C, o);
  15883. for(C = 1; C <= (lines[R].length - start)/10 + 1; ++C)
  15884. set_text_arr(lines[R].slice(start+(C-1)*10,start+C*10).trim(),arr,R,C,o);
  15885. }
  15886. if(o.sheetRows) arr = arr.slice(0, o.sheetRows);
  15887. return arr;
  15888. }
  15889. // List of accepted CSV separators
  15890. var guess_seps = {
  15891. 0x2C: ',',
  15892. 0x09: "\t",
  15893. 0x3B: ';'
  15894. };
  15895. // CSV separator weights to be used in case of equal numbers
  15896. var guess_sep_weights = {
  15897. 0x2C: 3,
  15898. 0x09: 2,
  15899. 0x3B: 1
  15900. };
  15901. function guess_sep(str) {
  15902. var cnt = {}, instr = false, end = 0, cc = 0;
  15903. for(;end < str.length;++end) {
  15904. if((cc=str.charCodeAt(end)) == 0x22) instr = !instr;
  15905. else if(!instr && cc in guess_seps) cnt[cc] = (cnt[cc]||0)+1;
  15906. }
  15907. cc = [];
  15908. for(end in cnt) if ( cnt.hasOwnProperty(end) ) {
  15909. cc.push([ cnt[end], end ]);
  15910. }
  15911. if ( !cc.length ) {
  15912. cnt = guess_sep_weights;
  15913. for(end in cnt) if ( cnt.hasOwnProperty(end) ) {
  15914. cc.push([ cnt[end], end ]);
  15915. }
  15916. }
  15917. cc.sort(function(a, b) { return a[0] - b[0] || guess_sep_weights[a[1]] - guess_sep_weights[b[1]]; });
  15918. return guess_seps[cc.pop()[1]];
  15919. }
  15920. function dsv_to_sheet_str(str, opts) {
  15921. var o = opts || {};
  15922. var sep = "";
  15923. if(DENSE != null && o.dense == null) o.dense = DENSE;
  15924. var ws = o.dense ? ([]) : ({});
  15925. var range = ({s: {c:0, r:0}, e: {c:0, r:0}});
  15926. if(str.slice(0,4) == "sep=" && str.charCodeAt(5) == 10) { sep = str.charAt(4); str = str.slice(6); }
  15927. else sep = guess_sep(str.slice(0,1024));
  15928. var R = 0, C = 0, v = 0;
  15929. var start = 0, end = 0, sepcc = sep.charCodeAt(0), instr = false, cc=0;
  15930. str = str.replace(/\r\n/mg, "\n");
  15931. var _re = o.dateNF != null ? dateNF_regex(o.dateNF) : null;
  15932. function finish_cell() {
  15933. var s = str.slice(start, end);
  15934. var cell = ({});
  15935. if(s.charAt(0) == '"' && s.charAt(s.length - 1) == '"') s = s.slice(1,-1).replace(/""/g,'"');
  15936. if(s.length === 0) cell.t = 'z';
  15937. else if(o.raw) { cell.t = 's'; cell.v = s; }
  15938. else if(s.trim().length === 0) { cell.t = 's'; cell.v = s; }
  15939. else if(s.charCodeAt(0) == 0x3D) {
  15940. if(s.charCodeAt(1) == 0x22 && s.charCodeAt(s.length - 1) == 0x22) { cell.t = 's'; cell.v = s.slice(2,-1).replace(/""/g,'"'); }
  15941. else if(fuzzyfmla(s)) { cell.t = 'n'; cell.f = s.slice(1); }
  15942. else { cell.t = 's'; cell.v = s; } }
  15943. else if(s == "TRUE") { cell.t = 'b'; cell.v = true; }
  15944. else if(s == "FALSE") { cell.t = 'b'; cell.v = false; }
  15945. else if(!isNaN(v = fuzzynum(s))) { cell.t = 'n'; if(o.cellText !== false) cell.w = s; cell.v = v; }
  15946. else if(!isNaN(fuzzydate(s).getDate()) || _re && s.match(_re)) {
  15947. cell.z = o.dateNF || SSF._table[14];
  15948. var k = 0;
  15949. if(_re && s.match(_re)){ s=dateNF_fix(s, o.dateNF, (s.match(_re)||[])); k=1; }
  15950. if(o.cellDates) { cell.t = 'd'; cell.v = parseDate(s, k); }
  15951. else { cell.t = 'n'; cell.v = datenum(parseDate(s, k)); }
  15952. if(o.cellText !== false) cell.w = SSF.format(cell.z, cell.v instanceof Date ? datenum(cell.v):cell.v);
  15953. if(!o.cellNF) delete cell.z;
  15954. } else {
  15955. cell.t = 's';
  15956. cell.v = s;
  15957. }
  15958. if(cell.t == 'z'){}
  15959. else if(o.dense) { if(!ws[R]) ws[R] = []; ws[R][C] = cell; }
  15960. else ws[encode_cell({c:C,r:R})] = cell;
  15961. start = end+1;
  15962. if(range.e.c < C) range.e.c = C;
  15963. if(range.e.r < R) range.e.r = R;
  15964. if(cc == sepcc) ++C; else { C = 0; ++R; if(o.sheetRows && o.sheetRows <= R) return true; }
  15965. }
  15966. outer: for(;end < str.length;++end) switch((cc=str.charCodeAt(end))) {
  15967. case 0x22: instr = !instr; break;
  15968. case sepcc: case 0x0a: case 0x0d: if(!instr && finish_cell()) break outer; break;
  15969. default: break;
  15970. }
  15971. if(end - start > 0) finish_cell();
  15972. ws['!ref'] = encode_range(range);
  15973. return ws;
  15974. }
  15975. function prn_to_sheet_str(str, opts) {
  15976. if(str.slice(0,4) == "sep=") return dsv_to_sheet_str(str, opts);
  15977. if(str.indexOf("\t") >= 0 || str.indexOf(",") >= 0 || str.indexOf(";") >= 0) return dsv_to_sheet_str(str, opts);
  15978. return aoa_to_sheet(prn_to_aoa_str(str, opts), opts);
  15979. }
  15980. function prn_to_sheet(d, opts) {
  15981. var str = "", bytes = opts.type == 'string' ? [0,0,0,0] : firstbyte(d, opts);
  15982. switch(opts.type) {
  15983. case 'base64': str = Base64.decode(d); break;
  15984. case 'binary': str = d; break;
  15985. case 'buffer':
  15986. if(opts.codepage == 65001) str = d.toString('utf8');
  15987. else if(opts.codepage && typeof cptable !== 'undefined') str = cptable.utils.decode(opts.codepage, d);
  15988. else str = d.toString('binary');
  15989. break;
  15990. case 'array': str = cc2str(d); break;
  15991. case 'string': str = d; break;
  15992. default: throw new Error("Unrecognized type " + opts.type);
  15993. }
  15994. if(bytes[0] == 0xEF && bytes[1] == 0xBB && bytes[2] == 0xBF) str = utf8read(str.slice(3));
  15995. else if((opts.type == 'binary') && typeof cptable !== 'undefined' && opts.codepage) str = cptable.utils.decode(opts.codepage, cptable.utils.encode(1252,str));
  15996. if(str.slice(0,19) == "socialcalc:version:") return ETH.to_sheet(opts.type == 'string' ? str : utf8read(str), opts);
  15997. return prn_to_sheet_str(str, opts);
  15998. }
  15999. function prn_to_workbook(d, opts) { return sheet_to_workbook(prn_to_sheet(d, opts), opts); }
  16000. function sheet_to_prn(ws) {
  16001. var o = [];
  16002. var r = safe_decode_range(ws['!ref']), cell;
  16003. var dense = Array.isArray(ws);
  16004. for(var R = r.s.r; R <= r.e.r; ++R) {
  16005. var oo = [];
  16006. for(var C = r.s.c; C <= r.e.c; ++C) {
  16007. var coord = encode_cell({r:R,c:C});
  16008. cell = dense ? (ws[R]||[])[C] : ws[coord];
  16009. if(!cell || cell.v == null) { oo.push(" "); continue; }
  16010. var w = (cell.w || (format_cell(cell), cell.w) || "").slice(0,10);
  16011. while(w.length < 10) w += " ";
  16012. oo.push(w + (C === 0 ? " " : ""));
  16013. }
  16014. o.push(oo.join(""));
  16015. }
  16016. return o.join("\n");
  16017. }
  16018. return {
  16019. to_workbook: prn_to_workbook,
  16020. to_sheet: prn_to_sheet,
  16021. from_sheet: sheet_to_prn
  16022. };
  16023. })();
  16024. /* Excel defaults to SYLK but warns if data is not valid */
  16025. function read_wb_ID(d, opts) {
  16026. var o = opts || {}, OLD_WTF = !!o.WTF; o.WTF = true;
  16027. try {
  16028. var out = SYLK.to_workbook(d, o);
  16029. o.WTF = OLD_WTF;
  16030. return out;
  16031. } catch(e) {
  16032. o.WTF = OLD_WTF;
  16033. if(!e.message.match(/SYLK bad record ID/) && OLD_WTF) throw e;
  16034. return PRN.to_workbook(d, opts);
  16035. }
  16036. }
  16037. var WK_ = (function() {
  16038. function lotushopper(data, cb, opts) {
  16039. if(!data) return;
  16040. prep_blob(data, data.l || 0);
  16041. var Enum = opts.Enum || WK1Enum;
  16042. while(data.l < data.length) {
  16043. var RT = data.read_shift(2);
  16044. var R = Enum[RT] || Enum[0xFF];
  16045. var length = data.read_shift(2);
  16046. var tgt = data.l + length;
  16047. var d = (R.f||parsenoop)(data, length, opts);
  16048. data.l = tgt;
  16049. if(cb(d, R.n, RT)) return;
  16050. }
  16051. }
  16052. function lotus_to_workbook(d, opts) {
  16053. switch(opts.type) {
  16054. case 'base64': return lotus_to_workbook_buf(s2a(Base64.decode(d)), opts);
  16055. case 'binary': return lotus_to_workbook_buf(s2a(d), opts);
  16056. case 'buffer':
  16057. case 'array': return lotus_to_workbook_buf(d, opts);
  16058. }
  16059. throw "Unsupported type " + opts.type;
  16060. }
  16061. function lotus_to_workbook_buf(d, opts) {
  16062. if(!d) return d;
  16063. var o = opts || {};
  16064. if(DENSE != null && o.dense == null) o.dense = DENSE;
  16065. var s = ((o.dense ? [] : {})), n = "Sheet1", sidx = 0;
  16066. var sheets = {}, snames = [n];
  16067. var refguess = {s: {r:0, c:0}, e: {r:0, c:0} };
  16068. var sheetRows = o.sheetRows || 0;
  16069. if(d[2] == 0x02) o.Enum = WK1Enum;
  16070. else if(d[2] == 0x1a) o.Enum = WK3Enum;
  16071. else if(d[2] == 0x0e) { o.Enum = WK3Enum; o.qpro = true; d.l = 0; }
  16072. else throw new Error("Unrecognized LOTUS BOF " + d[2]);
  16073. lotushopper(d, function(val, Rn, RT) {
  16074. if(d[2] == 0x02) switch(RT) {
  16075. case 0x00:
  16076. o.vers = val;
  16077. if(val >= 0x1000) o.qpro = true;
  16078. break;
  16079. case 0x06: refguess = val; break; /* RANGE */
  16080. case 0x0F: /* LABEL */
  16081. if(!o.qpro) val[1].v = val[1].v.slice(1);
  16082. /* falls through */
  16083. case 0x0D: /* INTEGER */
  16084. case 0x0E: /* NUMBER */
  16085. case 0x10: /* FORMULA */
  16086. case 0x33: /* STRING */
  16087. /* TODO: actual translation of the format code */
  16088. if(RT == 0x0E && (val[2] & 0x70) == 0x70 && (val[2] & 0x0F) > 1 && (val[2] & 0x0F) < 15) {
  16089. val[1].z = o.dateNF || SSF._table[14];
  16090. if(o.cellDates) { val[1].t = 'd'; val[1].v = numdate(val[1].v); }
  16091. }
  16092. if(o.dense) {
  16093. if(!s[val[0].r]) s[val[0].r] = [];
  16094. s[val[0].r][val[0].c] = val[1];
  16095. } else s[encode_cell(val[0])] = val[1];
  16096. break;
  16097. } else switch(RT) {
  16098. case 0x16: /* LABEL16 */
  16099. val[1].v = val[1].v.slice(1);
  16100. /* falls through */
  16101. case 0x17: /* NUMBER17 */
  16102. case 0x18: /* NUMBER18 */
  16103. case 0x19: /* FORMULA19 */
  16104. case 0x25: /* NUMBER25 */
  16105. case 0x27: /* NUMBER27 */
  16106. case 0x28: /* FORMULA28 */
  16107. if(val[3] > sidx) {
  16108. s["!ref"] = encode_range(refguess);
  16109. sheets[n] = s;
  16110. s = (o.dense ? [] : {});
  16111. refguess = {s: {r:0, c:0}, e: {r:0, c:0} };
  16112. sidx = val[3]; n = "Sheet" + (sidx + 1);
  16113. snames.push(n);
  16114. }
  16115. if(sheetRows > 0 && val[0].r >= sheetRows) break;
  16116. if(o.dense) {
  16117. if(!s[val[0].r]) s[val[0].r] = [];
  16118. s[val[0].r][val[0].c] = val[1];
  16119. } else s[encode_cell(val[0])] = val[1];
  16120. if(refguess.e.c < val[0].c) refguess.e.c = val[0].c;
  16121. if(refguess.e.r < val[0].r) refguess.e.r = val[0].r;
  16122. break;
  16123. default: break;
  16124. }
  16125. }, o);
  16126. s["!ref"] = encode_range(refguess);
  16127. sheets[n] = s;
  16128. return { SheetNames: snames, Sheets:sheets };
  16129. }
  16130. function parse_RANGE(blob) {
  16131. var o = {s:{c:0,r:0},e:{c:0,r:0}};
  16132. o.s.c = blob.read_shift(2);
  16133. o.s.r = blob.read_shift(2);
  16134. o.e.c = blob.read_shift(2);
  16135. o.e.r = blob.read_shift(2);
  16136. if(o.s.c == 0xFFFF) o.s.c = o.e.c = o.s.r = o.e.r = 0;
  16137. return o;
  16138. }
  16139. function parse_cell(blob, length, opts) {
  16140. var o = [{c:0,r:0}, {t:'n',v:0}, 0];
  16141. if(opts.qpro && opts.vers != 0x5120) {
  16142. o[0].c = blob.read_shift(1);
  16143. blob.l++;
  16144. o[0].r = blob.read_shift(2);
  16145. blob.l+=2;
  16146. } else {
  16147. o[2] = blob.read_shift(1);
  16148. o[0].c = blob.read_shift(2); o[0].r = blob.read_shift(2);
  16149. }
  16150. return o;
  16151. }
  16152. function parse_LABEL(blob, length, opts) {
  16153. var tgt = blob.l + length;
  16154. var o = parse_cell(blob, length, opts);
  16155. o[1].t = 's';
  16156. if(opts.vers == 0x5120) {
  16157. blob.l++;
  16158. var len = blob.read_shift(1);
  16159. o[1].v = blob.read_shift(len, 'utf8');
  16160. return o;
  16161. }
  16162. if(opts.qpro) blob.l++;
  16163. o[1].v = blob.read_shift(tgt - blob.l, 'cstr');
  16164. return o;
  16165. }
  16166. function parse_INTEGER(blob, length, opts) {
  16167. var o = parse_cell(blob, length, opts);
  16168. o[1].v = blob.read_shift(2, 'i');
  16169. return o;
  16170. }
  16171. function parse_NUMBER(blob, length, opts) {
  16172. var o = parse_cell(blob, length, opts);
  16173. o[1].v = blob.read_shift(8, 'f');
  16174. return o;
  16175. }
  16176. function parse_FORMULA(blob, length, opts) {
  16177. var tgt = blob.l + length;
  16178. var o = parse_cell(blob, length, opts);
  16179. /* TODO: formula */
  16180. o[1].v = blob.read_shift(8, 'f');
  16181. if(opts.qpro) blob.l = tgt;
  16182. else {
  16183. var flen = blob.read_shift(2);
  16184. blob.l += flen;
  16185. }
  16186. return o;
  16187. }
  16188. function parse_cell_3(blob) {
  16189. var o = [{c:0,r:0}, {t:'n',v:0}, 0];
  16190. o[0].r = blob.read_shift(2); o[3] = blob[blob.l++]; o[0].c = blob[blob.l++];
  16191. return o;
  16192. }
  16193. function parse_LABEL_16(blob, length) {
  16194. var o = parse_cell_3(blob, length);
  16195. o[1].t = 's';
  16196. o[1].v = blob.read_shift(length - 4, 'cstr');
  16197. return o;
  16198. }
  16199. function parse_NUMBER_18(blob, length) {
  16200. var o = parse_cell_3(blob, length);
  16201. o[1].v = blob.read_shift(2);
  16202. var v = o[1].v >> 1;
  16203. /* TODO: figure out all of the corner cases */
  16204. if(o[1].v & 0x1) {
  16205. switch(v & 0x07) {
  16206. case 1: v = (v >> 3) * 500; break;
  16207. case 2: v = (v >> 3) / 20; break;
  16208. case 4: v = (v >> 3) / 2000; break;
  16209. case 6: v = (v >> 3) / 16; break;
  16210. case 7: v = (v >> 3) / 64; break;
  16211. default: throw "unknown NUMBER_18 encoding " + (v & 0x07);
  16212. }
  16213. }
  16214. o[1].v = v;
  16215. return o;
  16216. }
  16217. function parse_NUMBER_17(blob, length) {
  16218. var o = parse_cell_3(blob, length);
  16219. var v1 = blob.read_shift(4);
  16220. var v2 = blob.read_shift(4);
  16221. var e = blob.read_shift(2);
  16222. if(e == 0xFFFF) { o[1].v = 0; return o; }
  16223. var s = e & 0x8000; e = (e&0x7FFF) - 16446;
  16224. o[1].v = (s*2 - 1) * ((e > 0 ? (v2 << e) : (v2 >>> -e)) + (e > -32 ? (v1 << (e + 32)) : (v1 >>> -(e + 32))));
  16225. return o;
  16226. }
  16227. function parse_FORMULA_19(blob, length) {
  16228. var o = parse_NUMBER_17(blob, 14);
  16229. blob.l += length - 14; /* TODO: formula */
  16230. return o;
  16231. }
  16232. function parse_NUMBER_25(blob, length) {
  16233. var o = parse_cell_3(blob, length);
  16234. var v1 = blob.read_shift(4);
  16235. o[1].v = v1 >> 6;
  16236. return o;
  16237. }
  16238. function parse_NUMBER_27(blob, length) {
  16239. var o = parse_cell_3(blob, length);
  16240. var v1 = blob.read_shift(8,'f');
  16241. o[1].v = v1;
  16242. return o;
  16243. }
  16244. function parse_FORMULA_28(blob, length) {
  16245. var o = parse_NUMBER_27(blob, 14);
  16246. blob.l += length - 10; /* TODO: formula */
  16247. return o;
  16248. }
  16249. var WK1Enum = {
  16250. 0x0000: { n:"BOF", f:parseuint16 },
  16251. 0x0001: { n:"EOF" },
  16252. 0x0002: { n:"CALCMODE" },
  16253. 0x0003: { n:"CALCORDER" },
  16254. 0x0004: { n:"SPLIT" },
  16255. 0x0005: { n:"SYNC" },
  16256. 0x0006: { n:"RANGE", f:parse_RANGE },
  16257. 0x0007: { n:"WINDOW1" },
  16258. 0x0008: { n:"COLW1" },
  16259. 0x0009: { n:"WINTWO" },
  16260. 0x000A: { n:"COLW2" },
  16261. 0x000B: { n:"NAME" },
  16262. 0x000C: { n:"BLANK" },
  16263. 0x000D: { n:"INTEGER", f:parse_INTEGER },
  16264. 0x000E: { n:"NUMBER", f:parse_NUMBER },
  16265. 0x000F: { n:"LABEL", f:parse_LABEL },
  16266. 0x0010: { n:"FORMULA", f:parse_FORMULA },
  16267. 0x0018: { n:"TABLE" },
  16268. 0x0019: { n:"ORANGE" },
  16269. 0x001A: { n:"PRANGE" },
  16270. 0x001B: { n:"SRANGE" },
  16271. 0x001C: { n:"FRANGE" },
  16272. 0x001D: { n:"KRANGE1" },
  16273. 0x0020: { n:"HRANGE" },
  16274. 0x0023: { n:"KRANGE2" },
  16275. 0x0024: { n:"PROTEC" },
  16276. 0x0025: { n:"FOOTER" },
  16277. 0x0026: { n:"HEADER" },
  16278. 0x0027: { n:"SETUP" },
  16279. 0x0028: { n:"MARGINS" },
  16280. 0x0029: { n:"LABELFMT" },
  16281. 0x002A: { n:"TITLES" },
  16282. 0x002B: { n:"SHEETJS" },
  16283. 0x002D: { n:"GRAPH" },
  16284. 0x002E: { n:"NGRAPH" },
  16285. 0x002F: { n:"CALCCOUNT" },
  16286. 0x0030: { n:"UNFORMATTED" },
  16287. 0x0031: { n:"CURSORW12" },
  16288. 0x0032: { n:"WINDOW" },
  16289. 0x0033: { n:"STRING", f:parse_LABEL },
  16290. 0x0037: { n:"PASSWORD" },
  16291. 0x0038: { n:"LOCKED" },
  16292. 0x003C: { n:"QUERY" },
  16293. 0x003D: { n:"QUERYNAME" },
  16294. 0x003E: { n:"PRINT" },
  16295. 0x003F: { n:"PRINTNAME" },
  16296. 0x0040: { n:"GRAPH2" },
  16297. 0x0041: { n:"GRAPHNAME" },
  16298. 0x0042: { n:"ZOOM" },
  16299. 0x0043: { n:"SYMSPLIT" },
  16300. 0x0044: { n:"NSROWS" },
  16301. 0x0045: { n:"NSCOLS" },
  16302. 0x0046: { n:"RULER" },
  16303. 0x0047: { n:"NNAME" },
  16304. 0x0048: { n:"ACOMM" },
  16305. 0x0049: { n:"AMACRO" },
  16306. 0x004A: { n:"PARSE" },
  16307. 0x00FF: { n:"", f:parsenoop }
  16308. };
  16309. var WK3Enum = {
  16310. 0x0000: { n:"BOF" },
  16311. 0x0001: { n:"EOF" },
  16312. 0x0003: { n:"??" },
  16313. 0x0004: { n:"??" },
  16314. 0x0005: { n:"??" },
  16315. 0x0006: { n:"??" },
  16316. 0x0007: { n:"??" },
  16317. 0x0009: { n:"??" },
  16318. 0x000a: { n:"??" },
  16319. 0x000b: { n:"??" },
  16320. 0x000c: { n:"??" },
  16321. 0x000e: { n:"??" },
  16322. 0x000f: { n:"??" },
  16323. 0x0010: { n:"??" },
  16324. 0x0011: { n:"??" },
  16325. 0x0012: { n:"??" },
  16326. 0x0013: { n:"??" },
  16327. 0x0015: { n:"??" },
  16328. 0x0016: { n:"LABEL16", f:parse_LABEL_16},
  16329. 0x0017: { n:"NUMBER17", f:parse_NUMBER_17 },
  16330. 0x0018: { n:"NUMBER18", f:parse_NUMBER_18 },
  16331. 0x0019: { n:"FORMULA19", f:parse_FORMULA_19},
  16332. 0x001a: { n:"??" },
  16333. 0x001b: { n:"??" },
  16334. 0x001c: { n:"??" },
  16335. 0x001d: { n:"??" },
  16336. 0x001e: { n:"??" },
  16337. 0x001f: { n:"??" },
  16338. 0x0021: { n:"??" },
  16339. 0x0025: { n:"NUMBER25", f:parse_NUMBER_25 },
  16340. 0x0027: { n:"NUMBER27", f:parse_NUMBER_27 },
  16341. 0x0028: { n:"FORMULA28", f:parse_FORMULA_28 },
  16342. 0x00FF: { n:"", f:parsenoop }
  16343. };
  16344. return {
  16345. to_workbook: lotus_to_workbook
  16346. };
  16347. })();
  16348. /* Parse a list of <r> tags */
  16349. var parse_rs = (function parse_rs_factory() {
  16350. var tregex = matchtag("t"), rpregex = matchtag("rPr"), rregex = /<(?:\w+:)?r>/g, rend = /<\/(?:\w+:)?r>/, nlregex = /\r\n/g;
  16351. /* 18.4.7 rPr CT_RPrElt */
  16352. var parse_rpr = function parse_rpr(rpr, intro, outro) {
  16353. var font = {}, cp = 65001, align = "";
  16354. var pass = false;
  16355. var m = rpr.match(tagregex), i = 0;
  16356. if(m) for(;i!=m.length; ++i) {
  16357. var y = parsexmltag(m[i]);
  16358. switch(y[0].replace(/\w*:/g,"")) {
  16359. /* 18.8.12 condense CT_BooleanProperty */
  16360. /* ** not required . */
  16361. case '<condense': break;
  16362. /* 18.8.17 extend CT_BooleanProperty */
  16363. /* ** not required . */
  16364. case '<extend': break;
  16365. /* 18.8.36 shadow CT_BooleanProperty */
  16366. /* ** not required . */
  16367. case '<shadow':
  16368. if(!y.val) break;
  16369. /* falls through */
  16370. case '<shadow>':
  16371. case '<shadow/>': font.shadow = 1; break;
  16372. case '</shadow>': break;
  16373. /* 18.4.1 charset CT_IntProperty TODO */
  16374. case '<charset':
  16375. if(y.val == '1') break;
  16376. cp = CS2CP[parseInt(y.val, 10)];
  16377. break;
  16378. /* 18.4.2 outline CT_BooleanProperty TODO */
  16379. case '<outline':
  16380. if(!y.val) break;
  16381. /* falls through */
  16382. case '<outline>':
  16383. case '<outline/>': font.outline = 1; break;
  16384. case '</outline>': break;
  16385. /* 18.4.5 rFont CT_FontName */
  16386. case '<rFont': font.name = y.val; break;
  16387. /* 18.4.11 sz CT_FontSize */
  16388. case '<sz': font.sz = y.val; break;
  16389. /* 18.4.10 strike CT_BooleanProperty */
  16390. case '<strike':
  16391. if(!y.val) break;
  16392. /* falls through */
  16393. case '<strike>':
  16394. case '<strike/>': font.strike = 1; break;
  16395. case '</strike>': break;
  16396. /* 18.4.13 u CT_UnderlineProperty */
  16397. case '<u':
  16398. if(!y.val) break;
  16399. switch(y.val) {
  16400. case 'double': font.uval = "double"; break;
  16401. case 'singleAccounting': font.uval = "single-accounting"; break;
  16402. case 'doubleAccounting': font.uval = "double-accounting"; break;
  16403. }
  16404. /* falls through */
  16405. case '<u>':
  16406. case '<u/>': font.u = 1; break;
  16407. case '</u>': break;
  16408. /* 18.8.2 b */
  16409. case '<b':
  16410. if(y.val == '0') break;
  16411. /* falls through */
  16412. case '<b>':
  16413. case '<b/>': font.b = 1; break;
  16414. case '</b>': break;
  16415. /* 18.8.26 i */
  16416. case '<i':
  16417. if(y.val == '0') break;
  16418. /* falls through */
  16419. case '<i>':
  16420. case '<i/>': font.i = 1; break;
  16421. case '</i>': break;
  16422. /* 18.3.1.15 color CT_Color TODO: tint, theme, auto, indexed */
  16423. case '<color':
  16424. if(y.rgb) font.color = y.rgb.slice(2,8);
  16425. break;
  16426. /* 18.8.18 family ST_FontFamily */
  16427. case '<family': font.family = y.val; break;
  16428. /* 18.4.14 vertAlign CT_VerticalAlignFontProperty TODO */
  16429. case '<vertAlign': align = y.val; break;
  16430. /* 18.8.35 scheme CT_FontScheme TODO */
  16431. case '<scheme': break;
  16432. /* 18.2.10 extLst CT_ExtensionList ? */
  16433. case '<extLst': case '<extLst>': case '</extLst>': break;
  16434. case '<ext': pass = true; break;
  16435. case '</ext>': pass = false; break;
  16436. default:
  16437. if(y[0].charCodeAt(1) !== 47 && !pass) throw new Error('Unrecognized rich format ' + y[0]);
  16438. }
  16439. }
  16440. var style = [];
  16441. if(font.u) style.push("text-decoration: underline;");
  16442. if(font.uval) style.push("text-underline-style:" + font.uval + ";");
  16443. if(font.sz) style.push("font-size:" + font.sz + "pt;");
  16444. if(font.outline) style.push("text-effect: outline;");
  16445. if(font.shadow) style.push("text-shadow: auto;");
  16446. intro.push('<span style="' + style.join("") + '">');
  16447. if(font.b) { intro.push("<b>"); outro.push("</b>"); }
  16448. if(font.i) { intro.push("<i>"); outro.push("</i>"); }
  16449. if(font.strike) { intro.push("<s>"); outro.push("</s>"); }
  16450. if(align == "superscript") align = "sup";
  16451. else if(align == "subscript") align = "sub";
  16452. if(align != "") { intro.push("<" + align + ">"); outro.push("</" + align + ">"); }
  16453. outro.push("</span>");
  16454. return cp;
  16455. };
  16456. /* 18.4.4 r CT_RElt */
  16457. function parse_r(r) {
  16458. var terms = [[],"",[]];
  16459. /* 18.4.12 t ST_Xstring */
  16460. var t = r.match(tregex)/*, cp = 65001*/;
  16461. if(!t) return "";
  16462. terms[1] = t[1];
  16463. var rpr = r.match(rpregex);
  16464. if(rpr) /*cp = */parse_rpr(rpr[1], terms[0], terms[2]);
  16465. return terms[0].join("") + terms[1].replace(nlregex,'<br/>') + terms[2].join("");
  16466. }
  16467. return function parse_rs(rs) {
  16468. return rs.replace(rregex,"").split(rend).map(parse_r).join("");
  16469. };
  16470. })();
  16471. /* 18.4.8 si CT_Rst */
  16472. var sitregex = /<(?:\w+:)?t[^>]*>([^<]*)<\/(?:\w+:)?t>/g, sirregex = /<(?:\w+:)?r>/;
  16473. var sirphregex = /<(?:\w+:)?rPh.*?>([\s\S]*?)<\/(?:\w+:)?rPh>/g;
  16474. function parse_si(x, opts) {
  16475. var html = opts ? opts.cellHTML : true;
  16476. var z = {};
  16477. if(!x) return null;
  16478. //var y;
  16479. /* 18.4.12 t ST_Xstring (Plaintext String) */
  16480. // TODO: is whitespace actually valid here?
  16481. if(x.match(/^\s*<(?:\w+:)?t[^>]*>/)) {
  16482. z.t = unescapexml(utf8read(x.slice(x.indexOf(">")+1).split(/<\/(?:\w+:)?t>/)[0]||""));
  16483. z.r = utf8read(x);
  16484. if(html) z.h = escapehtml(z.t);
  16485. }
  16486. /* 18.4.4 r CT_RElt (Rich Text Run) */
  16487. else if((/*y = */x.match(sirregex))) {
  16488. z.r = utf8read(x);
  16489. z.t = unescapexml(utf8read((x.replace(sirphregex, '').match(sitregex)||[]).join("").replace(tagregex,"")));
  16490. if(html) z.h = parse_rs(z.r);
  16491. }
  16492. /* 18.4.3 phoneticPr CT_PhoneticPr (TODO: needed for Asian support) */
  16493. /* 18.4.6 rPh CT_PhoneticRun (TODO: needed for Asian support) */
  16494. return z;
  16495. }
  16496. /* 18.4 Shared String Table */
  16497. var sstr0 = /<(?:\w+:)?sst([^>]*)>([\s\S]*)<\/(?:\w+:)?sst>/;
  16498. var sstr1 = /<(?:\w+:)?(?:si|sstItem)>/g;
  16499. var sstr2 = /<\/(?:\w+:)?(?:si|sstItem)>/;
  16500. function parse_sst_xml(data, opts) {
  16501. var s = ([]), ss = "";
  16502. if(!data) return s;
  16503. /* 18.4.9 sst CT_Sst */
  16504. var sst = data.match(sstr0);
  16505. if(sst) {
  16506. ss = sst[2].replace(sstr1,"").split(sstr2);
  16507. for(var i = 0; i != ss.length; ++i) {
  16508. var o = parse_si(ss[i].trim(), opts);
  16509. if(o != null) s[s.length] = o;
  16510. }
  16511. sst = parsexmltag(sst[1]); s.Count = sst.count; s.Unique = sst.uniqueCount;
  16512. }
  16513. return s;
  16514. }
  16515. RELS.SST = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/sharedStrings";
  16516. var straywsregex = /^\s|\s$|[\t\n\r]/;
  16517. function write_sst_xml(sst, opts) {
  16518. if(!opts.bookSST) return "";
  16519. var o = [XML_HEADER];
  16520. o[o.length] = (writextag('sst', null, {
  16521. xmlns: XMLNS.main[0],
  16522. count: sst.Count,
  16523. uniqueCount: sst.Unique
  16524. }));
  16525. for(var i = 0; i != sst.length; ++i) { if(sst[i] == null) continue;
  16526. var s = sst[i];
  16527. var sitag = "<si>";
  16528. if(s.r) sitag += s.r;
  16529. else {
  16530. sitag += "<t";
  16531. if(!s.t) s.t = "";
  16532. if(s.t.match(straywsregex)) sitag += ' xml:space="preserve"';
  16533. sitag += ">" + escapexml(s.t) + "</t>";
  16534. }
  16535. sitag += "</si>";
  16536. o[o.length] = (sitag);
  16537. }
  16538. if(o.length>2){ o[o.length] = ('</sst>'); o[1]=o[1].replace("/>",">"); }
  16539. return o.join("");
  16540. }
  16541. /* [MS-XLSB] 2.4.221 BrtBeginSst */
  16542. function parse_BrtBeginSst(data) {
  16543. return [data.read_shift(4), data.read_shift(4)];
  16544. }
  16545. /* [MS-XLSB] 2.1.7.45 Shared Strings */
  16546. function parse_sst_bin(data, opts) {
  16547. var s = ([]);
  16548. var pass = false;
  16549. recordhopper(data, function hopper_sst(val, R_n, RT) {
  16550. switch(RT) {
  16551. case 0x009F: /* 'BrtBeginSst' */
  16552. s.Count = val[0]; s.Unique = val[1]; break;
  16553. case 0x0013: /* 'BrtSSTItem' */
  16554. s.push(val); break;
  16555. case 0x00A0: /* 'BrtEndSst' */
  16556. return true;
  16557. case 0x0023: /* 'BrtFRTBegin' */
  16558. pass = true; break;
  16559. case 0x0024: /* 'BrtFRTEnd' */
  16560. pass = false; break;
  16561. default:
  16562. if(R_n.indexOf("Begin") > 0){/* empty */}
  16563. else if(R_n.indexOf("End") > 0){/* empty */}
  16564. if(!pass || opts.WTF) throw new Error("Unexpected record " + RT + " " + R_n);
  16565. }
  16566. });
  16567. return s;
  16568. }
  16569. function write_BrtBeginSst(sst, o) {
  16570. if(!o) o = new_buf(8);
  16571. o.write_shift(4, sst.Count);
  16572. o.write_shift(4, sst.Unique);
  16573. return o;
  16574. }
  16575. var write_BrtSSTItem = write_RichStr;
  16576. function write_sst_bin(sst) {
  16577. var ba = buf_array();
  16578. write_record(ba, "BrtBeginSst", write_BrtBeginSst(sst));
  16579. for(var i = 0; i < sst.length; ++i) write_record(ba, "BrtSSTItem", write_BrtSSTItem(sst[i]));
  16580. /* FRTSST */
  16581. write_record(ba, "BrtEndSst");
  16582. return ba.end();
  16583. }
  16584. function _JS2ANSI(str) {
  16585. if(typeof cptable !== 'undefined') return cptable.utils.encode(current_ansi, str);
  16586. var o = [], oo = str.split("");
  16587. for(var i = 0; i < oo.length; ++i) o[i] = oo[i].charCodeAt(0);
  16588. return o;
  16589. }
  16590. /* [MS-OFFCRYPTO] 2.1.4 Version */
  16591. function parse_CRYPTOVersion(blob, length) {
  16592. var o = {};
  16593. o.Major = blob.read_shift(2);
  16594. o.Minor = blob.read_shift(2);
  16595. if(length >= 4) blob.l += length - 4;
  16596. return o;
  16597. }
  16598. /* [MS-OFFCRYPTO] 2.1.5 DataSpaceVersionInfo */
  16599. function parse_DataSpaceVersionInfo(blob) {
  16600. var o = {};
  16601. o.id = blob.read_shift(0, 'lpp4');
  16602. o.R = parse_CRYPTOVersion(blob, 4);
  16603. o.U = parse_CRYPTOVersion(blob, 4);
  16604. o.W = parse_CRYPTOVersion(blob, 4);
  16605. return o;
  16606. }
  16607. /* [MS-OFFCRYPTO] 2.1.6.1 DataSpaceMapEntry Structure */
  16608. function parse_DataSpaceMapEntry(blob) {
  16609. var len = blob.read_shift(4);
  16610. var end = blob.l + len - 4;
  16611. var o = {};
  16612. var cnt = blob.read_shift(4);
  16613. var comps = [];
  16614. /* [MS-OFFCRYPTO] 2.1.6.2 DataSpaceReferenceComponent Structure */
  16615. while(cnt-- > 0) comps.push({ t: blob.read_shift(4), v: blob.read_shift(0, 'lpp4') });
  16616. o.name = blob.read_shift(0, 'lpp4');
  16617. o.comps = comps;
  16618. if(blob.l != end) throw new Error("Bad DataSpaceMapEntry: " + blob.l + " != " + end);
  16619. return o;
  16620. }
  16621. /* [MS-OFFCRYPTO] 2.1.6 DataSpaceMap */
  16622. function parse_DataSpaceMap(blob) {
  16623. var o = [];
  16624. blob.l += 4; // must be 0x8
  16625. var cnt = blob.read_shift(4);
  16626. while(cnt-- > 0) o.push(parse_DataSpaceMapEntry(blob));
  16627. return o;
  16628. }
  16629. /* [MS-OFFCRYPTO] 2.1.7 DataSpaceDefinition */
  16630. function parse_DataSpaceDefinition(blob) {
  16631. var o = [];
  16632. blob.l += 4; // must be 0x8
  16633. var cnt = blob.read_shift(4);
  16634. while(cnt-- > 0) o.push(blob.read_shift(0, 'lpp4'));
  16635. return o;
  16636. }
  16637. /* [MS-OFFCRYPTO] 2.1.8 DataSpaceDefinition */
  16638. function parse_TransformInfoHeader(blob) {
  16639. var o = {};
  16640. /*var len = */blob.read_shift(4);
  16641. blob.l += 4; // must be 0x1
  16642. o.id = blob.read_shift(0, 'lpp4');
  16643. o.name = blob.read_shift(0, 'lpp4');
  16644. o.R = parse_CRYPTOVersion(blob, 4);
  16645. o.U = parse_CRYPTOVersion(blob, 4);
  16646. o.W = parse_CRYPTOVersion(blob, 4);
  16647. return o;
  16648. }
  16649. function parse_Primary(blob) {
  16650. /* [MS-OFFCRYPTO] 2.2.6 IRMDSTransformInfo */
  16651. var hdr = parse_TransformInfoHeader(blob);
  16652. /* [MS-OFFCRYPTO] 2.1.9 EncryptionTransformInfo */
  16653. hdr.ename = blob.read_shift(0, '8lpp4');
  16654. hdr.blksz = blob.read_shift(4);
  16655. hdr.cmode = blob.read_shift(4);
  16656. if(blob.read_shift(4) != 0x04) throw new Error("Bad !Primary record");
  16657. return hdr;
  16658. }
  16659. /* [MS-OFFCRYPTO] 2.3.2 Encryption Header */
  16660. function parse_EncryptionHeader(blob, length) {
  16661. var tgt = blob.l + length;
  16662. var o = {};
  16663. o.Flags = (blob.read_shift(4) & 0x3F);
  16664. blob.l += 4;
  16665. o.AlgID = blob.read_shift(4);
  16666. var valid = false;
  16667. switch(o.AlgID) {
  16668. case 0x660E: case 0x660F: case 0x6610: valid = (o.Flags == 0x24); break;
  16669. case 0x6801: valid = (o.Flags == 0x04); break;
  16670. case 0: valid = (o.Flags == 0x10 || o.Flags == 0x04 || o.Flags == 0x24); break;
  16671. default: throw 'Unrecognized encryption algorithm: ' + o.AlgID;
  16672. }
  16673. if(!valid) throw new Error("Encryption Flags/AlgID mismatch");
  16674. o.AlgIDHash = blob.read_shift(4);
  16675. o.KeySize = blob.read_shift(4);
  16676. o.ProviderType = blob.read_shift(4);
  16677. blob.l += 8;
  16678. o.CSPName = blob.read_shift((tgt-blob.l)>>1, 'utf16le');
  16679. blob.l = tgt;
  16680. return o;
  16681. }
  16682. /* [MS-OFFCRYPTO] 2.3.3 Encryption Verifier */
  16683. function parse_EncryptionVerifier(blob, length) {
  16684. var o = {}, tgt = blob.l + length;
  16685. blob.l += 4; // SaltSize must be 0x10
  16686. o.Salt = blob.slice(blob.l, blob.l+16); blob.l += 16;
  16687. o.Verifier = blob.slice(blob.l, blob.l+16); blob.l += 16;
  16688. /*var sz = */blob.read_shift(4);
  16689. o.VerifierHash = blob.slice(blob.l, tgt); blob.l = tgt;
  16690. return o;
  16691. }
  16692. /* [MS-OFFCRYPTO] 2.3.4.* EncryptionInfo Stream */
  16693. function parse_EncryptionInfo(blob) {
  16694. var vers = parse_CRYPTOVersion(blob);
  16695. switch(vers.Minor) {
  16696. case 0x02: return [vers.Minor, parse_EncInfoStd(blob, vers)];
  16697. case 0x03: return [vers.Minor, parse_EncInfoExt(blob, vers)];
  16698. case 0x04: return [vers.Minor, parse_EncInfoAgl(blob, vers)];
  16699. }
  16700. throw new Error("ECMA-376 Encrypted file unrecognized Version: " + vers.Minor);
  16701. }
  16702. /* [MS-OFFCRYPTO] 2.3.4.5 EncryptionInfo Stream (Standard Encryption) */
  16703. function parse_EncInfoStd(blob) {
  16704. var flags = blob.read_shift(4);
  16705. if((flags & 0x3F) != 0x24) throw new Error("EncryptionInfo mismatch");
  16706. var sz = blob.read_shift(4);
  16707. //var tgt = blob.l + sz;
  16708. var hdr = parse_EncryptionHeader(blob, sz);
  16709. var verifier = parse_EncryptionVerifier(blob, blob.length - blob.l);
  16710. return { t:"Std", h:hdr, v:verifier };
  16711. }
  16712. /* [MS-OFFCRYPTO] 2.3.4.6 EncryptionInfo Stream (Extensible Encryption) */
  16713. function parse_EncInfoExt() { throw new Error("File is password-protected: ECMA-376 Extensible"); }
  16714. /* [MS-OFFCRYPTO] 2.3.4.10 EncryptionInfo Stream (Agile Encryption) */
  16715. function parse_EncInfoAgl(blob) {
  16716. var KeyData = ["saltSize","blockSize","keyBits","hashSize","cipherAlgorithm","cipherChaining","hashAlgorithm","saltValue"];
  16717. blob.l+=4;
  16718. var xml = blob.read_shift(blob.length - blob.l, 'utf8');
  16719. var o = {};
  16720. xml.replace(tagregex, function xml_agile(x) {
  16721. var y = parsexmltag(x);
  16722. switch(strip_ns(y[0])) {
  16723. case '<?xml': break;
  16724. case '<encryption': case '</encryption>': break;
  16725. case '<keyData': KeyData.forEach(function(k) { o[k] = y[k]; }); break;
  16726. case '<dataIntegrity': o.encryptedHmacKey = y.encryptedHmacKey; o.encryptedHmacValue = y.encryptedHmacValue; break;
  16727. case '<keyEncryptors>': case '<keyEncryptors': o.encs = []; break;
  16728. case '</keyEncryptors>': break;
  16729. case '<keyEncryptor': o.uri = y.uri; break;
  16730. case '</keyEncryptor>': break;
  16731. case '<encryptedKey': o.encs.push(y); break;
  16732. default: throw y[0];
  16733. }
  16734. });
  16735. return o;
  16736. }
  16737. /* [MS-OFFCRYPTO] 2.3.5.1 RC4 CryptoAPI Encryption Header */
  16738. function parse_RC4CryptoHeader(blob, length) {
  16739. var o = {};
  16740. var vers = o.EncryptionVersionInfo = parse_CRYPTOVersion(blob, 4); length -= 4;
  16741. if(vers.Minor != 2) throw new Error('unrecognized minor version code: ' + vers.Minor);
  16742. if(vers.Major > 4 || vers.Major < 2) throw new Error('unrecognized major version code: ' + vers.Major);
  16743. o.Flags = blob.read_shift(4); length -= 4;
  16744. var sz = blob.read_shift(4); length -= 4;
  16745. o.EncryptionHeader = parse_EncryptionHeader(blob, sz); length -= sz;
  16746. o.EncryptionVerifier = parse_EncryptionVerifier(blob, length);
  16747. return o;
  16748. }
  16749. /* [MS-OFFCRYPTO] 2.3.6.1 RC4 Encryption Header */
  16750. function parse_RC4Header(blob) {
  16751. var o = {};
  16752. var vers = o.EncryptionVersionInfo = parse_CRYPTOVersion(blob, 4);
  16753. if(vers.Major != 1 || vers.Minor != 1) throw 'unrecognized version code ' + vers.Major + ' : ' + vers.Minor;
  16754. o.Salt = blob.read_shift(16);
  16755. o.EncryptedVerifier = blob.read_shift(16);
  16756. o.EncryptedVerifierHash = blob.read_shift(16);
  16757. return o;
  16758. }
  16759. /* [MS-OFFCRYPTO] 2.3.7.1 Binary Document Password Verifier Derivation */
  16760. function crypto_CreatePasswordVerifier_Method1(Password) {
  16761. var Verifier = 0x0000, PasswordArray;
  16762. var PasswordDecoded = _JS2ANSI(Password);
  16763. var len = PasswordDecoded.length + 1, i, PasswordByte;
  16764. var Intermediate1, Intermediate2, Intermediate3;
  16765. PasswordArray = new_raw_buf(len);
  16766. PasswordArray[0] = PasswordDecoded.length;
  16767. for(i = 1; i != len; ++i) PasswordArray[i] = PasswordDecoded[i-1];
  16768. for(i = len-1; i >= 0; --i) {
  16769. PasswordByte = PasswordArray[i];
  16770. Intermediate1 = ((Verifier & 0x4000) === 0x0000) ? 0 : 1;
  16771. Intermediate2 = (Verifier << 1) & 0x7FFF;
  16772. Intermediate3 = Intermediate1 | Intermediate2;
  16773. Verifier = Intermediate3 ^ PasswordByte;
  16774. }
  16775. return Verifier ^ 0xCE4B;
  16776. }
  16777. /* [MS-OFFCRYPTO] 2.3.7.2 Binary Document XOR Array Initialization */
  16778. var crypto_CreateXorArray_Method1 = (function() {
  16779. var PadArray = [0xBB, 0xFF, 0xFF, 0xBA, 0xFF, 0xFF, 0xB9, 0x80, 0x00, 0xBE, 0x0F, 0x00, 0xBF, 0x0F, 0x00];
  16780. var InitialCode = [0xE1F0, 0x1D0F, 0xCC9C, 0x84C0, 0x110C, 0x0E10, 0xF1CE, 0x313E, 0x1872, 0xE139, 0xD40F, 0x84F9, 0x280C, 0xA96A, 0x4EC3];
  16781. var XorMatrix = [0xAEFC, 0x4DD9, 0x9BB2, 0x2745, 0x4E8A, 0x9D14, 0x2A09, 0x7B61, 0xF6C2, 0xFDA5, 0xEB6B, 0xC6F7, 0x9DCF, 0x2BBF, 0x4563, 0x8AC6, 0x05AD, 0x0B5A, 0x16B4, 0x2D68, 0x5AD0, 0x0375, 0x06EA, 0x0DD4, 0x1BA8, 0x3750, 0x6EA0, 0xDD40, 0xD849, 0xA0B3, 0x5147, 0xA28E, 0x553D, 0xAA7A, 0x44D5, 0x6F45, 0xDE8A, 0xAD35, 0x4A4B, 0x9496, 0x390D, 0x721A, 0xEB23, 0xC667, 0x9CEF, 0x29FF, 0x53FE, 0xA7FC, 0x5FD9, 0x47D3, 0x8FA6, 0x0F6D, 0x1EDA, 0x3DB4, 0x7B68, 0xF6D0, 0xB861, 0x60E3, 0xC1C6, 0x93AD, 0x377B, 0x6EF6, 0xDDEC, 0x45A0, 0x8B40, 0x06A1, 0x0D42, 0x1A84, 0x3508, 0x6A10, 0xAA51, 0x4483, 0x8906, 0x022D, 0x045A, 0x08B4, 0x1168, 0x76B4, 0xED68, 0xCAF1, 0x85C3, 0x1BA7, 0x374E, 0x6E9C, 0x3730, 0x6E60, 0xDCC0, 0xA9A1, 0x4363, 0x86C6, 0x1DAD, 0x3331, 0x6662, 0xCCC4, 0x89A9, 0x0373, 0x06E6, 0x0DCC, 0x1021, 0x2042, 0x4084, 0x8108, 0x1231, 0x2462, 0x48C4];
  16782. var Ror = function(Byte) { return ((Byte/2) | (Byte*128)) & 0xFF; };
  16783. var XorRor = function(byte1, byte2) { return Ror(byte1 ^ byte2); };
  16784. var CreateXorKey_Method1 = function(Password) {
  16785. var XorKey = InitialCode[Password.length - 1];
  16786. var CurrentElement = 0x68;
  16787. for(var i = Password.length-1; i >= 0; --i) {
  16788. var Char = Password[i];
  16789. for(var j = 0; j != 7; ++j) {
  16790. if(Char & 0x40) XorKey ^= XorMatrix[CurrentElement];
  16791. Char *= 2; --CurrentElement;
  16792. }
  16793. }
  16794. return XorKey;
  16795. };
  16796. return function(password) {
  16797. var Password = _JS2ANSI(password);
  16798. var XorKey = CreateXorKey_Method1(Password);
  16799. var Index = Password.length;
  16800. var ObfuscationArray = new_raw_buf(16);
  16801. for(var i = 0; i != 16; ++i) ObfuscationArray[i] = 0x00;
  16802. var Temp, PasswordLastChar, PadIndex;
  16803. if((Index & 1) === 1) {
  16804. Temp = XorKey >> 8;
  16805. ObfuscationArray[Index] = XorRor(PadArray[0], Temp);
  16806. --Index;
  16807. Temp = XorKey & 0xFF;
  16808. PasswordLastChar = Password[Password.length - 1];
  16809. ObfuscationArray[Index] = XorRor(PasswordLastChar, Temp);
  16810. }
  16811. while(Index > 0) {
  16812. --Index;
  16813. Temp = XorKey >> 8;
  16814. ObfuscationArray[Index] = XorRor(Password[Index], Temp);
  16815. --Index;
  16816. Temp = XorKey & 0xFF;
  16817. ObfuscationArray[Index] = XorRor(Password[Index], Temp);
  16818. }
  16819. Index = 15;
  16820. PadIndex = 15 - Password.length;
  16821. while(PadIndex > 0) {
  16822. Temp = XorKey >> 8;
  16823. ObfuscationArray[Index] = XorRor(PadArray[PadIndex], Temp);
  16824. --Index;
  16825. --PadIndex;
  16826. Temp = XorKey & 0xFF;
  16827. ObfuscationArray[Index] = XorRor(Password[Index], Temp);
  16828. --Index;
  16829. --PadIndex;
  16830. }
  16831. return ObfuscationArray;
  16832. };
  16833. })();
  16834. /* [MS-OFFCRYPTO] 2.3.7.3 Binary Document XOR Data Transformation Method 1 */
  16835. var crypto_DecryptData_Method1 = function(password, Data, XorArrayIndex, XorArray, O) {
  16836. /* If XorArray is set, use it; if O is not set, make changes in-place */
  16837. if(!O) O = Data;
  16838. if(!XorArray) XorArray = crypto_CreateXorArray_Method1(password);
  16839. var Index, Value;
  16840. for(Index = 0; Index != Data.length; ++Index) {
  16841. Value = Data[Index];
  16842. Value ^= XorArray[XorArrayIndex];
  16843. Value = ((Value>>5) | (Value<<3)) & 0xFF;
  16844. O[Index] = Value;
  16845. ++XorArrayIndex;
  16846. }
  16847. return [O, XorArrayIndex, XorArray];
  16848. };
  16849. var crypto_MakeXorDecryptor = function(password) {
  16850. var XorArrayIndex = 0, XorArray = crypto_CreateXorArray_Method1(password);
  16851. return function(Data) {
  16852. var O = crypto_DecryptData_Method1("", Data, XorArrayIndex, XorArray);
  16853. XorArrayIndex = O[1];
  16854. return O[0];
  16855. };
  16856. };
  16857. /* 2.5.343 */
  16858. function parse_XORObfuscation(blob, length, opts, out) {
  16859. var o = ({ key: parseuint16(blob), verificationBytes: parseuint16(blob) });
  16860. if(opts.password) o.verifier = crypto_CreatePasswordVerifier_Method1(opts.password);
  16861. out.valid = o.verificationBytes === o.verifier;
  16862. if(out.valid) out.insitu = crypto_MakeXorDecryptor(opts.password);
  16863. return o;
  16864. }
  16865. /* 2.4.117 */
  16866. function parse_FilePassHeader(blob, length, oo) {
  16867. var o = oo || {}; o.Info = blob.read_shift(2); blob.l -= 2;
  16868. if(o.Info === 1) o.Data = parse_RC4Header(blob, length);
  16869. else o.Data = parse_RC4CryptoHeader(blob, length);
  16870. return o;
  16871. }
  16872. function parse_FilePass(blob, length, opts) {
  16873. var o = ({ Type: opts.biff >= 8 ? blob.read_shift(2) : 0 }); /* wEncryptionType */
  16874. if(o.Type) parse_FilePassHeader(blob, length-2, o);
  16875. else parse_XORObfuscation(blob, opts.biff >= 8 ? length : length - 2, opts, o);
  16876. return o;
  16877. }
  16878. var RTF = (function() {
  16879. function rtf_to_sheet(d, opts) {
  16880. switch(opts.type) {
  16881. case 'base64': return rtf_to_sheet_str(Base64.decode(d), opts);
  16882. case 'binary': return rtf_to_sheet_str(d, opts);
  16883. case 'buffer': return rtf_to_sheet_str(d.toString('binary'), opts);
  16884. case 'array': return rtf_to_sheet_str(cc2str(d), opts);
  16885. }
  16886. throw new Error("Unrecognized type " + opts.type);
  16887. }
  16888. function rtf_to_sheet_str(str, opts) {
  16889. var o = opts || {};
  16890. var ws = o.dense ? ([]) : ({});
  16891. var range = ({s: {c:0, r:0}, e: {c:0, r:0}});
  16892. // TODO: parse
  16893. if(!str.match(/\\trowd/)) throw new Error("RTF missing table");
  16894. ws['!ref'] = encode_range(range);
  16895. return ws;
  16896. }
  16897. function rtf_to_workbook(d, opts) { return sheet_to_workbook(rtf_to_sheet(d, opts), opts); }
  16898. /* TODO: this is a stub */
  16899. function sheet_to_rtf(ws) {
  16900. var o = ["{\\rtf1\\ansi"];
  16901. var r = safe_decode_range(ws['!ref']), cell;
  16902. var dense = Array.isArray(ws);
  16903. for(var R = r.s.r; R <= r.e.r; ++R) {
  16904. o.push("\\trowd\\trautofit1");
  16905. for(var C = r.s.c; C <= r.e.c; ++C) o.push("\\cellx" + (C+1));
  16906. o.push("\\pard\\intbl");
  16907. for(C = r.s.c; C <= r.e.c; ++C) {
  16908. var coord = encode_cell({r:R,c:C});
  16909. cell = dense ? (ws[R]||[])[C]: ws[coord];
  16910. if(!cell || cell.v == null && (!cell.f || cell.F)) continue;
  16911. o.push(" " + (cell.w || (format_cell(cell), cell.w)));
  16912. o.push("\\cell");
  16913. }
  16914. o.push("\\pard\\intbl\\row");
  16915. }
  16916. return o.join("") + "}";
  16917. }
  16918. return {
  16919. to_workbook: rtf_to_workbook,
  16920. to_sheet: rtf_to_sheet,
  16921. from_sheet: sheet_to_rtf
  16922. };
  16923. })();
  16924. function hex2RGB(h) {
  16925. var o = h.slice(h[0]==="#"?1:0).slice(0,6);
  16926. return [parseInt(o.slice(0,2),16),parseInt(o.slice(2,4),16),parseInt(o.slice(4,6),16)];
  16927. }
  16928. function rgb2Hex(rgb) {
  16929. for(var i=0,o=1; i!=3; ++i) o = o*256 + (rgb[i]>255?255:rgb[i]<0?0:rgb[i]);
  16930. return o.toString(16).toUpperCase().slice(1);
  16931. }
  16932. function rgb2HSL(rgb) {
  16933. var R = rgb[0]/255, G = rgb[1]/255, B=rgb[2]/255;
  16934. var M = Math.max(R, G, B), m = Math.min(R, G, B), C = M - m;
  16935. if(C === 0) return [0, 0, R];
  16936. var H6 = 0, S = 0, L2 = (M + m);
  16937. S = C / (L2 > 1 ? 2 - L2 : L2);
  16938. switch(M){
  16939. case R: H6 = ((G - B) / C + 6)%6; break;
  16940. case G: H6 = ((B - R) / C + 2); break;
  16941. case B: H6 = ((R - G) / C + 4); break;
  16942. }
  16943. return [H6 / 6, S, L2 / 2];
  16944. }
  16945. function hsl2RGB(hsl){
  16946. var H = hsl[0], S = hsl[1], L = hsl[2];
  16947. var C = S * 2 * (L < 0.5 ? L : 1 - L), m = L - C/2;
  16948. var rgb = [m,m,m], h6 = 6*H;
  16949. var X;
  16950. if(S !== 0) switch(h6|0) {
  16951. case 0: case 6: X = C * h6; rgb[0] += C; rgb[1] += X; break;
  16952. case 1: X = C * (2 - h6); rgb[0] += X; rgb[1] += C; break;
  16953. case 2: X = C * (h6 - 2); rgb[1] += C; rgb[2] += X; break;
  16954. case 3: X = C * (4 - h6); rgb[1] += X; rgb[2] += C; break;
  16955. case 4: X = C * (h6 - 4); rgb[2] += C; rgb[0] += X; break;
  16956. case 5: X = C * (6 - h6); rgb[2] += X; rgb[0] += C; break;
  16957. }
  16958. for(var i = 0; i != 3; ++i) rgb[i] = Math.round(rgb[i]*255);
  16959. return rgb;
  16960. }
  16961. /* 18.8.3 bgColor tint algorithm */
  16962. function rgb_tint(hex, tint) {
  16963. if(tint === 0) return hex;
  16964. var hsl = rgb2HSL(hex2RGB(hex));
  16965. if (tint < 0) hsl[2] = hsl[2] * (1 + tint);
  16966. else hsl[2] = 1 - (1 - hsl[2]) * (1 - tint);
  16967. return rgb2Hex(hsl2RGB(hsl));
  16968. }
  16969. /* 18.3.1.13 width calculations */
  16970. /* [MS-OI29500] 2.1.595 Column Width & Formatting */
  16971. var DEF_MDW = 6, MAX_MDW = 15, MIN_MDW = 1, MDW = DEF_MDW;
  16972. function width2px(width) { return Math.floor(( width + (Math.round(128/MDW))/256 )* MDW ); }
  16973. function px2char(px) { return (Math.floor((px - 5)/MDW * 100 + 0.5))/100; }
  16974. function char2width(chr) { return (Math.round((chr * MDW + 5)/MDW*256))/256; }
  16975. //function px2char_(px) { return (((px - 5)/MDW * 100 + 0.5))/100; }
  16976. //function char2width_(chr) { return (((chr * MDW + 5)/MDW*256))/256; }
  16977. function cycle_width(collw) { return char2width(px2char(width2px(collw))); }
  16978. /* XLSX/XLSB/XLS specify width in units of MDW */
  16979. function find_mdw_colw(collw) {
  16980. var delta = Math.abs(collw - cycle_width(collw)), _MDW = MDW;
  16981. if(delta > 0.005) for(MDW=MIN_MDW; MDW<MAX_MDW; ++MDW) if(Math.abs(collw - cycle_width(collw)) <= delta) { delta = Math.abs(collw - cycle_width(collw)); _MDW = MDW; }
  16982. MDW = _MDW;
  16983. }
  16984. /* XLML specifies width in terms of pixels */
  16985. /*function find_mdw_wpx(wpx) {
  16986. var delta = Infinity, guess = 0, _MDW = MIN_MDW;
  16987. for(MDW=MIN_MDW; MDW<MAX_MDW; ++MDW) {
  16988. guess = char2width_(px2char_(wpx))*256;
  16989. guess = (guess) % 1;
  16990. if(guess > 0.5) guess--;
  16991. if(Math.abs(guess) < delta) { delta = Math.abs(guess); _MDW = MDW; }
  16992. }
  16993. MDW = _MDW;
  16994. }*/
  16995. function process_col(coll) {
  16996. if(coll.width) {
  16997. coll.wpx = width2px(coll.width);
  16998. coll.wch = px2char(coll.wpx);
  16999. coll.MDW = MDW;
  17000. } else if(coll.wpx) {
  17001. coll.wch = px2char(coll.wpx);
  17002. coll.width = char2width(coll.wch);
  17003. coll.MDW = MDW;
  17004. } else if(typeof coll.wch == 'number') {
  17005. coll.width = char2width(coll.wch);
  17006. coll.wpx = width2px(coll.width);
  17007. coll.MDW = MDW;
  17008. }
  17009. if(coll.customWidth) delete coll.customWidth;
  17010. }
  17011. var DEF_PPI = 96, PPI = DEF_PPI;
  17012. function px2pt(px) { return px * 96 / PPI; }
  17013. function pt2px(pt) { return pt * PPI / 96; }
  17014. /* [MS-EXSPXML3] 2.4.54 ST_enmPattern */
  17015. var XLMLPatternTypeMap = {
  17016. "None": "none",
  17017. "Solid": "solid",
  17018. "Gray50": "mediumGray",
  17019. "Gray75": "darkGray",
  17020. "Gray25": "lightGray",
  17021. "HorzStripe": "darkHorizontal",
  17022. "VertStripe": "darkVertical",
  17023. "ReverseDiagStripe": "darkDown",
  17024. "DiagStripe": "darkUp",
  17025. "DiagCross": "darkGrid",
  17026. "ThickDiagCross": "darkTrellis",
  17027. "ThinHorzStripe": "lightHorizontal",
  17028. "ThinVertStripe": "lightVertical",
  17029. "ThinReverseDiagStripe": "lightDown",
  17030. "ThinHorzCross": "lightGrid"
  17031. };
  17032. /* 18.8.5 borders CT_Borders */
  17033. function parse_borders(t, styles, themes, opts) {
  17034. styles.Borders = [];
  17035. var border = {}/*, sub_border = {}*/;
  17036. var pass = false;
  17037. t[0].match(tagregex).forEach(function(x) {
  17038. var y = parsexmltag(x);
  17039. switch(strip_ns(y[0])) {
  17040. case '<borders': case '<borders>': case '</borders>': break;
  17041. /* 18.8.4 border CT_Border */
  17042. case '<border': case '<border>': case '<border/>':
  17043. border = {};
  17044. if (y.diagonalUp) { border.diagonalUp = y.diagonalUp; }
  17045. if (y.diagonalDown) { border.diagonalDown = y.diagonalDown; }
  17046. styles.Borders.push(border);
  17047. break;
  17048. case '</border>': break;
  17049. /* note: not in spec, appears to be CT_BorderPr */
  17050. case '<left/>': break;
  17051. case '<left': case '<left>': break;
  17052. case '</left>': break;
  17053. /* note: not in spec, appears to be CT_BorderPr */
  17054. case '<right/>': break;
  17055. case '<right': case '<right>': break;
  17056. case '</right>': break;
  17057. /* 18.8.43 top CT_BorderPr */
  17058. case '<top/>': break;
  17059. case '<top': case '<top>': break;
  17060. case '</top>': break;
  17061. /* 18.8.6 bottom CT_BorderPr */
  17062. case '<bottom/>': break;
  17063. case '<bottom': case '<bottom>': break;
  17064. case '</bottom>': break;
  17065. /* 18.8.13 diagonal CT_BorderPr */
  17066. case '<diagonal': case '<diagonal>': case '<diagonal/>': break;
  17067. case '</diagonal>': break;
  17068. /* 18.8.25 horizontal CT_BorderPr */
  17069. case '<horizontal': case '<horizontal>': case '<horizontal/>': break;
  17070. case '</horizontal>': break;
  17071. /* 18.8.44 vertical CT_BorderPr */
  17072. case '<vertical': case '<vertical>': case '<vertical/>': break;
  17073. case '</vertical>': break;
  17074. /* 18.8.37 start CT_BorderPr */
  17075. case '<start': case '<start>': case '<start/>': break;
  17076. case '</start>': break;
  17077. /* 18.8.16 end CT_BorderPr */
  17078. case '<end': case '<end>': case '<end/>': break;
  17079. case '</end>': break;
  17080. /* 18.8.? color CT_Color */
  17081. case '<color': case '<color>': break;
  17082. case '<color/>': case '</color>': break;
  17083. /* 18.2.10 extLst CT_ExtensionList ? */
  17084. case '<extLst': case '<extLst>': case '</extLst>': break;
  17085. case '<ext': pass = true; break;
  17086. case '</ext>': pass = false; break;
  17087. default: if(opts && opts.WTF) {
  17088. if(!pass) throw new Error('unrecognized ' + y[0] + ' in borders');
  17089. }
  17090. }
  17091. });
  17092. }
  17093. /* 18.8.21 fills CT_Fills */
  17094. function parse_fills(t, styles, themes, opts) {
  17095. styles.Fills = [];
  17096. var fill = {};
  17097. var pass = false;
  17098. t[0].match(tagregex).forEach(function(x) {
  17099. var y = parsexmltag(x);
  17100. switch(strip_ns(y[0])) {
  17101. case '<fills': case '<fills>': case '</fills>': break;
  17102. /* 18.8.20 fill CT_Fill */
  17103. case '<fill>': case '<fill': case '<fill/>':
  17104. fill = {}; styles.Fills.push(fill); break;
  17105. case '</fill>': break;
  17106. /* 18.8.24 gradientFill CT_GradientFill */
  17107. case '<gradientFill>': break;
  17108. case '<gradientFill':
  17109. case '</gradientFill>': styles.Fills.push(fill); fill = {}; break;
  17110. /* 18.8.32 patternFill CT_PatternFill */
  17111. case '<patternFill': case '<patternFill>':
  17112. if(y.patternType) fill.patternType = y.patternType;
  17113. break;
  17114. case '<patternFill/>': case '</patternFill>': break;
  17115. /* 18.8.3 bgColor CT_Color */
  17116. case '<bgColor':
  17117. if(!fill.bgColor) fill.bgColor = {};
  17118. if(y.indexed) fill.bgColor.indexed = parseInt(y.indexed, 10);
  17119. if(y.theme) fill.bgColor.theme = parseInt(y.theme, 10);
  17120. if(y.tint) fill.bgColor.tint = parseFloat(y.tint);
  17121. /* Excel uses ARGB strings */
  17122. if(y.rgb) fill.bgColor.rgb = y.rgb.slice(-6);
  17123. break;
  17124. case '<bgColor/>': case '</bgColor>': break;
  17125. /* 18.8.19 fgColor CT_Color */
  17126. case '<fgColor':
  17127. if(!fill.fgColor) fill.fgColor = {};
  17128. if(y.theme) fill.fgColor.theme = parseInt(y.theme, 10);
  17129. if(y.tint) fill.fgColor.tint = parseFloat(y.tint);
  17130. /* Excel uses ARGB strings */
  17131. if(y.rgb) fill.fgColor.rgb = y.rgb.slice(-6);
  17132. break;
  17133. case '<fgColor/>': case '</fgColor>': break;
  17134. /* 18.8.38 stop CT_GradientStop */
  17135. case '<stop': case '<stop/>': break;
  17136. case '</stop>': break;
  17137. /* 18.8.? color CT_Color */
  17138. case '<color': case '<color/>': break;
  17139. case '</color>': break;
  17140. /* 18.2.10 extLst CT_ExtensionList ? */
  17141. case '<extLst': case '<extLst>': case '</extLst>': break;
  17142. case '<ext': pass = true; break;
  17143. case '</ext>': pass = false; break;
  17144. default: if(opts && opts.WTF) {
  17145. if(!pass) throw new Error('unrecognized ' + y[0] + ' in fills');
  17146. }
  17147. }
  17148. });
  17149. }
  17150. /* 18.8.23 fonts CT_Fonts */
  17151. function parse_fonts(t, styles, themes, opts) {
  17152. styles.Fonts = [];
  17153. var font = {};
  17154. var pass = false;
  17155. t[0].match(tagregex).forEach(function(x) {
  17156. var y = parsexmltag(x);
  17157. switch(strip_ns(y[0])) {
  17158. case '<fonts': case '<fonts>': case '</fonts>': break;
  17159. /* 18.8.22 font CT_Font */
  17160. case '<font': case '<font>': break;
  17161. case '</font>': case '<font/>':
  17162. styles.Fonts.push(font);
  17163. font = {};
  17164. break;
  17165. /* 18.8.29 name CT_FontName */
  17166. case '<name': if(y.val) font.name = y.val; break;
  17167. case '<name/>': case '</name>': break;
  17168. /* 18.8.2 b CT_BooleanProperty */
  17169. case '<b': font.bold = y.val ? parsexmlbool(y.val) : 1; break;
  17170. case '<b/>': font.bold = 1; break;
  17171. /* 18.8.26 i CT_BooleanProperty */
  17172. case '<i': font.italic = y.val ? parsexmlbool(y.val) : 1; break;
  17173. case '<i/>': font.italic = 1; break;
  17174. /* 18.4.13 u CT_UnderlineProperty */
  17175. case '<u':
  17176. switch(y.val) {
  17177. case "none": font.underline = 0x00; break;
  17178. case "single": font.underline = 0x01; break;
  17179. case "double": font.underline = 0x02; break;
  17180. case "singleAccounting": font.underline = 0x21; break;
  17181. case "doubleAccounting": font.underline = 0x22; break;
  17182. } break;
  17183. case '<u/>': font.underline = 1; break;
  17184. /* 18.4.10 strike CT_BooleanProperty */
  17185. case '<strike': font.strike = y.val ? parsexmlbool(y.val) : 1; break;
  17186. case '<strike/>': font.strike = 1; break;
  17187. /* 18.4.2 outline CT_BooleanProperty */
  17188. case '<outline': font.outline = y.val ? parsexmlbool(y.val) : 1; break;
  17189. case '<outline/>': font.outline = 1; break;
  17190. /* 18.8.36 shadow CT_BooleanProperty */
  17191. case '<shadow': font.shadow = y.val ? parsexmlbool(y.val) : 1; break;
  17192. case '<shadow/>': font.shadow = 1; break;
  17193. /* 18.8.12 condense CT_BooleanProperty */
  17194. case '<condense': font.condense = y.val ? parsexmlbool(y.val) : 1; break;
  17195. case '<condense/>': font.condense = 1; break;
  17196. /* 18.8.17 extend CT_BooleanProperty */
  17197. case '<extend': font.extend = y.val ? parsexmlbool(y.val) : 1; break;
  17198. case '<extend/>': font.extend = 1; break;
  17199. /* 18.4.11 sz CT_FontSize */
  17200. case '<sz': if(y.val) font.sz = +y.val; break;
  17201. case '<sz/>': case '</sz>': break;
  17202. /* 18.4.14 vertAlign CT_VerticalAlignFontProperty */
  17203. case '<vertAlign': if(y.val) font.vertAlign = y.val; break;
  17204. case '<vertAlign/>': case '</vertAlign>': break;
  17205. /* 18.8.18 family CT_FontFamily */
  17206. case '<family': if(y.val) font.family = parseInt(y.val,10); break;
  17207. case '<family/>': case '</family>': break;
  17208. /* 18.8.35 scheme CT_FontScheme */
  17209. case '<scheme': if(y.val) font.scheme = y.val; break;
  17210. case '<scheme/>': case '</scheme>': break;
  17211. /* 18.4.1 charset CT_IntProperty */
  17212. case '<charset':
  17213. if(y.val == '1') break;
  17214. y.codepage = CS2CP[parseInt(y.val, 10)];
  17215. break;
  17216. /* 18.?.? color CT_Color */
  17217. case '<color':
  17218. if(!font.color) font.color = {};
  17219. if(y.auto) font.color.auto = parsexmlbool(y.auto);
  17220. if(y.rgb) font.color.rgb = y.rgb.slice(-6);
  17221. else if(y.indexed) {
  17222. font.color.index = parseInt(y.indexed, 10);
  17223. var icv = XLSIcv[font.color.index];
  17224. if(font.color.index == 81) icv = XLSIcv[1];
  17225. if(!icv) throw new Error(x);
  17226. font.color.rgb = icv[0].toString(16) + icv[1].toString(16) + icv[2].toString(16);
  17227. } else if(y.theme) {
  17228. font.color.theme = parseInt(y.theme, 10);
  17229. if(y.tint) font.color.tint = parseFloat(y.tint);
  17230. if(y.theme && themes.themeElements && themes.themeElements.clrScheme) {
  17231. font.color.rgb = rgb_tint(themes.themeElements.clrScheme[font.color.theme].rgb, font.color.tint || 0);
  17232. }
  17233. }
  17234. break;
  17235. case '<color/>': case '</color>': break;
  17236. /* 18.2.10 extLst CT_ExtensionList ? */
  17237. case '<extLst': case '<extLst>': case '</extLst>': break;
  17238. case '<ext': pass = true; break;
  17239. case '</ext>': pass = false; break;
  17240. default: if(opts && opts.WTF) {
  17241. if(!pass) throw new Error('unrecognized ' + y[0] + ' in fonts');
  17242. }
  17243. }
  17244. });
  17245. }
  17246. /* 18.8.31 numFmts CT_NumFmts */
  17247. function parse_numFmts(t, styles, opts) {
  17248. styles.NumberFmt = [];
  17249. var k/*Array<number>*/ = (keys(SSF._table));
  17250. for(var i=0; i < k.length; ++i) styles.NumberFmt[k[i]] = SSF._table[k[i]];
  17251. var m = t[0].match(tagregex);
  17252. if(!m) return;
  17253. for(i=0; i < m.length; ++i) {
  17254. var y = parsexmltag(m[i]);
  17255. switch(strip_ns(y[0])) {
  17256. case '<numFmts': case '</numFmts>': case '<numFmts/>': case '<numFmts>': break;
  17257. case '<numFmt': {
  17258. var f=unescapexml(utf8read(y.formatCode)), j=parseInt(y.numFmtId,10);
  17259. styles.NumberFmt[j] = f;
  17260. if(j>0) {
  17261. if(j > 0x188) {
  17262. for(j = 0x188; j > 0x3c; --j) if(styles.NumberFmt[j] == null) break;
  17263. styles.NumberFmt[j] = f;
  17264. }
  17265. SSF.load(f,j);
  17266. }
  17267. } break;
  17268. case '</numFmt>': break;
  17269. default: if(opts.WTF) throw new Error('unrecognized ' + y[0] + ' in numFmts');
  17270. }
  17271. }
  17272. }
  17273. function write_numFmts(NF) {
  17274. var o = ["<numFmts>"];
  17275. [[5,8],[23,26],[41,44],[/*63*/50,/*66],[164,*/392]].forEach(function(r) {
  17276. for(var i = r[0]; i <= r[1]; ++i) if(NF[i] != null) o[o.length] = (writextag('numFmt',null,{numFmtId:i,formatCode:escapexml(NF[i])}));
  17277. });
  17278. if(o.length === 1) return "";
  17279. o[o.length] = ("</numFmts>");
  17280. o[0] = writextag('numFmts', null, { count:o.length-2 }).replace("/>", ">");
  17281. return o.join("");
  17282. }
  17283. /* 18.8.10 cellXfs CT_CellXfs */
  17284. var cellXF_uint = [ "numFmtId", "fillId", "fontId", "borderId", "xfId" ];
  17285. var cellXF_bool = [ "applyAlignment", "applyBorder", "applyFill", "applyFont", "applyNumberFormat", "applyProtection", "pivotButton", "quotePrefix" ];
  17286. function parse_cellXfs(t, styles, opts) {
  17287. styles.CellXf = [];
  17288. var xf;
  17289. var pass = false;
  17290. t[0].match(tagregex).forEach(function(x) {
  17291. var y = parsexmltag(x), i = 0;
  17292. switch(strip_ns(y[0])) {
  17293. case '<cellXfs': case '<cellXfs>': case '<cellXfs/>': case '</cellXfs>': break;
  17294. /* 18.8.45 xf CT_Xf */
  17295. case '<xf': case '<xf/>':
  17296. xf = y;
  17297. delete xf[0];
  17298. for(i = 0; i < cellXF_uint.length; ++i) if(xf[cellXF_uint[i]])
  17299. xf[cellXF_uint[i]] = parseInt(xf[cellXF_uint[i]], 10);
  17300. for(i = 0; i < cellXF_bool.length; ++i) if(xf[cellXF_bool[i]])
  17301. xf[cellXF_bool[i]] = parsexmlbool(xf[cellXF_bool[i]]);
  17302. if(xf.numFmtId > 0x188) {
  17303. for(i = 0x188; i > 0x3c; --i) if(styles.NumberFmt[xf.numFmtId] == styles.NumberFmt[i]) { xf.numFmtId = i; break; }
  17304. }
  17305. styles.CellXf.push(xf); break;
  17306. case '</xf>': break;
  17307. /* 18.8.1 alignment CT_CellAlignment */
  17308. case '<alignment': case '<alignment/>':
  17309. var alignment = {};
  17310. if(y.vertical) alignment.vertical = y.vertical;
  17311. if(y.horizontal) alignment.horizontal = y.horizontal;
  17312. if(y.textRotation != null) alignment.textRotation = y.textRotation;
  17313. if(y.indent) alignment.indent = y.indent;
  17314. if(y.wrapText) alignment.wrapText = y.wrapText;
  17315. xf.alignment = alignment;
  17316. break;
  17317. case '</alignment>': break;
  17318. /* 18.8.33 protection CT_CellProtection */
  17319. case '<protection': case '</protection>': case '<protection/>': break;
  17320. /* 18.2.10 extLst CT_ExtensionList ? */
  17321. case '<extLst': case '<extLst>': case '</extLst>': break;
  17322. case '<ext': pass = true; break;
  17323. case '</ext>': pass = false; break;
  17324. default: if(opts && opts.WTF) {
  17325. if(!pass) throw new Error('unrecognized ' + y[0] + ' in cellXfs');
  17326. }
  17327. }
  17328. });
  17329. }
  17330. function write_cellXfs(cellXfs) {
  17331. var o = [];
  17332. o[o.length] = (writextag('cellXfs',null));
  17333. cellXfs.forEach(function(c) { o[o.length] = (writextag('xf', null, c)); });
  17334. o[o.length] = ("</cellXfs>");
  17335. if(o.length === 2) return "";
  17336. o[0] = writextag('cellXfs',null, {count:o.length-2}).replace("/>",">");
  17337. return o.join("");
  17338. }
  17339. /* 18.8 Styles CT_Stylesheet*/
  17340. var parse_sty_xml= (function make_pstyx() {
  17341. var numFmtRegex = /<(?:\w+:)?numFmts([^>]*)>[\S\s]*?<\/(?:\w+:)?numFmts>/;
  17342. var cellXfRegex = /<(?:\w+:)?cellXfs([^>]*)>[\S\s]*?<\/(?:\w+:)?cellXfs>/;
  17343. var fillsRegex = /<(?:\w+:)?fills([^>]*)>[\S\s]*?<\/(?:\w+:)?fills>/;
  17344. var fontsRegex = /<(?:\w+:)?fonts([^>]*)>[\S\s]*?<\/(?:\w+:)?fonts>/;
  17345. var bordersRegex = /<(?:\w+:)?borders([^>]*)>[\S\s]*?<\/(?:\w+:)?borders>/;
  17346. return function parse_sty_xml(data, themes, opts) {
  17347. var styles = {};
  17348. if(!data) return styles;
  17349. data = data.replace(/<!--([\s\S]*?)-->/mg,"").replace(/<!DOCTYPE[^\[]*\[[^\]]*\]>/gm,"");
  17350. /* 18.8.39 styleSheet CT_Stylesheet */
  17351. var t;
  17352. /* 18.8.31 numFmts CT_NumFmts ? */
  17353. if((t=data.match(numFmtRegex))) parse_numFmts(t, styles, opts);
  17354. /* 18.8.23 fonts CT_Fonts ? */
  17355. if((t=data.match(fontsRegex))) parse_fonts(t, styles, themes, opts);
  17356. /* 18.8.21 fills CT_Fills ? */
  17357. if((t=data.match(fillsRegex))) parse_fills(t, styles, themes, opts);
  17358. /* 18.8.5 borders CT_Borders ? */
  17359. if((t=data.match(bordersRegex))) parse_borders(t, styles, themes, opts);
  17360. /* 18.8.9 cellStyleXfs CT_CellStyleXfs ? */
  17361. /* 18.8.10 cellXfs CT_CellXfs ? */
  17362. if((t=data.match(cellXfRegex))) parse_cellXfs(t, styles, opts);
  17363. /* 18.8.8 cellStyles CT_CellStyles ? */
  17364. /* 18.8.15 dxfs CT_Dxfs ? */
  17365. /* 18.8.42 tableStyles CT_TableStyles ? */
  17366. /* 18.8.11 colors CT_Colors ? */
  17367. /* 18.2.10 extLst CT_ExtensionList ? */
  17368. return styles;
  17369. };
  17370. })();
  17371. var STYLES_XML_ROOT = writextag('styleSheet', null, {
  17372. 'xmlns': XMLNS.main[0],
  17373. 'xmlns:vt': XMLNS.vt
  17374. });
  17375. RELS.STY = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles";
  17376. function write_sty_xml(wb, opts) {
  17377. if (typeof style_builder != 'undefined' && typeof 'require' != 'undefined') {
  17378. return style_builder.toXml();
  17379. }
  17380. var o = [XML_HEADER, STYLES_XML_ROOT], w;
  17381. if(wb.SSF && (w = write_numFmts(wb.SSF)) != null) o[o.length] = w;
  17382. o[o.length] = ('<fonts count="1"><font><sz val="12"/><color theme="1"/><name val="Calibri"/><family val="2"/><scheme val="minor"/></font></fonts>');
  17383. o[o.length] = ('<fills count="2"><fill><patternFill patternType="none"/></fill><fill><patternFill patternType="gray125"/></fill></fills>');
  17384. o[o.length] = ('<borders count="1"><border><left/><right/><top/><bottom/><diagonal/></border></borders>');
  17385. o[o.length] = ('<cellStyleXfs count="1"><xf numFmtId="0" fontId="0" fillId="0" borderId="0"/></cellStyleXfs>');
  17386. if((w = write_cellXfs(opts.cellXfs))) o[o.length] = (w);
  17387. o[o.length] = ('<cellStyles count="1"><cellStyle name="Normal" xfId="0" builtinId="0"/></cellStyles>');
  17388. o[o.length] = ('<dxfs count="0"/>');
  17389. o[o.length] = ('<tableStyles count="0" defaultTableStyle="TableStyleMedium9" defaultPivotStyle="PivotStyleMedium4"/>');
  17390. if(o.length>2){ o[o.length] = ('</styleSheet>'); o[1]=o[1].replace("/>",">"); }
  17391. return o.join("");
  17392. }
  17393. /* [MS-XLSB] 2.4.657 BrtFmt */
  17394. function parse_BrtFmt(data, length) {
  17395. var numFmtId = data.read_shift(2);
  17396. var stFmtCode = parse_XLWideString(data,length-2);
  17397. return [numFmtId, stFmtCode];
  17398. }
  17399. function write_BrtFmt(i, f, o) {
  17400. if(!o) o = new_buf(6 + 4 * f.length);
  17401. o.write_shift(2, i);
  17402. write_XLWideString(f, o);
  17403. var out = (o.length > o.l) ? o.slice(0, o.l) : o;
  17404. if(o.l == null) o.l = o.length;
  17405. return out;
  17406. }
  17407. /* [MS-XLSB] 2.4.659 BrtFont TODO */
  17408. function parse_BrtFont(data, length, opts) {
  17409. var out = ({});
  17410. out.sz = data.read_shift(2) / 20;
  17411. var grbit = parse_FontFlags(data, 2, opts);
  17412. if(grbit.fCondense) out.condense = 1;
  17413. if(grbit.fExtend) out.extend = 1;
  17414. if(grbit.fShadow) out.shadow = 1;
  17415. if(grbit.fOutline) out.outline = 1;
  17416. if(grbit.fStrikeout) out.strike = 1;
  17417. if(grbit.fItalic) out.italic = 1;
  17418. var bls = data.read_shift(2);
  17419. if(bls === 0x02BC) out.bold = 1;
  17420. switch(data.read_shift(2)) {
  17421. /* case 0: out.vertAlign = "baseline"; break; */
  17422. case 1: out.vertAlign = "superscript"; break;
  17423. case 2: out.vertAlign = "subscript"; break;
  17424. }
  17425. var underline = data.read_shift(1);
  17426. if(underline != 0) out.underline = underline;
  17427. var family = data.read_shift(1);
  17428. if(family > 0) out.family = family;
  17429. var bCharSet = data.read_shift(1);
  17430. if(bCharSet > 0) out.charset = bCharSet;
  17431. data.l++;
  17432. out.color = parse_BrtColor(data, 8);
  17433. switch(data.read_shift(1)) {
  17434. /* case 0: out.scheme = "none": break; */
  17435. case 1: out.scheme = "major"; break;
  17436. case 2: out.scheme = "minor"; break;
  17437. }
  17438. out.name = parse_XLWideString(data, length - 21);
  17439. return out;
  17440. }
  17441. function write_BrtFont(font, o) {
  17442. if(!o) o = new_buf(25+4*32);
  17443. o.write_shift(2, font.sz * 20);
  17444. write_FontFlags(font, o);
  17445. o.write_shift(2, font.bold ? 0x02BC : 0x0190);
  17446. var sss = 0;
  17447. if(font.vertAlign == "superscript") sss = 1;
  17448. else if(font.vertAlign == "subscript") sss = 2;
  17449. o.write_shift(2, sss);
  17450. o.write_shift(1, font.underline || 0);
  17451. o.write_shift(1, font.family || 0);
  17452. o.write_shift(1, font.charset || 0);
  17453. o.write_shift(1, 0);
  17454. write_BrtColor(font.color, o);
  17455. var scheme = 0;
  17456. if(font.scheme == "major") scheme = 1;
  17457. if(font.scheme == "minor") scheme = 2;
  17458. o.write_shift(1, scheme);
  17459. write_XLWideString(font.name, o);
  17460. return o.length > o.l ? o.slice(0, o.l) : o;
  17461. }
  17462. /* [MS-XLSB] 2.4.650 BrtFill */
  17463. var XLSBFillPTNames = [
  17464. "none",
  17465. "solid",
  17466. "mediumGray",
  17467. "darkGray",
  17468. "lightGray",
  17469. "darkHorizontal",
  17470. "darkVertical",
  17471. "darkDown",
  17472. "darkUp",
  17473. "darkGrid",
  17474. "darkTrellis",
  17475. "lightHorizontal",
  17476. "lightVertical",
  17477. "lightDown",
  17478. "lightUp",
  17479. "lightGrid",
  17480. "lightTrellis",
  17481. "gray125",
  17482. "gray0625"
  17483. ];
  17484. var rev_XLSBFillPTNames = (evert(XLSBFillPTNames));
  17485. /* TODO: gradient fill representation */
  17486. var parse_BrtFill = parsenoop;
  17487. function write_BrtFill(fill, o) {
  17488. if(!o) o = new_buf(4*3 + 8*7 + 16*1);
  17489. var fls = rev_XLSBFillPTNames[fill.patternType];
  17490. if(fls == null) fls = 0x28;
  17491. o.write_shift(4, fls);
  17492. var j = 0;
  17493. if(fls != 0x28) {
  17494. /* TODO: custom FG Color */
  17495. write_BrtColor({auto:1}, o);
  17496. /* TODO: custom BG Color */
  17497. write_BrtColor({auto:1}, o);
  17498. for(; j < 12; ++j) o.write_shift(4, 0);
  17499. } else {
  17500. for(; j < 4; ++j) o.write_shift(4, 0);
  17501. for(; j < 12; ++j) o.write_shift(4, 0); /* TODO */
  17502. /* iGradientType */
  17503. /* xnumDegree */
  17504. /* xnumFillToLeft */
  17505. /* xnumFillToRight */
  17506. /* xnumFillToTop */
  17507. /* xnumFillToBottom */
  17508. /* cNumStop */
  17509. /* xfillGradientStop */
  17510. }
  17511. return o.length > o.l ? o.slice(0, o.l) : o;
  17512. }
  17513. /* [MS-XLSB] 2.4.824 BrtXF */
  17514. function parse_BrtXF(data, length) {
  17515. var tgt = data.l + length;
  17516. var ixfeParent = data.read_shift(2);
  17517. var ifmt = data.read_shift(2);
  17518. data.l = tgt;
  17519. return {ixfe:ixfeParent, numFmtId:ifmt };
  17520. }
  17521. function write_BrtXF(data, ixfeP, o) {
  17522. if(!o) o = new_buf(16);
  17523. o.write_shift(2, ixfeP||0);
  17524. o.write_shift(2, data.numFmtId||0);
  17525. o.write_shift(2, 0); /* iFont */
  17526. o.write_shift(2, 0); /* iFill */
  17527. o.write_shift(2, 0); /* ixBorder */
  17528. o.write_shift(1, 0); /* trot */
  17529. o.write_shift(1, 0); /* indent */
  17530. o.write_shift(1, 0); /* flags */
  17531. o.write_shift(1, 0); /* flags */
  17532. o.write_shift(1, 0); /* xfGrbitAtr */
  17533. o.write_shift(1, 0);
  17534. return o;
  17535. }
  17536. /* [MS-XLSB] 2.5.4 Blxf TODO */
  17537. function write_Blxf(data, o) {
  17538. if(!o) o = new_buf(10);
  17539. o.write_shift(1, 0); /* dg */
  17540. o.write_shift(1, 0);
  17541. o.write_shift(4, 0); /* color */
  17542. o.write_shift(4, 0); /* color */
  17543. return o;
  17544. }
  17545. /* [MS-XLSB] 2.4.302 BrtBorder TODO */
  17546. var parse_BrtBorder = parsenoop;
  17547. function write_BrtBorder(border, o) {
  17548. if(!o) o = new_buf(51);
  17549. o.write_shift(1, 0); /* diagonal */
  17550. write_Blxf(null, o); /* top */
  17551. write_Blxf(null, o); /* bottom */
  17552. write_Blxf(null, o); /* left */
  17553. write_Blxf(null, o); /* right */
  17554. write_Blxf(null, o); /* diag */
  17555. return o.length > o.l ? o.slice(0, o.l) : o;
  17556. }
  17557. /* [MS-XLSB] 2.4.763 BrtStyle TODO */
  17558. function write_BrtStyle(style, o) {
  17559. if(!o) o = new_buf(12+4*10);
  17560. o.write_shift(4, style.xfId);
  17561. o.write_shift(2, 1);
  17562. o.write_shift(1, +style.builtinId);
  17563. o.write_shift(1, 0); /* iLevel */
  17564. write_XLNullableWideString(style.name || "", o);
  17565. return o.length > o.l ? o.slice(0, o.l) : o;
  17566. }
  17567. /* [MS-XLSB] 2.4.272 BrtBeginTableStyles */
  17568. function write_BrtBeginTableStyles(cnt, defTableStyle, defPivotStyle) {
  17569. var o = new_buf(4+256*2*4);
  17570. o.write_shift(4, cnt);
  17571. write_XLNullableWideString(defTableStyle, o);
  17572. write_XLNullableWideString(defPivotStyle, o);
  17573. return o.length > o.l ? o.slice(0, o.l) : o;
  17574. }
  17575. /* [MS-XLSB] 2.1.7.50 Styles */
  17576. function parse_sty_bin(data, themes, opts) {
  17577. var styles = {};
  17578. styles.NumberFmt = ([]);
  17579. for(var y in SSF._table) styles.NumberFmt[y] = SSF._table[y];
  17580. styles.CellXf = [];
  17581. styles.Fonts = [];
  17582. var state = [];
  17583. var pass = false;
  17584. recordhopper(data, function hopper_sty(val, R_n, RT) {
  17585. switch(RT) {
  17586. case 0x002C: /* 'BrtFmt' */
  17587. styles.NumberFmt[val[0]] = val[1]; SSF.load(val[1], val[0]);
  17588. break;
  17589. case 0x002B: /* 'BrtFont' */
  17590. styles.Fonts.push(val);
  17591. if(val.color.theme != null && themes && themes.themeElements && themes.themeElements.clrScheme) {
  17592. val.color.rgb = rgb_tint(themes.themeElements.clrScheme[val.color.theme].rgb, val.color.tint || 0);
  17593. }
  17594. break;
  17595. case 0x0401: /* 'BrtKnownFonts' */ break;
  17596. case 0x002D: /* 'BrtFill' */ break;
  17597. case 0x002E: /* 'BrtBorder' */ break;
  17598. case 0x002F: /* 'BrtXF' */
  17599. if(state[state.length - 1] == "BrtBeginCellXFs") {
  17600. styles.CellXf.push(val);
  17601. }
  17602. break;
  17603. case 0x0030: /* 'BrtStyle' */
  17604. case 0x01FB: /* 'BrtDXF' */
  17605. case 0x023C: /* 'BrtMRUColor' */
  17606. case 0x01DB: /* 'BrtIndexedColor': */
  17607. break;
  17608. case 0x0493: /* 'BrtDXF14' */
  17609. case 0x0836: /* 'BrtDXF15' */
  17610. case 0x046A: /* 'BrtSlicerStyleElement' */
  17611. case 0x0200: /* 'BrtTableStyleElement' */
  17612. case 0x082F: /* 'BrtTimelineStyleElement' */
  17613. case 0x0C00: /* 'BrtUid' */
  17614. break;
  17615. case 0x0023: /* 'BrtFRTBegin' */
  17616. pass = true; break;
  17617. case 0x0024: /* 'BrtFRTEnd' */
  17618. pass = false; break;
  17619. case 0x0025: /* 'BrtACBegin' */
  17620. state.push(R_n); break;
  17621. case 0x0026: /* 'BrtACEnd' */
  17622. state.pop(); break;
  17623. default:
  17624. if((R_n||"").indexOf("Begin") > 0) state.push(R_n);
  17625. else if((R_n||"").indexOf("End") > 0) state.pop();
  17626. else if(!pass || opts.WTF) throw new Error("Unexpected record " + RT + " " + R_n);
  17627. }
  17628. });
  17629. return styles;
  17630. }
  17631. function write_FMTS_bin(ba, NF) {
  17632. if(!NF) return;
  17633. var cnt = 0;
  17634. [[5,8],[23,26],[41,44],[/*63*/50,/*66],[164,*/392]].forEach(function(r) {
  17635. for(var i = r[0]; i <= r[1]; ++i) if(NF[i] != null) ++cnt;
  17636. });
  17637. if(cnt == 0) return;
  17638. write_record(ba, "BrtBeginFmts", write_UInt32LE(cnt));
  17639. [[5,8],[23,26],[41,44],[/*63*/50,/*66],[164,*/392]].forEach(function(r) {
  17640. for(var i = r[0]; i <= r[1]; ++i) if(NF[i] != null) write_record(ba, "BrtFmt", write_BrtFmt(i, NF[i]));
  17641. });
  17642. write_record(ba, "BrtEndFmts");
  17643. }
  17644. function write_FONTS_bin(ba) {
  17645. var cnt = 1;
  17646. if(cnt == 0) return;
  17647. write_record(ba, "BrtBeginFonts", write_UInt32LE(cnt));
  17648. write_record(ba, "BrtFont", write_BrtFont({
  17649. sz:12,
  17650. color: {theme:1},
  17651. name: "Calibri",
  17652. family: 2,
  17653. scheme: "minor"
  17654. }));
  17655. /* 1*65491BrtFont [ACFONTS] */
  17656. write_record(ba, "BrtEndFonts");
  17657. }
  17658. function write_FILLS_bin(ba) {
  17659. var cnt = 2;
  17660. if(cnt == 0) return;
  17661. write_record(ba, "BrtBeginFills", write_UInt32LE(cnt));
  17662. write_record(ba, "BrtFill", write_BrtFill({patternType:"none"}));
  17663. write_record(ba, "BrtFill", write_BrtFill({patternType:"gray125"}));
  17664. /* 1*65431BrtFill */
  17665. write_record(ba, "BrtEndFills");
  17666. }
  17667. function write_BORDERS_bin(ba) {
  17668. var cnt = 1;
  17669. if(cnt == 0) return;
  17670. write_record(ba, "BrtBeginBorders", write_UInt32LE(cnt));
  17671. write_record(ba, "BrtBorder", write_BrtBorder({}));
  17672. /* 1*65430BrtBorder */
  17673. write_record(ba, "BrtEndBorders");
  17674. }
  17675. function write_CELLSTYLEXFS_bin(ba) {
  17676. var cnt = 1;
  17677. write_record(ba, "BrtBeginCellStyleXFs", write_UInt32LE(cnt));
  17678. write_record(ba, "BrtXF", write_BrtXF({
  17679. numFmtId:0,
  17680. fontId:0,
  17681. fillId:0,
  17682. borderId:0
  17683. }, 0xFFFF));
  17684. /* 1*65430(BrtXF *FRT) */
  17685. write_record(ba, "BrtEndCellStyleXFs");
  17686. }
  17687. function write_CELLXFS_bin(ba, data) {
  17688. write_record(ba, "BrtBeginCellXFs", write_UInt32LE(data.length));
  17689. data.forEach(function(c) { write_record(ba, "BrtXF", write_BrtXF(c,0)); });
  17690. /* 1*65430(BrtXF *FRT) */
  17691. write_record(ba, "BrtEndCellXFs");
  17692. }
  17693. function write_STYLES_bin(ba) {
  17694. var cnt = 1;
  17695. write_record(ba, "BrtBeginStyles", write_UInt32LE(cnt));
  17696. write_record(ba, "BrtStyle", write_BrtStyle({
  17697. xfId:0,
  17698. builtinId:0,
  17699. name:"Normal"
  17700. }));
  17701. /* 1*65430(BrtStyle *FRT) */
  17702. write_record(ba, "BrtEndStyles");
  17703. }
  17704. function write_DXFS_bin(ba) {
  17705. var cnt = 0;
  17706. write_record(ba, "BrtBeginDXFs", write_UInt32LE(cnt));
  17707. /* *2147483647(BrtDXF *FRT) */
  17708. write_record(ba, "BrtEndDXFs");
  17709. }
  17710. function write_TABLESTYLES_bin(ba) {
  17711. var cnt = 0;
  17712. write_record(ba, "BrtBeginTableStyles", write_BrtBeginTableStyles(cnt, "TableStyleMedium9", "PivotStyleMedium4"));
  17713. /* *TABLESTYLE */
  17714. write_record(ba, "BrtEndTableStyles");
  17715. }
  17716. function write_COLORPALETTE_bin() {
  17717. return;
  17718. /* BrtBeginColorPalette [INDEXEDCOLORS] [MRUCOLORS] BrtEndColorPalette */
  17719. }
  17720. /* [MS-XLSB] 2.1.7.50 Styles */
  17721. function write_sty_bin(wb, opts) {
  17722. var ba = buf_array();
  17723. write_record(ba, "BrtBeginStyleSheet");
  17724. write_FMTS_bin(ba, wb.SSF);
  17725. write_FONTS_bin(ba, wb);
  17726. write_FILLS_bin(ba, wb);
  17727. write_BORDERS_bin(ba, wb);
  17728. write_CELLSTYLEXFS_bin(ba, wb);
  17729. write_CELLXFS_bin(ba, opts.cellXfs);
  17730. write_STYLES_bin(ba, wb);
  17731. write_DXFS_bin(ba, wb);
  17732. write_TABLESTYLES_bin(ba, wb);
  17733. write_COLORPALETTE_bin(ba, wb);
  17734. /* FRTSTYLESHEET*/
  17735. write_record(ba, "BrtEndStyleSheet");
  17736. return ba.end();
  17737. }
  17738. RELS.THEME = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/theme";
  17739. /* 20.1.6.2 clrScheme CT_ColorScheme */
  17740. function parse_clrScheme(t, themes, opts) {
  17741. themes.themeElements.clrScheme = [];
  17742. var color = {};
  17743. (t[0].match(tagregex)||[]).forEach(function(x) {
  17744. var y = parsexmltag(x);
  17745. switch(y[0]) {
  17746. /* 20.1.6.2 clrScheme (Color Scheme) CT_ColorScheme */
  17747. case '<a:clrScheme': case '</a:clrScheme>': break;
  17748. /* 20.1.2.3.32 srgbClr CT_SRgbColor */
  17749. case '<a:srgbClr':
  17750. color.rgb = y.val; break;
  17751. /* 20.1.2.3.33 sysClr CT_SystemColor */
  17752. case '<a:sysClr':
  17753. color.rgb = y.lastClr; break;
  17754. /* 20.1.4.1.1 accent1 (Accent 1) */
  17755. /* 20.1.4.1.2 accent2 (Accent 2) */
  17756. /* 20.1.4.1.3 accent3 (Accent 3) */
  17757. /* 20.1.4.1.4 accent4 (Accent 4) */
  17758. /* 20.1.4.1.5 accent5 (Accent 5) */
  17759. /* 20.1.4.1.6 accent6 (Accent 6) */
  17760. /* 20.1.4.1.9 dk1 (Dark 1) */
  17761. /* 20.1.4.1.10 dk2 (Dark 2) */
  17762. /* 20.1.4.1.15 folHlink (Followed Hyperlink) */
  17763. /* 20.1.4.1.19 hlink (Hyperlink) */
  17764. /* 20.1.4.1.22 lt1 (Light 1) */
  17765. /* 20.1.4.1.23 lt2 (Light 2) */
  17766. case '<a:dk1>': case '</a:dk1>':
  17767. case '<a:lt1>': case '</a:lt1>':
  17768. case '<a:dk2>': case '</a:dk2>':
  17769. case '<a:lt2>': case '</a:lt2>':
  17770. case '<a:accent1>': case '</a:accent1>':
  17771. case '<a:accent2>': case '</a:accent2>':
  17772. case '<a:accent3>': case '</a:accent3>':
  17773. case '<a:accent4>': case '</a:accent4>':
  17774. case '<a:accent5>': case '</a:accent5>':
  17775. case '<a:accent6>': case '</a:accent6>':
  17776. case '<a:hlink>': case '</a:hlink>':
  17777. case '<a:folHlink>': case '</a:folHlink>':
  17778. if (y[0].charAt(1) === '/') {
  17779. themes.themeElements.clrScheme.push(color);
  17780. color = {};
  17781. } else {
  17782. color.name = y[0].slice(3, y[0].length - 1);
  17783. }
  17784. break;
  17785. default: if(opts && opts.WTF) throw new Error('Unrecognized ' + y[0] + ' in clrScheme');
  17786. }
  17787. });
  17788. }
  17789. /* 20.1.4.1.18 fontScheme CT_FontScheme */
  17790. function parse_fontScheme() { }
  17791. /* 20.1.4.1.15 fmtScheme CT_StyleMatrix */
  17792. function parse_fmtScheme() { }
  17793. var clrsregex = /<a:clrScheme([^>]*)>[\s\S]*<\/a:clrScheme>/;
  17794. var fntsregex = /<a:fontScheme([^>]*)>[\s\S]*<\/a:fontScheme>/;
  17795. var fmtsregex = /<a:fmtScheme([^>]*)>[\s\S]*<\/a:fmtScheme>/;
  17796. /* 20.1.6.10 themeElements CT_BaseStyles */
  17797. function parse_themeElements(data, themes, opts) {
  17798. themes.themeElements = {};
  17799. var t;
  17800. [
  17801. /* clrScheme CT_ColorScheme */
  17802. ['clrScheme', clrsregex, parse_clrScheme],
  17803. /* fontScheme CT_FontScheme */
  17804. ['fontScheme', fntsregex, parse_fontScheme],
  17805. /* fmtScheme CT_StyleMatrix */
  17806. ['fmtScheme', fmtsregex, parse_fmtScheme]
  17807. ].forEach(function(m) {
  17808. if(!(t=data.match(m[1]))) throw new Error(m[0] + ' not found in themeElements');
  17809. m[2](t, themes, opts);
  17810. });
  17811. }
  17812. var themeltregex = /<a:themeElements([^>]*)>[\s\S]*<\/a:themeElements>/;
  17813. /* 14.2.7 Theme Part */
  17814. function parse_theme_xml(data, opts) {
  17815. /* 20.1.6.9 theme CT_OfficeStyleSheet */
  17816. if(!data || data.length === 0) return parse_theme_xml(write_theme());
  17817. var t;
  17818. var themes = {};
  17819. /* themeElements CT_BaseStyles */
  17820. if(!(t=data.match(themeltregex))) throw new Error('themeElements not found in theme');
  17821. parse_themeElements(t[0], themes, opts);
  17822. return themes;
  17823. }
  17824. function write_theme(Themes, opts) {
  17825. if(opts && opts.themeXLSX) return opts.themeXLSX;
  17826. var o = [XML_HEADER];
  17827. o[o.length] = '<a:theme xmlns:a="http://schemas.openxmlformats.org/drawingml/2006/main" name="Office Theme">';
  17828. o[o.length] = '<a:themeElements>';
  17829. o[o.length] = '<a:clrScheme name="Office">';
  17830. o[o.length] = '<a:dk1><a:sysClr val="windowText" lastClr="000000"/></a:dk1>';
  17831. o[o.length] = '<a:lt1><a:sysClr val="window" lastClr="FFFFFF"/></a:lt1>';
  17832. o[o.length] = '<a:dk2><a:srgbClr val="1F497D"/></a:dk2>';
  17833. o[o.length] = '<a:lt2><a:srgbClr val="EEECE1"/></a:lt2>';
  17834. o[o.length] = '<a:accent1><a:srgbClr val="4F81BD"/></a:accent1>';
  17835. o[o.length] = '<a:accent2><a:srgbClr val="C0504D"/></a:accent2>';
  17836. o[o.length] = '<a:accent3><a:srgbClr val="9BBB59"/></a:accent3>';
  17837. o[o.length] = '<a:accent4><a:srgbClr val="8064A2"/></a:accent4>';
  17838. o[o.length] = '<a:accent5><a:srgbClr val="4BACC6"/></a:accent5>';
  17839. o[o.length] = '<a:accent6><a:srgbClr val="F79646"/></a:accent6>';
  17840. o[o.length] = '<a:hlink><a:srgbClr val="0000FF"/></a:hlink>';
  17841. o[o.length] = '<a:folHlink><a:srgbClr val="800080"/></a:folHlink>';
  17842. o[o.length] = '</a:clrScheme>';
  17843. o[o.length] = '<a:fontScheme name="Office">';
  17844. o[o.length] = '<a:majorFont>';
  17845. o[o.length] = '<a:latin typeface="Cambria"/>';
  17846. o[o.length] = '<a:ea typeface=""/>';
  17847. o[o.length] = '<a:cs typeface=""/>';
  17848. o[o.length] = '<a:font script="Jpan" typeface="MS Pゴシック"/>';
  17849. o[o.length] = '<a:font script="Hang" typeface="맑은 고딕"/>';
  17850. o[o.length] = '<a:font script="Hans" typeface="宋体"/>';
  17851. o[o.length] = '<a:font script="Hant" typeface="新細明體"/>';
  17852. o[o.length] = '<a:font script="Arab" typeface="Times New Roman"/>';
  17853. o[o.length] = '<a:font script="Hebr" typeface="Times New Roman"/>';
  17854. o[o.length] = '<a:font script="Thai" typeface="Tahoma"/>';
  17855. o[o.length] = '<a:font script="Ethi" typeface="Nyala"/>';
  17856. o[o.length] = '<a:font script="Beng" typeface="Vrinda"/>';
  17857. o[o.length] = '<a:font script="Gujr" typeface="Shruti"/>';
  17858. o[o.length] = '<a:font script="Khmr" typeface="MoolBoran"/>';
  17859. o[o.length] = '<a:font script="Knda" typeface="Tunga"/>';
  17860. o[o.length] = '<a:font script="Guru" typeface="Raavi"/>';
  17861. o[o.length] = '<a:font script="Cans" typeface="Euphemia"/>';
  17862. o[o.length] = '<a:font script="Cher" typeface="Plantagenet Cherokee"/>';
  17863. o[o.length] = '<a:font script="Yiii" typeface="Microsoft Yi Baiti"/>';
  17864. o[o.length] = '<a:font script="Tibt" typeface="Microsoft Himalaya"/>';
  17865. o[o.length] = '<a:font script="Thaa" typeface="MV Boli"/>';
  17866. o[o.length] = '<a:font script="Deva" typeface="Mangal"/>';
  17867. o[o.length] = '<a:font script="Telu" typeface="Gautami"/>';
  17868. o[o.length] = '<a:font script="Taml" typeface="Latha"/>';
  17869. o[o.length] = '<a:font script="Syrc" typeface="Estrangelo Edessa"/>';
  17870. o[o.length] = '<a:font script="Orya" typeface="Kalinga"/>';
  17871. o[o.length] = '<a:font script="Mlym" typeface="Kartika"/>';
  17872. o[o.length] = '<a:font script="Laoo" typeface="DokChampa"/>';
  17873. o[o.length] = '<a:font script="Sinh" typeface="Iskoola Pota"/>';
  17874. o[o.length] = '<a:font script="Mong" typeface="Mongolian Baiti"/>';
  17875. o[o.length] = '<a:font script="Viet" typeface="Times New Roman"/>';
  17876. o[o.length] = '<a:font script="Uigh" typeface="Microsoft Uighur"/>';
  17877. o[o.length] = '<a:font script="Geor" typeface="Sylfaen"/>';
  17878. o[o.length] = '</a:majorFont>';
  17879. o[o.length] = '<a:minorFont>';
  17880. o[o.length] = '<a:latin typeface="Calibri"/>';
  17881. o[o.length] = '<a:ea typeface=""/>';
  17882. o[o.length] = '<a:cs typeface=""/>';
  17883. o[o.length] = '<a:font script="Jpan" typeface="MS Pゴシック"/>';
  17884. o[o.length] = '<a:font script="Hang" typeface="맑은 고딕"/>';
  17885. o[o.length] = '<a:font script="Hans" typeface="宋体"/>';
  17886. o[o.length] = '<a:font script="Hant" typeface="新細明體"/>';
  17887. o[o.length] = '<a:font script="Arab" typeface="Arial"/>';
  17888. o[o.length] = '<a:font script="Hebr" typeface="Arial"/>';
  17889. o[o.length] = '<a:font script="Thai" typeface="Tahoma"/>';
  17890. o[o.length] = '<a:font script="Ethi" typeface="Nyala"/>';
  17891. o[o.length] = '<a:font script="Beng" typeface="Vrinda"/>';
  17892. o[o.length] = '<a:font script="Gujr" typeface="Shruti"/>';
  17893. o[o.length] = '<a:font script="Khmr" typeface="DaunPenh"/>';
  17894. o[o.length] = '<a:font script="Knda" typeface="Tunga"/>';
  17895. o[o.length] = '<a:font script="Guru" typeface="Raavi"/>';
  17896. o[o.length] = '<a:font script="Cans" typeface="Euphemia"/>';
  17897. o[o.length] = '<a:font script="Cher" typeface="Plantagenet Cherokee"/>';
  17898. o[o.length] = '<a:font script="Yiii" typeface="Microsoft Yi Baiti"/>';
  17899. o[o.length] = '<a:font script="Tibt" typeface="Microsoft Himalaya"/>';
  17900. o[o.length] = '<a:font script="Thaa" typeface="MV Boli"/>';
  17901. o[o.length] = '<a:font script="Deva" typeface="Mangal"/>';
  17902. o[o.length] = '<a:font script="Telu" typeface="Gautami"/>';
  17903. o[o.length] = '<a:font script="Taml" typeface="Latha"/>';
  17904. o[o.length] = '<a:font script="Syrc" typeface="Estrangelo Edessa"/>';
  17905. o[o.length] = '<a:font script="Orya" typeface="Kalinga"/>';
  17906. o[o.length] = '<a:font script="Mlym" typeface="Kartika"/>';
  17907. o[o.length] = '<a:font script="Laoo" typeface="DokChampa"/>';
  17908. o[o.length] = '<a:font script="Sinh" typeface="Iskoola Pota"/>';
  17909. o[o.length] = '<a:font script="Mong" typeface="Mongolian Baiti"/>';
  17910. o[o.length] = '<a:font script="Viet" typeface="Arial"/>';
  17911. o[o.length] = '<a:font script="Uigh" typeface="Microsoft Uighur"/>';
  17912. o[o.length] = '<a:font script="Geor" typeface="Sylfaen"/>';
  17913. o[o.length] = '</a:minorFont>';
  17914. o[o.length] = '</a:fontScheme>';
  17915. o[o.length] = '<a:fmtScheme name="Office">';
  17916. o[o.length] = '<a:fillStyleLst>';
  17917. o[o.length] = '<a:solidFill><a:schemeClr val="phClr"/></a:solidFill>';
  17918. o[o.length] = '<a:gradFill rotWithShape="1">';
  17919. o[o.length] = '<a:gsLst>';
  17920. o[o.length] = '<a:gs pos="0"><a:schemeClr val="phClr"><a:tint val="50000"/><a:satMod val="300000"/></a:schemeClr></a:gs>';
  17921. o[o.length] = '<a:gs pos="35000"><a:schemeClr val="phClr"><a:tint val="37000"/><a:satMod val="300000"/></a:schemeClr></a:gs>';
  17922. o[o.length] = '<a:gs pos="100000"><a:schemeClr val="phClr"><a:tint val="15000"/><a:satMod val="350000"/></a:schemeClr></a:gs>';
  17923. o[o.length] = '</a:gsLst>';
  17924. o[o.length] = '<a:lin ang="16200000" scaled="1"/>';
  17925. o[o.length] = '</a:gradFill>';
  17926. o[o.length] = '<a:gradFill rotWithShape="1">';
  17927. o[o.length] = '<a:gsLst>';
  17928. o[o.length] = '<a:gs pos="0"><a:schemeClr val="phClr"><a:tint val="100000"/><a:shade val="100000"/><a:satMod val="130000"/></a:schemeClr></a:gs>';
  17929. o[o.length] = '<a:gs pos="100000"><a:schemeClr val="phClr"><a:tint val="50000"/><a:shade val="100000"/><a:satMod val="350000"/></a:schemeClr></a:gs>';
  17930. o[o.length] = '</a:gsLst>';
  17931. o[o.length] = '<a:lin ang="16200000" scaled="0"/>';
  17932. o[o.length] = '</a:gradFill>';
  17933. o[o.length] = '</a:fillStyleLst>';
  17934. o[o.length] = '<a:lnStyleLst>';
  17935. o[o.length] = '<a:ln w="9525" cap="flat" cmpd="sng" algn="ctr"><a:solidFill><a:schemeClr val="phClr"><a:shade val="95000"/><a:satMod val="105000"/></a:schemeClr></a:solidFill><a:prstDash val="solid"/></a:ln>';
  17936. o[o.length] = '<a:ln w="25400" cap="flat" cmpd="sng" algn="ctr"><a:solidFill><a:schemeClr val="phClr"/></a:solidFill><a:prstDash val="solid"/></a:ln>';
  17937. o[o.length] = '<a:ln w="38100" cap="flat" cmpd="sng" algn="ctr"><a:solidFill><a:schemeClr val="phClr"/></a:solidFill><a:prstDash val="solid"/></a:ln>';
  17938. o[o.length] = '</a:lnStyleLst>';
  17939. o[o.length] = '<a:effectStyleLst>';
  17940. o[o.length] = '<a:effectStyle>';
  17941. o[o.length] = '<a:effectLst>';
  17942. o[o.length] = '<a:outerShdw blurRad="40000" dist="20000" dir="5400000" rotWithShape="0"><a:srgbClr val="000000"><a:alpha val="38000"/></a:srgbClr></a:outerShdw>';
  17943. o[o.length] = '</a:effectLst>';
  17944. o[o.length] = '</a:effectStyle>';
  17945. o[o.length] = '<a:effectStyle>';
  17946. o[o.length] = '<a:effectLst>';
  17947. o[o.length] = '<a:outerShdw blurRad="40000" dist="23000" dir="5400000" rotWithShape="0"><a:srgbClr val="000000"><a:alpha val="35000"/></a:srgbClr></a:outerShdw>';
  17948. o[o.length] = '</a:effectLst>';
  17949. o[o.length] = '</a:effectStyle>';
  17950. o[o.length] = '<a:effectStyle>';
  17951. o[o.length] = '<a:effectLst>';
  17952. o[o.length] = '<a:outerShdw blurRad="40000" dist="23000" dir="5400000" rotWithShape="0"><a:srgbClr val="000000"><a:alpha val="35000"/></a:srgbClr></a:outerShdw>';
  17953. o[o.length] = '</a:effectLst>';
  17954. o[o.length] = '<a:scene3d><a:camera prst="orthographicFront"><a:rot lat="0" lon="0" rev="0"/></a:camera><a:lightRig rig="threePt" dir="t"><a:rot lat="0" lon="0" rev="1200000"/></a:lightRig></a:scene3d>';
  17955. o[o.length] = '<a:sp3d><a:bevelT w="63500" h="25400"/></a:sp3d>';
  17956. o[o.length] = '</a:effectStyle>';
  17957. o[o.length] = '</a:effectStyleLst>';
  17958. o[o.length] = '<a:bgFillStyleLst>';
  17959. o[o.length] = '<a:solidFill><a:schemeClr val="phClr"/></a:solidFill>';
  17960. o[o.length] = '<a:gradFill rotWithShape="1">';
  17961. o[o.length] = '<a:gsLst>';
  17962. o[o.length] = '<a:gs pos="0"><a:schemeClr val="phClr"><a:tint val="40000"/><a:satMod val="350000"/></a:schemeClr></a:gs>';
  17963. o[o.length] = '<a:gs pos="40000"><a:schemeClr val="phClr"><a:tint val="45000"/><a:shade val="99000"/><a:satMod val="350000"/></a:schemeClr></a:gs>';
  17964. o[o.length] = '<a:gs pos="100000"><a:schemeClr val="phClr"><a:shade val="20000"/><a:satMod val="255000"/></a:schemeClr></a:gs>';
  17965. o[o.length] = '</a:gsLst>';
  17966. o[o.length] = '<a:path path="circle"><a:fillToRect l="50000" t="-80000" r="50000" b="180000"/></a:path>';
  17967. o[o.length] = '</a:gradFill>';
  17968. o[o.length] = '<a:gradFill rotWithShape="1">';
  17969. o[o.length] = '<a:gsLst>';
  17970. o[o.length] = '<a:gs pos="0"><a:schemeClr val="phClr"><a:tint val="80000"/><a:satMod val="300000"/></a:schemeClr></a:gs>';
  17971. o[o.length] = '<a:gs pos="100000"><a:schemeClr val="phClr"><a:shade val="30000"/><a:satMod val="200000"/></a:schemeClr></a:gs>';
  17972. o[o.length] = '</a:gsLst>';
  17973. o[o.length] = '<a:path path="circle"><a:fillToRect l="50000" t="50000" r="50000" b="50000"/></a:path>';
  17974. o[o.length] = '</a:gradFill>';
  17975. o[o.length] = '</a:bgFillStyleLst>';
  17976. o[o.length] = '</a:fmtScheme>';
  17977. o[o.length] = '</a:themeElements>';
  17978. o[o.length] = '<a:objectDefaults>';
  17979. o[o.length] = '<a:spDef>';
  17980. o[o.length] = '<a:spPr/><a:bodyPr/><a:lstStyle/><a:style><a:lnRef idx="1"><a:schemeClr val="accent1"/></a:lnRef><a:fillRef idx="3"><a:schemeClr val="accent1"/></a:fillRef><a:effectRef idx="2"><a:schemeClr val="accent1"/></a:effectRef><a:fontRef idx="minor"><a:schemeClr val="lt1"/></a:fontRef></a:style>';
  17981. o[o.length] = '</a:spDef>';
  17982. o[o.length] = '<a:lnDef>';
  17983. o[o.length] = '<a:spPr/><a:bodyPr/><a:lstStyle/><a:style><a:lnRef idx="2"><a:schemeClr val="accent1"/></a:lnRef><a:fillRef idx="0"><a:schemeClr val="accent1"/></a:fillRef><a:effectRef idx="1"><a:schemeClr val="accent1"/></a:effectRef><a:fontRef idx="minor"><a:schemeClr val="tx1"/></a:fontRef></a:style>';
  17984. o[o.length] = '</a:lnDef>';
  17985. o[o.length] = '</a:objectDefaults>';
  17986. o[o.length] = '<a:extraClrSchemeLst/>';
  17987. o[o.length] = '</a:theme>';
  17988. return o.join("");
  17989. }
  17990. /* [MS-XLS] 2.4.326 TODO: payload is a zip file */
  17991. function parse_Theme(blob, length, opts) {
  17992. var end = blob.l + length;
  17993. var dwThemeVersion = blob.read_shift(4);
  17994. if(dwThemeVersion === 124226) return;
  17995. if(!opts.cellStyles || !jszip) { blob.l = end; return; }
  17996. var data = blob.slice(blob.l);
  17997. blob.l = end;
  17998. var zip; try { zip = new jszip(data); } catch(e) { return; }
  17999. var themeXML = getzipstr(zip, "theme/theme/theme1.xml", true);
  18000. if(!themeXML) return;
  18001. return parse_theme_xml(themeXML, opts);
  18002. }
  18003. /* 2.5.49 */
  18004. function parse_ColorTheme(blob) { return blob.read_shift(4); }
  18005. /* 2.5.155 */
  18006. function parse_FullColorExt(blob) {
  18007. var o = {};
  18008. o.xclrType = blob.read_shift(2);
  18009. o.nTintShade = blob.read_shift(2);
  18010. switch(o.xclrType) {
  18011. case 0: blob.l += 4; break;
  18012. case 1: o.xclrValue = parse_IcvXF(blob, 4); break;
  18013. case 2: o.xclrValue = parse_LongRGBA(blob, 4); break;
  18014. case 3: o.xclrValue = parse_ColorTheme(blob, 4); break;
  18015. case 4: blob.l += 4; break;
  18016. }
  18017. blob.l += 8;
  18018. return o;
  18019. }
  18020. /* 2.5.164 TODO: read 7 bits*/
  18021. function parse_IcvXF(blob, length) {
  18022. return parsenoop(blob, length);
  18023. }
  18024. /* 2.5.280 */
  18025. function parse_XFExtGradient(blob, length) {
  18026. return parsenoop(blob, length);
  18027. }
  18028. /* [MS-XLS] 2.5.108 */
  18029. function parse_ExtProp(blob) {
  18030. var extType = blob.read_shift(2);
  18031. var cb = blob.read_shift(2) - 4;
  18032. var o = [extType];
  18033. switch(extType) {
  18034. case 0x04: case 0x05: case 0x07: case 0x08:
  18035. case 0x09: case 0x0A: case 0x0B: case 0x0D:
  18036. o[1] = parse_FullColorExt(blob, cb); break;
  18037. case 0x06: o[1] = parse_XFExtGradient(blob, cb); break;
  18038. case 0x0E: case 0x0F: o[1] = blob.read_shift(cb === 1 ? 1 : 2); break;
  18039. default: throw new Error("Unrecognized ExtProp type: " + extType + " " + cb);
  18040. }
  18041. return o;
  18042. }
  18043. /* 2.4.355 */
  18044. function parse_XFExt(blob, length) {
  18045. var end = blob.l + length;
  18046. blob.l += 2;
  18047. var ixfe = blob.read_shift(2);
  18048. blob.l += 2;
  18049. var cexts = blob.read_shift(2);
  18050. var ext = [];
  18051. while(cexts-- > 0) ext.push(parse_ExtProp(blob, end-blob.l));
  18052. return {ixfe:ixfe, ext:ext};
  18053. }
  18054. /* xf is an XF, see parse_XFExt for xfext */
  18055. function update_xfext(xf, xfext) {
  18056. xfext.forEach(function(xfe) {
  18057. switch(xfe[0]) { /* 2.5.108 extPropData */
  18058. case 0x04: break; /* foreground color */
  18059. case 0x05: break; /* background color */
  18060. case 0x06: break; /* gradient fill */
  18061. case 0x07: break; /* top cell border color */
  18062. case 0x08: break; /* bottom cell border color */
  18063. case 0x09: break; /* left cell border color */
  18064. case 0x0a: break; /* right cell border color */
  18065. case 0x0b: break; /* diagonal cell border color */
  18066. case 0x0d: break; /* text color */
  18067. case 0x0e: break; /* font scheme */
  18068. case 0x0f: break; /* indentation level */
  18069. }
  18070. });
  18071. }
  18072. /* 18.6 Calculation Chain */
  18073. function parse_cc_xml(data) {
  18074. var d = [];
  18075. if(!data) return d;
  18076. var i = 1;
  18077. (data.match(tagregex)||[]).forEach(function(x) {
  18078. var y = parsexmltag(x);
  18079. switch(y[0]) {
  18080. case '<?xml': break;
  18081. /* 18.6.2 calcChain CT_CalcChain 1 */
  18082. case '<calcChain': case '<calcChain>': case '</calcChain>': break;
  18083. /* 18.6.1 c CT_CalcCell 1 */
  18084. case '<c': delete y[0]; if(y.i) i = y.i; else y.i = i; d.push(y); break;
  18085. }
  18086. });
  18087. return d;
  18088. }
  18089. //function write_cc_xml(data, opts) { }
  18090. /* [MS-XLSB] 2.6.4.1 */
  18091. function parse_BrtCalcChainItem$(data) {
  18092. var out = {};
  18093. out.i = data.read_shift(4);
  18094. var cell = {};
  18095. cell.r = data.read_shift(4);
  18096. cell.c = data.read_shift(4);
  18097. out.r = encode_cell(cell);
  18098. var flags = data.read_shift(1);
  18099. if(flags & 0x2) out.l = '1';
  18100. if(flags & 0x8) out.a = '1';
  18101. return out;
  18102. }
  18103. /* 18.6 Calculation Chain */
  18104. function parse_cc_bin(data, name, opts) {
  18105. var out = [];
  18106. var pass = false;
  18107. recordhopper(data, function hopper_cc(val, R_n, RT) {
  18108. switch(RT) {
  18109. case 0x003F: /* 'BrtCalcChainItem$' */
  18110. out.push(val); break;
  18111. default:
  18112. if((R_n||"").indexOf("Begin") > 0){/* empty */}
  18113. else if((R_n||"").indexOf("End") > 0){/* empty */}
  18114. else if(!pass || opts.WTF) throw new Error("Unexpected record " + RT + " " + R_n);
  18115. }
  18116. });
  18117. return out;
  18118. }
  18119. //function write_cc_bin(data, opts) { }
  18120. /* 18.14 Supplementary Workbook Data */
  18121. function parse_xlink_xml() {
  18122. //var opts = _opts || {};
  18123. //if(opts.WTF) throw "XLSX External Link";
  18124. }
  18125. /* [MS-XLSB] 2.1.7.25 External Link */
  18126. function parse_xlink_bin(data, name, _opts) {
  18127. if(!data) return data;
  18128. var opts = _opts || {};
  18129. var pass = false, end = false;
  18130. recordhopper(data, function xlink_parse(val, R_n, RT) {
  18131. if(end) return;
  18132. switch(RT) {
  18133. case 0x0167: /* 'BrtSupTabs' */
  18134. case 0x016B: /* 'BrtExternTableStart' */
  18135. case 0x016C: /* 'BrtExternTableEnd' */
  18136. case 0x016E: /* 'BrtExternRowHdr' */
  18137. case 0x016F: /* 'BrtExternCellBlank' */
  18138. case 0x0170: /* 'BrtExternCellReal' */
  18139. case 0x0171: /* 'BrtExternCellBool' */
  18140. case 0x0172: /* 'BrtExternCellError' */
  18141. case 0x0173: /* 'BrtExternCellString' */
  18142. case 0x01D8: /* 'BrtExternValueMeta' */
  18143. case 0x0241: /* 'BrtSupNameStart' */
  18144. case 0x0242: /* 'BrtSupNameValueStart' */
  18145. case 0x0243: /* 'BrtSupNameValueEnd' */
  18146. case 0x0244: /* 'BrtSupNameNum' */
  18147. case 0x0245: /* 'BrtSupNameErr' */
  18148. case 0x0246: /* 'BrtSupNameSt' */
  18149. case 0x0247: /* 'BrtSupNameNil' */
  18150. case 0x0248: /* 'BrtSupNameBool' */
  18151. case 0x0249: /* 'BrtSupNameFmla' */
  18152. case 0x024A: /* 'BrtSupNameBits' */
  18153. case 0x024B: /* 'BrtSupNameEnd' */
  18154. break;
  18155. case 0x0023: /* 'BrtFRTBegin' */
  18156. pass = true; break;
  18157. case 0x0024: /* 'BrtFRTEnd' */
  18158. pass = false; break;
  18159. default:
  18160. if((R_n||"").indexOf("Begin") > 0){/* empty */}
  18161. else if((R_n||"").indexOf("End") > 0){/* empty */}
  18162. else if(!pass || opts.WTF) throw new Error("Unexpected record " + RT.toString(16) + " " + R_n);
  18163. }
  18164. }, opts);
  18165. }
  18166. RELS.IMG = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/image";
  18167. RELS.DRAW = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/drawing";
  18168. /* 20.5 DrawingML - SpreadsheetML Drawing */
  18169. function parse_drawing(data, rels) {
  18170. if(!data) return "??";
  18171. /*
  18172. Chartsheet Drawing:
  18173. - 20.5.2.35 wsDr CT_Drawing
  18174. - 20.5.2.1 absoluteAnchor CT_AbsoluteAnchor
  18175. - 20.5.2.16 graphicFrame CT_GraphicalObjectFrame
  18176. - 20.1.2.2.16 graphic CT_GraphicalObject
  18177. - 20.1.2.2.17 graphicData CT_GraphicalObjectData
  18178. - chart reference
  18179. the actual type is based on the URI of the graphicData
  18180. TODO: handle embedded charts and other types of graphics
  18181. */
  18182. var id = (data.match(/<c:chart [^>]*r:id="([^"]*)"/)||["",""])[1];
  18183. return rels['!id'][id].Target;
  18184. }
  18185. /* L.5.5.2 SpreadsheetML Comments + VML Schema */
  18186. var _shapeid = 1024;
  18187. function write_comments_vml(rId, comments) {
  18188. var csize = [21600, 21600];
  18189. /* L.5.2.1.2 Path Attribute */
  18190. var bbox = ["m0,0l0",csize[1],csize[0],csize[1],csize[0],"0xe"].join(",");
  18191. var o = [
  18192. writextag("xml", null, { 'xmlns:v': XLMLNS.v, 'xmlns:o': XLMLNS.o, 'xmlns:x': XLMLNS.x, 'xmlns:mv': XLMLNS.mv }).replace(/\/>/,">"),
  18193. writextag("o:shapelayout", writextag("o:idmap", null, {'v:ext':"edit", 'data':rId}), {'v:ext':"edit"}),
  18194. writextag("v:shapetype", [
  18195. writextag("v:stroke", null, {joinstyle:"miter"}),
  18196. writextag("v:path", null, {gradientshapeok:"t", 'o:connecttype':"rect"})
  18197. ].join(""), {id:"_x0000_t202", 'o:spt':202, coordsize:csize.join(","),path:bbox})
  18198. ];
  18199. while(_shapeid < rId * 1000) _shapeid += 1000;
  18200. comments.forEach(function(x) { var c = decode_cell(x[0]);
  18201. o = o.concat([
  18202. '<v:shape' + wxt_helper({
  18203. id:'_x0000_s' + (++_shapeid),
  18204. type:"#_x0000_t202",
  18205. style:"position:absolute; margin-left:80pt;margin-top:5pt;width:104pt;height:64pt;z-index:10" + (x[1].hidden ? ";visibility:hidden" : "") ,
  18206. fillcolor:"#ECFAD4",
  18207. strokecolor:"#edeaa1"
  18208. }) + '>',
  18209. writextag('v:fill', writextag("o:fill", null, {type:"gradientUnscaled", 'v:ext':"view"}), {'color2':"#BEFF82", 'angle':"-180", 'type':"gradient"}),
  18210. writextag("v:shadow", null, {on:"t", 'obscured':"t"}),
  18211. writextag("v:path", null, {'o:connecttype':"none"}),
  18212. '<v:textbox><div style="text-align:left"></div></v:textbox>',
  18213. '<x:ClientData ObjectType="Note">',
  18214. '<x:MoveWithCells/>',
  18215. '<x:SizeWithCells/>',
  18216. /* Part 4 19.4.2.3 Anchor (Anchor) */
  18217. writetag('x:Anchor', [c.c, 0, c.r, 0, c.c+3, 100, c.r+5, 100].join(",")),
  18218. writetag('x:AutoFill', "False"),
  18219. writetag('x:Row', String(c.r)),
  18220. writetag('x:Column', String(c.c)),
  18221. x[1].hidden ? '' : '<x:Visible/>',
  18222. '</x:ClientData>',
  18223. '</v:shape>'
  18224. ]); });
  18225. o.push('</xml>');
  18226. return o.join("");
  18227. }
  18228. RELS.CMNT = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/comments";
  18229. function parse_comments(zip, dirComments, sheets, sheetRels, opts) {
  18230. for(var i = 0; i != dirComments.length; ++i) {
  18231. var canonicalpath=dirComments[i];
  18232. var comments=parse_cmnt(getzipdata(zip, canonicalpath.replace(/^\//,''), true), canonicalpath, opts);
  18233. if(!comments || !comments.length) continue;
  18234. // find the sheets targeted by these comments
  18235. var sheetNames = keys(sheets);
  18236. for(var j = 0; j != sheetNames.length; ++j) {
  18237. var sheetName = sheetNames[j];
  18238. var rels = sheetRels[sheetName];
  18239. if(rels) {
  18240. var rel = rels[canonicalpath];
  18241. if(rel) insertCommentsIntoSheet(sheetName, sheets[sheetName], comments);
  18242. }
  18243. }
  18244. }
  18245. }
  18246. function insertCommentsIntoSheet(sheetName, sheet, comments) {
  18247. var dense = Array.isArray(sheet);
  18248. var cell;
  18249. comments.forEach(function(comment) {
  18250. var r = decode_cell(comment.ref);
  18251. if(dense) {
  18252. if(!sheet[r.r]) sheet[r.r] = [];
  18253. cell = sheet[r.r][r.c];
  18254. } else cell = sheet[comment.ref];
  18255. if (!cell) {
  18256. cell = {};
  18257. if(dense) sheet[r.r][r.c] = cell;
  18258. else sheet[comment.ref] = cell;
  18259. var range = safe_decode_range(sheet["!ref"]||"BDWGO1000001:A1");
  18260. if(range.s.r > r.r) range.s.r = r.r;
  18261. if(range.e.r < r.r) range.e.r = r.r;
  18262. if(range.s.c > r.c) range.s.c = r.c;
  18263. if(range.e.c < r.c) range.e.c = r.c;
  18264. var encoded = encode_range(range);
  18265. if (encoded !== sheet["!ref"]) sheet["!ref"] = encoded;
  18266. }
  18267. if (!cell.c) cell.c = [];
  18268. var o = ({a: comment.author, t: comment.t, r: comment.r});
  18269. if(comment.h) o.h = comment.h;
  18270. cell.c.push(o);
  18271. });
  18272. }
  18273. /* 18.7 Comments */
  18274. function parse_comments_xml(data, opts) {
  18275. /* 18.7.6 CT_Comments */
  18276. if(data.match(/<(?:\w+:)?comments *\/>/)) return [];
  18277. var authors = [];
  18278. var commentList = [];
  18279. var authtag = data.match(/<(?:\w+:)?authors>([\s\S]*)<\/(?:\w+:)?authors>/);
  18280. if(authtag && authtag[1]) authtag[1].split(/<\/\w*:?author>/).forEach(function(x) {
  18281. if(x === "" || x.trim() === "") return;
  18282. var a = x.match(/<(?:\w+:)?author[^>]*>(.*)/);
  18283. if(a) authors.push(a[1]);
  18284. });
  18285. var cmnttag = data.match(/<(?:\w+:)?commentList>([\s\S]*)<\/(?:\w+:)?commentList>/);
  18286. if(cmnttag && cmnttag[1]) cmnttag[1].split(/<\/\w*:?comment>/).forEach(function(x) {
  18287. if(x === "" || x.trim() === "") return;
  18288. var cm = x.match(/<(?:\w+:)?comment[^>]*>/);
  18289. if(!cm) return;
  18290. var y = parsexmltag(cm[0]);
  18291. var comment = ({ author: y.authorId && authors[y.authorId] || "sheetjsghost", ref: y.ref, guid: y.guid });
  18292. var cell = decode_cell(y.ref);
  18293. if(opts.sheetRows && opts.sheetRows <= cell.r) return;
  18294. var textMatch = x.match(/<(?:\w+:)?text>([\s\S]*)<\/(?:\w+:)?text>/);
  18295. var rt = !!textMatch && !!textMatch[1] && parse_si(textMatch[1]) || {r:"",t:"",h:""};
  18296. comment.r = rt.r;
  18297. if(rt.r == "<t></t>") rt.t = rt.h = "";
  18298. comment.t = rt.t.replace(/\r\n/g,"\n").replace(/\r/g,"\n");
  18299. if(opts.cellHTML) comment.h = rt.h;
  18300. commentList.push(comment);
  18301. });
  18302. return commentList;
  18303. }
  18304. var CMNT_XML_ROOT = writextag('comments', null, { 'xmlns': XMLNS.main[0] });
  18305. function write_comments_xml(data) {
  18306. var o = [XML_HEADER, CMNT_XML_ROOT];
  18307. var iauthor = [];
  18308. o.push("<authors>");
  18309. data.forEach(function(x) { x[1].forEach(function(w) { var a = escapexml(w.a);
  18310. if(iauthor.indexOf(a) > -1) return;
  18311. iauthor.push(a);
  18312. o.push("<author>" + a + "</author>");
  18313. }); });
  18314. o.push("</authors>");
  18315. o.push("<commentList>");
  18316. data.forEach(function(d) {
  18317. d[1].forEach(function(c) {
  18318. /* 18.7.3 CT_Comment */
  18319. o.push('<comment ref="' + d[0] + '" authorId="' + iauthor.indexOf(escapexml(c.a)) + '"><text>');
  18320. o.push(writetag("t", c.t == null ? "" : escapexml(c.t)));
  18321. o.push('</text></comment>');
  18322. });
  18323. });
  18324. o.push("</commentList>");
  18325. if(o.length>2) { o[o.length] = ('</comments>'); o[1]=o[1].replace("/>",">"); }
  18326. return o.join("");
  18327. }
  18328. /* [MS-XLSB] 2.4.28 BrtBeginComment */
  18329. function parse_BrtBeginComment(data) {
  18330. var out = {};
  18331. out.iauthor = data.read_shift(4);
  18332. var rfx = parse_UncheckedRfX(data, 16);
  18333. out.rfx = rfx.s;
  18334. out.ref = encode_cell(rfx.s);
  18335. data.l += 16; /*var guid = parse_GUID(data); */
  18336. return out;
  18337. }
  18338. function write_BrtBeginComment(data, o) {
  18339. if(o == null) o = new_buf(36);
  18340. o.write_shift(4, data[1].iauthor);
  18341. write_UncheckedRfX((data[0]), o);
  18342. o.write_shift(4, 0);
  18343. o.write_shift(4, 0);
  18344. o.write_shift(4, 0);
  18345. o.write_shift(4, 0);
  18346. return o;
  18347. }
  18348. /* [MS-XLSB] 2.4.327 BrtCommentAuthor */
  18349. var parse_BrtCommentAuthor = parse_XLWideString;
  18350. function write_BrtCommentAuthor(data) { return write_XLWideString(data.slice(0, 54)); }
  18351. /* [MS-XLSB] 2.1.7.8 Comments */
  18352. function parse_comments_bin(data, opts) {
  18353. var out = [];
  18354. var authors = [];
  18355. var c = {};
  18356. var pass = false;
  18357. recordhopper(data, function hopper_cmnt(val, R_n, RT) {
  18358. switch(RT) {
  18359. case 0x0278: /* 'BrtCommentAuthor' */
  18360. authors.push(val); break;
  18361. case 0x027B: /* 'BrtBeginComment' */
  18362. c = val; break;
  18363. case 0x027D: /* 'BrtCommentText' */
  18364. c.t = val.t; c.h = val.h; c.r = val.r; break;
  18365. case 0x027C: /* 'BrtEndComment' */
  18366. c.author = authors[c.iauthor];
  18367. delete c.iauthor;
  18368. if(opts.sheetRows && opts.sheetRows <= c.rfx.r) break;
  18369. if(!c.t) c.t = "";
  18370. delete c.rfx; out.push(c); break;
  18371. case 0x0C00: /* 'BrtUid' */
  18372. break;
  18373. case 0x0023: /* 'BrtFRTBegin' */
  18374. pass = true; break;
  18375. case 0x0024: /* 'BrtFRTEnd' */
  18376. pass = false; break;
  18377. case 0x0025: /* 'BrtACBegin' */ break;
  18378. case 0x0026: /* 'BrtACEnd' */ break;
  18379. default:
  18380. if((R_n||"").indexOf("Begin") > 0){/* empty */}
  18381. else if((R_n||"").indexOf("End") > 0){/* empty */}
  18382. else if(!pass || opts.WTF) throw new Error("Unexpected record " + RT + " " + R_n);
  18383. }
  18384. });
  18385. return out;
  18386. }
  18387. function write_comments_bin(data) {
  18388. var ba = buf_array();
  18389. var iauthor = [];
  18390. write_record(ba, "BrtBeginComments");
  18391. write_record(ba, "BrtBeginCommentAuthors");
  18392. data.forEach(function(comment) {
  18393. comment[1].forEach(function(c) {
  18394. if(iauthor.indexOf(c.a) > -1) return;
  18395. iauthor.push(c.a.slice(0,54));
  18396. write_record(ba, "BrtCommentAuthor", write_BrtCommentAuthor(c.a));
  18397. });
  18398. });
  18399. write_record(ba, "BrtEndCommentAuthors");
  18400. write_record(ba, "BrtBeginCommentList");
  18401. data.forEach(function(comment) {
  18402. comment[1].forEach(function(c) {
  18403. c.iauthor = iauthor.indexOf(c.a);
  18404. var range = {s:decode_cell(comment[0]),e:decode_cell(comment[0])};
  18405. write_record(ba, "BrtBeginComment", write_BrtBeginComment([range, c]));
  18406. if(c.t && c.t.length > 0) write_record(ba, "BrtCommentText", write_BrtCommentText(c));
  18407. write_record(ba, "BrtEndComment");
  18408. delete c.iauthor;
  18409. });
  18410. });
  18411. write_record(ba, "BrtEndCommentList");
  18412. write_record(ba, "BrtEndComments");
  18413. return ba.end();
  18414. }
  18415. var CT_VBA = "application/vnd.ms-office.vbaProject";
  18416. function make_vba_xls(cfb) {
  18417. var newcfb = CFB.utils.cfb_new({root:"R"});
  18418. cfb.FullPaths.forEach(function(p, i) {
  18419. if(p.slice(-1) === "/" || !p.match(/_VBA_PROJECT_CUR/)) return;
  18420. var newpath = p.replace(/^[^\/]*/,"R").replace(/\/_VBA_PROJECT_CUR\u0000*/, "");
  18421. CFB.utils.cfb_add(newcfb, newpath, cfb.FileIndex[i].content);
  18422. });
  18423. return CFB.write(newcfb);
  18424. }
  18425. function fill_vba_xls(cfb, vba) {
  18426. vba.FullPaths.forEach(function(p, i) {
  18427. if(i == 0) return;
  18428. var newpath = p.replace(/[^\/]*[\/]/, "/_VBA_PROJECT_CUR/");
  18429. if(newpath.slice(-1) !== "/") CFB.utils.cfb_add(cfb, newpath, vba.FileIndex[i].content);
  18430. });
  18431. }
  18432. var VBAFMTS = [ "xlsb", "xlsm", "xlam", "biff8", "xla" ];
  18433. RELS.DS = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/dialogsheet";
  18434. RELS.MS = "http://schemas.microsoft.com/office/2006/relationships/xlMacrosheet";
  18435. /* macro and dialog sheet stubs */
  18436. function parse_ds_bin() { return {'!type':'dialog'}; }
  18437. function parse_ds_xml() { return {'!type':'dialog'}; }
  18438. function parse_ms_bin() { return {'!type':'macro'}; }
  18439. function parse_ms_xml() { return {'!type':'macro'}; }
  18440. /* TODO: it will be useful to parse the function str */
  18441. var rc_to_a1 = (function(){
  18442. var rcregex = /(^|[^A-Za-z])R(\[?)(-?\d+|)\]?C(\[?)(-?\d+|)\]?/g;
  18443. var rcbase = ({r:0,c:0});
  18444. function rcfunc($$,$1,$2,$3,$4,$5) {
  18445. var R = $3.length>0?parseInt($3,10)|0:0, C = $5.length>0?parseInt($5,10)|0:0;
  18446. if(C<0 && $4.length === 0) C=0;
  18447. var cRel = false, rRel = false;
  18448. if($4.length > 0 || $5.length == 0) cRel = true; if(cRel) C += rcbase.c; else --C;
  18449. if($2.length > 0 || $3.length == 0) rRel = true; if(rRel) R += rcbase.r; else --R;
  18450. return $1 + (cRel ? "" : "$") + encode_col(C) + (rRel ? "" : "$") + encode_row(R);
  18451. }
  18452. return function rc_to_a1(fstr, base) {
  18453. rcbase = base;
  18454. return fstr.replace(rcregex, rcfunc);
  18455. };
  18456. })();
  18457. var crefregex = /(^|[^._A-Z0-9])([$]?)([A-Z]{1,2}|[A-W][A-Z]{2}|X[A-E][A-Z]|XF[A-D])([$]?)([1-9]\d{0,5}|10[0-3]\d{4}|104[0-7]\d{3}|1048[0-4]\d{2}|10485[0-6]\d|104857[0-6])(?![_.\(A-Za-z0-9])/g;
  18458. var a1_to_rc =(function(){
  18459. return function a1_to_rc(fstr, base) {
  18460. return fstr.replace(crefregex, function($0, $1, $2, $3, $4, $5) {
  18461. var c = decode_col($3) - ($2 ? 0 : base.c);
  18462. var r = decode_row($5) - ($4 ? 0 : base.r);
  18463. var R = (r == 0 ? "" : !$4 ? "[" + r + "]" : (r+1));
  18464. var C = (c == 0 ? "" : !$2 ? "[" + c + "]" : (c+1));
  18465. return $1 + "R" + R + "C" + C;
  18466. });
  18467. };
  18468. })();
  18469. /* no defined name can collide with a valid cell address A1:XFD1048576 ... except LOG10! */
  18470. function shift_formula_str(f, delta) {
  18471. return f.replace(crefregex, function($0, $1, $2, $3, $4, $5) {
  18472. return $1+($2=="$" ? $2+$3 : encode_col(decode_col($3)+delta.c))+($4=="$" ? $4+$5 : encode_row(decode_row($5) + delta.r));
  18473. });
  18474. }
  18475. function shift_formula_xlsx(f, range, cell) {
  18476. var r = decode_range(range), s = r.s, c = decode_cell(cell);
  18477. var delta = {r:c.r - s.r, c:c.c - s.c};
  18478. return shift_formula_str(f, delta);
  18479. }
  18480. /* TODO: parse formula */
  18481. function fuzzyfmla(f) {
  18482. if(f.length == 1) return false;
  18483. return true;
  18484. }
  18485. function _xlfn(f) {
  18486. return f.replace(/_xlfn\./g,"");
  18487. }
  18488. function parseread1(blob) { blob.l+=1; return; }
  18489. /* [MS-XLS] 2.5.51 */
  18490. function parse_ColRelU(blob, length) {
  18491. var c = blob.read_shift(length == 1 ? 1 : 2);
  18492. return [c & 0x3FFF, (c >> 14) & 1, (c >> 15) & 1];
  18493. }
  18494. /* [MS-XLS] 2.5.198.105 ; [MS-XLSB] 2.5.97.89 */
  18495. function parse_RgceArea(blob, length, opts) {
  18496. var w = 2;
  18497. if(opts) {
  18498. if(opts.biff >= 2 && opts.biff <= 5) return parse_RgceArea_BIFF2(blob, length, opts);
  18499. else if(opts.biff == 12) w = 4;
  18500. }
  18501. var r=blob.read_shift(w), R=blob.read_shift(w);
  18502. var c=parse_ColRelU(blob, 2);
  18503. var C=parse_ColRelU(blob, 2);
  18504. return { s:{r:r, c:c[0], cRel:c[1], rRel:c[2]}, e:{r:R, c:C[0], cRel:C[1], rRel:C[2]} };
  18505. }
  18506. /* BIFF 2-5 encodes flags in the row field */
  18507. function parse_RgceArea_BIFF2(blob) {
  18508. var r=parse_ColRelU(blob, 2), R=parse_ColRelU(blob, 2);
  18509. var c=blob.read_shift(1);
  18510. var C=blob.read_shift(1);
  18511. return { s:{r:r[0], c:c, cRel:r[1], rRel:r[2]}, e:{r:R[0], c:C, cRel:R[1], rRel:R[2]} };
  18512. }
  18513. /* [MS-XLS] 2.5.198.105 ; [MS-XLSB] 2.5.97.90 */
  18514. function parse_RgceAreaRel(blob, length, opts) {
  18515. if(opts.biff < 8) return parse_RgceArea_BIFF2(blob, length, opts);
  18516. var r=blob.read_shift(opts.biff == 12 ? 4 : 2), R=blob.read_shift(opts.biff == 12 ? 4 : 2);
  18517. var c=parse_ColRelU(blob, 2);
  18518. var C=parse_ColRelU(blob, 2);
  18519. return { s:{r:r, c:c[0], cRel:c[1], rRel:c[2]}, e:{r:R, c:C[0], cRel:C[1], rRel:C[2]} };
  18520. }
  18521. /* [MS-XLS] 2.5.198.109 ; [MS-XLSB] 2.5.97.91 */
  18522. function parse_RgceLoc(blob, length, opts) {
  18523. if(opts && opts.biff >= 2 && opts.biff <= 5) return parse_RgceLoc_BIFF2(blob, length, opts);
  18524. var r = blob.read_shift(opts && opts.biff == 12 ? 4 : 2);
  18525. var c = parse_ColRelU(blob, 2);
  18526. return {r:r, c:c[0], cRel:c[1], rRel:c[2]};
  18527. }
  18528. function parse_RgceLoc_BIFF2(blob) {
  18529. var r = parse_ColRelU(blob, 2);
  18530. var c = blob.read_shift(1);
  18531. return {r:r[0], c:c, cRel:r[1], rRel:r[2]};
  18532. }
  18533. /* [MS-XLS] 2.5.198.107, 2.5.47 */
  18534. function parse_RgceElfLoc(blob) {
  18535. var r = blob.read_shift(2);
  18536. var c = blob.read_shift(2);
  18537. return {r:r, c:c & 0xFF, fQuoted:!!(c & 0x4000), cRel:c>>15, rRel:c>>15 };
  18538. }
  18539. /* [MS-XLS] 2.5.198.111 ; [MS-XLSB] 2.5.97.92 TODO */
  18540. function parse_RgceLocRel(blob, length, opts) {
  18541. var biff = opts && opts.biff ? opts.biff : 8;
  18542. if(biff >= 2 && biff <= 5) return parse_RgceLocRel_BIFF2(blob, length, opts);
  18543. var r = blob.read_shift(biff >= 12 ? 4 : 2);
  18544. var cl = blob.read_shift(2);
  18545. var cRel = (cl & 0x4000) >> 14, rRel = (cl & 0x8000) >> 15;
  18546. cl &= 0x3FFF;
  18547. if(rRel == 1) while(r > 0x7FFFF) r -= 0x100000;
  18548. if(cRel == 1) while(cl > 0x1FFF) cl = cl - 0x4000;
  18549. return {r:r,c:cl,cRel:cRel,rRel:rRel};
  18550. }
  18551. function parse_RgceLocRel_BIFF2(blob) {
  18552. var rl = blob.read_shift(2);
  18553. var c = blob.read_shift(1);
  18554. var rRel = (rl & 0x8000) >> 15, cRel = (rl & 0x4000) >> 14;
  18555. rl &= 0x3FFF;
  18556. if(rRel == 1 && rl >= 0x2000) rl = rl - 0x4000;
  18557. if(cRel == 1 && c >= 0x80) c = c - 0x100;
  18558. return {r:rl,c:c,cRel:cRel,rRel:rRel};
  18559. }
  18560. /* [MS-XLS] 2.5.198.27 ; [MS-XLSB] 2.5.97.18 */
  18561. function parse_PtgArea(blob, length, opts) {
  18562. var type = (blob[blob.l++] & 0x60) >> 5;
  18563. var area = parse_RgceArea(blob, opts.biff >= 2 && opts.biff <= 5 ? 6 : 8, opts);
  18564. return [type, area];
  18565. }
  18566. /* [MS-XLS] 2.5.198.28 ; [MS-XLSB] 2.5.97.19 */
  18567. function parse_PtgArea3d(blob, length, opts) {
  18568. var type = (blob[blob.l++] & 0x60) >> 5;
  18569. var ixti = blob.read_shift(2, 'i');
  18570. var w = 8;
  18571. if(opts) switch(opts.biff) {
  18572. case 5: blob.l += 12; w = 6; break;
  18573. case 12: w = 12; break;
  18574. }
  18575. var area = parse_RgceArea(blob, w, opts);
  18576. return [type, ixti, area];
  18577. }
  18578. /* [MS-XLS] 2.5.198.29 ; [MS-XLSB] 2.5.97.20 */
  18579. function parse_PtgAreaErr(blob, length, opts) {
  18580. var type = (blob[blob.l++] & 0x60) >> 5;
  18581. blob.l += opts && (opts.biff > 8) ? 12 : (opts.biff < 8 ? 6 : 8);
  18582. return [type];
  18583. }
  18584. /* [MS-XLS] 2.5.198.30 ; [MS-XLSB] 2.5.97.21 */
  18585. function parse_PtgAreaErr3d(blob, length, opts) {
  18586. var type = (blob[blob.l++] & 0x60) >> 5;
  18587. var ixti = blob.read_shift(2);
  18588. var w = 8;
  18589. if(opts) switch(opts.biff) {
  18590. case 5: blob.l += 12; w = 6; break;
  18591. case 12: w = 12; break;
  18592. }
  18593. blob.l += w;
  18594. return [type, ixti];
  18595. }
  18596. /* [MS-XLS] 2.5.198.31 ; [MS-XLSB] 2.5.97.22 */
  18597. function parse_PtgAreaN(blob, length, opts) {
  18598. var type = (blob[blob.l++] & 0x60) >> 5;
  18599. var area = parse_RgceAreaRel(blob, length - 1, opts);
  18600. return [type, area];
  18601. }
  18602. /* [MS-XLS] 2.5.198.32 ; [MS-XLSB] 2.5.97.23 */
  18603. function parse_PtgArray(blob, length, opts) {
  18604. var type = (blob[blob.l++] & 0x60) >> 5;
  18605. blob.l += opts.biff == 2 ? 6 : opts.biff == 12 ? 14 : 7;
  18606. return [type];
  18607. }
  18608. /* [MS-XLS] 2.5.198.33 ; [MS-XLSB] 2.5.97.24 */
  18609. function parse_PtgAttrBaxcel(blob) {
  18610. var bitSemi = blob[blob.l+1] & 0x01; /* 1 = volatile */
  18611. var bitBaxcel = 1;
  18612. blob.l += 4;
  18613. return [bitSemi, bitBaxcel];
  18614. }
  18615. /* [MS-XLS] 2.5.198.34 ; [MS-XLSB] 2.5.97.25 */
  18616. function parse_PtgAttrChoose(blob, length, opts) {
  18617. blob.l +=2;
  18618. var offset = blob.read_shift(opts && opts.biff == 2 ? 1 : 2);
  18619. var o = [];
  18620. /* offset is 1 less than the number of elements */
  18621. for(var i = 0; i <= offset; ++i) o.push(blob.read_shift(opts && opts.biff == 2 ? 1 : 2));
  18622. return o;
  18623. }
  18624. /* [MS-XLS] 2.5.198.35 ; [MS-XLSB] 2.5.97.26 */
  18625. function parse_PtgAttrGoto(blob, length, opts) {
  18626. var bitGoto = (blob[blob.l+1] & 0xFF) ? 1 : 0;
  18627. blob.l += 2;
  18628. return [bitGoto, blob.read_shift(opts && opts.biff == 2 ? 1 : 2)];
  18629. }
  18630. /* [MS-XLS] 2.5.198.36 ; [MS-XLSB] 2.5.97.27 */
  18631. function parse_PtgAttrIf(blob, length, opts) {
  18632. var bitIf = (blob[blob.l+1] & 0xFF) ? 1 : 0;
  18633. blob.l += 2;
  18634. return [bitIf, blob.read_shift(opts && opts.biff == 2 ? 1 : 2)];
  18635. }
  18636. /* [MS-XLSB] 2.5.97.28 */
  18637. function parse_PtgAttrIfError(blob) {
  18638. var bitIf = (blob[blob.l+1] & 0xFF) ? 1 : 0;
  18639. blob.l += 2;
  18640. return [bitIf, blob.read_shift(2)];
  18641. }
  18642. /* [MS-XLS] 2.5.198.37 ; [MS-XLSB] 2.5.97.29 */
  18643. function parse_PtgAttrSemi(blob, length, opts) {
  18644. var bitSemi = (blob[blob.l+1] & 0xFF) ? 1 : 0;
  18645. blob.l += opts && opts.biff == 2 ? 3 : 4;
  18646. return [bitSemi];
  18647. }
  18648. /* [MS-XLS] 2.5.198.40 ; [MS-XLSB] 2.5.97.32 */
  18649. function parse_PtgAttrSpaceType(blob) {
  18650. var type = blob.read_shift(1), cch = blob.read_shift(1);
  18651. return [type, cch];
  18652. }
  18653. /* [MS-XLS] 2.5.198.38 ; [MS-XLSB] 2.5.97.30 */
  18654. function parse_PtgAttrSpace(blob) {
  18655. blob.read_shift(2);
  18656. return parse_PtgAttrSpaceType(blob, 2);
  18657. }
  18658. /* [MS-XLS] 2.5.198.39 ; [MS-XLSB] 2.5.97.31 */
  18659. function parse_PtgAttrSpaceSemi(blob) {
  18660. blob.read_shift(2);
  18661. return parse_PtgAttrSpaceType(blob, 2);
  18662. }
  18663. /* [MS-XLS] 2.5.198.84 ; [MS-XLSB] 2.5.97.68 TODO */
  18664. function parse_PtgRef(blob, length, opts) {
  18665. //var ptg = blob[blob.l] & 0x1F;
  18666. var type = (blob[blob.l] & 0x60)>>5;
  18667. blob.l += 1;
  18668. var loc = parse_RgceLoc(blob, 0, opts);
  18669. return [type, loc];
  18670. }
  18671. /* [MS-XLS] 2.5.198.88 ; [MS-XLSB] 2.5.97.72 TODO */
  18672. function parse_PtgRefN(blob, length, opts) {
  18673. var type = (blob[blob.l] & 0x60)>>5;
  18674. blob.l += 1;
  18675. var loc = parse_RgceLocRel(blob, 0, opts);
  18676. return [type, loc];
  18677. }
  18678. /* [MS-XLS] 2.5.198.85 ; [MS-XLSB] 2.5.97.69 TODO */
  18679. function parse_PtgRef3d(blob, length, opts) {
  18680. var type = (blob[blob.l] & 0x60)>>5;
  18681. blob.l += 1;
  18682. var ixti = blob.read_shift(2); // XtiIndex
  18683. if(opts && opts.biff == 5) blob.l += 12;
  18684. var loc = parse_RgceLoc(blob, 0, opts); // TODO: or RgceLocRel
  18685. return [type, ixti, loc];
  18686. }
  18687. /* [MS-XLS] 2.5.198.62 ; [MS-XLSB] 2.5.97.45 TODO */
  18688. function parse_PtgFunc(blob, length, opts) {
  18689. //var ptg = blob[blob.l] & 0x1F;
  18690. var type = (blob[blob.l] & 0x60)>>5;
  18691. blob.l += 1;
  18692. var iftab = blob.read_shift(opts && opts.biff <= 3 ? 1 : 2);
  18693. return [FtabArgc[iftab], Ftab[iftab], type];
  18694. }
  18695. /* [MS-XLS] 2.5.198.63 ; [MS-XLSB] 2.5.97.46 TODO */
  18696. function parse_PtgFuncVar(blob, length, opts) {
  18697. var type = blob[blob.l++];
  18698. var cparams = blob.read_shift(1), tab = opts && opts.biff <= 3 ? [(type == 0x58 ? -1 : 0), blob.read_shift(1)]: parsetab(blob);
  18699. return [cparams, (tab[0] === 0 ? Ftab : Cetab)[tab[1]]];
  18700. }
  18701. function parsetab(blob) {
  18702. return [blob[blob.l+1]>>7, blob.read_shift(2) & 0x7FFF];
  18703. }
  18704. /* [MS-XLS] 2.5.198.41 ; [MS-XLSB] 2.5.97.33 */
  18705. function parse_PtgAttrSum(blob, length, opts) {
  18706. blob.l += opts && opts.biff == 2 ? 3 : 4; return;
  18707. }
  18708. /* [MS-XLS] 2.5.198.58 ; [MS-XLSB] 2.5.97.40 */
  18709. function parse_PtgExp(blob, length, opts) {
  18710. blob.l++;
  18711. if(opts && opts.biff == 12) return [blob.read_shift(4, 'i'), 0];
  18712. var row = blob.read_shift(2);
  18713. var col = blob.read_shift(opts && opts.biff == 2 ? 1 : 2);
  18714. return [row, col];
  18715. }
  18716. /* [MS-XLS] 2.5.198.57 ; [MS-XLSB] 2.5.97.39 */
  18717. function parse_PtgErr(blob) { blob.l++; return BErr[blob.read_shift(1)]; }
  18718. /* [MS-XLS] 2.5.198.66 ; [MS-XLSB] 2.5.97.49 */
  18719. function parse_PtgInt(blob) { blob.l++; return blob.read_shift(2); }
  18720. /* [MS-XLS] 2.5.198.42 ; [MS-XLSB] 2.5.97.34 */
  18721. function parse_PtgBool(blob) { blob.l++; return blob.read_shift(1)!==0;}
  18722. /* [MS-XLS] 2.5.198.79 ; [MS-XLSB] 2.5.97.63 */
  18723. function parse_PtgNum(blob) { blob.l++; return parse_Xnum(blob, 8); }
  18724. /* [MS-XLS] 2.5.198.89 ; [MS-XLSB] 2.5.97.74 */
  18725. function parse_PtgStr(blob, length, opts) { blob.l++; return parse_ShortXLUnicodeString(blob, length-1, opts); }
  18726. /* [MS-XLS] 2.5.192.112 + 2.5.192.11{3,4,5,6,7} */
  18727. /* [MS-XLSB] 2.5.97.93 + 2.5.97.9{4,5,6,7} */
  18728. function parse_SerAr(blob, biff) {
  18729. var val = [blob.read_shift(1)];
  18730. if(biff == 12) switch(val[0]) {
  18731. case 0x02: val[0] = 0x04; break; /* SerBool */
  18732. case 0x04: val[0] = 0x10; break; /* SerErr */
  18733. case 0x00: val[0] = 0x01; break; /* SerNum */
  18734. case 0x01: val[0] = 0x02; break; /* SerStr */
  18735. }
  18736. switch(val[0]) {
  18737. case 0x04: /* SerBool -- boolean */
  18738. val[1] = parsebool(blob, 1) ? 'TRUE' : 'FALSE';
  18739. if(biff != 12) blob.l += 7; break;
  18740. case 0x25: /* appears to be an alias */
  18741. case 0x10: /* SerErr -- error */
  18742. val[1] = BErr[blob[blob.l]];
  18743. blob.l += ((biff == 12) ? 4 : 8); break;
  18744. case 0x00: /* SerNil -- honestly, I'm not sure how to reproduce this */
  18745. blob.l += 8; break;
  18746. case 0x01: /* SerNum -- Xnum */
  18747. val[1] = parse_Xnum(blob, 8); break;
  18748. case 0x02: /* SerStr -- XLUnicodeString (<256 chars) */
  18749. val[1] = parse_XLUnicodeString2(blob, 0, {biff:biff > 0 && biff < 8 ? 2 : biff}); break;
  18750. default: throw new Error("Bad SerAr: " + val[0]); /* Unreachable */
  18751. }
  18752. return val;
  18753. }
  18754. /* [MS-XLS] 2.5.198.61 ; [MS-XLSB] 2.5.97.44 */
  18755. function parse_PtgExtraMem(blob, cce, opts) {
  18756. var count = blob.read_shift((opts.biff == 12) ? 4 : 2);
  18757. var out = [];
  18758. for(var i = 0; i != count; ++i) out.push(((opts.biff == 12) ? parse_UncheckedRfX : parse_Ref8U)(blob, 8));
  18759. return out;
  18760. }
  18761. /* [MS-XLS] 2.5.198.59 ; [MS-XLSB] 2.5.97.41 */
  18762. function parse_PtgExtraArray(blob, length, opts) {
  18763. var rows = 0, cols = 0;
  18764. if(opts.biff == 12) {
  18765. rows = blob.read_shift(4); // DRw
  18766. cols = blob.read_shift(4); // DCol
  18767. } else {
  18768. cols = 1 + blob.read_shift(1); //DColByteU
  18769. rows = 1 + blob.read_shift(2); //DRw
  18770. }
  18771. if(opts.biff >= 2 && opts.biff < 8) { --rows; if(--cols == 0) cols = 0x100; }
  18772. // $FlowIgnore
  18773. for(var i = 0, o = []; i != rows && (o[i] = []); ++i)
  18774. for(var j = 0; j != cols; ++j) o[i][j] = parse_SerAr(blob, opts.biff);
  18775. return o;
  18776. }
  18777. /* [MS-XLS] 2.5.198.76 ; [MS-XLSB] 2.5.97.60 */
  18778. function parse_PtgName(blob, length, opts) {
  18779. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18780. var w = (!opts || (opts.biff >= 8)) ? 4 : 2;
  18781. var nameindex = blob.read_shift(w);
  18782. switch(opts.biff) {
  18783. case 2: blob.l += 5; break;
  18784. case 3: case 4: blob.l += 8; break;
  18785. case 5: blob.l += 12; break;
  18786. }
  18787. return [type, 0, nameindex];
  18788. }
  18789. /* [MS-XLS] 2.5.198.77 ; [MS-XLSB] 2.5.97.61 */
  18790. function parse_PtgNameX(blob, length, opts) {
  18791. if(opts.biff == 5) return parse_PtgNameX_BIFF5(blob, length, opts);
  18792. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18793. var ixti = blob.read_shift(2); // XtiIndex
  18794. var nameindex = blob.read_shift(4);
  18795. return [type, ixti, nameindex];
  18796. }
  18797. function parse_PtgNameX_BIFF5(blob) {
  18798. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18799. var ixti = blob.read_shift(2, 'i'); // XtiIndex
  18800. blob.l += 8;
  18801. var nameindex = blob.read_shift(2);
  18802. blob.l += 12;
  18803. return [type, ixti, nameindex];
  18804. }
  18805. /* [MS-XLS] 2.5.198.70 ; [MS-XLSB] 2.5.97.54 */
  18806. function parse_PtgMemArea(blob, length, opts) {
  18807. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18808. blob.l += (opts && opts.biff == 2 ? 3 : 4);
  18809. var cce = blob.read_shift(opts && opts.biff == 2 ? 1 : 2);
  18810. return [type, cce];
  18811. }
  18812. /* [MS-XLS] 2.5.198.72 ; [MS-XLSB] 2.5.97.56 */
  18813. function parse_PtgMemFunc(blob, length, opts) {
  18814. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18815. var cce = blob.read_shift(opts && opts.biff == 2 ? 1 : 2);
  18816. return [type, cce];
  18817. }
  18818. /* [MS-XLS] 2.5.198.86 ; [MS-XLSB] 2.5.97.69 */
  18819. function parse_PtgRefErr(blob, length, opts) {
  18820. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18821. blob.l += 4;
  18822. if(opts.biff < 8) blob.l--;
  18823. if(opts.biff == 12) blob.l += 2;
  18824. return [type];
  18825. }
  18826. /* [MS-XLS] 2.5.198.87 ; [MS-XLSB] 2.5.97.71 */
  18827. function parse_PtgRefErr3d(blob, length, opts) {
  18828. var type = (blob[blob.l++] & 0x60) >> 5;
  18829. var ixti = blob.read_shift(2);
  18830. var w = 4;
  18831. if(opts) switch(opts.biff) {
  18832. case 5: w = 15; break;
  18833. case 12: w = 6; break;
  18834. }
  18835. blob.l += w;
  18836. return [type, ixti];
  18837. }
  18838. /* [MS-XLS] 2.5.198.71 ; [MS-XLSB] 2.5.97.55 */
  18839. var parse_PtgMemErr = parsenoop;
  18840. /* [MS-XLS] 2.5.198.73 ; [MS-XLSB] 2.5.97.57 */
  18841. var parse_PtgMemNoMem = parsenoop;
  18842. /* [MS-XLS] 2.5.198.92 */
  18843. var parse_PtgTbl = parsenoop;
  18844. function parse_PtgElfLoc(blob, length, opts) {
  18845. blob.l += 2;
  18846. return [parse_RgceElfLoc(blob, 4, opts)];
  18847. }
  18848. function parse_PtgElfNoop(blob) {
  18849. blob.l += 6;
  18850. return [];
  18851. }
  18852. /* [MS-XLS] 2.5.198.46 */
  18853. var parse_PtgElfCol = parse_PtgElfLoc;
  18854. /* [MS-XLS] 2.5.198.47 */
  18855. var parse_PtgElfColS = parse_PtgElfNoop;
  18856. /* [MS-XLS] 2.5.198.48 */
  18857. var parse_PtgElfColSV = parse_PtgElfNoop;
  18858. /* [MS-XLS] 2.5.198.49 */
  18859. var parse_PtgElfColV = parse_PtgElfLoc;
  18860. /* [MS-XLS] 2.5.198.50 */
  18861. function parse_PtgElfLel(blob) {
  18862. blob.l += 2;
  18863. return [parseuint16(blob), blob.read_shift(2) & 0x01];
  18864. }
  18865. /* [MS-XLS] 2.5.198.51 */
  18866. var parse_PtgElfRadical = parse_PtgElfLoc;
  18867. /* [MS-XLS] 2.5.198.52 */
  18868. var parse_PtgElfRadicalLel = parse_PtgElfLel;
  18869. /* [MS-XLS] 2.5.198.53 */
  18870. var parse_PtgElfRadicalS = parse_PtgElfNoop;
  18871. /* [MS-XLS] 2.5.198.54 */
  18872. var parse_PtgElfRw = parse_PtgElfLoc;
  18873. /* [MS-XLS] 2.5.198.55 */
  18874. var parse_PtgElfRwV = parse_PtgElfLoc;
  18875. /* [MS-XLSB] 2.5.97.52 TODO */
  18876. var PtgListRT = [
  18877. "Data",
  18878. "All",
  18879. "Headers",
  18880. "??",
  18881. "?Data2",
  18882. "??",
  18883. "?DataHeaders",
  18884. "??",
  18885. "Totals",
  18886. "??",
  18887. "??",
  18888. "??",
  18889. "?DataTotals",
  18890. "??",
  18891. "??",
  18892. "??",
  18893. "?Current"
  18894. ];
  18895. function parse_PtgList(blob) {
  18896. blob.l += 2;
  18897. var ixti = blob.read_shift(2);
  18898. var flags = blob.read_shift(2);
  18899. var idx = blob.read_shift(4);
  18900. var c = blob.read_shift(2);
  18901. var C = blob.read_shift(2);
  18902. var rt = PtgListRT[(flags >> 2) & 0x1F];
  18903. return {ixti: ixti, coltype:(flags&0x3), rt:rt, idx:idx, c:c, C:C};
  18904. }
  18905. /* [MS-XLS] 2.5.198.91 ; [MS-XLSB] 2.5.97.76 */
  18906. function parse_PtgSxName(blob) {
  18907. blob.l += 2;
  18908. return [blob.read_shift(4)];
  18909. }
  18910. /* [XLS] old spec */
  18911. function parse_PtgSheet(blob, length, opts) {
  18912. blob.l += 5;
  18913. blob.l += 2;
  18914. blob.l += (opts.biff == 2 ? 1 : 4);
  18915. return ["PTGSHEET"];
  18916. }
  18917. function parse_PtgEndSheet(blob, length, opts) {
  18918. blob.l += (opts.biff == 2 ? 4 : 5);
  18919. return ["PTGENDSHEET"];
  18920. }
  18921. function parse_PtgMemAreaN(blob) {
  18922. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18923. var cce = blob.read_shift(2);
  18924. return [type, cce];
  18925. }
  18926. function parse_PtgMemNoMemN(blob) {
  18927. var type = (blob.read_shift(1) >>> 5) & 0x03;
  18928. var cce = blob.read_shift(2);
  18929. return [type, cce];
  18930. }
  18931. function parse_PtgAttrNoop(blob) {
  18932. blob.l += 4;
  18933. return [0, 0];
  18934. }
  18935. /* [MS-XLS] 2.5.198.25 ; [MS-XLSB] 2.5.97.16 */
  18936. var PtgTypes = {
  18937. 0x01: { n:'PtgExp', f:parse_PtgExp },
  18938. 0x02: { n:'PtgTbl', f:parse_PtgTbl },
  18939. 0x03: { n:'PtgAdd', f:parseread1 },
  18940. 0x04: { n:'PtgSub', f:parseread1 },
  18941. 0x05: { n:'PtgMul', f:parseread1 },
  18942. 0x06: { n:'PtgDiv', f:parseread1 },
  18943. 0x07: { n:'PtgPower', f:parseread1 },
  18944. 0x08: { n:'PtgConcat', f:parseread1 },
  18945. 0x09: { n:'PtgLt', f:parseread1 },
  18946. 0x0A: { n:'PtgLe', f:parseread1 },
  18947. 0x0B: { n:'PtgEq', f:parseread1 },
  18948. 0x0C: { n:'PtgGe', f:parseread1 },
  18949. 0x0D: { n:'PtgGt', f:parseread1 },
  18950. 0x0E: { n:'PtgNe', f:parseread1 },
  18951. 0x0F: { n:'PtgIsect', f:parseread1 },
  18952. 0x10: { n:'PtgUnion', f:parseread1 },
  18953. 0x11: { n:'PtgRange', f:parseread1 },
  18954. 0x12: { n:'PtgUplus', f:parseread1 },
  18955. 0x13: { n:'PtgUminus', f:parseread1 },
  18956. 0x14: { n:'PtgPercent', f:parseread1 },
  18957. 0x15: { n:'PtgParen', f:parseread1 },
  18958. 0x16: { n:'PtgMissArg', f:parseread1 },
  18959. 0x17: { n:'PtgStr', f:parse_PtgStr },
  18960. 0x1A: { n:'PtgSheet', f:parse_PtgSheet },
  18961. 0x1B: { n:'PtgEndSheet', f:parse_PtgEndSheet },
  18962. 0x1C: { n:'PtgErr', f:parse_PtgErr },
  18963. 0x1D: { n:'PtgBool', f:parse_PtgBool },
  18964. 0x1E: { n:'PtgInt', f:parse_PtgInt },
  18965. 0x1F: { n:'PtgNum', f:parse_PtgNum },
  18966. 0x20: { n:'PtgArray', f:parse_PtgArray },
  18967. 0x21: { n:'PtgFunc', f:parse_PtgFunc },
  18968. 0x22: { n:'PtgFuncVar', f:parse_PtgFuncVar },
  18969. 0x23: { n:'PtgName', f:parse_PtgName },
  18970. 0x24: { n:'PtgRef', f:parse_PtgRef },
  18971. 0x25: { n:'PtgArea', f:parse_PtgArea },
  18972. 0x26: { n:'PtgMemArea', f:parse_PtgMemArea },
  18973. 0x27: { n:'PtgMemErr', f:parse_PtgMemErr },
  18974. 0x28: { n:'PtgMemNoMem', f:parse_PtgMemNoMem },
  18975. 0x29: { n:'PtgMemFunc', f:parse_PtgMemFunc },
  18976. 0x2A: { n:'PtgRefErr', f:parse_PtgRefErr },
  18977. 0x2B: { n:'PtgAreaErr', f:parse_PtgAreaErr },
  18978. 0x2C: { n:'PtgRefN', f:parse_PtgRefN },
  18979. 0x2D: { n:'PtgAreaN', f:parse_PtgAreaN },
  18980. 0x2E: { n:'PtgMemAreaN', f:parse_PtgMemAreaN },
  18981. 0x2F: { n:'PtgMemNoMemN', f:parse_PtgMemNoMemN },
  18982. 0x39: { n:'PtgNameX', f:parse_PtgNameX },
  18983. 0x3A: { n:'PtgRef3d', f:parse_PtgRef3d },
  18984. 0x3B: { n:'PtgArea3d', f:parse_PtgArea3d },
  18985. 0x3C: { n:'PtgRefErr3d', f:parse_PtgRefErr3d },
  18986. 0x3D: { n:'PtgAreaErr3d', f:parse_PtgAreaErr3d },
  18987. 0xFF: {}
  18988. };
  18989. /* These are duplicated in the PtgTypes table */
  18990. var PtgDupes = {
  18991. 0x40: 0x20, 0x60: 0x20,
  18992. 0x41: 0x21, 0x61: 0x21,
  18993. 0x42: 0x22, 0x62: 0x22,
  18994. 0x43: 0x23, 0x63: 0x23,
  18995. 0x44: 0x24, 0x64: 0x24,
  18996. 0x45: 0x25, 0x65: 0x25,
  18997. 0x46: 0x26, 0x66: 0x26,
  18998. 0x47: 0x27, 0x67: 0x27,
  18999. 0x48: 0x28, 0x68: 0x28,
  19000. 0x49: 0x29, 0x69: 0x29,
  19001. 0x4A: 0x2A, 0x6A: 0x2A,
  19002. 0x4B: 0x2B, 0x6B: 0x2B,
  19003. 0x4C: 0x2C, 0x6C: 0x2C,
  19004. 0x4D: 0x2D, 0x6D: 0x2D,
  19005. 0x4E: 0x2E, 0x6E: 0x2E,
  19006. 0x4F: 0x2F, 0x6F: 0x2F,
  19007. 0x58: 0x22, 0x78: 0x22,
  19008. 0x59: 0x39, 0x79: 0x39,
  19009. 0x5A: 0x3A, 0x7A: 0x3A,
  19010. 0x5B: 0x3B, 0x7B: 0x3B,
  19011. 0x5C: 0x3C, 0x7C: 0x3C,
  19012. 0x5D: 0x3D, 0x7D: 0x3D
  19013. };
  19014. (function(){for(var y in PtgDupes) PtgTypes[y] = PtgTypes[PtgDupes[y]];})();
  19015. var Ptg18 = {
  19016. 0x01: { n:'PtgElfLel', f:parse_PtgElfLel },
  19017. 0x02: { n:'PtgElfRw', f:parse_PtgElfRw },
  19018. 0x03: { n:'PtgElfCol', f:parse_PtgElfCol },
  19019. 0x06: { n:'PtgElfRwV', f:parse_PtgElfRwV },
  19020. 0x07: { n:'PtgElfColV', f:parse_PtgElfColV },
  19021. 0x0A: { n:'PtgElfRadical', f:parse_PtgElfRadical },
  19022. 0x0B: { n:'PtgElfRadicalS', f:parse_PtgElfRadicalS },
  19023. 0x0D: { n:'PtgElfColS', f:parse_PtgElfColS },
  19024. 0x0F: { n:'PtgElfColSV', f:parse_PtgElfColSV },
  19025. 0x10: { n:'PtgElfRadicalLel', f:parse_PtgElfRadicalLel },
  19026. 0x19: { n:'PtgList', f:parse_PtgList },
  19027. 0x1D: { n:'PtgSxName', f:parse_PtgSxName },
  19028. 0xFF: {}
  19029. };
  19030. var Ptg19 = {
  19031. 0x00: { n:'PtgAttrNoop', f:parse_PtgAttrNoop },
  19032. 0x01: { n:'PtgAttrSemi', f:parse_PtgAttrSemi },
  19033. 0x02: { n:'PtgAttrIf', f:parse_PtgAttrIf },
  19034. 0x04: { n:'PtgAttrChoose', f:parse_PtgAttrChoose },
  19035. 0x08: { n:'PtgAttrGoto', f:parse_PtgAttrGoto },
  19036. 0x10: { n:'PtgAttrSum', f:parse_PtgAttrSum },
  19037. 0x20: { n:'PtgAttrBaxcel', f:parse_PtgAttrBaxcel },
  19038. 0x40: { n:'PtgAttrSpace', f:parse_PtgAttrSpace },
  19039. 0x41: { n:'PtgAttrSpaceSemi', f:parse_PtgAttrSpaceSemi },
  19040. 0x80: { n:'PtgAttrIfError', f:parse_PtgAttrIfError },
  19041. 0xFF: {}
  19042. };
  19043. Ptg19[0x21] = Ptg19[0x20];
  19044. /* [MS-XLS] 2.5.198.103 ; [MS-XLSB] 2.5.97.87 */
  19045. function parse_RgbExtra(blob, length, rgce, opts) {
  19046. if(opts.biff < 8) return parsenoop(blob, length);
  19047. var target = blob.l + length;
  19048. var o = [];
  19049. for(var i = 0; i !== rgce.length; ++i) {
  19050. switch(rgce[i][0]) {
  19051. case 'PtgArray': /* PtgArray -> PtgExtraArray */
  19052. rgce[i][1] = parse_PtgExtraArray(blob, 0, opts);
  19053. o.push(rgce[i][1]);
  19054. break;
  19055. case 'PtgMemArea': /* PtgMemArea -> PtgExtraMem */
  19056. rgce[i][2] = parse_PtgExtraMem(blob, rgce[i][1], opts);
  19057. o.push(rgce[i][2]);
  19058. break;
  19059. case 'PtgExp': /* PtgExp -> PtgExtraCol */
  19060. if(opts && opts.biff == 12) {
  19061. rgce[i][1][1] = blob.read_shift(4);
  19062. o.push(rgce[i][1]);
  19063. } break;
  19064. case 'PtgList': /* TODO: PtgList -> PtgExtraList */
  19065. case 'PtgElfRadicalS': /* TODO: PtgElfRadicalS -> PtgExtraElf */
  19066. case 'PtgElfColS': /* TODO: PtgElfColS -> PtgExtraElf */
  19067. case 'PtgElfColSV': /* TODO: PtgElfColSV -> PtgExtraElf */
  19068. throw "Unsupported " + rgce[i][0];
  19069. default: break;
  19070. }
  19071. }
  19072. length = target - blob.l;
  19073. /* note: this is technically an error but Excel disregards */
  19074. //if(target !== blob.l && blob.l !== target - length) throw new error(target + " != " + blob.l);
  19075. if(length !== 0) o.push(parsenoop(blob, length));
  19076. return o;
  19077. }
  19078. /* [MS-XLS] 2.5.198.104 ; [MS-XLSB] 2.5.97.88 */
  19079. function parse_Rgce(blob, length, opts) {
  19080. var target = blob.l + length;
  19081. var R, id, ptgs = [];
  19082. while(target != blob.l) {
  19083. length = target - blob.l;
  19084. id = blob[blob.l];
  19085. R = PtgTypes[id];
  19086. if(id === 0x18 || id === 0x19) R = (id === 0x18 ? Ptg18 : Ptg19)[blob[blob.l + 1]];
  19087. if(!R || !R.f) { /*ptgs.push*/(parsenoop(blob, length)); }
  19088. else { ptgs.push([R.n, R.f(blob, length, opts)]); }
  19089. }
  19090. return ptgs;
  19091. }
  19092. function stringify_array(f) {
  19093. var o = [];
  19094. for(var i = 0; i < f.length; ++i) {
  19095. var x = f[i], r = [];
  19096. for(var j = 0; j < x.length; ++j) {
  19097. var y = x[j];
  19098. if(y) switch(y[0]) {
  19099. // TODO: handle embedded quotes
  19100. case 0x02:
  19101. r.push('"' + y[1].replace(/"/g,'""') + '"'); break;
  19102. default: r.push(y[1]);
  19103. } else r.push("");
  19104. }
  19105. o.push(r.join(","));
  19106. }
  19107. return o.join(";");
  19108. }
  19109. /* [MS-XLS] 2.2.2 ; [MS-XLSB] 2.2.2 TODO */
  19110. var PtgBinOp = {
  19111. PtgAdd: "+",
  19112. PtgConcat: "&",
  19113. PtgDiv: "/",
  19114. PtgEq: "=",
  19115. PtgGe: ">=",
  19116. PtgGt: ">",
  19117. PtgLe: "<=",
  19118. PtgLt: "<",
  19119. PtgMul: "*",
  19120. PtgNe: "<>",
  19121. PtgPower: "^",
  19122. PtgSub: "-"
  19123. };
  19124. function formula_quote_sheet_name(sname, opts) {
  19125. if(!sname && !(opts && opts.biff <= 5 && opts.biff >= 2)) throw new Error("empty sheet name");
  19126. if(sname.indexOf(" ") > -1) return "'" + sname + "'";
  19127. return sname;
  19128. }
  19129. function get_ixti_raw(supbooks, ixti, opts) {
  19130. if(!supbooks) return "SH33TJSERR0";
  19131. if(opts.biff > 8 && (!supbooks.XTI || !supbooks.XTI[ixti])) return supbooks.SheetNames[ixti];
  19132. if(!supbooks.XTI) return "SH33TJSERR6";
  19133. var XTI = supbooks.XTI[ixti];
  19134. if(opts.biff < 8) {
  19135. if(ixti > 10000) ixti-= 65536;
  19136. if(ixti < 0) ixti = -ixti;
  19137. return ixti == 0 ? "" : supbooks.XTI[ixti - 1];
  19138. }
  19139. if(!XTI) return "SH33TJSERR1";
  19140. var o = "";
  19141. if(opts.biff > 8) switch(supbooks[XTI[0]][0]) {
  19142. case 0x0165: /* 'BrtSupSelf' */
  19143. o = XTI[1] == -1 ? "#REF" : supbooks.SheetNames[XTI[1]];
  19144. return XTI[1] == XTI[2] ? o : o + ":" + supbooks.SheetNames[XTI[2]];
  19145. case 0x0166: /* 'BrtSupSame' */
  19146. if(opts.SID != null) return supbooks.SheetNames[opts.SID];
  19147. return "SH33TJSSAME" + supbooks[XTI[0]][0];
  19148. case 0x0163: /* 'BrtSupBookSrc' */
  19149. /* falls through */
  19150. default: return "SH33TJSSRC" + supbooks[XTI[0]][0];
  19151. }
  19152. switch(supbooks[XTI[0]][0][0]) {
  19153. case 0x0401:
  19154. o = XTI[1] == -1 ? "#REF" : (supbooks.SheetNames[XTI[1]] || "SH33TJSERR3");
  19155. return XTI[1] == XTI[2] ? o : o + ":" + supbooks.SheetNames[XTI[2]];
  19156. case 0x3A01: return "SH33TJSERR8";
  19157. default:
  19158. if(!supbooks[XTI[0]][0][3]) return "SH33TJSERR2";
  19159. o = XTI[1] == -1 ? "#REF" : (supbooks[XTI[0]][0][3][XTI[1]] || "SH33TJSERR4");
  19160. return XTI[1] == XTI[2] ? o : o + ":" + supbooks[XTI[0]][0][3][XTI[2]];
  19161. }
  19162. }
  19163. function get_ixti(supbooks, ixti, opts) {
  19164. return formula_quote_sheet_name(get_ixti_raw(supbooks, ixti, opts), opts);
  19165. }
  19166. function stringify_formula(formula/*Array<any>*/, range, cell, supbooks, opts) {
  19167. var biff = (opts && opts.biff) || 8;
  19168. var _range = /*range != null ? range :*/ {s:{c:0, r:0},e:{c:0, r:0}};
  19169. var stack = [], e1, e2, c, ixti=0, nameidx=0, r, sname="";
  19170. if(!formula[0] || !formula[0][0]) return "";
  19171. var last_sp = -1, sp = "";
  19172. for(var ff = 0, fflen = formula[0].length; ff < fflen; ++ff) {
  19173. var f = formula[0][ff];
  19174. switch(f[0]) {
  19175. case 'PtgUminus': /* [MS-XLS] 2.5.198.93 */
  19176. stack.push("-" + stack.pop()); break;
  19177. case 'PtgUplus': /* [MS-XLS] 2.5.198.95 */
  19178. stack.push("+" + stack.pop()); break;
  19179. case 'PtgPercent': /* [MS-XLS] 2.5.198.81 */
  19180. stack.push(stack.pop() + "%"); break;
  19181. case 'PtgAdd': /* [MS-XLS] 2.5.198.26 */
  19182. case 'PtgConcat': /* [MS-XLS] 2.5.198.43 */
  19183. case 'PtgDiv': /* [MS-XLS] 2.5.198.45 */
  19184. case 'PtgEq': /* [MS-XLS] 2.5.198.56 */
  19185. case 'PtgGe': /* [MS-XLS] 2.5.198.64 */
  19186. case 'PtgGt': /* [MS-XLS] 2.5.198.65 */
  19187. case 'PtgLe': /* [MS-XLS] 2.5.198.68 */
  19188. case 'PtgLt': /* [MS-XLS] 2.5.198.69 */
  19189. case 'PtgMul': /* [MS-XLS] 2.5.198.75 */
  19190. case 'PtgNe': /* [MS-XLS] 2.5.198.78 */
  19191. case 'PtgPower': /* [MS-XLS] 2.5.198.82 */
  19192. case 'PtgSub': /* [MS-XLS] 2.5.198.90 */
  19193. e1 = stack.pop(); e2 = stack.pop();
  19194. if(last_sp >= 0) {
  19195. switch(formula[0][last_sp][1][0]) {
  19196. case 0:
  19197. // $FlowIgnore
  19198. sp = fill(" ", formula[0][last_sp][1][1]); break;
  19199. case 1:
  19200. // $FlowIgnore
  19201. sp = fill("\r", formula[0][last_sp][1][1]); break;
  19202. default:
  19203. sp = "";
  19204. // $FlowIgnore
  19205. if(opts.WTF) throw new Error("Unexpected PtgAttrSpaceType " + formula[0][last_sp][1][0]);
  19206. }
  19207. e2 = e2 + sp;
  19208. last_sp = -1;
  19209. }
  19210. stack.push(e2+PtgBinOp[f[0]]+e1);
  19211. break;
  19212. case 'PtgIsect': /* [MS-XLS] 2.5.198.67 */
  19213. e1 = stack.pop(); e2 = stack.pop();
  19214. stack.push(e2+" "+e1);
  19215. break;
  19216. case 'PtgUnion': /* [MS-XLS] 2.5.198.94 */
  19217. e1 = stack.pop(); e2 = stack.pop();
  19218. stack.push(e2+","+e1);
  19219. break;
  19220. case 'PtgRange': /* [MS-XLS] 2.5.198.83 */
  19221. e1 = stack.pop(); e2 = stack.pop();
  19222. stack.push(e2+":"+e1);
  19223. break;
  19224. case 'PtgAttrChoose': /* [MS-XLS] 2.5.198.34 */
  19225. break;
  19226. case 'PtgAttrGoto': /* [MS-XLS] 2.5.198.35 */
  19227. break;
  19228. case 'PtgAttrIf': /* [MS-XLS] 2.5.198.36 */
  19229. break;
  19230. case 'PtgAttrIfError': /* [MS-XLSB] 2.5.97.28 */
  19231. break;
  19232. case 'PtgRef': /* [MS-XLS] 2.5.198.84 */
  19233. c = shift_cell_xls((f[1][1]), _range, opts);
  19234. stack.push(encode_cell_xls(c, biff));
  19235. break;
  19236. case 'PtgRefN': /* [MS-XLS] 2.5.198.88 */
  19237. c = cell ? shift_cell_xls((f[1][1]), cell, opts) : (f[1][1]);
  19238. stack.push(encode_cell_xls(c, biff));
  19239. break;
  19240. case 'PtgRef3d': /* [MS-XLS] 2.5.198.85 */
  19241. ixti = f[1][1]; c = shift_cell_xls((f[1][2]), _range, opts);
  19242. sname = get_ixti(supbooks, ixti, opts);
  19243. var w = sname; /* IE9 fails on defined names */ // eslint-disable-line no-unused-vars
  19244. stack.push(sname + "!" + encode_cell_xls(c, biff));
  19245. break;
  19246. case 'PtgFunc': /* [MS-XLS] 2.5.198.62 */
  19247. case 'PtgFuncVar': /* [MS-XLS] 2.5.198.63 */
  19248. /* f[1] = [argc, func, type] */
  19249. var argc = (f[1][0]), func = (f[1][1]);
  19250. if(!argc) argc = 0;
  19251. argc &= 0x7F;
  19252. var args = argc == 0 ? [] : stack.slice(-argc);
  19253. stack.length -= argc;
  19254. if(func === 'User') func = args.shift();
  19255. stack.push(func + "(" + args.join(",") + ")");
  19256. break;
  19257. case 'PtgBool': /* [MS-XLS] 2.5.198.42 */
  19258. stack.push(f[1] ? "TRUE" : "FALSE"); break;
  19259. case 'PtgInt': /* [MS-XLS] 2.5.198.66 */
  19260. stack.push(f[1]); break;
  19261. case 'PtgNum': /* [MS-XLS] 2.5.198.79 TODO: precision? */
  19262. stack.push(String(f[1])); break;
  19263. case 'PtgStr': /* [MS-XLS] 2.5.198.89 */
  19264. // $FlowIgnore
  19265. stack.push('"' + f[1] + '"'); break;
  19266. case 'PtgErr': /* [MS-XLS] 2.5.198.57 */
  19267. stack.push(f[1]); break;
  19268. case 'PtgAreaN': /* [MS-XLS] 2.5.198.31 TODO */
  19269. r = shift_range_xls(f[1][1], cell ? {s:cell} : _range, opts);
  19270. stack.push(encode_range_xls((r), opts));
  19271. break;
  19272. case 'PtgArea': /* [MS-XLS] 2.5.198.27 TODO: fixed points */
  19273. r = shift_range_xls(f[1][1], _range, opts);
  19274. stack.push(encode_range_xls((r), opts));
  19275. break;
  19276. case 'PtgArea3d': /* [MS-XLS] 2.5.198.28 TODO */
  19277. ixti = f[1][1]; r = f[1][2];
  19278. sname = get_ixti(supbooks, ixti, opts);
  19279. stack.push(sname + "!" + encode_range_xls((r), opts));
  19280. break;
  19281. case 'PtgAttrSum': /* [MS-XLS] 2.5.198.41 */
  19282. stack.push("SUM(" + stack.pop() + ")");
  19283. break;
  19284. case 'PtgAttrBaxcel': /* [MS-XLS] 2.5.198.33 */
  19285. case 'PtgAttrSemi': /* [MS-XLS] 2.5.198.37 */
  19286. break;
  19287. case 'PtgName': /* [MS-XLS] 2.5.198.76 ; [MS-XLSB] 2.5.97.60 TODO: revisions */
  19288. /* f[1] = type, 0, nameindex */
  19289. nameidx = (f[1][2]);
  19290. var lbl = (supbooks.names||[])[nameidx-1] || (supbooks[0]||[])[nameidx];
  19291. var name = lbl ? lbl.Name : "SH33TJSNAME" + String(nameidx);
  19292. if(name in XLSXFutureFunctions) name = XLSXFutureFunctions[name];
  19293. stack.push(name);
  19294. break;
  19295. case 'PtgNameX': /* [MS-XLS] 2.5.198.77 ; [MS-XLSB] 2.5.97.61 TODO: revisions */
  19296. /* f[1] = type, ixti, nameindex */
  19297. var bookidx = (f[1][1]); nameidx = (f[1][2]); var externbook;
  19298. /* TODO: Properly handle missing values */
  19299. if(opts.biff <= 5) {
  19300. if(bookidx < 0) bookidx = -bookidx;
  19301. if(supbooks[bookidx]) externbook = supbooks[bookidx][nameidx];
  19302. } else {
  19303. var o = "";
  19304. if(((supbooks[bookidx]||[])[0]||[])[0] == 0x3A01){/* empty */}
  19305. else if(((supbooks[bookidx]||[])[0]||[])[0] == 0x0401){
  19306. if(supbooks[bookidx][nameidx] && supbooks[bookidx][nameidx].itab > 0) {
  19307. o = supbooks.SheetNames[supbooks[bookidx][nameidx].itab-1] + "!";
  19308. }
  19309. }
  19310. else o = supbooks.SheetNames[nameidx-1]+ "!";
  19311. if(supbooks[bookidx] && supbooks[bookidx][nameidx]) o += supbooks[bookidx][nameidx].Name;
  19312. else if(supbooks[0] && supbooks[0][nameidx]) o += supbooks[0][nameidx].Name;
  19313. else o += "SH33TJSERRX";
  19314. stack.push(o);
  19315. break;
  19316. }
  19317. if(!externbook) externbook = {Name: "SH33TJSERRY"};
  19318. stack.push(externbook.Name);
  19319. break;
  19320. case 'PtgParen': /* [MS-XLS] 2.5.198.80 */
  19321. var lp = '(', rp = ')';
  19322. if(last_sp >= 0) {
  19323. sp = "";
  19324. switch(formula[0][last_sp][1][0]) {
  19325. // $FlowIgnore
  19326. case 2: lp = fill(" ", formula[0][last_sp][1][1]) + lp; break;
  19327. // $FlowIgnore
  19328. case 3: lp = fill("\r", formula[0][last_sp][1][1]) + lp; break;
  19329. // $FlowIgnore
  19330. case 4: rp = fill(" ", formula[0][last_sp][1][1]) + rp; break;
  19331. // $FlowIgnore
  19332. case 5: rp = fill("\r", formula[0][last_sp][1][1]) + rp; break;
  19333. default:
  19334. // $FlowIgnore
  19335. if(opts.WTF) throw new Error("Unexpected PtgAttrSpaceType " + formula[0][last_sp][1][0]);
  19336. }
  19337. last_sp = -1;
  19338. }
  19339. stack.push(lp + stack.pop() + rp); break;
  19340. case 'PtgRefErr': /* [MS-XLS] 2.5.198.86 */
  19341. stack.push('#REF!'); break;
  19342. case 'PtgRefErr3d': /* [MS-XLS] 2.5.198.87 */
  19343. stack.push('#REF!'); break;
  19344. case 'PtgExp': /* [MS-XLS] 2.5.198.58 TODO */
  19345. c = {c:(f[1][1]),r:(f[1][0])};
  19346. var q = ({c: cell.c, r:cell.r});
  19347. if(supbooks.sharedf[encode_cell(c)]) {
  19348. var parsedf = (supbooks.sharedf[encode_cell(c)]);
  19349. stack.push(stringify_formula(parsedf, _range, q, supbooks, opts));
  19350. }
  19351. else {
  19352. var fnd = false;
  19353. for(e1=0;e1!=supbooks.arrayf.length; ++e1) {
  19354. /* TODO: should be something like range_has */
  19355. e2 = supbooks.arrayf[e1];
  19356. if(c.c < e2[0].s.c || c.c > e2[0].e.c) continue;
  19357. if(c.r < e2[0].s.r || c.r > e2[0].e.r) continue;
  19358. stack.push(stringify_formula(e2[1], _range, q, supbooks, opts));
  19359. fnd = true;
  19360. break;
  19361. }
  19362. if(!fnd) stack.push(f[1]);
  19363. }
  19364. break;
  19365. case 'PtgArray': /* [MS-XLS] 2.5.198.32 TODO */
  19366. stack.push("{" + stringify_array(f[1]) + "}");
  19367. break;
  19368. case 'PtgMemArea': /* [MS-XLS] 2.5.198.70 TODO: confirm this is a non-display */
  19369. //stack.push("(" + f[2].map(encode_range).join(",") + ")");
  19370. break;
  19371. case 'PtgAttrSpace': /* [MS-XLS] 2.5.198.38 */
  19372. case 'PtgAttrSpaceSemi': /* [MS-XLS] 2.5.198.39 */
  19373. last_sp = ff;
  19374. break;
  19375. case 'PtgTbl': /* [MS-XLS] 2.5.198.92 TODO */
  19376. break;
  19377. case 'PtgMemErr': /* [MS-XLS] 2.5.198.71 */
  19378. break;
  19379. case 'PtgMissArg': /* [MS-XLS] 2.5.198.74 */
  19380. stack.push("");
  19381. break;
  19382. case 'PtgAreaErr': /* [MS-XLS] 2.5.198.29 */
  19383. stack.push("#REF!"); break;
  19384. case 'PtgAreaErr3d': /* [MS-XLS] 2.5.198.30 */
  19385. stack.push("#REF!"); break;
  19386. case 'PtgList': /* [MS-XLSB] 2.5.97.52 */
  19387. // $FlowIgnore
  19388. stack.push("Table" + f[1].idx + "[#" + f[1].rt + "]");
  19389. break;
  19390. case 'PtgMemAreaN':
  19391. case 'PtgMemNoMemN':
  19392. case 'PtgAttrNoop':
  19393. case 'PtgSheet':
  19394. case 'PtgEndSheet':
  19395. break;
  19396. case 'PtgMemFunc': /* [MS-XLS] 2.5.198.72 TODO */
  19397. break;
  19398. case 'PtgMemNoMem': /* [MS-XLS] 2.5.198.73 TODO */
  19399. break;
  19400. case 'PtgElfCol': /* [MS-XLS] 2.5.198.46 */
  19401. case 'PtgElfColS': /* [MS-XLS] 2.5.198.47 */
  19402. case 'PtgElfColSV': /* [MS-XLS] 2.5.198.48 */
  19403. case 'PtgElfColV': /* [MS-XLS] 2.5.198.49 */
  19404. case 'PtgElfLel': /* [MS-XLS] 2.5.198.50 */
  19405. case 'PtgElfRadical': /* [MS-XLS] 2.5.198.51 */
  19406. case 'PtgElfRadicalLel': /* [MS-XLS] 2.5.198.52 */
  19407. case 'PtgElfRadicalS': /* [MS-XLS] 2.5.198.53 */
  19408. case 'PtgElfRw': /* [MS-XLS] 2.5.198.54 */
  19409. case 'PtgElfRwV': /* [MS-XLS] 2.5.198.55 */
  19410. throw new Error("Unsupported ELFs");
  19411. case 'PtgSxName': /* [MS-XLS] 2.5.198.91 TODO -- find a test case */
  19412. throw new Error('Unrecognized Formula Token: ' + String(f));
  19413. default: throw new Error('Unrecognized Formula Token: ' + String(f));
  19414. }
  19415. var PtgNonDisp = ['PtgAttrSpace', 'PtgAttrSpaceSemi', 'PtgAttrGoto'];
  19416. if(opts.biff != 3) if(last_sp >= 0 && PtgNonDisp.indexOf(formula[0][ff][0]) == -1) {
  19417. f = formula[0][last_sp];
  19418. var _left = true;
  19419. switch(f[1][0]) {
  19420. /* note: some bad XLSB files omit the PtgParen */
  19421. case 4: _left = false;
  19422. /* falls through */
  19423. case 0:
  19424. // $FlowIgnore
  19425. sp = fill(" ", f[1][1]); break;
  19426. case 5: _left = false;
  19427. /* falls through */
  19428. case 1:
  19429. // $FlowIgnore
  19430. sp = fill("\r", f[1][1]); break;
  19431. default:
  19432. sp = "";
  19433. // $FlowIgnore
  19434. if(opts.WTF) throw new Error("Unexpected PtgAttrSpaceType " + f[1][0]);
  19435. }
  19436. stack.push((_left ? sp : "") + stack.pop() + (_left ? "" : sp));
  19437. last_sp = -1;
  19438. }
  19439. }
  19440. if(stack.length > 1 && opts.WTF) throw new Error("bad formula stack");
  19441. return stack[0];
  19442. }
  19443. /* [MS-XLS] 2.5.198.1 TODO */
  19444. function parse_ArrayParsedFormula(blob, length, opts) {
  19445. var target = blob.l + length, len = opts.biff == 2 ? 1 : 2;
  19446. var rgcb, cce = blob.read_shift(len); // length of rgce
  19447. if(cce == 0xFFFF) return [[],parsenoop(blob, length-2)];
  19448. var rgce = parse_Rgce(blob, cce, opts);
  19449. if(length !== cce + len) rgcb = parse_RgbExtra(blob, length - cce - len, rgce, opts);
  19450. blob.l = target;
  19451. return [rgce, rgcb];
  19452. }
  19453. /* [MS-XLS] 2.5.198.3 TODO */
  19454. function parse_XLSCellParsedFormula(blob, length, opts) {
  19455. var target = blob.l + length, len = opts.biff == 2 ? 1 : 2;
  19456. var rgcb, cce = blob.read_shift(len); // length of rgce
  19457. if(cce == 0xFFFF) return [[],parsenoop(blob, length-2)];
  19458. var rgce = parse_Rgce(blob, cce, opts);
  19459. if(length !== cce + len) rgcb = parse_RgbExtra(blob, length - cce - len, rgce, opts);
  19460. blob.l = target;
  19461. return [rgce, rgcb];
  19462. }
  19463. /* [MS-XLS] 2.5.198.21 */
  19464. function parse_NameParsedFormula(blob, length, opts, cce) {
  19465. var target = blob.l + length;
  19466. var rgce = parse_Rgce(blob, cce, opts);
  19467. var rgcb;
  19468. if(target !== blob.l) rgcb = parse_RgbExtra(blob, target - blob.l, rgce, opts);
  19469. return [rgce, rgcb];
  19470. }
  19471. /* [MS-XLS] 2.5.198.118 TODO */
  19472. function parse_SharedParsedFormula(blob, length, opts) {
  19473. var target = blob.l + length;
  19474. var rgcb, cce = blob.read_shift(2); // length of rgce
  19475. var rgce = parse_Rgce(blob, cce, opts);
  19476. if(cce == 0xFFFF) return [[],parsenoop(blob, length-2)];
  19477. if(length !== cce + 2) rgcb = parse_RgbExtra(blob, target - cce - 2, rgce, opts);
  19478. return [rgce, rgcb];
  19479. }
  19480. /* [MS-XLS] 2.5.133 TODO: how to emit empty strings? */
  19481. function parse_FormulaValue(blob) {
  19482. var b;
  19483. if(__readUInt16LE(blob,blob.l + 6) !== 0xFFFF) return [parse_Xnum(blob),'n'];
  19484. switch(blob[blob.l]) {
  19485. case 0x00: blob.l += 8; return ["String", 's'];
  19486. case 0x01: b = blob[blob.l+2] === 0x1; blob.l += 8; return [b,'b'];
  19487. case 0x02: b = blob[blob.l+2]; blob.l += 8; return [b,'e'];
  19488. case 0x03: blob.l += 8; return ["",'s'];
  19489. }
  19490. return [];
  19491. }
  19492. /* [MS-XLS] 2.4.127 TODO */
  19493. function parse_Formula(blob, length, opts) {
  19494. var end = blob.l + length;
  19495. var cell = parse_XLSCell(blob, 6);
  19496. if(opts.biff == 2) ++blob.l;
  19497. var val = parse_FormulaValue(blob,8);
  19498. var flags = blob.read_shift(1);
  19499. if(opts.biff != 2) {
  19500. blob.read_shift(1);
  19501. if(opts.biff >= 5) {
  19502. /*var chn = */blob.read_shift(4);
  19503. }
  19504. }
  19505. var cbf = parse_XLSCellParsedFormula(blob, end - blob.l, opts);
  19506. return {cell:cell, val:val[0], formula:cbf, shared: (flags >> 3) & 1, tt:val[1]};
  19507. }
  19508. /* XLSB Parsed Formula records have the same shape */
  19509. function parse_XLSBParsedFormula(data, length, opts) {
  19510. var cce = data.read_shift(4);
  19511. var rgce = parse_Rgce(data, cce, opts);
  19512. var cb = data.read_shift(4);
  19513. var rgcb = cb > 0 ? parse_RgbExtra(data, cb, rgce, opts) : null;
  19514. return [rgce, rgcb];
  19515. }
  19516. /* [MS-XLSB] 2.5.97.1 ArrayParsedFormula */
  19517. var parse_XLSBArrayParsedFormula = parse_XLSBParsedFormula;
  19518. /* [MS-XLSB] 2.5.97.4 CellParsedFormula */
  19519. var parse_XLSBCellParsedFormula = parse_XLSBParsedFormula;
  19520. /* [MS-XLSB] 2.5.97.12 NameParsedFormula */
  19521. var parse_XLSBNameParsedFormula = parse_XLSBParsedFormula;
  19522. /* [MS-XLSB] 2.5.97.98 SharedParsedFormula */
  19523. var parse_XLSBSharedParsedFormula = parse_XLSBParsedFormula;
  19524. /* [MS-XLS] 2.5.198.4 */
  19525. var Cetab = {
  19526. 0x0000: 'BEEP',
  19527. 0x0001: 'OPEN',
  19528. 0x0002: 'OPEN.LINKS',
  19529. 0x0003: 'CLOSE.ALL',
  19530. 0x0004: 'SAVE',
  19531. 0x0005: 'SAVE.AS',
  19532. 0x0006: 'FILE.DELETE',
  19533. 0x0007: 'PAGE.SETUP',
  19534. 0x0008: 'PRINT',
  19535. 0x0009: 'PRINTER.SETUP',
  19536. 0x000A: 'QUIT',
  19537. 0x000B: 'NEW.WINDOW',
  19538. 0x000C: 'ARRANGE.ALL',
  19539. 0x000D: 'WINDOW.SIZE',
  19540. 0x000E: 'WINDOW.MOVE',
  19541. 0x000F: 'FULL',
  19542. 0x0010: 'CLOSE',
  19543. 0x0011: 'RUN',
  19544. 0x0016: 'SET.PRINT.AREA',
  19545. 0x0017: 'SET.PRINT.TITLES',
  19546. 0x0018: 'SET.PAGE.BREAK',
  19547. 0x0019: 'REMOVE.PAGE.BREAK',
  19548. 0x001A: 'FONT',
  19549. 0x001B: 'DISPLAY',
  19550. 0x001C: 'PROTECT.DOCUMENT',
  19551. 0x001D: 'PRECISION',
  19552. 0x001E: 'A1.R1C1',
  19553. 0x001F: 'CALCULATE.NOW',
  19554. 0x0020: 'CALCULATION',
  19555. 0x0022: 'DATA.FIND',
  19556. 0x0023: 'EXTRACT',
  19557. 0x0024: 'DATA.DELETE',
  19558. 0x0025: 'SET.DATABASE',
  19559. 0x0026: 'SET.CRITERIA',
  19560. 0x0027: 'SORT',
  19561. 0x0028: 'DATA.SERIES',
  19562. 0x0029: 'TABLE',
  19563. 0x002A: 'FORMAT.NUMBER',
  19564. 0x002B: 'ALIGNMENT',
  19565. 0x002C: 'STYLE',
  19566. 0x002D: 'BORDER',
  19567. 0x002E: 'CELL.PROTECTION',
  19568. 0x002F: 'COLUMN.WIDTH',
  19569. 0x0030: 'UNDO',
  19570. 0x0031: 'CUT',
  19571. 0x0032: 'COPY',
  19572. 0x0033: 'PASTE',
  19573. 0x0034: 'CLEAR',
  19574. 0x0035: 'PASTE.SPECIAL',
  19575. 0x0036: 'EDIT.DELETE',
  19576. 0x0037: 'INSERT',
  19577. 0x0038: 'FILL.RIGHT',
  19578. 0x0039: 'FILL.DOWN',
  19579. 0x003D: 'DEFINE.NAME',
  19580. 0x003E: 'CREATE.NAMES',
  19581. 0x003F: 'FORMULA.GOTO',
  19582. 0x0040: 'FORMULA.FIND',
  19583. 0x0041: 'SELECT.LAST.CELL',
  19584. 0x0042: 'SHOW.ACTIVE.CELL',
  19585. 0x0043: 'GALLERY.AREA',
  19586. 0x0044: 'GALLERY.BAR',
  19587. 0x0045: 'GALLERY.COLUMN',
  19588. 0x0046: 'GALLERY.LINE',
  19589. 0x0047: 'GALLERY.PIE',
  19590. 0x0048: 'GALLERY.SCATTER',
  19591. 0x0049: 'COMBINATION',
  19592. 0x004A: 'PREFERRED',
  19593. 0x004B: 'ADD.OVERLAY',
  19594. 0x004C: 'GRIDLINES',
  19595. 0x004D: 'SET.PREFERRED',
  19596. 0x004E: 'AXES',
  19597. 0x004F: 'LEGEND',
  19598. 0x0050: 'ATTACH.TEXT',
  19599. 0x0051: 'ADD.ARROW',
  19600. 0x0052: 'SELECT.CHART',
  19601. 0x0053: 'SELECT.PLOT.AREA',
  19602. 0x0054: 'PATTERNS',
  19603. 0x0055: 'MAIN.CHART',
  19604. 0x0056: 'OVERLAY',
  19605. 0x0057: 'SCALE',
  19606. 0x0058: 'FORMAT.LEGEND',
  19607. 0x0059: 'FORMAT.TEXT',
  19608. 0x005A: 'EDIT.REPEAT',
  19609. 0x005B: 'PARSE',
  19610. 0x005C: 'JUSTIFY',
  19611. 0x005D: 'HIDE',
  19612. 0x005E: 'UNHIDE',
  19613. 0x005F: 'WORKSPACE',
  19614. 0x0060: 'FORMULA',
  19615. 0x0061: 'FORMULA.FILL',
  19616. 0x0062: 'FORMULA.ARRAY',
  19617. 0x0063: 'DATA.FIND.NEXT',
  19618. 0x0064: 'DATA.FIND.PREV',
  19619. 0x0065: 'FORMULA.FIND.NEXT',
  19620. 0x0066: 'FORMULA.FIND.PREV',
  19621. 0x0067: 'ACTIVATE',
  19622. 0x0068: 'ACTIVATE.NEXT',
  19623. 0x0069: 'ACTIVATE.PREV',
  19624. 0x006A: 'UNLOCKED.NEXT',
  19625. 0x006B: 'UNLOCKED.PREV',
  19626. 0x006C: 'COPY.PICTURE',
  19627. 0x006D: 'SELECT',
  19628. 0x006E: 'DELETE.NAME',
  19629. 0x006F: 'DELETE.FORMAT',
  19630. 0x0070: 'VLINE',
  19631. 0x0071: 'HLINE',
  19632. 0x0072: 'VPAGE',
  19633. 0x0073: 'HPAGE',
  19634. 0x0074: 'VSCROLL',
  19635. 0x0075: 'HSCROLL',
  19636. 0x0076: 'ALERT',
  19637. 0x0077: 'NEW',
  19638. 0x0078: 'CANCEL.COPY',
  19639. 0x0079: 'SHOW.CLIPBOARD',
  19640. 0x007A: 'MESSAGE',
  19641. 0x007C: 'PASTE.LINK',
  19642. 0x007D: 'APP.ACTIVATE',
  19643. 0x007E: 'DELETE.ARROW',
  19644. 0x007F: 'ROW.HEIGHT',
  19645. 0x0080: 'FORMAT.MOVE',
  19646. 0x0081: 'FORMAT.SIZE',
  19647. 0x0082: 'FORMULA.REPLACE',
  19648. 0x0083: 'SEND.KEYS',
  19649. 0x0084: 'SELECT.SPECIAL',
  19650. 0x0085: 'APPLY.NAMES',
  19651. 0x0086: 'REPLACE.FONT',
  19652. 0x0087: 'FREEZE.PANES',
  19653. 0x0088: 'SHOW.INFO',
  19654. 0x0089: 'SPLIT',
  19655. 0x008A: 'ON.WINDOW',
  19656. 0x008B: 'ON.DATA',
  19657. 0x008C: 'DISABLE.INPUT',
  19658. 0x008E: 'OUTLINE',
  19659. 0x008F: 'LIST.NAMES',
  19660. 0x0090: 'FILE.CLOSE',
  19661. 0x0091: 'SAVE.WORKBOOK',
  19662. 0x0092: 'DATA.FORM',
  19663. 0x0093: 'COPY.CHART',
  19664. 0x0094: 'ON.TIME',
  19665. 0x0095: 'WAIT',
  19666. 0x0096: 'FORMAT.FONT',
  19667. 0x0097: 'FILL.UP',
  19668. 0x0098: 'FILL.LEFT',
  19669. 0x0099: 'DELETE.OVERLAY',
  19670. 0x009B: 'SHORT.MENUS',
  19671. 0x009F: 'SET.UPDATE.STATUS',
  19672. 0x00A1: 'COLOR.PALETTE',
  19673. 0x00A2: 'DELETE.STYLE',
  19674. 0x00A3: 'WINDOW.RESTORE',
  19675. 0x00A4: 'WINDOW.MAXIMIZE',
  19676. 0x00A6: 'CHANGE.LINK',
  19677. 0x00A7: 'CALCULATE.DOCUMENT',
  19678. 0x00A8: 'ON.KEY',
  19679. 0x00A9: 'APP.RESTORE',
  19680. 0x00AA: 'APP.MOVE',
  19681. 0x00AB: 'APP.SIZE',
  19682. 0x00AC: 'APP.MINIMIZE',
  19683. 0x00AD: 'APP.MAXIMIZE',
  19684. 0x00AE: 'BRING.TO.FRONT',
  19685. 0x00AF: 'SEND.TO.BACK',
  19686. 0x00B9: 'MAIN.CHART.TYPE',
  19687. 0x00BA: 'OVERLAY.CHART.TYPE',
  19688. 0x00BB: 'SELECT.END',
  19689. 0x00BC: 'OPEN.MAIL',
  19690. 0x00BD: 'SEND.MAIL',
  19691. 0x00BE: 'STANDARD.FONT',
  19692. 0x00BF: 'CONSOLIDATE',
  19693. 0x00C0: 'SORT.SPECIAL',
  19694. 0x00C1: 'GALLERY.3D.AREA',
  19695. 0x00C2: 'GALLERY.3D.COLUMN',
  19696. 0x00C3: 'GALLERY.3D.LINE',
  19697. 0x00C4: 'GALLERY.3D.PIE',
  19698. 0x00C5: 'VIEW.3D',
  19699. 0x00C6: 'GOAL.SEEK',
  19700. 0x00C7: 'WORKGROUP',
  19701. 0x00C8: 'FILL.GROUP',
  19702. 0x00C9: 'UPDATE.LINK',
  19703. 0x00CA: 'PROMOTE',
  19704. 0x00CB: 'DEMOTE',
  19705. 0x00CC: 'SHOW.DETAIL',
  19706. 0x00CE: 'UNGROUP',
  19707. 0x00CF: 'OBJECT.PROPERTIES',
  19708. 0x00D0: 'SAVE.NEW.OBJECT',
  19709. 0x00D1: 'SHARE',
  19710. 0x00D2: 'SHARE.NAME',
  19711. 0x00D3: 'DUPLICATE',
  19712. 0x00D4: 'APPLY.STYLE',
  19713. 0x00D5: 'ASSIGN.TO.OBJECT',
  19714. 0x00D6: 'OBJECT.PROTECTION',
  19715. 0x00D7: 'HIDE.OBJECT',
  19716. 0x00D8: 'SET.EXTRACT',
  19717. 0x00D9: 'CREATE.PUBLISHER',
  19718. 0x00DA: 'SUBSCRIBE.TO',
  19719. 0x00DB: 'ATTRIBUTES',
  19720. 0x00DC: 'SHOW.TOOLBAR',
  19721. 0x00DE: 'PRINT.PREVIEW',
  19722. 0x00DF: 'EDIT.COLOR',
  19723. 0x00E0: 'SHOW.LEVELS',
  19724. 0x00E1: 'FORMAT.MAIN',
  19725. 0x00E2: 'FORMAT.OVERLAY',
  19726. 0x00E3: 'ON.RECALC',
  19727. 0x00E4: 'EDIT.SERIES',
  19728. 0x00E5: 'DEFINE.STYLE',
  19729. 0x00F0: 'LINE.PRINT',
  19730. 0x00F3: 'ENTER.DATA',
  19731. 0x00F9: 'GALLERY.RADAR',
  19732. 0x00FA: 'MERGE.STYLES',
  19733. 0x00FB: 'EDITION.OPTIONS',
  19734. 0x00FC: 'PASTE.PICTURE',
  19735. 0x00FD: 'PASTE.PICTURE.LINK',
  19736. 0x00FE: 'SPELLING',
  19737. 0x0100: 'ZOOM',
  19738. 0x0103: 'INSERT.OBJECT',
  19739. 0x0104: 'WINDOW.MINIMIZE',
  19740. 0x0109: 'SOUND.NOTE',
  19741. 0x010A: 'SOUND.PLAY',
  19742. 0x010B: 'FORMAT.SHAPE',
  19743. 0x010C: 'EXTEND.POLYGON',
  19744. 0x010D: 'FORMAT.AUTO',
  19745. 0x0110: 'GALLERY.3D.BAR',
  19746. 0x0111: 'GALLERY.3D.SURFACE',
  19747. 0x0112: 'FILL.AUTO',
  19748. 0x0114: 'CUSTOMIZE.TOOLBAR',
  19749. 0x0115: 'ADD.TOOL',
  19750. 0x0116: 'EDIT.OBJECT',
  19751. 0x0117: 'ON.DOUBLECLICK',
  19752. 0x0118: 'ON.ENTRY',
  19753. 0x0119: 'WORKBOOK.ADD',
  19754. 0x011A: 'WORKBOOK.MOVE',
  19755. 0x011B: 'WORKBOOK.COPY',
  19756. 0x011C: 'WORKBOOK.OPTIONS',
  19757. 0x011D: 'SAVE.WORKSPACE',
  19758. 0x0120: 'CHART.WIZARD',
  19759. 0x0121: 'DELETE.TOOL',
  19760. 0x0122: 'MOVE.TOOL',
  19761. 0x0123: 'WORKBOOK.SELECT',
  19762. 0x0124: 'WORKBOOK.ACTIVATE',
  19763. 0x0125: 'ASSIGN.TO.TOOL',
  19764. 0x0127: 'COPY.TOOL',
  19765. 0x0128: 'RESET.TOOL',
  19766. 0x0129: 'CONSTRAIN.NUMERIC',
  19767. 0x012A: 'PASTE.TOOL',
  19768. 0x012E: 'WORKBOOK.NEW',
  19769. 0x0131: 'SCENARIO.CELLS',
  19770. 0x0132: 'SCENARIO.DELETE',
  19771. 0x0133: 'SCENARIO.ADD',
  19772. 0x0134: 'SCENARIO.EDIT',
  19773. 0x0135: 'SCENARIO.SHOW',
  19774. 0x0136: 'SCENARIO.SHOW.NEXT',
  19775. 0x0137: 'SCENARIO.SUMMARY',
  19776. 0x0138: 'PIVOT.TABLE.WIZARD',
  19777. 0x0139: 'PIVOT.FIELD.PROPERTIES',
  19778. 0x013A: 'PIVOT.FIELD',
  19779. 0x013B: 'PIVOT.ITEM',
  19780. 0x013C: 'PIVOT.ADD.FIELDS',
  19781. 0x013E: 'OPTIONS.CALCULATION',
  19782. 0x013F: 'OPTIONS.EDIT',
  19783. 0x0140: 'OPTIONS.VIEW',
  19784. 0x0141: 'ADDIN.MANAGER',
  19785. 0x0142: 'MENU.EDITOR',
  19786. 0x0143: 'ATTACH.TOOLBARS',
  19787. 0x0144: 'VBAActivate',
  19788. 0x0145: 'OPTIONS.CHART',
  19789. 0x0148: 'VBA.INSERT.FILE',
  19790. 0x014A: 'VBA.PROCEDURE.DEFINITION',
  19791. 0x0150: 'ROUTING.SLIP',
  19792. 0x0152: 'ROUTE.DOCUMENT',
  19793. 0x0153: 'MAIL.LOGON',
  19794. 0x0156: 'INSERT.PICTURE',
  19795. 0x0157: 'EDIT.TOOL',
  19796. 0x0158: 'GALLERY.DOUGHNUT',
  19797. 0x015E: 'CHART.TREND',
  19798. 0x0160: 'PIVOT.ITEM.PROPERTIES',
  19799. 0x0162: 'WORKBOOK.INSERT',
  19800. 0x0163: 'OPTIONS.TRANSITION',
  19801. 0x0164: 'OPTIONS.GENERAL',
  19802. 0x0172: 'FILTER.ADVANCED',
  19803. 0x0175: 'MAIL.ADD.MAILER',
  19804. 0x0176: 'MAIL.DELETE.MAILER',
  19805. 0x0177: 'MAIL.REPLY',
  19806. 0x0178: 'MAIL.REPLY.ALL',
  19807. 0x0179: 'MAIL.FORWARD',
  19808. 0x017A: 'MAIL.NEXT.LETTER',
  19809. 0x017B: 'DATA.LABEL',
  19810. 0x017C: 'INSERT.TITLE',
  19811. 0x017D: 'FONT.PROPERTIES',
  19812. 0x017E: 'MACRO.OPTIONS',
  19813. 0x017F: 'WORKBOOK.HIDE',
  19814. 0x0180: 'WORKBOOK.UNHIDE',
  19815. 0x0181: 'WORKBOOK.DELETE',
  19816. 0x0182: 'WORKBOOK.NAME',
  19817. 0x0184: 'GALLERY.CUSTOM',
  19818. 0x0186: 'ADD.CHART.AUTOFORMAT',
  19819. 0x0187: 'DELETE.CHART.AUTOFORMAT',
  19820. 0x0188: 'CHART.ADD.DATA',
  19821. 0x0189: 'AUTO.OUTLINE',
  19822. 0x018A: 'TAB.ORDER',
  19823. 0x018B: 'SHOW.DIALOG',
  19824. 0x018C: 'SELECT.ALL',
  19825. 0x018D: 'UNGROUP.SHEETS',
  19826. 0x018E: 'SUBTOTAL.CREATE',
  19827. 0x018F: 'SUBTOTAL.REMOVE',
  19828. 0x0190: 'RENAME.OBJECT',
  19829. 0x019C: 'WORKBOOK.SCROLL',
  19830. 0x019D: 'WORKBOOK.NEXT',
  19831. 0x019E: 'WORKBOOK.PREV',
  19832. 0x019F: 'WORKBOOK.TAB.SPLIT',
  19833. 0x01A0: 'FULL.SCREEN',
  19834. 0x01A1: 'WORKBOOK.PROTECT',
  19835. 0x01A4: 'SCROLLBAR.PROPERTIES',
  19836. 0x01A5: 'PIVOT.SHOW.PAGES',
  19837. 0x01A6: 'TEXT.TO.COLUMNS',
  19838. 0x01A7: 'FORMAT.CHARTTYPE',
  19839. 0x01A8: 'LINK.FORMAT',
  19840. 0x01A9: 'TRACER.DISPLAY',
  19841. 0x01AE: 'TRACER.NAVIGATE',
  19842. 0x01AF: 'TRACER.CLEAR',
  19843. 0x01B0: 'TRACER.ERROR',
  19844. 0x01B1: 'PIVOT.FIELD.GROUP',
  19845. 0x01B2: 'PIVOT.FIELD.UNGROUP',
  19846. 0x01B3: 'CHECKBOX.PROPERTIES',
  19847. 0x01B4: 'LABEL.PROPERTIES',
  19848. 0x01B5: 'LISTBOX.PROPERTIES',
  19849. 0x01B6: 'EDITBOX.PROPERTIES',
  19850. 0x01B7: 'PIVOT.REFRESH',
  19851. 0x01B8: 'LINK.COMBO',
  19852. 0x01B9: 'OPEN.TEXT',
  19853. 0x01BA: 'HIDE.DIALOG',
  19854. 0x01BB: 'SET.DIALOG.FOCUS',
  19855. 0x01BC: 'ENABLE.OBJECT',
  19856. 0x01BD: 'PUSHBUTTON.PROPERTIES',
  19857. 0x01BE: 'SET.DIALOG.DEFAULT',
  19858. 0x01BF: 'FILTER',
  19859. 0x01C0: 'FILTER.SHOW.ALL',
  19860. 0x01C1: 'CLEAR.OUTLINE',
  19861. 0x01C2: 'FUNCTION.WIZARD',
  19862. 0x01C3: 'ADD.LIST.ITEM',
  19863. 0x01C4: 'SET.LIST.ITEM',
  19864. 0x01C5: 'REMOVE.LIST.ITEM',
  19865. 0x01C6: 'SELECT.LIST.ITEM',
  19866. 0x01C7: 'SET.CONTROL.VALUE',
  19867. 0x01C8: 'SAVE.COPY.AS',
  19868. 0x01CA: 'OPTIONS.LISTS.ADD',
  19869. 0x01CB: 'OPTIONS.LISTS.DELETE',
  19870. 0x01CC: 'SERIES.AXES',
  19871. 0x01CD: 'SERIES.X',
  19872. 0x01CE: 'SERIES.Y',
  19873. 0x01CF: 'ERRORBAR.X',
  19874. 0x01D0: 'ERRORBAR.Y',
  19875. 0x01D1: 'FORMAT.CHART',
  19876. 0x01D2: 'SERIES.ORDER',
  19877. 0x01D3: 'MAIL.LOGOFF',
  19878. 0x01D4: 'CLEAR.ROUTING.SLIP',
  19879. 0x01D5: 'APP.ACTIVATE.MICROSOFT',
  19880. 0x01D6: 'MAIL.EDIT.MAILER',
  19881. 0x01D7: 'ON.SHEET',
  19882. 0x01D8: 'STANDARD.WIDTH',
  19883. 0x01D9: 'SCENARIO.MERGE',
  19884. 0x01DA: 'SUMMARY.INFO',
  19885. 0x01DB: 'FIND.FILE',
  19886. 0x01DC: 'ACTIVE.CELL.FONT',
  19887. 0x01DD: 'ENABLE.TIPWIZARD',
  19888. 0x01DE: 'VBA.MAKE.ADDIN',
  19889. 0x01E0: 'INSERTDATATABLE',
  19890. 0x01E1: 'WORKGROUP.OPTIONS',
  19891. 0x01E2: 'MAIL.SEND.MAILER',
  19892. 0x01E5: 'AUTOCORRECT',
  19893. 0x01E9: 'POST.DOCUMENT',
  19894. 0x01EB: 'PICKLIST',
  19895. 0x01ED: 'VIEW.SHOW',
  19896. 0x01EE: 'VIEW.DEFINE',
  19897. 0x01EF: 'VIEW.DELETE',
  19898. 0x01FD: 'SHEET.BACKGROUND',
  19899. 0x01FE: 'INSERT.MAP.OBJECT',
  19900. 0x01FF: 'OPTIONS.MENONO',
  19901. 0x0205: 'MSOCHECKS',
  19902. 0x0206: 'NORMAL',
  19903. 0x0207: 'LAYOUT',
  19904. 0x0208: 'RM.PRINT.AREA',
  19905. 0x0209: 'CLEAR.PRINT.AREA',
  19906. 0x020A: 'ADD.PRINT.AREA',
  19907. 0x020B: 'MOVE.BRK',
  19908. 0x0221: 'HIDECURR.NOTE',
  19909. 0x0222: 'HIDEALL.NOTES',
  19910. 0x0223: 'DELETE.NOTE',
  19911. 0x0224: 'TRAVERSE.NOTES',
  19912. 0x0225: 'ACTIVATE.NOTES',
  19913. 0x026C: 'PROTECT.REVISIONS',
  19914. 0x026D: 'UNPROTECT.REVISIONS',
  19915. 0x0287: 'OPTIONS.ME',
  19916. 0x028D: 'WEB.PUBLISH',
  19917. 0x029B: 'NEWWEBQUERY',
  19918. 0x02A1: 'PIVOT.TABLE.CHART',
  19919. 0x02F1: 'OPTIONS.SAVE',
  19920. 0x02F3: 'OPTIONS.SPELL',
  19921. 0x0328: 'HIDEALL.INKANNOTS'
  19922. };
  19923. /* [MS-XLS] 2.5.198.17 */
  19924. /* [MS-XLSB] 2.5.97.10 */
  19925. var Ftab = {
  19926. 0x0000: 'COUNT',
  19927. 0x0001: 'IF',
  19928. 0x0002: 'ISNA',
  19929. 0x0003: 'ISERROR',
  19930. 0x0004: 'SUM',
  19931. 0x0005: 'AVERAGE',
  19932. 0x0006: 'MIN',
  19933. 0x0007: 'MAX',
  19934. 0x0008: 'ROW',
  19935. 0x0009: 'COLUMN',
  19936. 0x000A: 'NA',
  19937. 0x000B: 'NPV',
  19938. 0x000C: 'STDEV',
  19939. 0x000D: 'DOLLAR',
  19940. 0x000E: 'FIXED',
  19941. 0x000F: 'SIN',
  19942. 0x0010: 'COS',
  19943. 0x0011: 'TAN',
  19944. 0x0012: 'ATAN',
  19945. 0x0013: 'PI',
  19946. 0x0014: 'SQRT',
  19947. 0x0015: 'EXP',
  19948. 0x0016: 'LN',
  19949. 0x0017: 'LOG10',
  19950. 0x0018: 'ABS',
  19951. 0x0019: 'INT',
  19952. 0x001A: 'SIGN',
  19953. 0x001B: 'ROUND',
  19954. 0x001C: 'LOOKUP',
  19955. 0x001D: 'INDEX',
  19956. 0x001E: 'REPT',
  19957. 0x001F: 'MID',
  19958. 0x0020: 'LEN',
  19959. 0x0021: 'VALUE',
  19960. 0x0022: 'TRUE',
  19961. 0x0023: 'FALSE',
  19962. 0x0024: 'AND',
  19963. 0x0025: 'OR',
  19964. 0x0026: 'NOT',
  19965. 0x0027: 'MOD',
  19966. 0x0028: 'DCOUNT',
  19967. 0x0029: 'DSUM',
  19968. 0x002A: 'DAVERAGE',
  19969. 0x002B: 'DMIN',
  19970. 0x002C: 'DMAX',
  19971. 0x002D: 'DSTDEV',
  19972. 0x002E: 'VAR',
  19973. 0x002F: 'DVAR',
  19974. 0x0030: 'TEXT',
  19975. 0x0031: 'LINEST',
  19976. 0x0032: 'TREND',
  19977. 0x0033: 'LOGEST',
  19978. 0x0034: 'GROWTH',
  19979. 0x0035: 'GOTO',
  19980. 0x0036: 'HALT',
  19981. 0x0037: 'RETURN',
  19982. 0x0038: 'PV',
  19983. 0x0039: 'FV',
  19984. 0x003A: 'NPER',
  19985. 0x003B: 'PMT',
  19986. 0x003C: 'RATE',
  19987. 0x003D: 'MIRR',
  19988. 0x003E: 'IRR',
  19989. 0x003F: 'RAND',
  19990. 0x0040: 'MATCH',
  19991. 0x0041: 'DATE',
  19992. 0x0042: 'TIME',
  19993. 0x0043: 'DAY',
  19994. 0x0044: 'MONTH',
  19995. 0x0045: 'YEAR',
  19996. 0x0046: 'WEEKDAY',
  19997. 0x0047: 'HOUR',
  19998. 0x0048: 'MINUTE',
  19999. 0x0049: 'SECOND',
  20000. 0x004A: 'NOW',
  20001. 0x004B: 'AREAS',
  20002. 0x004C: 'ROWS',
  20003. 0x004D: 'COLUMNS',
  20004. 0x004E: 'OFFSET',
  20005. 0x004F: 'ABSREF',
  20006. 0x0050: 'RELREF',
  20007. 0x0051: 'ARGUMENT',
  20008. 0x0052: 'SEARCH',
  20009. 0x0053: 'TRANSPOSE',
  20010. 0x0054: 'ERROR',
  20011. 0x0055: 'STEP',
  20012. 0x0056: 'TYPE',
  20013. 0x0057: 'ECHO',
  20014. 0x0058: 'SET.NAME',
  20015. 0x0059: 'CALLER',
  20016. 0x005A: 'DEREF',
  20017. 0x005B: 'WINDOWS',
  20018. 0x005C: 'SERIES',
  20019. 0x005D: 'DOCUMENTS',
  20020. 0x005E: 'ACTIVE.CELL',
  20021. 0x005F: 'SELECTION',
  20022. 0x0060: 'RESULT',
  20023. 0x0061: 'ATAN2',
  20024. 0x0062: 'ASIN',
  20025. 0x0063: 'ACOS',
  20026. 0x0064: 'CHOOSE',
  20027. 0x0065: 'HLOOKUP',
  20028. 0x0066: 'VLOOKUP',
  20029. 0x0067: 'LINKS',
  20030. 0x0068: 'INPUT',
  20031. 0x0069: 'ISREF',
  20032. 0x006A: 'GET.FORMULA',
  20033. 0x006B: 'GET.NAME',
  20034. 0x006C: 'SET.VALUE',
  20035. 0x006D: 'LOG',
  20036. 0x006E: 'EXEC',
  20037. 0x006F: 'CHAR',
  20038. 0x0070: 'LOWER',
  20039. 0x0071: 'UPPER',
  20040. 0x0072: 'PROPER',
  20041. 0x0073: 'LEFT',
  20042. 0x0074: 'RIGHT',
  20043. 0x0075: 'EXACT',
  20044. 0x0076: 'TRIM',
  20045. 0x0077: 'REPLACE',
  20046. 0x0078: 'SUBSTITUTE',
  20047. 0x0079: 'CODE',
  20048. 0x007A: 'NAMES',
  20049. 0x007B: 'DIRECTORY',
  20050. 0x007C: 'FIND',
  20051. 0x007D: 'CELL',
  20052. 0x007E: 'ISERR',
  20053. 0x007F: 'ISTEXT',
  20054. 0x0080: 'ISNUMBER',
  20055. 0x0081: 'ISBLANK',
  20056. 0x0082: 'T',
  20057. 0x0083: 'N',
  20058. 0x0084: 'FOPEN',
  20059. 0x0085: 'FCLOSE',
  20060. 0x0086: 'FSIZE',
  20061. 0x0087: 'FREADLN',
  20062. 0x0088: 'FREAD',
  20063. 0x0089: 'FWRITELN',
  20064. 0x008A: 'FWRITE',
  20065. 0x008B: 'FPOS',
  20066. 0x008C: 'DATEVALUE',
  20067. 0x008D: 'TIMEVALUE',
  20068. 0x008E: 'SLN',
  20069. 0x008F: 'SYD',
  20070. 0x0090: 'DDB',
  20071. 0x0091: 'GET.DEF',
  20072. 0x0092: 'REFTEXT',
  20073. 0x0093: 'TEXTREF',
  20074. 0x0094: 'INDIRECT',
  20075. 0x0095: 'REGISTER',
  20076. 0x0096: 'CALL',
  20077. 0x0097: 'ADD.BAR',
  20078. 0x0098: 'ADD.MENU',
  20079. 0x0099: 'ADD.COMMAND',
  20080. 0x009A: 'ENABLE.COMMAND',
  20081. 0x009B: 'CHECK.COMMAND',
  20082. 0x009C: 'RENAME.COMMAND',
  20083. 0x009D: 'SHOW.BAR',
  20084. 0x009E: 'DELETE.MENU',
  20085. 0x009F: 'DELETE.COMMAND',
  20086. 0x00A0: 'GET.CHART.ITEM',
  20087. 0x00A1: 'DIALOG.BOX',
  20088. 0x00A2: 'CLEAN',
  20089. 0x00A3: 'MDETERM',
  20090. 0x00A4: 'MINVERSE',
  20091. 0x00A5: 'MMULT',
  20092. 0x00A6: 'FILES',
  20093. 0x00A7: 'IPMT',
  20094. 0x00A8: 'PPMT',
  20095. 0x00A9: 'COUNTA',
  20096. 0x00AA: 'CANCEL.KEY',
  20097. 0x00AB: 'FOR',
  20098. 0x00AC: 'WHILE',
  20099. 0x00AD: 'BREAK',
  20100. 0x00AE: 'NEXT',
  20101. 0x00AF: 'INITIATE',
  20102. 0x00B0: 'REQUEST',
  20103. 0x00B1: 'POKE',
  20104. 0x00B2: 'EXECUTE',
  20105. 0x00B3: 'TERMINATE',
  20106. 0x00B4: 'RESTART',
  20107. 0x00B5: 'HELP',
  20108. 0x00B6: 'GET.BAR',
  20109. 0x00B7: 'PRODUCT',
  20110. 0x00B8: 'FACT',
  20111. 0x00B9: 'GET.CELL',
  20112. 0x00BA: 'GET.WORKSPACE',
  20113. 0x00BB: 'GET.WINDOW',
  20114. 0x00BC: 'GET.DOCUMENT',
  20115. 0x00BD: 'DPRODUCT',
  20116. 0x00BE: 'ISNONTEXT',
  20117. 0x00BF: 'GET.NOTE',
  20118. 0x00C0: 'NOTE',
  20119. 0x00C1: 'STDEVP',
  20120. 0x00C2: 'VARP',
  20121. 0x00C3: 'DSTDEVP',
  20122. 0x00C4: 'DVARP',
  20123. 0x00C5: 'TRUNC',
  20124. 0x00C6: 'ISLOGICAL',
  20125. 0x00C7: 'DCOUNTA',
  20126. 0x00C8: 'DELETE.BAR',
  20127. 0x00C9: 'UNREGISTER',
  20128. 0x00CC: 'USDOLLAR',
  20129. 0x00CD: 'FINDB',
  20130. 0x00CE: 'SEARCHB',
  20131. 0x00CF: 'REPLACEB',
  20132. 0x00D0: 'LEFTB',
  20133. 0x00D1: 'RIGHTB',
  20134. 0x00D2: 'MIDB',
  20135. 0x00D3: 'LENB',
  20136. 0x00D4: 'ROUNDUP',
  20137. 0x00D5: 'ROUNDDOWN',
  20138. 0x00D6: 'ASC',
  20139. 0x00D7: 'DBCS',
  20140. 0x00D8: 'RANK',
  20141. 0x00DB: 'ADDRESS',
  20142. 0x00DC: 'DAYS360',
  20143. 0x00DD: 'TODAY',
  20144. 0x00DE: 'VDB',
  20145. 0x00DF: 'ELSE',
  20146. 0x00E0: 'ELSE.IF',
  20147. 0x00E1: 'END.IF',
  20148. 0x00E2: 'FOR.CELL',
  20149. 0x00E3: 'MEDIAN',
  20150. 0x00E4: 'SUMPRODUCT',
  20151. 0x00E5: 'SINH',
  20152. 0x00E6: 'COSH',
  20153. 0x00E7: 'TANH',
  20154. 0x00E8: 'ASINH',
  20155. 0x00E9: 'ACOSH',
  20156. 0x00EA: 'ATANH',
  20157. 0x00EB: 'DGET',
  20158. 0x00EC: 'CREATE.OBJECT',
  20159. 0x00ED: 'VOLATILE',
  20160. 0x00EE: 'LAST.ERROR',
  20161. 0x00EF: 'CUSTOM.UNDO',
  20162. 0x00F0: 'CUSTOM.REPEAT',
  20163. 0x00F1: 'FORMULA.CONVERT',
  20164. 0x00F2: 'GET.LINK.INFO',
  20165. 0x00F3: 'TEXT.BOX',
  20166. 0x00F4: 'INFO',
  20167. 0x00F5: 'GROUP',
  20168. 0x00F6: 'GET.OBJECT',
  20169. 0x00F7: 'DB',
  20170. 0x00F8: 'PAUSE',
  20171. 0x00FB: 'RESUME',
  20172. 0x00FC: 'FREQUENCY',
  20173. 0x00FD: 'ADD.TOOLBAR',
  20174. 0x00FE: 'DELETE.TOOLBAR',
  20175. 0x00FF: 'User',
  20176. 0x0100: 'RESET.TOOLBAR',
  20177. 0x0101: 'EVALUATE',
  20178. 0x0102: 'GET.TOOLBAR',
  20179. 0x0103: 'GET.TOOL',
  20180. 0x0104: 'SPELLING.CHECK',
  20181. 0x0105: 'ERROR.TYPE',
  20182. 0x0106: 'APP.TITLE',
  20183. 0x0107: 'WINDOW.TITLE',
  20184. 0x0108: 'SAVE.TOOLBAR',
  20185. 0x0109: 'ENABLE.TOOL',
  20186. 0x010A: 'PRESS.TOOL',
  20187. 0x010B: 'REGISTER.ID',
  20188. 0x010C: 'GET.WORKBOOK',
  20189. 0x010D: 'AVEDEV',
  20190. 0x010E: 'BETADIST',
  20191. 0x010F: 'GAMMALN',
  20192. 0x0110: 'BETAINV',
  20193. 0x0111: 'BINOMDIST',
  20194. 0x0112: 'CHIDIST',
  20195. 0x0113: 'CHIINV',
  20196. 0x0114: 'COMBIN',
  20197. 0x0115: 'CONFIDENCE',
  20198. 0x0116: 'CRITBINOM',
  20199. 0x0117: 'EVEN',
  20200. 0x0118: 'EXPONDIST',
  20201. 0x0119: 'FDIST',
  20202. 0x011A: 'FINV',
  20203. 0x011B: 'FISHER',
  20204. 0x011C: 'FISHERINV',
  20205. 0x011D: 'FLOOR',
  20206. 0x011E: 'GAMMADIST',
  20207. 0x011F: 'GAMMAINV',
  20208. 0x0120: 'CEILING',
  20209. 0x0121: 'HYPGEOMDIST',
  20210. 0x0122: 'LOGNORMDIST',
  20211. 0x0123: 'LOGINV',
  20212. 0x0124: 'NEGBINOMDIST',
  20213. 0x0125: 'NORMDIST',
  20214. 0x0126: 'NORMSDIST',
  20215. 0x0127: 'NORMINV',
  20216. 0x0128: 'NORMSINV',
  20217. 0x0129: 'STANDARDIZE',
  20218. 0x012A: 'ODD',
  20219. 0x012B: 'PERMUT',
  20220. 0x012C: 'POISSON',
  20221. 0x012D: 'TDIST',
  20222. 0x012E: 'WEIBULL',
  20223. 0x012F: 'SUMXMY2',
  20224. 0x0130: 'SUMX2MY2',
  20225. 0x0131: 'SUMX2PY2',
  20226. 0x0132: 'CHITEST',
  20227. 0x0133: 'CORREL',
  20228. 0x0134: 'COVAR',
  20229. 0x0135: 'FORECAST',
  20230. 0x0136: 'FTEST',
  20231. 0x0137: 'INTERCEPT',
  20232. 0x0138: 'PEARSON',
  20233. 0x0139: 'RSQ',
  20234. 0x013A: 'STEYX',
  20235. 0x013B: 'SLOPE',
  20236. 0x013C: 'TTEST',
  20237. 0x013D: 'PROB',
  20238. 0x013E: 'DEVSQ',
  20239. 0x013F: 'GEOMEAN',
  20240. 0x0140: 'HARMEAN',
  20241. 0x0141: 'SUMSQ',
  20242. 0x0142: 'KURT',
  20243. 0x0143: 'SKEW',
  20244. 0x0144: 'ZTEST',
  20245. 0x0145: 'LARGE',
  20246. 0x0146: 'SMALL',
  20247. 0x0147: 'QUARTILE',
  20248. 0x0148: 'PERCENTILE',
  20249. 0x0149: 'PERCENTRANK',
  20250. 0x014A: 'MODE',
  20251. 0x014B: 'TRIMMEAN',
  20252. 0x014C: 'TINV',
  20253. 0x014E: 'MOVIE.COMMAND',
  20254. 0x014F: 'GET.MOVIE',
  20255. 0x0150: 'CONCATENATE',
  20256. 0x0151: 'POWER',
  20257. 0x0152: 'PIVOT.ADD.DATA',
  20258. 0x0153: 'GET.PIVOT.TABLE',
  20259. 0x0154: 'GET.PIVOT.FIELD',
  20260. 0x0155: 'GET.PIVOT.ITEM',
  20261. 0x0156: 'RADIANS',
  20262. 0x0157: 'DEGREES',
  20263. 0x0158: 'SUBTOTAL',
  20264. 0x0159: 'SUMIF',
  20265. 0x015A: 'COUNTIF',
  20266. 0x015B: 'COUNTBLANK',
  20267. 0x015C: 'SCENARIO.GET',
  20268. 0x015D: 'OPTIONS.LISTS.GET',
  20269. 0x015E: 'ISPMT',
  20270. 0x015F: 'DATEDIF',
  20271. 0x0160: 'DATESTRING',
  20272. 0x0161: 'NUMBERSTRING',
  20273. 0x0162: 'ROMAN',
  20274. 0x0163: 'OPEN.DIALOG',
  20275. 0x0164: 'SAVE.DIALOG',
  20276. 0x0165: 'VIEW.GET',
  20277. 0x0166: 'GETPIVOTDATA',
  20278. 0x0167: 'HYPERLINK',
  20279. 0x0168: 'PHONETIC',
  20280. 0x0169: 'AVERAGEA',
  20281. 0x016A: 'MAXA',
  20282. 0x016B: 'MINA',
  20283. 0x016C: 'STDEVPA',
  20284. 0x016D: 'VARPA',
  20285. 0x016E: 'STDEVA',
  20286. 0x016F: 'VARA',
  20287. 0x0170: 'BAHTTEXT',
  20288. 0x0171: 'THAIDAYOFWEEK',
  20289. 0x0172: 'THAIDIGIT',
  20290. 0x0173: 'THAIMONTHOFYEAR',
  20291. 0x0174: 'THAINUMSOUND',
  20292. 0x0175: 'THAINUMSTRING',
  20293. 0x0176: 'THAISTRINGLENGTH',
  20294. 0x0177: 'ISTHAIDIGIT',
  20295. 0x0178: 'ROUNDBAHTDOWN',
  20296. 0x0179: 'ROUNDBAHTUP',
  20297. 0x017A: 'THAIYEAR',
  20298. 0x017B: 'RTD',
  20299. 0x017C: 'CUBEVALUE',
  20300. 0x017D: 'CUBEMEMBER',
  20301. 0x017E: 'CUBEMEMBERPROPERTY',
  20302. 0x017F: 'CUBERANKEDMEMBER',
  20303. 0x0180: 'HEX2BIN',
  20304. 0x0181: 'HEX2DEC',
  20305. 0x0182: 'HEX2OCT',
  20306. 0x0183: 'DEC2BIN',
  20307. 0x0184: 'DEC2HEX',
  20308. 0x0185: 'DEC2OCT',
  20309. 0x0186: 'OCT2BIN',
  20310. 0x0187: 'OCT2HEX',
  20311. 0x0188: 'OCT2DEC',
  20312. 0x0189: 'BIN2DEC',
  20313. 0x018A: 'BIN2OCT',
  20314. 0x018B: 'BIN2HEX',
  20315. 0x018C: 'IMSUB',
  20316. 0x018D: 'IMDIV',
  20317. 0x018E: 'IMPOWER',
  20318. 0x018F: 'IMABS',
  20319. 0x0190: 'IMSQRT',
  20320. 0x0191: 'IMLN',
  20321. 0x0192: 'IMLOG2',
  20322. 0x0193: 'IMLOG10',
  20323. 0x0194: 'IMSIN',
  20324. 0x0195: 'IMCOS',
  20325. 0x0196: 'IMEXP',
  20326. 0x0197: 'IMARGUMENT',
  20327. 0x0198: 'IMCONJUGATE',
  20328. 0x0199: 'IMAGINARY',
  20329. 0x019A: 'IMREAL',
  20330. 0x019B: 'COMPLEX',
  20331. 0x019C: 'IMSUM',
  20332. 0x019D: 'IMPRODUCT',
  20333. 0x019E: 'SERIESSUM',
  20334. 0x019F: 'FACTDOUBLE',
  20335. 0x01A0: 'SQRTPI',
  20336. 0x01A1: 'QUOTIENT',
  20337. 0x01A2: 'DELTA',
  20338. 0x01A3: 'GESTEP',
  20339. 0x01A4: 'ISEVEN',
  20340. 0x01A5: 'ISODD',
  20341. 0x01A6: 'MROUND',
  20342. 0x01A7: 'ERF',
  20343. 0x01A8: 'ERFC',
  20344. 0x01A9: 'BESSELJ',
  20345. 0x01AA: 'BESSELK',
  20346. 0x01AB: 'BESSELY',
  20347. 0x01AC: 'BESSELI',
  20348. 0x01AD: 'XIRR',
  20349. 0x01AE: 'XNPV',
  20350. 0x01AF: 'PRICEMAT',
  20351. 0x01B0: 'YIELDMAT',
  20352. 0x01B1: 'INTRATE',
  20353. 0x01B2: 'RECEIVED',
  20354. 0x01B3: 'DISC',
  20355. 0x01B4: 'PRICEDISC',
  20356. 0x01B5: 'YIELDDISC',
  20357. 0x01B6: 'TBILLEQ',
  20358. 0x01B7: 'TBILLPRICE',
  20359. 0x01B8: 'TBILLYIELD',
  20360. 0x01B9: 'PRICE',
  20361. 0x01BA: 'YIELD',
  20362. 0x01BB: 'DOLLARDE',
  20363. 0x01BC: 'DOLLARFR',
  20364. 0x01BD: 'NOMINAL',
  20365. 0x01BE: 'EFFECT',
  20366. 0x01BF: 'CUMPRINC',
  20367. 0x01C0: 'CUMIPMT',
  20368. 0x01C1: 'EDATE',
  20369. 0x01C2: 'EOMONTH',
  20370. 0x01C3: 'YEARFRAC',
  20371. 0x01C4: 'COUPDAYBS',
  20372. 0x01C5: 'COUPDAYS',
  20373. 0x01C6: 'COUPDAYSNC',
  20374. 0x01C7: 'COUPNCD',
  20375. 0x01C8: 'COUPNUM',
  20376. 0x01C9: 'COUPPCD',
  20377. 0x01CA: 'DURATION',
  20378. 0x01CB: 'MDURATION',
  20379. 0x01CC: 'ODDLPRICE',
  20380. 0x01CD: 'ODDLYIELD',
  20381. 0x01CE: 'ODDFPRICE',
  20382. 0x01CF: 'ODDFYIELD',
  20383. 0x01D0: 'RANDBETWEEN',
  20384. 0x01D1: 'WEEKNUM',
  20385. 0x01D2: 'AMORDEGRC',
  20386. 0x01D3: 'AMORLINC',
  20387. 0x01D4: 'CONVERT',
  20388. 0x02D4: 'SHEETJS',
  20389. 0x01D5: 'ACCRINT',
  20390. 0x01D6: 'ACCRINTM',
  20391. 0x01D7: 'WORKDAY',
  20392. 0x01D8: 'NETWORKDAYS',
  20393. 0x01D9: 'GCD',
  20394. 0x01DA: 'MULTINOMIAL',
  20395. 0x01DB: 'LCM',
  20396. 0x01DC: 'FVSCHEDULE',
  20397. 0x01DD: 'CUBEKPIMEMBER',
  20398. 0x01DE: 'CUBESET',
  20399. 0x01DF: 'CUBESETCOUNT',
  20400. 0x01E0: 'IFERROR',
  20401. 0x01E1: 'COUNTIFS',
  20402. 0x01E2: 'SUMIFS',
  20403. 0x01E3: 'AVERAGEIF',
  20404. 0x01E4: 'AVERAGEIFS'
  20405. };
  20406. var FtabArgc = {
  20407. 0x0002: 1, /* ISNA */
  20408. 0x0003: 1, /* ISERROR */
  20409. 0x000A: 0, /* NA */
  20410. 0x000F: 1, /* SIN */
  20411. 0x0010: 1, /* COS */
  20412. 0x0011: 1, /* TAN */
  20413. 0x0012: 1, /* ATAN */
  20414. 0x0013: 0, /* PI */
  20415. 0x0014: 1, /* SQRT */
  20416. 0x0015: 1, /* EXP */
  20417. 0x0016: 1, /* LN */
  20418. 0x0017: 1, /* LOG10 */
  20419. 0x0018: 1, /* ABS */
  20420. 0x0019: 1, /* INT */
  20421. 0x001A: 1, /* SIGN */
  20422. 0x001B: 2, /* ROUND */
  20423. 0x001E: 2, /* REPT */
  20424. 0x001F: 3, /* MID */
  20425. 0x0020: 1, /* LEN */
  20426. 0x0021: 1, /* VALUE */
  20427. 0x0022: 0, /* TRUE */
  20428. 0x0023: 0, /* FALSE */
  20429. 0x0026: 1, /* NOT */
  20430. 0x0027: 2, /* MOD */
  20431. 0x0028: 3, /* DCOUNT */
  20432. 0x0029: 3, /* DSUM */
  20433. 0x002A: 3, /* DAVERAGE */
  20434. 0x002B: 3, /* DMIN */
  20435. 0x002C: 3, /* DMAX */
  20436. 0x002D: 3, /* DSTDEV */
  20437. 0x002F: 3, /* DVAR */
  20438. 0x0030: 2, /* TEXT */
  20439. 0x0035: 1, /* GOTO */
  20440. 0x003D: 3, /* MIRR */
  20441. 0x003F: 0, /* RAND */
  20442. 0x0041: 3, /* DATE */
  20443. 0x0042: 3, /* TIME */
  20444. 0x0043: 1, /* DAY */
  20445. 0x0044: 1, /* MONTH */
  20446. 0x0045: 1, /* YEAR */
  20447. 0x0046: 1, /* WEEKDAY */
  20448. 0x0047: 1, /* HOUR */
  20449. 0x0048: 1, /* MINUTE */
  20450. 0x0049: 1, /* SECOND */
  20451. 0x004A: 0, /* NOW */
  20452. 0x004B: 1, /* AREAS */
  20453. 0x004C: 1, /* ROWS */
  20454. 0x004D: 1, /* COLUMNS */
  20455. 0x004F: 2, /* ABSREF */
  20456. 0x0050: 2, /* RELREF */
  20457. 0x0053: 1, /* TRANSPOSE */
  20458. 0x0055: 0, /* STEP */
  20459. 0x0056: 1, /* TYPE */
  20460. 0x0059: 0, /* CALLER */
  20461. 0x005A: 1, /* DEREF */
  20462. 0x005E: 0, /* ACTIVE.CELL */
  20463. 0x005F: 0, /* SELECTION */
  20464. 0x0061: 2, /* ATAN2 */
  20465. 0x0062: 1, /* ASIN */
  20466. 0x0063: 1, /* ACOS */
  20467. 0x0065: 3, /* HLOOKUP */
  20468. 0x0066: 3, /* VLOOKUP */
  20469. 0x0069: 1, /* ISREF */
  20470. 0x006A: 1, /* GET.FORMULA */
  20471. 0x006C: 2, /* SET.VALUE */
  20472. 0x006F: 1, /* CHAR */
  20473. 0x0070: 1, /* LOWER */
  20474. 0x0071: 1, /* UPPER */
  20475. 0x0072: 1, /* PROPER */
  20476. 0x0075: 2, /* EXACT */
  20477. 0x0076: 1, /* TRIM */
  20478. 0x0077: 4, /* REPLACE */
  20479. 0x0079: 1, /* CODE */
  20480. 0x007E: 1, /* ISERR */
  20481. 0x007F: 1, /* ISTEXT */
  20482. 0x0080: 1, /* ISNUMBER */
  20483. 0x0081: 1, /* ISBLANK */
  20484. 0x0082: 1, /* T */
  20485. 0x0083: 1, /* N */
  20486. 0x0085: 1, /* FCLOSE */
  20487. 0x0086: 1, /* FSIZE */
  20488. 0x0087: 1, /* FREADLN */
  20489. 0x0088: 2, /* FREAD */
  20490. 0x0089: 2, /* FWRITELN */
  20491. 0x008A: 2, /* FWRITE */
  20492. 0x008C: 1, /* DATEVALUE */
  20493. 0x008D: 1, /* TIMEVALUE */
  20494. 0x008E: 3, /* SLN */
  20495. 0x008F: 4, /* SYD */
  20496. 0x0090: 4, /* DDB */
  20497. 0x00A1: 1, /* DIALOG.BOX */
  20498. 0x00A2: 1, /* CLEAN */
  20499. 0x00A3: 1, /* MDETERM */
  20500. 0x00A4: 1, /* MINVERSE */
  20501. 0x00A5: 2, /* MMULT */
  20502. 0x00AC: 1, /* WHILE */
  20503. 0x00AF: 2, /* INITIATE */
  20504. 0x00B0: 2, /* REQUEST */
  20505. 0x00B1: 3, /* POKE */
  20506. 0x00B2: 2, /* EXECUTE */
  20507. 0x00B3: 1, /* TERMINATE */
  20508. 0x00B8: 1, /* FACT */
  20509. 0x00BA: 1, /* GET.WORKSPACE */
  20510. 0x00BD: 3, /* DPRODUCT */
  20511. 0x00BE: 1, /* ISNONTEXT */
  20512. 0x00C3: 3, /* DSTDEVP */
  20513. 0x00C4: 3, /* DVARP */
  20514. 0x00C5: 1, /* TRUNC */
  20515. 0x00C6: 1, /* ISLOGICAL */
  20516. 0x00C7: 3, /* DCOUNTA */
  20517. 0x00C9: 1, /* UNREGISTER */
  20518. 0x00CF: 4, /* REPLACEB */
  20519. 0x00D2: 3, /* MIDB */
  20520. 0x00D3: 1, /* LENB */
  20521. 0x00D4: 2, /* ROUNDUP */
  20522. 0x00D5: 2, /* ROUNDDOWN */
  20523. 0x00D6: 1, /* ASC */
  20524. 0x00D7: 1, /* DBCS */
  20525. 0x00E1: 0, /* END.IF */
  20526. 0x00E5: 1, /* SINH */
  20527. 0x00E6: 1, /* COSH */
  20528. 0x00E7: 1, /* TANH */
  20529. 0x00E8: 1, /* ASINH */
  20530. 0x00E9: 1, /* ACOSH */
  20531. 0x00EA: 1, /* ATANH */
  20532. 0x00EB: 3, /* DGET */
  20533. 0x00F4: 1, /* INFO */
  20534. 0x00F7: 4, /* DB */
  20535. 0x00FC: 2, /* FREQUENCY */
  20536. 0x0101: 1, /* EVALUATE */
  20537. 0x0105: 1, /* ERROR.TYPE */
  20538. 0x010F: 1, /* GAMMALN */
  20539. 0x0111: 4, /* BINOMDIST */
  20540. 0x0112: 2, /* CHIDIST */
  20541. 0x0113: 2, /* CHIINV */
  20542. 0x0114: 2, /* COMBIN */
  20543. 0x0115: 3, /* CONFIDENCE */
  20544. 0x0116: 3, /* CRITBINOM */
  20545. 0x0117: 1, /* EVEN */
  20546. 0x0118: 3, /* EXPONDIST */
  20547. 0x0119: 3, /* FDIST */
  20548. 0x011A: 3, /* FINV */
  20549. 0x011B: 1, /* FISHER */
  20550. 0x011C: 1, /* FISHERINV */
  20551. 0x011D: 2, /* FLOOR */
  20552. 0x011E: 4, /* GAMMADIST */
  20553. 0x011F: 3, /* GAMMAINV */
  20554. 0x0120: 2, /* CEILING */
  20555. 0x0121: 4, /* HYPGEOMDIST */
  20556. 0x0122: 3, /* LOGNORMDIST */
  20557. 0x0123: 3, /* LOGINV */
  20558. 0x0124: 3, /* NEGBINOMDIST */
  20559. 0x0125: 4, /* NORMDIST */
  20560. 0x0126: 1, /* NORMSDIST */
  20561. 0x0127: 3, /* NORMINV */
  20562. 0x0128: 1, /* NORMSINV */
  20563. 0x0129: 3, /* STANDARDIZE */
  20564. 0x012A: 1, /* ODD */
  20565. 0x012B: 2, /* PERMUT */
  20566. 0x012C: 3, /* POISSON */
  20567. 0x012D: 3, /* TDIST */
  20568. 0x012E: 4, /* WEIBULL */
  20569. 0x012F: 2, /* SUMXMY2 */
  20570. 0x0130: 2, /* SUMX2MY2 */
  20571. 0x0131: 2, /* SUMX2PY2 */
  20572. 0x0132: 2, /* CHITEST */
  20573. 0x0133: 2, /* CORREL */
  20574. 0x0134: 2, /* COVAR */
  20575. 0x0135: 3, /* FORECAST */
  20576. 0x0136: 2, /* FTEST */
  20577. 0x0137: 2, /* INTERCEPT */
  20578. 0x0138: 2, /* PEARSON */
  20579. 0x0139: 2, /* RSQ */
  20580. 0x013A: 2, /* STEYX */
  20581. 0x013B: 2, /* SLOPE */
  20582. 0x013C: 4, /* TTEST */
  20583. 0x0145: 2, /* LARGE */
  20584. 0x0146: 2, /* SMALL */
  20585. 0x0147: 2, /* QUARTILE */
  20586. 0x0148: 2, /* PERCENTILE */
  20587. 0x014B: 2, /* TRIMMEAN */
  20588. 0x014C: 2, /* TINV */
  20589. 0x0151: 2, /* POWER */
  20590. 0x0156: 1, /* RADIANS */
  20591. 0x0157: 1, /* DEGREES */
  20592. 0x015A: 2, /* COUNTIF */
  20593. 0x015B: 1, /* COUNTBLANK */
  20594. 0x015E: 4, /* ISPMT */
  20595. 0x015F: 3, /* DATEDIF */
  20596. 0x0160: 1, /* DATESTRING */
  20597. 0x0161: 2, /* NUMBERSTRING */
  20598. 0x0168: 1, /* PHONETIC */
  20599. 0x0170: 1, /* BAHTTEXT */
  20600. 0x0171: 1, /* THAIDAYOFWEEK */
  20601. 0x0172: 1, /* THAIDIGIT */
  20602. 0x0173: 1, /* THAIMONTHOFYEAR */
  20603. 0x0174: 1, /* THAINUMSOUND */
  20604. 0x0175: 1, /* THAINUMSTRING */
  20605. 0x0176: 1, /* THAISTRINGLENGTH */
  20606. 0x0177: 1, /* ISTHAIDIGIT */
  20607. 0x0178: 1, /* ROUNDBAHTDOWN */
  20608. 0x0179: 1, /* ROUNDBAHTUP */
  20609. 0x017A: 1, /* THAIYEAR */
  20610. 0x017E: 3, /* CUBEMEMBERPROPERTY */
  20611. 0x0181: 1, /* HEX2DEC */
  20612. 0x0188: 1, /* OCT2DEC */
  20613. 0x0189: 1, /* BIN2DEC */
  20614. 0x018C: 2, /* IMSUB */
  20615. 0x018D: 2, /* IMDIV */
  20616. 0x018E: 2, /* IMPOWER */
  20617. 0x018F: 1, /* IMABS */
  20618. 0x0190: 1, /* IMSQRT */
  20619. 0x0191: 1, /* IMLN */
  20620. 0x0192: 1, /* IMLOG2 */
  20621. 0x0193: 1, /* IMLOG10 */
  20622. 0x0194: 1, /* IMSIN */
  20623. 0x0195: 1, /* IMCOS */
  20624. 0x0196: 1, /* IMEXP */
  20625. 0x0197: 1, /* IMARGUMENT */
  20626. 0x0198: 1, /* IMCONJUGATE */
  20627. 0x0199: 1, /* IMAGINARY */
  20628. 0x019A: 1, /* IMREAL */
  20629. 0x019E: 4, /* SERIESSUM */
  20630. 0x019F: 1, /* FACTDOUBLE */
  20631. 0x01A0: 1, /* SQRTPI */
  20632. 0x01A1: 2, /* QUOTIENT */
  20633. 0x01A4: 1, /* ISEVEN */
  20634. 0x01A5: 1, /* ISODD */
  20635. 0x01A6: 2, /* MROUND */
  20636. 0x01A8: 1, /* ERFC */
  20637. 0x01A9: 2, /* BESSELJ */
  20638. 0x01AA: 2, /* BESSELK */
  20639. 0x01AB: 2, /* BESSELY */
  20640. 0x01AC: 2, /* BESSELI */
  20641. 0x01AE: 3, /* XNPV */
  20642. 0x01B6: 3, /* TBILLEQ */
  20643. 0x01B7: 3, /* TBILLPRICE */
  20644. 0x01B8: 3, /* TBILLYIELD */
  20645. 0x01BB: 2, /* DOLLARDE */
  20646. 0x01BC: 2, /* DOLLARFR */
  20647. 0x01BD: 2, /* NOMINAL */
  20648. 0x01BE: 2, /* EFFECT */
  20649. 0x01BF: 6, /* CUMPRINC */
  20650. 0x01C0: 6, /* CUMIPMT */
  20651. 0x01C1: 2, /* EDATE */
  20652. 0x01C2: 2, /* EOMONTH */
  20653. 0x01D0: 2, /* RANDBETWEEN */
  20654. 0x01D4: 3, /* CONVERT */
  20655. 0x01DC: 2, /* FVSCHEDULE */
  20656. 0x01DF: 1, /* CUBESETCOUNT */
  20657. 0x01E0: 2, /* IFERROR */
  20658. 0xFFFF: 0
  20659. };
  20660. /* [MS-XLSX] 2.2.3 Functions */
  20661. /* [MS-XLSB] 2.5.97.10 Ftab */
  20662. var XLSXFutureFunctions = {
  20663. "_xlfn.ACOT": "ACOT",
  20664. "_xlfn.ACOTH": "ACOTH",
  20665. "_xlfn.AGGREGATE": "AGGREGATE",
  20666. "_xlfn.ARABIC": "ARABIC",
  20667. "_xlfn.AVERAGEIF": "AVERAGEIF",
  20668. "_xlfn.AVERAGEIFS": "AVERAGEIFS",
  20669. "_xlfn.BASE": "BASE",
  20670. "_xlfn.BETA.DIST": "BETA.DIST",
  20671. "_xlfn.BETA.INV": "BETA.INV",
  20672. "_xlfn.BINOM.DIST": "BINOM.DIST",
  20673. "_xlfn.BINOM.DIST.RANGE": "BINOM.DIST.RANGE",
  20674. "_xlfn.BINOM.INV": "BINOM.INV",
  20675. "_xlfn.BITAND": "BITAND",
  20676. "_xlfn.BITLSHIFT": "BITLSHIFT",
  20677. "_xlfn.BITOR": "BITOR",
  20678. "_xlfn.BITRSHIFT": "BITRSHIFT",
  20679. "_xlfn.BITXOR": "BITXOR",
  20680. "_xlfn.CEILING.MATH": "CEILING.MATH",
  20681. "_xlfn.CEILING.PRECISE": "CEILING.PRECISE",
  20682. "_xlfn.CHISQ.DIST": "CHISQ.DIST",
  20683. "_xlfn.CHISQ.DIST.RT": "CHISQ.DIST.RT",
  20684. "_xlfn.CHISQ.INV": "CHISQ.INV",
  20685. "_xlfn.CHISQ.INV.RT": "CHISQ.INV.RT",
  20686. "_xlfn.CHISQ.TEST": "CHISQ.TEST",
  20687. "_xlfn.COMBINA": "COMBINA",
  20688. "_xlfn.CONCAT": "CONCAT",
  20689. "_xlfn.CONFIDENCE.NORM": "CONFIDENCE.NORM",
  20690. "_xlfn.CONFIDENCE.T": "CONFIDENCE.T",
  20691. "_xlfn.COT": "COT",
  20692. "_xlfn.COTH": "COTH",
  20693. "_xlfn.COUNTIFS": "COUNTIFS",
  20694. "_xlfn.COVARIANCE.P": "COVARIANCE.P",
  20695. "_xlfn.COVARIANCE.S": "COVARIANCE.S",
  20696. "_xlfn.CSC": "CSC",
  20697. "_xlfn.CSCH": "CSCH",
  20698. "_xlfn.DAYS": "DAYS",
  20699. "_xlfn.DECIMAL": "DECIMAL",
  20700. "_xlfn.ECMA.CEILING": "ECMA.CEILING",
  20701. "_xlfn.ERF.PRECISE": "ERF.PRECISE",
  20702. "_xlfn.ERFC.PRECISE": "ERFC.PRECISE",
  20703. "_xlfn.EXPON.DIST": "EXPON.DIST",
  20704. "_xlfn.F.DIST": "F.DIST",
  20705. "_xlfn.F.DIST.RT": "F.DIST.RT",
  20706. "_xlfn.F.INV": "F.INV",
  20707. "_xlfn.F.INV.RT": "F.INV.RT",
  20708. "_xlfn.F.TEST": "F.TEST",
  20709. "_xlfn.FILTERXML": "FILTERXML",
  20710. "_xlfn.FLOOR.MATH": "FLOOR.MATH",
  20711. "_xlfn.FLOOR.PRECISE": "FLOOR.PRECISE",
  20712. "_xlfn.FORECAST.ETS": "FORECAST.ETS",
  20713. "_xlfn.FORECAST.ETS.CONFINT": "FORECAST.ETS.CONFINT",
  20714. "_xlfn.FORECAST.ETS.SEASONALITY": "FORECAST.ETS.SEASONALITY",
  20715. "_xlfn.FORECAST.ETS.STAT": "FORECAST.ETS.STAT",
  20716. "_xlfn.FORECAST.LINEAR": "FORECAST.LINEAR",
  20717. "_xlfn.FORMULATEXT": "FORMULATEXT",
  20718. "_xlfn.GAMMA": "GAMMA",
  20719. "_xlfn.GAMMA.DIST": "GAMMA.DIST",
  20720. "_xlfn.GAMMA.INV": "GAMMA.INV",
  20721. "_xlfn.GAMMALN.PRECISE": "GAMMALN.PRECISE",
  20722. "_xlfn.GAUSS": "GAUSS",
  20723. "_xlfn.HYPGEOM.DIST": "HYPGEOM.DIST",
  20724. "_xlfn.IFERROR": "IFERROR",
  20725. "_xlfn.IFNA": "IFNA",
  20726. "_xlfn.IFS": "IFS",
  20727. "_xlfn.IMCOSH": "IMCOSH",
  20728. "_xlfn.IMCOT": "IMCOT",
  20729. "_xlfn.IMCSC": "IMCSC",
  20730. "_xlfn.IMCSCH": "IMCSCH",
  20731. "_xlfn.IMSEC": "IMSEC",
  20732. "_xlfn.IMSECH": "IMSECH",
  20733. "_xlfn.IMSINH": "IMSINH",
  20734. "_xlfn.IMTAN": "IMTAN",
  20735. "_xlfn.ISFORMULA": "ISFORMULA",
  20736. "_xlfn.ISO.CEILING": "ISO.CEILING",
  20737. "_xlfn.ISOWEEKNUM": "ISOWEEKNUM",
  20738. "_xlfn.LOGNORM.DIST": "LOGNORM.DIST",
  20739. "_xlfn.LOGNORM.INV": "LOGNORM.INV",
  20740. "_xlfn.MAXIFS": "MAXIFS",
  20741. "_xlfn.MINIFS": "MINIFS",
  20742. "_xlfn.MODE.MULT": "MODE.MULT",
  20743. "_xlfn.MODE.SNGL": "MODE.SNGL",
  20744. "_xlfn.MUNIT": "MUNIT",
  20745. "_xlfn.NEGBINOM.DIST": "NEGBINOM.DIST",
  20746. "_xlfn.NETWORKDAYS.INTL": "NETWORKDAYS.INTL",
  20747. "_xlfn.NIGBINOM": "NIGBINOM",
  20748. "_xlfn.NORM.DIST": "NORM.DIST",
  20749. "_xlfn.NORM.INV": "NORM.INV",
  20750. "_xlfn.NORM.S.DIST": "NORM.S.DIST",
  20751. "_xlfn.NORM.S.INV": "NORM.S.INV",
  20752. "_xlfn.NUMBERVALUE": "NUMBERVALUE",
  20753. "_xlfn.PDURATION": "PDURATION",
  20754. "_xlfn.PERCENTILE.EXC": "PERCENTILE.EXC",
  20755. "_xlfn.PERCENTILE.INC": "PERCENTILE.INC",
  20756. "_xlfn.PERCENTRANK.EXC": "PERCENTRANK.EXC",
  20757. "_xlfn.PERCENTRANK.INC": "PERCENTRANK.INC",
  20758. "_xlfn.PERMUTATIONA": "PERMUTATIONA",
  20759. "_xlfn.PHI": "PHI",
  20760. "_xlfn.POISSON.DIST": "POISSON.DIST",
  20761. "_xlfn.QUARTILE.EXC": "QUARTILE.EXC",
  20762. "_xlfn.QUARTILE.INC": "QUARTILE.INC",
  20763. "_xlfn.QUERYSTRING": "QUERYSTRING",
  20764. "_xlfn.RANK.AVG": "RANK.AVG",
  20765. "_xlfn.RANK.EQ": "RANK.EQ",
  20766. "_xlfn.RRI": "RRI",
  20767. "_xlfn.SEC": "SEC",
  20768. "_xlfn.SECH": "SECH",
  20769. "_xlfn.SHEET": "SHEET",
  20770. "_xlfn.SHEETS": "SHEETS",
  20771. "_xlfn.SKEW.P": "SKEW.P",
  20772. "_xlfn.STDEV.P": "STDEV.P",
  20773. "_xlfn.STDEV.S": "STDEV.S",
  20774. "_xlfn.SUMIFS": "SUMIFS",
  20775. "_xlfn.SWITCH": "SWITCH",
  20776. "_xlfn.T.DIST": "T.DIST",
  20777. "_xlfn.T.DIST.2T": "T.DIST.2T",
  20778. "_xlfn.T.DIST.RT": "T.DIST.RT",
  20779. "_xlfn.T.INV": "T.INV",
  20780. "_xlfn.T.INV.2T": "T.INV.2T",
  20781. "_xlfn.T.TEST": "T.TEST",
  20782. "_xlfn.TEXTJOIN": "TEXTJOIN",
  20783. "_xlfn.UNICHAR": "UNICHAR",
  20784. "_xlfn.UNICODE": "UNICODE",
  20785. "_xlfn.VAR.P": "VAR.P",
  20786. "_xlfn.VAR.S": "VAR.S",
  20787. "_xlfn.WEBSERVICE": "WEBSERVICE",
  20788. "_xlfn.WEIBULL.DIST": "WEIBULL.DIST",
  20789. "_xlfn.WORKDAY.INTL": "WORKDAY.INTL",
  20790. "_xlfn.XOR": "XOR",
  20791. "_xlfn.Z.TEST": "Z.TEST"
  20792. };
  20793. /* Part 3 TODO: actually parse formulae */
  20794. function ods_to_csf_formula(f) {
  20795. if(f.slice(0,3) == "of:") f = f.slice(3);
  20796. /* 5.2 Basic Expressions */
  20797. if(f.charCodeAt(0) == 61) {
  20798. f = f.slice(1);
  20799. if(f.charCodeAt(0) == 61) f = f.slice(1);
  20800. }
  20801. f = f.replace(/COM\.MICROSOFT\./g, "");
  20802. /* Part 3 Section 5.8 References */
  20803. f = f.replace(/\[((?:\.[A-Z]+[0-9]+)(?::\.[A-Z]+[0-9]+)?)\]/g, function($$, $1) { return $1.replace(/\./g,""); });
  20804. /* TODO: something other than this */
  20805. f = f.replace(/\[.(#[A-Z]*[?!])\]/g, "$1");
  20806. return f.replace(/[;~]/g,",").replace(/\|/g,";");
  20807. }
  20808. function csf_to_ods_formula(f) {
  20809. var o = "of:=" + f.replace(crefregex, "$1[.$2$3$4$5]").replace(/\]:\[/g,":");
  20810. /* TODO: something other than this */
  20811. return o.replace(/;/g, "|").replace(/,/g,";");
  20812. }
  20813. function ods_to_csf_3D(r) {
  20814. var a = r.split(":");
  20815. var s = a[0].split(".")[0];
  20816. return [s, a[0].split(".")[1] + (a.length > 1 ? (":" + (a[1].split(".")[1] || a[1].split(".")[0])) : "")];
  20817. }
  20818. function csf_to_ods_3D(r) {
  20819. return r.replace(/\./,"!");
  20820. }
  20821. var strs = {}; // shared strings
  20822. var _ssfopts = {}; // spreadsheet formatting options
  20823. RELS.WS = [
  20824. "http://schemas.openxmlformats.org/officeDocument/2006/relationships/worksheet",
  20825. "http://purl.oclc.org/ooxml/officeDocument/relationships/worksheet"
  20826. ];
  20827. /*global Map */
  20828. var browser_has_Map = typeof Map !== 'undefined';
  20829. function get_sst_id(sst, str, rev) {
  20830. var i = 0, len = sst.length;
  20831. if(rev) {
  20832. if(browser_has_Map ? rev.has(str) : rev.hasOwnProperty(str)) {
  20833. var revarr = browser_has_Map ? rev.get(str) : rev[str];
  20834. for(; i < revarr.length; ++i) {
  20835. if(sst[revarr[i]].t === str) { sst.Count ++; return revarr[i]; }
  20836. }
  20837. }
  20838. } else for(; i < len; ++i) {
  20839. if(sst[i].t === str) { sst.Count ++; return i; }
  20840. }
  20841. sst[len] = ({t:str}); sst.Count ++; sst.Unique ++;
  20842. if(rev) {
  20843. if(browser_has_Map) {
  20844. if(!rev.has(str)) rev.set(str, []);
  20845. rev.get(str).push(len);
  20846. } else {
  20847. if(!rev.hasOwnProperty(str)) rev[str] = [];
  20848. rev[str].push(len);
  20849. }
  20850. }
  20851. return len;
  20852. }
  20853. function col_obj_w(C, col) {
  20854. var p = ({min:C+1,max:C+1});
  20855. /* wch (chars), wpx (pixels) */
  20856. var wch = -1;
  20857. if(col.MDW) MDW = col.MDW;
  20858. if(col.width != null) p.customWidth = 1;
  20859. else if(col.wpx != null) wch = px2char(col.wpx);
  20860. else if(col.wch != null) wch = col.wch;
  20861. if(wch > -1) { p.width = char2width(wch); p.customWidth = 1; }
  20862. else if(col.width != null) p.width = col.width;
  20863. if(col.hidden) p.hidden = true;
  20864. return p;
  20865. }
  20866. function default_margins(margins, mode) {
  20867. if(!margins) return;
  20868. var defs = [0.7, 0.7, 0.75, 0.75, 0.3, 0.3];
  20869. if(mode == 'xlml') defs = [1, 1, 1, 1, 0.5, 0.5];
  20870. if(margins.left == null) margins.left = defs[0];
  20871. if(margins.right == null) margins.right = defs[1];
  20872. if(margins.top == null) margins.top = defs[2];
  20873. if(margins.bottom == null) margins.bottom = defs[3];
  20874. if(margins.header == null) margins.header = defs[4];
  20875. if(margins.footer == null) margins.footer = defs[5];
  20876. }
  20877. function get_cell_style(styles, cell, opts) {
  20878. if (typeof style_builder != 'undefined') {
  20879. if (/^\d+$/.exec(cell.s)) { return cell.s} // if its already an integer index, let it be
  20880. if (cell.s && (cell.s == +cell.s)) { return cell.s} // if its already an integer index, let it be
  20881. var s = cell.s || {};
  20882. if (cell.z) s.numFmt = cell.z;
  20883. return style_builder.addStyle(s);
  20884. }
  20885. else {
  20886. var z = opts.revssf[cell.z != null ? cell.z : "General"];
  20887. var i = 0x3c, len = styles.length;
  20888. if (z == null && opts.ssf) {
  20889. for (; i < 0x188; ++i) if (opts.ssf[i] == null) {
  20890. SSF.load(cell.z, i);
  20891. // $FlowIgnore
  20892. opts.ssf[i] = cell.z;
  20893. opts.revssf[cell.z] = z = i;
  20894. break;
  20895. }
  20896. }
  20897. for (i = 0; i != len; ++i) if (styles[i].numFmtId === z) return i;
  20898. styles[len] = {
  20899. numFmtId: z,
  20900. fontId: 0,
  20901. fillId: 0,
  20902. borderId: 0,
  20903. xfId: 0,
  20904. applyNumberFormat: 1
  20905. };
  20906. return len;
  20907. }
  20908. }
  20909. function safe_format(p, fmtid, fillid, opts, themes, styles) {
  20910. if(p.t === 'z') return;
  20911. if(p.t === 'd' && typeof p.v === 'string') p.v = parseDate(p.v);
  20912. try {
  20913. if(opts.cellNF) p.z = SSF._table[fmtid];
  20914. } catch(e) { if(opts.WTF) throw e; }
  20915. if(!opts || opts.cellText !== false) try {
  20916. if(SSF._table[fmtid] == null) SSF.load(SSFImplicit[fmtid] || "General", fmtid);
  20917. if(p.t === 'e') p.w = p.w || BErr[p.v];
  20918. else if(fmtid === 0) {
  20919. if(p.t === 'n') {
  20920. if((p.v|0) === p.v) p.w = SSF._general_int(p.v);
  20921. else p.w = SSF._general_num(p.v);
  20922. }
  20923. else if(p.t === 'd') {
  20924. var dd = datenum(p.v);
  20925. if((dd|0) === dd) p.w = SSF._general_int(dd);
  20926. else p.w = SSF._general_num(dd);
  20927. }
  20928. else if(p.v === undefined) return "";
  20929. else p.w = SSF._general(p.v,_ssfopts);
  20930. }
  20931. else if(p.t === 'd') p.w = SSF.format(fmtid,datenum(p.v),_ssfopts);
  20932. else p.w = SSF.format(fmtid,p.v,_ssfopts);
  20933. } catch(e) { if(opts.WTF) throw e; }
  20934. if(!opts.cellStyles) return;
  20935. if(fillid != null) try {
  20936. p.s = styles.Fills[fillid];
  20937. if (p.s.fgColor && p.s.fgColor.theme && !p.s.fgColor.rgb) {
  20938. p.s.fgColor.rgb = rgb_tint(themes.themeElements.clrScheme[p.s.fgColor.theme].rgb, p.s.fgColor.tint || 0);
  20939. if(opts.WTF) p.s.fgColor.raw_rgb = themes.themeElements.clrScheme[p.s.fgColor.theme].rgb;
  20940. }
  20941. if (p.s.bgColor && p.s.bgColor.theme) {
  20942. p.s.bgColor.rgb = rgb_tint(themes.themeElements.clrScheme[p.s.bgColor.theme].rgb, p.s.bgColor.tint || 0);
  20943. if(opts.WTF) p.s.bgColor.raw_rgb = themes.themeElements.clrScheme[p.s.bgColor.theme].rgb;
  20944. }
  20945. } catch(e) { if(opts.WTF && styles.Fills) throw e; }
  20946. }
  20947. function check_ws(ws, sname, i) {
  20948. if(ws && ws['!ref']) {
  20949. var range = safe_decode_range(ws['!ref']);
  20950. if(range.e.c < range.s.c || range.e.r < range.s.r) throw new Error("Bad range (" + i + "): " + ws['!ref']);
  20951. }
  20952. }
  20953. function parse_ws_xml_dim(ws, s) {
  20954. var d = safe_decode_range(s);
  20955. if(d.s.r<=d.e.r && d.s.c<=d.e.c && d.s.r>=0 && d.s.c>=0) ws["!ref"] = encode_range(d);
  20956. }
  20957. var mergecregex = /<(?:\w:)?mergeCell ref="[A-Z0-9:]+"\s*[\/]?>/g;
  20958. var sheetdataregex = /<(?:\w+:)?sheetData>([\s\S]*)<\/(?:\w+:)?sheetData>/;
  20959. var hlinkregex = /<(?:\w:)?hyperlink [^>]*>/mg;
  20960. var dimregex = /"(\w*:\w*)"/;
  20961. var colregex = /<(?:\w:)?col\b[^>]*[\/]?>/g;
  20962. var afregex = /<(?:\w:)?autoFilter[^>]*([\/]|>([\s\S]*)<\/(?:\w:)?autoFilter)>/g;
  20963. var marginregex= /<(?:\w:)?pageMargins[^>]*\/>/g;
  20964. var sheetprregex = /<(?:\w:)?sheetPr\b(?:[^>a-z][^>]*)?\/>/;
  20965. var svsregex = /<(?:\w:)?sheetViews[^>]*(?:[\/]|>([\s\S]*)<\/(?:\w:)?sheetViews)>/;
  20966. /* 18.3 Worksheets */
  20967. function parse_ws_xml(data, opts, idx, rels, wb, themes, styles) {
  20968. if(!data) return data;
  20969. if(DENSE != null && opts.dense == null) opts.dense = DENSE;
  20970. /* 18.3.1.99 worksheet CT_Worksheet */
  20971. var s = opts.dense ? ([]) : ({});
  20972. var refguess = ({s: {r:2000000, c:2000000}, e: {r:0, c:0} });
  20973. var data1 = "", data2 = "";
  20974. var mtch = data.match(sheetdataregex);
  20975. if(mtch) {
  20976. data1 = data.slice(0, mtch.index);
  20977. data2 = data.slice(mtch.index + mtch[0].length);
  20978. } else data1 = data2 = data;
  20979. /* 18.3.1.82 sheetPr CT_SheetPr */
  20980. var sheetPr = data1.match(sheetprregex);
  20981. if(sheetPr) parse_ws_xml_sheetpr(sheetPr[0], s, wb, idx);
  20982. /* 18.3.1.35 dimension CT_SheetDimension */
  20983. // $FlowIgnore
  20984. var ridx = (data1.match(/<(?:\w*:)?dimension/)||{index:-1}).index;
  20985. if(ridx > 0) {
  20986. var ref = data1.slice(ridx,ridx+50).match(dimregex);
  20987. if(ref) parse_ws_xml_dim(s, ref[1]);
  20988. }
  20989. /* 18.3.1.88 sheetViews CT_SheetViews */
  20990. var svs = data1.match(svsregex);
  20991. if(svs && svs[1]) parse_ws_xml_sheetviews(svs[1], wb);
  20992. /* 18.3.1.17 cols CT_Cols */
  20993. var columns = [];
  20994. if(opts.cellStyles) {
  20995. /* 18.3.1.13 col CT_Col */
  20996. var cols = data1.match(colregex);
  20997. if(cols) parse_ws_xml_cols(columns, cols);
  20998. }
  20999. /* 18.3.1.80 sheetData CT_SheetData ? */
  21000. if(mtch) parse_ws_xml_data(mtch[1], s, opts, refguess, themes, styles);
  21001. /* 18.3.1.2 autoFilter CT_AutoFilter */
  21002. var afilter = data2.match(afregex);
  21003. if(afilter) s['!autofilter'] = parse_ws_xml_autofilter(afilter[0]);
  21004. /* 18.3.1.55 mergeCells CT_MergeCells */
  21005. var merges = [];
  21006. var _merge = data2.match(mergecregex);
  21007. if(_merge) for(ridx = 0; ridx != _merge.length; ++ridx)
  21008. merges[ridx] = safe_decode_range(_merge[ridx].slice(_merge[ridx].indexOf("\"")+1));
  21009. /* 18.3.1.48 hyperlinks CT_Hyperlinks */
  21010. var hlink = data2.match(hlinkregex);
  21011. if(hlink) parse_ws_xml_hlinks(s, hlink, rels);
  21012. /* 18.3.1.62 pageMargins CT_PageMargins */
  21013. var margins = data2.match(marginregex);
  21014. if(margins) s['!margins'] = parse_ws_xml_margins(parsexmltag(margins[0]));
  21015. if(!s["!ref"] && refguess.e.c >= refguess.s.c && refguess.e.r >= refguess.s.r) s["!ref"] = encode_range(refguess);
  21016. if(opts.sheetRows > 0 && s["!ref"]) {
  21017. var tmpref = safe_decode_range(s["!ref"]);
  21018. if(opts.sheetRows <= +tmpref.e.r) {
  21019. tmpref.e.r = opts.sheetRows - 1;
  21020. if(tmpref.e.r > refguess.e.r) tmpref.e.r = refguess.e.r;
  21021. if(tmpref.e.r < tmpref.s.r) tmpref.s.r = tmpref.e.r;
  21022. if(tmpref.e.c > refguess.e.c) tmpref.e.c = refguess.e.c;
  21023. if(tmpref.e.c < tmpref.s.c) tmpref.s.c = tmpref.e.c;
  21024. s["!fullref"] = s["!ref"];
  21025. s["!ref"] = encode_range(tmpref);
  21026. }
  21027. }
  21028. if(columns.length > 0) s["!cols"] = columns;
  21029. if(merges.length > 0) s["!merges"] = merges;
  21030. return s;
  21031. }
  21032. function write_ws_xml_merges(merges) {
  21033. if(merges.length === 0) return "";
  21034. var o = '<mergeCells count="' + merges.length + '">';
  21035. for(var i = 0; i != merges.length; ++i) o += '<mergeCell ref="' + encode_range(merges[i]) + '"/>';
  21036. return o + '</mergeCells>';
  21037. }
  21038. /* 18.3.1.82-3 sheetPr CT_ChartsheetPr / CT_SheetPr */
  21039. function parse_ws_xml_sheetpr(sheetPr, s, wb, idx) {
  21040. var data = parsexmltag(sheetPr);
  21041. if(!wb.Sheets[idx]) wb.Sheets[idx] = {};
  21042. if(data.codeName) wb.Sheets[idx].CodeName = data.codeName;
  21043. }
  21044. /* 18.3.1.85 sheetProtection CT_SheetProtection */
  21045. function write_ws_xml_protection(sp) {
  21046. // algorithmName, hashValue, saltValue, spinCountpassword
  21047. var o = ({sheet:1});
  21048. var deffalse = ["objects", "scenarios", "selectLockedCells", "selectUnlockedCells"];
  21049. var deftrue = [
  21050. "formatColumns", "formatRows", "formatCells",
  21051. "insertColumns", "insertRows", "insertHyperlinks",
  21052. "deleteColumns", "deleteRows",
  21053. "sort", "autoFilter", "pivotTables"
  21054. ];
  21055. deffalse.forEach(function(n) { if(sp[n] != null && sp[n]) o[n] = "1"; });
  21056. deftrue.forEach(function(n) { if(sp[n] != null && !sp[n]) o[n] = "0"; });
  21057. /* TODO: algorithm */
  21058. if(sp.password) o.password = crypto_CreatePasswordVerifier_Method1(sp.password).toString(16).toUpperCase();
  21059. return writextag('sheetProtection', null, o);
  21060. }
  21061. function parse_ws_xml_hlinks(s, data, rels) {
  21062. var dense = Array.isArray(s);
  21063. for(var i = 0; i != data.length; ++i) {
  21064. var val = parsexmltag(utf8read(data[i]), true);
  21065. if(!val.ref) return;
  21066. var rel = ((rels || {})['!id']||[])[val.id];
  21067. if(rel) {
  21068. val.Target = rel.Target;
  21069. if(val.location) val.Target += "#"+val.location;
  21070. } else {
  21071. val.Target = "#" + val.location;
  21072. rel = {Target: val.Target, TargetMode: 'Internal'};
  21073. }
  21074. val.Rel = rel;
  21075. if(val.tooltip) { val.Tooltip = val.tooltip; delete val.tooltip; }
  21076. var rng = safe_decode_range(val.ref);
  21077. for(var R=rng.s.r;R<=rng.e.r;++R) for(var C=rng.s.c;C<=rng.e.c;++C) {
  21078. var addr = encode_cell({c:C,r:R});
  21079. if(dense) {
  21080. if(!s[R]) s[R] = [];
  21081. if(!s[R][C]) s[R][C] = {t:"z",v:undefined};
  21082. s[R][C].l = val;
  21083. } else {
  21084. if(!s[addr]) s[addr] = {t:"z",v:undefined};
  21085. s[addr].l = val;
  21086. }
  21087. }
  21088. }
  21089. }
  21090. function parse_ws_xml_margins(margin) {
  21091. var o = {};
  21092. ["left", "right", "top", "bottom", "header", "footer"].forEach(function(k) {
  21093. if(margin[k]) o[k] = parseFloat(margin[k]);
  21094. });
  21095. return o;
  21096. }
  21097. function write_ws_xml_margins(margin) {
  21098. default_margins(margin);
  21099. return writextag('pageMargins', null, margin);
  21100. }
  21101. function parse_ws_xml_cols(columns, cols) {
  21102. var seencol = false;
  21103. for(var coli = 0; coli != cols.length; ++coli) {
  21104. var coll = parsexmltag(cols[coli], true);
  21105. if(coll.hidden) coll.hidden = parsexmlbool(coll.hidden);
  21106. var colm=parseInt(coll.min, 10)-1, colM=parseInt(coll.max,10)-1;
  21107. delete coll.min; delete coll.max; coll.width = +coll.width;
  21108. if(!seencol && coll.width) { seencol = true; find_mdw_colw(coll.width); }
  21109. process_col(coll);
  21110. while(colm <= colM) columns[colm++] = dup(coll);
  21111. }
  21112. }
  21113. function write_ws_xml_cols(ws, cols) {
  21114. var o = ["<cols>"], col;
  21115. for(var i = 0; i != cols.length; ++i) {
  21116. if(!(col = cols[i])) continue;
  21117. o[o.length] = (writextag('col', null, col_obj_w(i, col)));
  21118. }
  21119. o[o.length] = "</cols>";
  21120. return o.join("");
  21121. }
  21122. function parse_ws_xml_autofilter(data) {
  21123. var o = { ref: (data.match(/ref="([^"]*)"/)||[])[1]};
  21124. return o;
  21125. }
  21126. function write_ws_xml_autofilter(data, ws, wb, idx) {
  21127. var ref = typeof data.ref == "string" ? data.ref : encode_range(data.ref);
  21128. if(!wb.Workbook) wb.Workbook = {};
  21129. if(!wb.Workbook.Names) wb.Workbook.Names = [];
  21130. var names = wb.Workbook.Names;
  21131. var range = decode_range(ref);
  21132. if(range.s.r == range.e.r) { range.e.r = decode_range(ws["!ref"]).e.r; ref = encode_range(range); }
  21133. for(var i = 0; i < names.length; ++i) {
  21134. var name = names[i];
  21135. if(name.Name != '_xlnm._FilterDatabase') continue;
  21136. if(name.Sheet != idx) continue;
  21137. name.Ref = "'" + wb.SheetNames[idx] + "'!" + ref; break;
  21138. }
  21139. if(i == names.length) names.push({ Name: '_xlnm._FilterDatabase', Sheet: idx, Ref: "'" + wb.SheetNames[idx] + "'!" + ref });
  21140. return writextag("autoFilter", null, {ref:ref});
  21141. }
  21142. /* 18.3.1.88 sheetViews CT_SheetViews */
  21143. /* 18.3.1.87 sheetView CT_SheetView */
  21144. var sviewregex = /<(?:\w:)?sheetView(?:[^>a-z][^>]*)?\/>/;
  21145. function parse_ws_xml_sheetviews(data, wb) {
  21146. (data.match(sviewregex)||[]).forEach(function(r) {
  21147. var tag = parsexmltag(r);
  21148. if(parsexmlbool(tag.rightToLeft)) {
  21149. if(!wb.Views) wb.Views = [{}];
  21150. if(!wb.Views[0]) wb.Views[0] = {};
  21151. wb.Views[0].RTL = true;
  21152. }
  21153. });
  21154. }
  21155. function write_ws_xml_sheetviews(ws, opts, idx, wb) {
  21156. var sview = {workbookViewId:"0"};
  21157. // $FlowIgnore
  21158. if( (((wb||{}).Workbook||{}).Views||[])[0] ) sview.rightToLeft = wb.Workbook.Views[0].RTL ? "1" : "0";
  21159. return writextag("sheetViews", writextag("sheetView", null, sview), {});
  21160. }
  21161. function write_ws_xml_cell(cell, ref, ws, opts) {
  21162. if(cell.v === undefined && cell.f === undefined || cell.t === 'z') return "";
  21163. var vv = "";
  21164. var oldt = cell.t, oldv = cell.v;
  21165. switch(cell.t) {
  21166. case 'b': vv = cell.v ? "1" : "0"; break;
  21167. case 'n': vv = ''+cell.v; break;
  21168. case 'e': vv = BErr[cell.v]; break;
  21169. case 'd':
  21170. if(opts.cellDates) vv = parseDate(cell.v, -1).toISOString();
  21171. else {
  21172. cell = dup(cell);
  21173. cell.t = 'n';
  21174. vv = ''+(cell.v = datenum(parseDate(cell.v)));
  21175. }
  21176. if(typeof cell.z === 'undefined') cell.z = SSF._table[14];
  21177. break;
  21178. default: vv = cell.v; break;
  21179. }
  21180. var v = writetag('v', escapexml(vv)), o = ({r:ref});
  21181. /* TODO: cell style */
  21182. var os = get_cell_style(opts.cellXfs, cell, opts);
  21183. if(os !== 0) o.s = os;
  21184. switch(cell.t) {
  21185. case 'n': break;
  21186. case 'd': o.t = "d"; break;
  21187. case 'b': o.t = "b"; break;
  21188. case 'e': o.t = "e"; break;
  21189. default: if(cell.v == null) { delete cell.t; break; }
  21190. if(opts.bookSST) {
  21191. v = writetag('v', ''+get_sst_id(opts.Strings, cell.v, opts.revStrings));
  21192. o.t = "s"; break;
  21193. }
  21194. o.t = "str"; break;
  21195. }
  21196. if(cell.t != oldt) { cell.t = oldt; cell.v = oldv; }
  21197. if(cell.f) {
  21198. var ff = cell.F && cell.F.slice(0, ref.length) == ref ? {t:"array", ref:cell.F} : null;
  21199. v = writextag('f', escapexml(cell.f), ff) + (cell.v != null ? v : "");
  21200. }
  21201. if(cell.l) ws['!links'].push([ref, cell.l]);
  21202. if(cell.c) ws['!comments'].push([ref, cell.c]);
  21203. return writextag('c', v, o);
  21204. }
  21205. var parse_ws_xml_data = (function() {
  21206. var cellregex = /<(?:\w+:)?c[ >]/, rowregex = /<\/(?:\w+:)?row>/;
  21207. var rregex = /r=["']([^"']*)["']/, isregex = /<(?:\w+:)?is>([\S\s]*?)<\/(?:\w+:)?is>/;
  21208. var refregex = /ref=["']([^"']*)["']/;
  21209. var match_v = matchtag("v"), match_f = matchtag("f");
  21210. return function parse_ws_xml_data(sdata, s, opts, guess, themes, styles) {
  21211. var ri = 0, x = "", cells = [], cref = [], idx=0, i=0, cc=0, d="", p;
  21212. var tag, tagr = 0, tagc = 0;
  21213. var sstr, ftag;
  21214. var fmtid = 0, fillid = 0;
  21215. var do_format = Array.isArray(styles.CellXf), cf;
  21216. var arrayf = [];
  21217. var sharedf = [];
  21218. var dense = Array.isArray(s);
  21219. var rows = [], rowobj = {}, rowrite = false;
  21220. for(var marr = sdata.split(rowregex), mt = 0, marrlen = marr.length; mt != marrlen; ++mt) {
  21221. x = marr[mt].trim();
  21222. var xlen = x.length;
  21223. if(xlen === 0) continue;
  21224. /* 18.3.1.73 row CT_Row */
  21225. for(ri = 0; ri < xlen; ++ri) if(x.charCodeAt(ri) === 62) break; ++ri;
  21226. tag = parsexmltag(x.slice(0,ri), true);
  21227. tagr = tag.r != null ? parseInt(tag.r, 10) : tagr+1; tagc = -1;
  21228. if(opts.sheetRows && opts.sheetRows < tagr) continue;
  21229. if(guess.s.r > tagr - 1) guess.s.r = tagr - 1;
  21230. if(guess.e.r < tagr - 1) guess.e.r = tagr - 1;
  21231. if(opts && opts.cellStyles) {
  21232. rowobj = {}; rowrite = false;
  21233. if(tag.ht) { rowrite = true; rowobj.hpt = parseFloat(tag.ht); rowobj.hpx = pt2px(rowobj.hpt); }
  21234. if(tag.hidden == "1") { rowrite = true; rowobj.hidden = true; }
  21235. if(tag.outlineLevel != null) { rowrite = true; rowobj.level = +tag.outlineLevel; }
  21236. if(rowrite) rows[tagr-1] = rowobj;
  21237. }
  21238. /* 18.3.1.4 c CT_Cell */
  21239. cells = x.slice(ri).split(cellregex);
  21240. for(ri = 0; ri != cells.length; ++ri) {
  21241. x = cells[ri].trim();
  21242. if(x.length === 0) continue;
  21243. cref = x.match(rregex); idx = ri; i=0; cc=0;
  21244. x = "<c " + (x.slice(0,1)=="<"?">":"") + x;
  21245. if(cref != null && cref.length === 2) {
  21246. idx = 0; d=cref[1];
  21247. for(i=0; i != d.length; ++i) {
  21248. if((cc=d.charCodeAt(i)-64) < 1 || cc > 26) break;
  21249. idx = 26*idx + cc;
  21250. }
  21251. --idx;
  21252. tagc = idx;
  21253. } else ++tagc;
  21254. for(i = 0; i != x.length; ++i) if(x.charCodeAt(i) === 62) break; ++i;
  21255. tag = parsexmltag(x.slice(0,i), true);
  21256. if(!tag.r) tag.r = encode_cell({r:tagr-1, c:tagc});
  21257. d = x.slice(i);
  21258. p = ({t:""});
  21259. if((cref=d.match(match_v))!= null && cref[1] !== '') p.v=unescapexml(cref[1]);
  21260. if(opts.cellFormula) {
  21261. if((cref=d.match(match_f))!= null && cref[1] !== '') {
  21262. /* TODO: match against XLSXFutureFunctions */
  21263. p.f=_xlfn(unescapexml(utf8read(cref[1])));
  21264. if(cref[0].indexOf('t="array"') > -1) {
  21265. p.F = (d.match(refregex)||[])[1];
  21266. if(p.F.indexOf(":") > -1) arrayf.push([safe_decode_range(p.F), p.F]);
  21267. } else if(cref[0].indexOf('t="shared"') > -1) {
  21268. // TODO: parse formula
  21269. ftag = parsexmltag(cref[0]);
  21270. sharedf[parseInt(ftag.si, 10)] = [ftag, _xlfn(unescapexml(utf8read(cref[1]))), tag.r];
  21271. }
  21272. } else if((cref=d.match(/<f[^>]*\/>/))) {
  21273. ftag = parsexmltag(cref[0]);
  21274. if(sharedf[ftag.si]) p.f = shift_formula_xlsx(sharedf[ftag.si][1], sharedf[ftag.si][2]/*[0].ref*/, tag.r);
  21275. }
  21276. /* TODO: factor out contains logic */
  21277. var _tag = decode_cell(tag.r);
  21278. for(i = 0; i < arrayf.length; ++i)
  21279. if(_tag.r >= arrayf[i][0].s.r && _tag.r <= arrayf[i][0].e.r)
  21280. if(_tag.c >= arrayf[i][0].s.c && _tag.c <= arrayf[i][0].e.c)
  21281. p.F = arrayf[i][1];
  21282. }
  21283. if(tag.t == null && p.v === undefined) {
  21284. if(p.f || p.F) {
  21285. p.v = 0; p.t = "n";
  21286. } else if(!opts.sheetStubs) continue;
  21287. else p.t = "z";
  21288. }
  21289. else p.t = tag.t || "n";
  21290. if(guess.s.c > tagc) guess.s.c = tagc;
  21291. if(guess.e.c < tagc) guess.e.c = tagc;
  21292. /* 18.18.11 t ST_CellType */
  21293. switch(p.t) {
  21294. case 'n':
  21295. if(p.v == "" || p.v == null) {
  21296. if(!opts.sheetStubs) continue;
  21297. p.t = 'z';
  21298. } else p.v = parseFloat(p.v);
  21299. break;
  21300. case 's':
  21301. if(typeof p.v == 'undefined') {
  21302. if(!opts.sheetStubs) continue;
  21303. p.t = 'z';
  21304. } else {
  21305. sstr = strs[parseInt(p.v, 10)];
  21306. p.v = sstr.t;
  21307. p.r = sstr.r;
  21308. if(opts.cellHTML) p.h = sstr.h;
  21309. }
  21310. break;
  21311. case 'str':
  21312. p.t = "s";
  21313. p.v = (p.v!=null) ? utf8read(p.v) : '';
  21314. if(opts.cellHTML) p.h = escapehtml(p.v);
  21315. break;
  21316. case 'inlineStr':
  21317. cref = d.match(isregex);
  21318. p.t = 's';
  21319. if(cref != null && (sstr = parse_si(cref[1]))) p.v = sstr.t; else p.v = "";
  21320. break;
  21321. case 'b': p.v = parsexmlbool(p.v); break;
  21322. case 'd':
  21323. if(opts.cellDates) p.v = parseDate(p.v, 1);
  21324. else { p.v = datenum(parseDate(p.v, 1)); p.t = 'n'; }
  21325. break;
  21326. /* error string in .w, number in .v */
  21327. case 'e':
  21328. if(!opts || opts.cellText !== false) p.w = p.v;
  21329. p.v = RBErr[p.v]; break;
  21330. }
  21331. /* formatting */
  21332. fmtid = fillid = 0;
  21333. if(do_format && tag.s !== undefined) {
  21334. cf = styles.CellXf[tag.s];
  21335. if(cf != null) {
  21336. if(cf.numFmtId != null) fmtid = cf.numFmtId;
  21337. if(opts.cellStyles) {
  21338. if(cf.fillId != null) fillid = cf.fillId;
  21339. }
  21340. }
  21341. }
  21342. safe_format(p, fmtid, fillid, opts, themes, styles);
  21343. if(opts.cellDates && do_format && p.t == 'n' && SSF.is_date(SSF._table[fmtid])) { p.t = 'd'; p.v = numdate(p.v); }
  21344. if(dense) {
  21345. var _r = decode_cell(tag.r);
  21346. if(!s[_r.r]) s[_r.r] = [];
  21347. s[_r.r][_r.c] = p;
  21348. } else s[tag.r] = p;
  21349. }
  21350. }
  21351. if(rows.length > 0) s['!rows'] = rows;
  21352. }; })();
  21353. function write_ws_xml_data(ws, opts, idx, wb) {
  21354. var o = [], r = [], range = safe_decode_range(ws['!ref']), cell="", ref, rr = "", cols = [], R=0, C=0, rows = ws['!rows'];
  21355. var dense = Array.isArray(ws);
  21356. var params = ({r:rr}), row, height = -1;
  21357. for(C = range.s.c; C <= range.e.c; ++C) cols[C] = encode_col(C);
  21358. for(R = range.s.r; R <= range.e.r; ++R) {
  21359. r = [];
  21360. rr = encode_row(R);
  21361. for(C = range.s.c; C <= range.e.c; ++C) {
  21362. ref = cols[C] + rr;
  21363. var _cell = dense ? (ws[R]||[])[C]: ws[ref];
  21364. if(_cell === undefined) continue;
  21365. if((cell = write_ws_xml_cell(_cell, ref, ws, opts, idx, wb)) != null) r.push(cell);
  21366. }
  21367. if(r.length > 0 || (rows && rows[R])) {
  21368. params = ({r:rr});
  21369. if(rows && rows[R]) {
  21370. row = rows[R];
  21371. if(row.hidden) params.hidden = 1;
  21372. height = -1;
  21373. if(row.hpx) height = px2pt(row.hpx);
  21374. else if(row.hpt) height = row.hpt;
  21375. if(height > -1) { params.ht = height; params.customHeight = 1; }
  21376. if(row.level) { params.outlineLevel = row.level; }
  21377. }
  21378. o[o.length] = (writextag('row', r.join(""), params));
  21379. }
  21380. }
  21381. if(rows) for(; R < rows.length; ++R) {
  21382. if(rows && rows[R]) {
  21383. params = ({r:R+1});
  21384. row = rows[R];
  21385. if(row.hidden) params.hidden = 1;
  21386. height = -1;
  21387. if (row.hpx) height = px2pt(row.hpx);
  21388. else if (row.hpt) height = row.hpt;
  21389. if (height > -1) { params.ht = height; params.customHeight = 1; }
  21390. if (row.level) { params.outlineLevel = row.level; }
  21391. o[o.length] = (writextag('row', "", params));
  21392. }
  21393. }
  21394. return o.join("");
  21395. }
  21396. var WS_XML_ROOT = writextag('worksheet', null, {
  21397. 'xmlns': XMLNS.main[0],
  21398. 'xmlns:r': XMLNS.r
  21399. });
  21400. function write_ws_xml(idx, opts, wb, rels) {
  21401. var o = [XML_HEADER, WS_XML_ROOT];
  21402. var s = wb.SheetNames[idx], sidx = 0, rdata = "";
  21403. var ws = wb.Sheets[s];
  21404. if(ws == null) ws = {};
  21405. var ref = ws['!ref'] || 'A1';
  21406. var range = safe_decode_range(ref);
  21407. if(range.e.c > 0x3FFF || range.e.r > 0xFFFFF) {
  21408. if(opts.WTF) throw new Error("Range " + ref + " exceeds format limit A1:XFD1048576");
  21409. range.e.c = Math.min(range.e.c, 0x3FFF);
  21410. range.e.r = Math.min(range.e.c, 0xFFFFF);
  21411. ref = encode_range(range);
  21412. }
  21413. if(!rels) rels = {};
  21414. ws['!comments'] = [];
  21415. ws['!drawing'] = [];
  21416. if(opts.bookType !== 'xlsx' && wb.vbaraw) {
  21417. var cname = wb.SheetNames[idx];
  21418. try { if(wb.Workbook) cname = wb.Workbook.Sheets[idx].CodeName || cname; } catch(e) {}
  21419. o[o.length] = (writextag('sheetPr', null, {'codeName': escapexml(cname)}));
  21420. }
  21421. o[o.length] = (writextag('dimension', null, {'ref': ref}));
  21422. o[o.length] = write_ws_xml_sheetviews(ws, opts, idx, wb);
  21423. /* TODO: store in WB, process styles */
  21424. if(opts.sheetFormat) o[o.length] = (writextag('sheetFormatPr', null, {
  21425. defaultRowHeight:opts.sheetFormat.defaultRowHeight||'16',
  21426. baseColWidth:opts.sheetFormat.baseColWidth||'10',
  21427. outlineLevelRow:opts.sheetFormat.outlineLevelRow||'7'
  21428. }));
  21429. if(ws['!cols'] != null && ws['!cols'].length > 0) o[o.length] = (write_ws_xml_cols(ws, ws['!cols']));
  21430. o[sidx = o.length] = '<sheetData/>';
  21431. ws['!links'] = [];
  21432. if(ws['!ref'] != null) {
  21433. rdata = write_ws_xml_data(ws, opts, idx, wb, rels);
  21434. if(rdata.length > 0) o[o.length] = (rdata);
  21435. }
  21436. if(o.length>sidx+1) { o[o.length] = ('</sheetData>'); o[sidx]=o[sidx].replace("/>",">"); }
  21437. /* sheetCalcPr */
  21438. if(ws['!protect'] != null) o[o.length] = write_ws_xml_protection(ws['!protect']);
  21439. /* protectedRanges */
  21440. /* scenarios */
  21441. if(ws['!autofilter'] != null) o[o.length] = write_ws_xml_autofilter(ws['!autofilter'], ws, wb, idx);
  21442. /* sortState */
  21443. /* dataConsolidate */
  21444. /* customSheetViews */
  21445. if(ws['!merges'] != null && ws['!merges'].length > 0) o[o.length] = (write_ws_xml_merges(ws['!merges']));
  21446. /* phoneticPr */
  21447. /* conditionalFormatting */
  21448. /* dataValidations */
  21449. var relc = -1, rel, rId = -1;
  21450. if(ws['!links'].length > 0) {
  21451. o[o.length] = "<hyperlinks>";
  21452. ws['!links'].forEach(function(l) {
  21453. if(!l[1].Target) return;
  21454. rel = ({"ref":l[0]});
  21455. if(l[1].Target.charAt(0) != "#") {
  21456. rId = add_rels(rels, -1, escapexml(l[1].Target).replace(/#.*$/, ""), RELS.HLINK);
  21457. rel["r:id"] = "rId"+rId;
  21458. }
  21459. if((relc = l[1].Target.indexOf("#")) > -1) rel.location = escapexml(l[1].Target.slice(relc+1));
  21460. if(l[1].Tooltip) rel.tooltip = escapexml(l[1].Tooltip);
  21461. o[o.length] = writextag("hyperlink",null,rel);
  21462. });
  21463. o[o.length] = "</hyperlinks>";
  21464. }
  21465. delete ws['!links'];
  21466. /* printOptions */
  21467. if (ws['!margins'] != null) o[o.length] = write_ws_xml_margins(ws['!margins']);
  21468. /* pageSetup */
  21469. //var hfidx = o.length;
  21470. o[o.length] = "";
  21471. /* rowBreaks */
  21472. /* colBreaks */
  21473. /* customProperties */
  21474. /* cellWatches */
  21475. if(!opts || opts.ignoreEC || (opts.ignoreEC == (void 0))) o[o.length] = writetag("ignoredErrors", writextag("ignoredError", null, {numberStoredAsText:1, sqref:ref}));
  21476. /* smartTags */
  21477. if(ws['!drawing'].length > 0) {
  21478. rId = add_rels(rels, -1, "../drawings/drawing" + (idx+1) + ".xml", RELS.DRAW);
  21479. o[o.length] = writextag("drawing", null, {"r:id":"rId" + rId});
  21480. }
  21481. else delete ws['!drawing'];
  21482. if(ws['!comments'].length > 0) {
  21483. rId = add_rels(rels, -1, "../drawings/vmlDrawing" + (idx+1) + ".vml", RELS.VML);
  21484. o[o.length] = writextag("legacyDrawing", null, {"r:id":"rId" + rId});
  21485. ws['!legacy'] = rId;
  21486. }
  21487. /* drawingHF */
  21488. /* picture */
  21489. /* oleObjects */
  21490. /* controls */
  21491. /* webPublishItems */
  21492. /* tableParts */
  21493. /* extList */
  21494. if(o.length>2) { o[o.length] = ('</worksheet>'); o[1]=o[1].replace("/>",">"); }
  21495. return o.join("");
  21496. }
  21497. /* [MS-XLSB] 2.4.726 BrtRowHdr */
  21498. function parse_BrtRowHdr(data, length) {
  21499. var z = ({});
  21500. var tgt = data.l + length;
  21501. z.r = data.read_shift(4);
  21502. data.l += 4; // TODO: ixfe
  21503. var miyRw = data.read_shift(2);
  21504. data.l += 1; // TODO: top/bot padding
  21505. var flags = data.read_shift(1);
  21506. data.l = tgt;
  21507. if(flags & 0x07) z.level = flags & 0x07;
  21508. if(flags & 0x10) z.hidden = true;
  21509. if(flags & 0x20) z.hpt = miyRw / 20;
  21510. return z;
  21511. }
  21512. function write_BrtRowHdr(R, range, ws) {
  21513. var o = new_buf(17+8*16);
  21514. var row = (ws['!rows']||[])[R]||{};
  21515. o.write_shift(4, R);
  21516. o.write_shift(4, 0); /* TODO: ixfe */
  21517. var miyRw = 0x0140;
  21518. if(row.hpx) miyRw = px2pt(row.hpx) * 20;
  21519. else if(row.hpt) miyRw = row.hpt * 20;
  21520. o.write_shift(2, miyRw);
  21521. o.write_shift(1, 0); /* top/bot padding */
  21522. var flags = 0x0;
  21523. if(row.level) flags |= row.level;
  21524. if(row.hidden) flags |= 0x10;
  21525. if(row.hpx || row.hpt) flags |= 0x20;
  21526. o.write_shift(1, flags);
  21527. o.write_shift(1, 0); /* phonetic guide */
  21528. /* [MS-XLSB] 2.5.8 BrtColSpan explains the mechanism */
  21529. var ncolspan = 0, lcs = o.l;
  21530. o.l += 4;
  21531. var caddr = {r:R, c:0};
  21532. for(var i = 0; i < 16; ++i) {
  21533. if((range.s.c > ((i+1) << 10)) || (range.e.c < (i << 10))) continue;
  21534. var first = -1, last = -1;
  21535. for(var j = (i<<10); j < ((i+1)<<10); ++j) {
  21536. caddr.c = j;
  21537. var cell = Array.isArray(ws) ? (ws[caddr.r]||[])[caddr.c] : ws[encode_cell(caddr)];
  21538. if(cell) { if(first < 0) first = j; last = j; }
  21539. }
  21540. if(first < 0) continue;
  21541. ++ncolspan;
  21542. o.write_shift(4, first);
  21543. o.write_shift(4, last);
  21544. }
  21545. var l = o.l;
  21546. o.l = lcs;
  21547. o.write_shift(4, ncolspan);
  21548. o.l = l;
  21549. return o.length > o.l ? o.slice(0, o.l) : o;
  21550. }
  21551. function write_row_header(ba, ws, range, R) {
  21552. var o = write_BrtRowHdr(R, range, ws);
  21553. if((o.length > 17) || (ws['!rows']||[])[R]) write_record(ba, 'BrtRowHdr', o);
  21554. }
  21555. /* [MS-XLSB] 2.4.820 BrtWsDim */
  21556. var parse_BrtWsDim = parse_UncheckedRfX;
  21557. var write_BrtWsDim = write_UncheckedRfX;
  21558. /* [MS-XLSB] 2.4.821 BrtWsFmtInfo */
  21559. function parse_BrtWsFmtInfo() {
  21560. }
  21561. //function write_BrtWsFmtInfo(ws, o) { }
  21562. /* [MS-XLSB] 2.4.823 BrtWsProp */
  21563. function parse_BrtWsProp(data, length) {
  21564. var z = {};
  21565. /* TODO: pull flags */
  21566. data.l += 19;
  21567. z.name = parse_XLSBCodeName(data, length - 19);
  21568. return z;
  21569. }
  21570. function write_BrtWsProp(str, o) {
  21571. if(o == null) o = new_buf(84+4*str.length);
  21572. for(var i = 0; i < 3; ++i) o.write_shift(1,0);
  21573. write_BrtColor({auto:1}, o);
  21574. o.write_shift(-4,-1);
  21575. o.write_shift(-4,-1);
  21576. write_XLSBCodeName(str, o);
  21577. return o.slice(0, o.l);
  21578. }
  21579. /* [MS-XLSB] 2.4.306 BrtCellBlank */
  21580. function parse_BrtCellBlank(data) {
  21581. var cell = parse_XLSBCell(data);
  21582. return [cell];
  21583. }
  21584. function write_BrtCellBlank(cell, ncell, o) {
  21585. if(o == null) o = new_buf(8);
  21586. return write_XLSBCell(ncell, o);
  21587. }
  21588. /* [MS-XLSB] 2.4.307 BrtCellBool */
  21589. function parse_BrtCellBool(data) {
  21590. var cell = parse_XLSBCell(data);
  21591. var fBool = data.read_shift(1);
  21592. return [cell, fBool, 'b'];
  21593. }
  21594. function write_BrtCellBool(cell, ncell, o) {
  21595. if(o == null) o = new_buf(9);
  21596. write_XLSBCell(ncell, o);
  21597. o.write_shift(1, cell.v ? 1 : 0);
  21598. return o;
  21599. }
  21600. /* [MS-XLSB] 2.4.308 BrtCellError */
  21601. function parse_BrtCellError(data) {
  21602. var cell = parse_XLSBCell(data);
  21603. var bError = data.read_shift(1);
  21604. return [cell, bError, 'e'];
  21605. }
  21606. /* [MS-XLSB] 2.4.311 BrtCellIsst */
  21607. function parse_BrtCellIsst(data) {
  21608. var cell = parse_XLSBCell(data);
  21609. var isst = data.read_shift(4);
  21610. return [cell, isst, 's'];
  21611. }
  21612. function write_BrtCellIsst(cell, ncell, o) {
  21613. if(o == null) o = new_buf(12);
  21614. write_XLSBCell(ncell, o);
  21615. o.write_shift(4, ncell.v);
  21616. return o;
  21617. }
  21618. /* [MS-XLSB] 2.4.313 BrtCellReal */
  21619. function parse_BrtCellReal(data) {
  21620. var cell = parse_XLSBCell(data);
  21621. var value = parse_Xnum(data);
  21622. return [cell, value, 'n'];
  21623. }
  21624. function write_BrtCellReal(cell, ncell, o) {
  21625. if(o == null) o = new_buf(16);
  21626. write_XLSBCell(ncell, o);
  21627. write_Xnum(cell.v, o);
  21628. return o;
  21629. }
  21630. /* [MS-XLSB] 2.4.314 BrtCellRk */
  21631. function parse_BrtCellRk(data) {
  21632. var cell = parse_XLSBCell(data);
  21633. var value = parse_RkNumber(data);
  21634. return [cell, value, 'n'];
  21635. }
  21636. function write_BrtCellRk(cell, ncell, o) {
  21637. if(o == null) o = new_buf(12);
  21638. write_XLSBCell(ncell, o);
  21639. write_RkNumber(cell.v, o);
  21640. return o;
  21641. }
  21642. /* [MS-XLSB] 2.4.317 BrtCellSt */
  21643. function parse_BrtCellSt(data) {
  21644. var cell = parse_XLSBCell(data);
  21645. var value = parse_XLWideString(data);
  21646. return [cell, value, 'str'];
  21647. }
  21648. function write_BrtCellSt(cell, ncell, o) {
  21649. if(o == null) o = new_buf(12 + 4 * cell.v.length);
  21650. write_XLSBCell(ncell, o);
  21651. write_XLWideString(cell.v, o);
  21652. return o.length > o.l ? o.slice(0, o.l) : o;
  21653. }
  21654. /* [MS-XLSB] 2.4.653 BrtFmlaBool */
  21655. function parse_BrtFmlaBool(data, length, opts) {
  21656. var end = data.l + length;
  21657. var cell = parse_XLSBCell(data);
  21658. cell.r = opts['!row'];
  21659. var value = data.read_shift(1);
  21660. var o = [cell, value, 'b'];
  21661. if(opts.cellFormula) {
  21662. data.l += 2;
  21663. var formula = parse_XLSBCellParsedFormula(data, end - data.l, opts);
  21664. o[3] = stringify_formula(formula, null/*range*/, cell, opts.supbooks, opts);/* TODO */
  21665. }
  21666. else data.l = end;
  21667. return o;
  21668. }
  21669. /* [MS-XLSB] 2.4.654 BrtFmlaError */
  21670. function parse_BrtFmlaError(data, length, opts) {
  21671. var end = data.l + length;
  21672. var cell = parse_XLSBCell(data);
  21673. cell.r = opts['!row'];
  21674. var value = data.read_shift(1);
  21675. var o = [cell, value, 'e'];
  21676. if(opts.cellFormula) {
  21677. data.l += 2;
  21678. var formula = parse_XLSBCellParsedFormula(data, end - data.l, opts);
  21679. o[3] = stringify_formula(formula, null/*range*/, cell, opts.supbooks, opts);/* TODO */
  21680. }
  21681. else data.l = end;
  21682. return o;
  21683. }
  21684. /* [MS-XLSB] 2.4.655 BrtFmlaNum */
  21685. function parse_BrtFmlaNum(data, length, opts) {
  21686. var end = data.l + length;
  21687. var cell = parse_XLSBCell(data);
  21688. cell.r = opts['!row'];
  21689. var value = parse_Xnum(data);
  21690. var o = [cell, value, 'n'];
  21691. if(opts.cellFormula) {
  21692. data.l += 2;
  21693. var formula = parse_XLSBCellParsedFormula(data, end - data.l, opts);
  21694. o[3] = stringify_formula(formula, null/*range*/, cell, opts.supbooks, opts);/* TODO */
  21695. }
  21696. else data.l = end;
  21697. return o;
  21698. }
  21699. /* [MS-XLSB] 2.4.656 BrtFmlaString */
  21700. function parse_BrtFmlaString(data, length, opts) {
  21701. var end = data.l + length;
  21702. var cell = parse_XLSBCell(data);
  21703. cell.r = opts['!row'];
  21704. var value = parse_XLWideString(data);
  21705. var o = [cell, value, 'str'];
  21706. if(opts.cellFormula) {
  21707. data.l += 2;
  21708. var formula = parse_XLSBCellParsedFormula(data, end - data.l, opts);
  21709. o[3] = stringify_formula(formula, null/*range*/, cell, opts.supbooks, opts);/* TODO */
  21710. }
  21711. else data.l = end;
  21712. return o;
  21713. }
  21714. /* [MS-XLSB] 2.4.682 BrtMergeCell */
  21715. var parse_BrtMergeCell = parse_UncheckedRfX;
  21716. var write_BrtMergeCell = write_UncheckedRfX;
  21717. /* [MS-XLSB] 2.4.107 BrtBeginMergeCells */
  21718. function write_BrtBeginMergeCells(cnt, o) {
  21719. if(o == null) o = new_buf(4);
  21720. o.write_shift(4, cnt);
  21721. return o;
  21722. }
  21723. /* [MS-XLSB] 2.4.662 BrtHLink */
  21724. function parse_BrtHLink(data, length) {
  21725. var end = data.l + length;
  21726. var rfx = parse_UncheckedRfX(data, 16);
  21727. var relId = parse_XLNullableWideString(data);
  21728. var loc = parse_XLWideString(data);
  21729. var tooltip = parse_XLWideString(data);
  21730. var display = parse_XLWideString(data);
  21731. data.l = end;
  21732. var o = ({rfx:rfx, relId:relId, loc:loc, display:display});
  21733. if(tooltip) o.Tooltip = tooltip;
  21734. return o;
  21735. }
  21736. function write_BrtHLink(l, rId) {
  21737. var o = new_buf(50+4*(l[1].Target.length + (l[1].Tooltip || "").length));
  21738. write_UncheckedRfX({s:decode_cell(l[0]), e:decode_cell(l[0])}, o);
  21739. write_RelID("rId" + rId, o);
  21740. var locidx = l[1].Target.indexOf("#");
  21741. var loc = locidx == -1 ? "" : l[1].Target.slice(locidx+1);
  21742. write_XLWideString(loc || "", o);
  21743. write_XLWideString(l[1].Tooltip || "", o);
  21744. write_XLWideString("", o);
  21745. return o.slice(0, o.l);
  21746. }
  21747. /* [MS-XLSB] 2.4.6 BrtArrFmla */
  21748. function parse_BrtArrFmla(data, length, opts) {
  21749. var end = data.l + length;
  21750. var rfx = parse_RfX(data, 16);
  21751. var fAlwaysCalc = data.read_shift(1);
  21752. var o = [rfx]; o[2] = fAlwaysCalc;
  21753. if(opts.cellFormula) {
  21754. var formula = parse_XLSBArrayParsedFormula(data, end - data.l, opts);
  21755. o[1] = formula;
  21756. } else data.l = end;
  21757. return o;
  21758. }
  21759. /* [MS-XLSB] 2.4.750 BrtShrFmla */
  21760. function parse_BrtShrFmla(data, length, opts) {
  21761. var end = data.l + length;
  21762. var rfx = parse_UncheckedRfX(data, 16);
  21763. var o = [rfx];
  21764. if(opts.cellFormula) {
  21765. var formula = parse_XLSBSharedParsedFormula(data, end - data.l, opts);
  21766. o[1] = formula;
  21767. data.l = end;
  21768. } else data.l = end;
  21769. return o;
  21770. }
  21771. /* [MS-XLSB] 2.4.323 BrtColInfo */
  21772. /* TODO: once XLS ColInfo is set, combine the functions */
  21773. function write_BrtColInfo(C, col, o) {
  21774. if(o == null) o = new_buf(18);
  21775. var p = col_obj_w(C, col);
  21776. o.write_shift(-4, C);
  21777. o.write_shift(-4, C);
  21778. o.write_shift(4, (p.width || 10) * 256);
  21779. o.write_shift(4, 0/*ixfe*/); // style
  21780. var flags = 0;
  21781. if(col.hidden) flags |= 0x01;
  21782. if(typeof p.width == 'number') flags |= 0x02;
  21783. o.write_shift(1, flags); // bit flag
  21784. o.write_shift(1, 0); // bit flag
  21785. return o;
  21786. }
  21787. /* [MS-XLSB] 2.4.678 BrtMargins */
  21788. var BrtMarginKeys = ["left","right","top","bottom","header","footer"];
  21789. function parse_BrtMargins(data) {
  21790. var margins = ({});
  21791. BrtMarginKeys.forEach(function(k) { margins[k] = parse_Xnum(data, 8); });
  21792. return margins;
  21793. }
  21794. function write_BrtMargins(margins, o) {
  21795. if(o == null) o = new_buf(6*8);
  21796. default_margins(margins);
  21797. BrtMarginKeys.forEach(function(k) { write_Xnum((margins)[k], o); });
  21798. return o;
  21799. }
  21800. /* [MS-XLSB] 2.4.299 BrtBeginWsView */
  21801. function parse_BrtBeginWsView(data) {
  21802. var f = data.read_shift(2);
  21803. data.l += 28;
  21804. return { RTL: f & 0x20 };
  21805. }
  21806. function write_BrtBeginWsView(ws, Workbook, o) {
  21807. if(o == null) o = new_buf(30);
  21808. var f = 0x39c;
  21809. if((((Workbook||{}).Views||[])[0]||{}).RTL) f |= 0x20;
  21810. o.write_shift(2, f); // bit flag
  21811. o.write_shift(4, 0);
  21812. o.write_shift(4, 0); // view first row
  21813. o.write_shift(4, 0); // view first col
  21814. o.write_shift(1, 0); // gridline color ICV
  21815. o.write_shift(1, 0);
  21816. o.write_shift(2, 0);
  21817. o.write_shift(2, 100); // zoom scale
  21818. o.write_shift(2, 0);
  21819. o.write_shift(2, 0);
  21820. o.write_shift(2, 0);
  21821. o.write_shift(4, 0); // workbook view id
  21822. return o;
  21823. }
  21824. /* [MS-XLSB] 2.4.309 BrtCellIgnoreEC */
  21825. function write_BrtCellIgnoreEC(ref) {
  21826. var o = new_buf(24);
  21827. o.write_shift(4, 4);
  21828. o.write_shift(4, 1);
  21829. write_UncheckedRfX(ref, o);
  21830. return o;
  21831. }
  21832. /* [MS-XLSB] 2.4.748 BrtSheetProtection */
  21833. function write_BrtSheetProtection(sp, o) {
  21834. if(o == null) o = new_buf(16*4+2);
  21835. o.write_shift(2, sp.password ? crypto_CreatePasswordVerifier_Method1(sp.password) : 0);
  21836. o.write_shift(4, 1); // this record should not be written if no protection
  21837. [
  21838. ["objects", false], // fObjects
  21839. ["scenarios", false], // fScenarios
  21840. ["formatCells", true], // fFormatCells
  21841. ["formatColumns", true], // fFormatColumns
  21842. ["formatRows", true], // fFormatRows
  21843. ["insertColumns", true], // fInsertColumns
  21844. ["insertRows", true], // fInsertRows
  21845. ["insertHyperlinks", true], // fInsertHyperlinks
  21846. ["deleteColumns", true], // fDeleteColumns
  21847. ["deleteRows", true], // fDeleteRows
  21848. ["selectLockedCells", false], // fSelLockedCells
  21849. ["sort", true], // fSort
  21850. ["autoFilter", true], // fAutoFilter
  21851. ["pivotTables", true], // fPivotTables
  21852. ["selectUnlockedCells", false] // fSelUnlockedCells
  21853. ].forEach(function(n) {
  21854. if(n[1]) o.write_shift(4, sp[n[0]] != null && !sp[n[0]] ? 1 : 0);
  21855. else o.write_shift(4, sp[n[0]] != null && sp[n[0]] ? 0 : 1);
  21856. });
  21857. return o;
  21858. }
  21859. /* [MS-XLSB] 2.1.7.61 Worksheet */
  21860. function parse_ws_bin(data, _opts, idx, rels, wb, themes, styles) {
  21861. if(!data) return data;
  21862. var opts = _opts || {};
  21863. if(!rels) rels = {'!id':{}};
  21864. if(DENSE != null && opts.dense == null) opts.dense = DENSE;
  21865. var s = (opts.dense ? [] : {});
  21866. var ref;
  21867. var refguess = {s: {r:2000000, c:2000000}, e: {r:0, c:0} };
  21868. var pass = false, end = false;
  21869. var row, p, cf, R, C, addr, sstr, rr, cell;
  21870. var merges = [];
  21871. opts.biff = 12;
  21872. opts['!row'] = 0;
  21873. var ai = 0, af = false;
  21874. var arrayf = [];
  21875. var sharedf = {};
  21876. var supbooks = opts.supbooks || wb.supbooks || ([[]]);
  21877. supbooks.sharedf = sharedf;
  21878. supbooks.arrayf = arrayf;
  21879. supbooks.SheetNames = wb.SheetNames || wb.Sheets.map(function(x) { return x.name; });
  21880. if(!opts.supbooks) {
  21881. opts.supbooks = supbooks;
  21882. if(wb.Names) for(var i = 0; i < wb.Names.length; ++i) supbooks[0][i+1] = wb.Names[i];
  21883. }
  21884. var colinfo = [], rowinfo = [];
  21885. var seencol = false;
  21886. recordhopper(data, function ws_parse(val, R_n, RT) {
  21887. if(end) return;
  21888. switch(RT) {
  21889. case 0x0094: /* 'BrtWsDim' */
  21890. ref = val; break;
  21891. case 0x0000: /* 'BrtRowHdr' */
  21892. row = val;
  21893. if(opts.sheetRows && opts.sheetRows <= row.r) end=true;
  21894. rr = encode_row(R = row.r);
  21895. opts['!row'] = row.r;
  21896. if(val.hidden || val.hpt || val.level != null) {
  21897. if(val.hpt) val.hpx = pt2px(val.hpt);
  21898. rowinfo[val.r] = val;
  21899. }
  21900. break;
  21901. case 0x0002: /* 'BrtCellRk' */
  21902. case 0x0003: /* 'BrtCellError' */
  21903. case 0x0004: /* 'BrtCellBool' */
  21904. case 0x0005: /* 'BrtCellReal' */
  21905. case 0x0006: /* 'BrtCellSt' */
  21906. case 0x0007: /* 'BrtCellIsst' */
  21907. case 0x0008: /* 'BrtFmlaString' */
  21908. case 0x0009: /* 'BrtFmlaNum' */
  21909. case 0x000A: /* 'BrtFmlaBool' */
  21910. case 0x000B: /* 'BrtFmlaError' */
  21911. p = ({t:val[2]});
  21912. switch(val[2]) {
  21913. case 'n': p.v = val[1]; break;
  21914. case 's': sstr = strs[val[1]]; p.v = sstr.t; p.r = sstr.r; break;
  21915. case 'b': p.v = val[1] ? true : false; break;
  21916. case 'e': p.v = val[1]; if(opts.cellText !== false) p.w = BErr[p.v]; break;
  21917. case 'str': p.t = 's'; p.v = val[1]; break;
  21918. }
  21919. if((cf = styles.CellXf[val[0].iStyleRef])) safe_format(p,cf.numFmtId,null,opts, themes, styles);
  21920. C = val[0].c;
  21921. if(opts.dense) { if(!s[R]) s[R] = []; s[R][C] = p; }
  21922. else s[encode_col(C) + rr] = p;
  21923. if(opts.cellFormula) {
  21924. af = false;
  21925. for(ai = 0; ai < arrayf.length; ++ai) {
  21926. var aii = arrayf[ai];
  21927. if(row.r >= aii[0].s.r && row.r <= aii[0].e.r)
  21928. if(C >= aii[0].s.c && C <= aii[0].e.c) {
  21929. p.F = encode_range(aii[0]); af = true;
  21930. }
  21931. }
  21932. if(!af && val.length > 3) p.f = val[3];
  21933. }
  21934. if(refguess.s.r > row.r) refguess.s.r = row.r;
  21935. if(refguess.s.c > C) refguess.s.c = C;
  21936. if(refguess.e.r < row.r) refguess.e.r = row.r;
  21937. if(refguess.e.c < C) refguess.e.c = C;
  21938. if(opts.cellDates && cf && p.t == 'n' && SSF.is_date(SSF._table[cf.numFmtId])) {
  21939. var _d = SSF.parse_date_code(p.v); if(_d) { p.t = 'd'; p.v = new Date(_d.y, _d.m-1,_d.d,_d.H,_d.M,_d.S,_d.u); }
  21940. }
  21941. break;
  21942. case 0x0001: /* 'BrtCellBlank' */
  21943. if(!opts.sheetStubs || pass) break;
  21944. p = ({t:'z',v:undefined});
  21945. C = val[0].c;
  21946. if(opts.dense) { if(!s[R]) s[R] = []; s[R][C] = p; }
  21947. else s[encode_col(C) + rr] = p;
  21948. if(refguess.s.r > row.r) refguess.s.r = row.r;
  21949. if(refguess.s.c > C) refguess.s.c = C;
  21950. if(refguess.e.r < row.r) refguess.e.r = row.r;
  21951. if(refguess.e.c < C) refguess.e.c = C;
  21952. break;
  21953. case 0x00B0: /* 'BrtMergeCell' */
  21954. merges.push(val); break;
  21955. case 0x01EE: /* 'BrtHLink' */
  21956. var rel = rels['!id'][val.relId];
  21957. if(rel) {
  21958. val.Target = rel.Target;
  21959. if(val.loc) val.Target += "#"+val.loc;
  21960. val.Rel = rel;
  21961. } else if(val.relId == '') {
  21962. val.Target = "#" + val.loc;
  21963. }
  21964. for(R=val.rfx.s.r;R<=val.rfx.e.r;++R) for(C=val.rfx.s.c;C<=val.rfx.e.c;++C) {
  21965. if(opts.dense) {
  21966. if(!s[R]) s[R] = [];
  21967. if(!s[R][C]) s[R][C] = {t:'z',v:undefined};
  21968. s[R][C].l = val;
  21969. } else {
  21970. addr = encode_cell({c:C,r:R});
  21971. if(!s[addr]) s[addr] = {t:'z',v:undefined};
  21972. s[addr].l = val;
  21973. }
  21974. }
  21975. break;
  21976. case 0x01AA: /* 'BrtArrFmla' */
  21977. if(!opts.cellFormula) break;
  21978. arrayf.push(val);
  21979. cell = ((opts.dense ? s[R][C] : s[encode_col(C) + rr]));
  21980. cell.f = stringify_formula(val[1], refguess, {r:row.r, c:C}, supbooks, opts);
  21981. cell.F = encode_range(val[0]);
  21982. break;
  21983. case 0x01AB: /* 'BrtShrFmla' */
  21984. if(!opts.cellFormula) break;
  21985. sharedf[encode_cell(val[0].s)] = val[1];
  21986. cell = (opts.dense ? s[R][C] : s[encode_col(C) + rr]);
  21987. cell.f = stringify_formula(val[1], refguess, {r:row.r, c:C}, supbooks, opts);
  21988. break;
  21989. /* identical to 'ColInfo' in XLS */
  21990. case 0x003C: /* 'BrtColInfo' */
  21991. if(!opts.cellStyles) break;
  21992. while(val.e >= val.s) {
  21993. colinfo[val.e--] = { width: val.w/256, hidden: !!(val.flags & 0x01) };
  21994. if(!seencol) { seencol = true; find_mdw_colw(val.w/256); }
  21995. process_col(colinfo[val.e+1]);
  21996. }
  21997. break;
  21998. case 0x00A1: /* 'BrtBeginAFilter' */
  21999. s['!autofilter'] = { ref:encode_range(val) };
  22000. break;
  22001. case 0x01DC: /* 'BrtMargins' */
  22002. s['!margins'] = val;
  22003. break;
  22004. case 0x0093: /* 'BrtWsProp' */
  22005. if(!wb.Sheets[idx]) wb.Sheets[idx] = {};
  22006. if(val.name) wb.Sheets[idx].CodeName = val.name;
  22007. break;
  22008. case 0x0089: /* 'BrtBeginWsView' */
  22009. if(!wb.Views) wb.Views = [{}];
  22010. if(!wb.Views[0]) wb.Views[0] = {};
  22011. if(val.RTL) wb.Views[0].RTL = true;
  22012. break;
  22013. case 0x01E5: /* 'BrtWsFmtInfo' */
  22014. break;
  22015. case 0x00AF: /* 'BrtAFilterDateGroupItem' */
  22016. case 0x0284: /* 'BrtActiveX' */
  22017. case 0x0271: /* 'BrtBigName' */
  22018. case 0x0232: /* 'BrtBkHim' */
  22019. case 0x018C: /* 'BrtBrk' */
  22020. case 0x0458: /* 'BrtCFIcon' */
  22021. case 0x047A: /* 'BrtCFRuleExt' */
  22022. case 0x01D7: /* 'BrtCFVO' */
  22023. case 0x041A: /* 'BrtCFVO14' */
  22024. case 0x0289: /* 'BrtCellIgnoreEC' */
  22025. case 0x0451: /* 'BrtCellIgnoreEC14' */
  22026. case 0x0031: /* 'BrtCellMeta' */
  22027. case 0x024D: /* 'BrtCellSmartTagProperty' */
  22028. case 0x025F: /* 'BrtCellWatch' */
  22029. case 0x0234: /* 'BrtColor' */
  22030. case 0x041F: /* 'BrtColor14' */
  22031. case 0x00A8: /* 'BrtColorFilter' */
  22032. case 0x00AE: /* 'BrtCustomFilter' */
  22033. case 0x049C: /* 'BrtCustomFilter14' */
  22034. case 0x01F3: /* 'BrtDRef' */
  22035. case 0x0040: /* 'BrtDVal' */
  22036. case 0x041D: /* 'BrtDVal14' */
  22037. case 0x0226: /* 'BrtDrawing' */
  22038. case 0x00AB: /* 'BrtDynamicFilter' */
  22039. case 0x00A7: /* 'BrtFilter' */
  22040. case 0x0499: /* 'BrtFilter14' */
  22041. case 0x00A9: /* 'BrtIconFilter' */
  22042. case 0x049D: /* 'BrtIconFilter14' */
  22043. case 0x0227: /* 'BrtLegacyDrawing' */
  22044. case 0x0228: /* 'BrtLegacyDrawingHF' */
  22045. case 0x0295: /* 'BrtListPart' */
  22046. case 0x027F: /* 'BrtOleObject' */
  22047. case 0x01DE: /* 'BrtPageSetup' */
  22048. case 0x0097: /* 'BrtPane' */
  22049. case 0x0219: /* 'BrtPhoneticInfo' */
  22050. case 0x01DD: /* 'BrtPrintOptions' */
  22051. case 0x0218: /* 'BrtRangeProtection' */
  22052. case 0x044F: /* 'BrtRangeProtection14' */
  22053. case 0x02A8: /* 'BrtRangeProtectionIso' */
  22054. case 0x0450: /* 'BrtRangeProtectionIso14' */
  22055. case 0x0400: /* 'BrtRwDescent' */
  22056. case 0x0098: /* 'BrtSel' */
  22057. case 0x0297: /* 'BrtSheetCalcProp' */
  22058. case 0x0217: /* 'BrtSheetProtection' */
  22059. case 0x02A6: /* 'BrtSheetProtectionIso' */
  22060. case 0x01F8: /* 'BrtSlc' */
  22061. case 0x0413: /* 'BrtSparkline' */
  22062. case 0x01AC: /* 'BrtTable' */
  22063. case 0x00AA: /* 'BrtTop10Filter' */
  22064. case 0x0C00: /* 'BrtUid' */
  22065. case 0x0032: /* 'BrtValueMeta' */
  22066. case 0x0816: /* 'BrtWebExtension' */
  22067. case 0x0415: /* 'BrtWsFmtInfoEx14' */
  22068. break;
  22069. case 0x0023: /* 'BrtFRTBegin' */
  22070. pass = true; break;
  22071. case 0x0024: /* 'BrtFRTEnd' */
  22072. pass = false; break;
  22073. case 0x0025: /* 'BrtACBegin' */ break;
  22074. case 0x0026: /* 'BrtACEnd' */ break;
  22075. default:
  22076. if((R_n||"").indexOf("Begin") > 0){/* empty */}
  22077. else if((R_n||"").indexOf("End") > 0){/* empty */}
  22078. else if(!pass || opts.WTF) throw new Error("Unexpected record " + RT + " " + R_n);
  22079. }
  22080. }, opts);
  22081. delete opts.supbooks;
  22082. delete opts['!row'];
  22083. if(!s["!ref"] && (refguess.s.r < 2000000 || ref && (ref.e.r > 0 || ref.e.c > 0 || ref.s.r > 0 || ref.s.c > 0))) s["!ref"] = encode_range(ref || refguess);
  22084. if(opts.sheetRows && s["!ref"]) {
  22085. var tmpref = safe_decode_range(s["!ref"]);
  22086. if(opts.sheetRows <= +tmpref.e.r) {
  22087. tmpref.e.r = opts.sheetRows - 1;
  22088. if(tmpref.e.r > refguess.e.r) tmpref.e.r = refguess.e.r;
  22089. if(tmpref.e.r < tmpref.s.r) tmpref.s.r = tmpref.e.r;
  22090. if(tmpref.e.c > refguess.e.c) tmpref.e.c = refguess.e.c;
  22091. if(tmpref.e.c < tmpref.s.c) tmpref.s.c = tmpref.e.c;
  22092. s["!fullref"] = s["!ref"];
  22093. s["!ref"] = encode_range(tmpref);
  22094. }
  22095. }
  22096. if(merges.length > 0) s["!merges"] = merges;
  22097. if(colinfo.length > 0) s["!cols"] = colinfo;
  22098. if(rowinfo.length > 0) s["!rows"] = rowinfo;
  22099. return s;
  22100. }
  22101. /* TODO: something useful -- this is a stub */
  22102. function write_ws_bin_cell(ba, cell, R, C, opts, ws) {
  22103. if(cell.v === undefined) return "";
  22104. var vv = "";
  22105. switch(cell.t) {
  22106. case 'b': vv = cell.v ? "1" : "0"; break;
  22107. case 'd': // no BrtCellDate :(
  22108. cell = dup(cell);
  22109. cell.z = cell.z || SSF._table[14];
  22110. cell.v = datenum(parseDate(cell.v)); cell.t = 'n';
  22111. break;
  22112. /* falls through */
  22113. case 'n': case 'e': vv = ''+cell.v; break;
  22114. default: vv = cell.v; break;
  22115. }
  22116. var o = ({r:R, c:C});
  22117. /* TODO: cell style */
  22118. o.s = get_cell_style(opts.cellXfs, cell, opts);
  22119. if(cell.l) ws['!links'].push([encode_cell(o), cell.l]);
  22120. if(cell.c) ws['!comments'].push([encode_cell(o), cell.c]);
  22121. switch(cell.t) {
  22122. case 's': case 'str':
  22123. if(opts.bookSST) {
  22124. vv = get_sst_id(opts.Strings, (cell.v), opts.revStrings);
  22125. o.t = "s"; o.v = vv;
  22126. write_record(ba, "BrtCellIsst", write_BrtCellIsst(cell, o));
  22127. } else {
  22128. o.t = "str";
  22129. write_record(ba, "BrtCellSt", write_BrtCellSt(cell, o));
  22130. }
  22131. return;
  22132. case 'n':
  22133. /* TODO: determine threshold for Real vs RK */
  22134. if(cell.v == (cell.v | 0) && cell.v > -1000 && cell.v < 1000) write_record(ba, "BrtCellRk", write_BrtCellRk(cell, o));
  22135. else write_record(ba, "BrtCellReal", write_BrtCellReal(cell, o));
  22136. return;
  22137. case 'b':
  22138. o.t = "b";
  22139. write_record(ba, "BrtCellBool", write_BrtCellBool(cell, o));
  22140. return;
  22141. case 'e': /* TODO: error */ o.t = "e"; break;
  22142. }
  22143. write_record(ba, "BrtCellBlank", write_BrtCellBlank(cell, o));
  22144. }
  22145. function write_CELLTABLE(ba, ws, idx, opts) {
  22146. var range = safe_decode_range(ws['!ref'] || "A1"), ref, rr = "", cols = [];
  22147. write_record(ba, 'BrtBeginSheetData');
  22148. var dense = Array.isArray(ws);
  22149. var cap = range.e.r;
  22150. if(ws['!rows']) cap = Math.max(range.e.r, ws['!rows'].length - 1);
  22151. for(var R = range.s.r; R <= cap; ++R) {
  22152. rr = encode_row(R);
  22153. /* [ACCELLTABLE] */
  22154. /* BrtRowHdr */
  22155. write_row_header(ba, ws, range, R);
  22156. if(R <= range.e.r) for(var C = range.s.c; C <= range.e.c; ++C) {
  22157. /* *16384CELL */
  22158. if(R === range.s.r) cols[C] = encode_col(C);
  22159. ref = cols[C] + rr;
  22160. var cell = dense ? (ws[R]||[])[C] : ws[ref];
  22161. if(!cell) continue;
  22162. /* write cell */
  22163. write_ws_bin_cell(ba, cell, R, C, opts, ws);
  22164. }
  22165. }
  22166. write_record(ba, 'BrtEndSheetData');
  22167. }
  22168. function write_MERGECELLS(ba, ws) {
  22169. if(!ws || !ws['!merges']) return;
  22170. write_record(ba, 'BrtBeginMergeCells', write_BrtBeginMergeCells(ws['!merges'].length));
  22171. ws['!merges'].forEach(function(m) { write_record(ba, 'BrtMergeCell', write_BrtMergeCell(m)); });
  22172. write_record(ba, 'BrtEndMergeCells');
  22173. }
  22174. function write_COLINFOS(ba, ws) {
  22175. if(!ws || !ws['!cols']) return;
  22176. write_record(ba, 'BrtBeginColInfos');
  22177. ws['!cols'].forEach(function(m, i) { if(m) write_record(ba, 'BrtColInfo', write_BrtColInfo(i, m)); });
  22178. write_record(ba, 'BrtEndColInfos');
  22179. }
  22180. function write_IGNOREECS(ba, ws) {
  22181. if(!ws || !ws['!ref']) return;
  22182. write_record(ba, 'BrtBeginCellIgnoreECs');
  22183. write_record(ba, 'BrtCellIgnoreEC', write_BrtCellIgnoreEC(safe_decode_range(ws['!ref'])));
  22184. write_record(ba, 'BrtEndCellIgnoreECs');
  22185. }
  22186. function write_HLINKS(ba, ws, rels) {
  22187. /* *BrtHLink */
  22188. ws['!links'].forEach(function(l) {
  22189. if(!l[1].Target) return;
  22190. var rId = add_rels(rels, -1, l[1].Target.replace(/#.*$/, ""), RELS.HLINK);
  22191. write_record(ba, "BrtHLink", write_BrtHLink(l, rId));
  22192. });
  22193. delete ws['!links'];
  22194. }
  22195. function write_LEGACYDRAWING(ba, ws, idx, rels) {
  22196. /* [BrtLegacyDrawing] */
  22197. if(ws['!comments'].length > 0) {
  22198. var rId = add_rels(rels, -1, "../drawings/vmlDrawing" + (idx+1) + ".vml", RELS.VML);
  22199. write_record(ba, "BrtLegacyDrawing", write_RelID("rId" + rId));
  22200. ws['!legacy'] = rId;
  22201. }
  22202. }
  22203. function write_AUTOFILTER(ba, ws) {
  22204. if(!ws['!autofilter']) return;
  22205. write_record(ba, "BrtBeginAFilter", write_UncheckedRfX(safe_decode_range(ws['!autofilter'].ref)));
  22206. /* *FILTERCOLUMN */
  22207. /* [SORTSTATE] */
  22208. /* BrtEndAFilter */
  22209. write_record(ba, "BrtEndAFilter");
  22210. }
  22211. function write_WSVIEWS2(ba, ws, Workbook) {
  22212. write_record(ba, "BrtBeginWsViews");
  22213. { /* 1*WSVIEW2 */
  22214. /* [ACUID] */
  22215. write_record(ba, "BrtBeginWsView", write_BrtBeginWsView(ws, Workbook));
  22216. /* [BrtPane] */
  22217. /* *4BrtSel */
  22218. /* *4SXSELECT */
  22219. /* *FRT */
  22220. write_record(ba, "BrtEndWsView");
  22221. }
  22222. /* *FRT */
  22223. write_record(ba, "BrtEndWsViews");
  22224. }
  22225. function write_WSFMTINFO() {
  22226. /* [ACWSFMTINFO] */
  22227. //write_record(ba, "BrtWsFmtInfo", write_BrtWsFmtInfo(ws));
  22228. }
  22229. function write_SHEETPROTECT(ba, ws) {
  22230. if(!ws['!protect']) return;
  22231. /* [BrtSheetProtectionIso] */
  22232. write_record(ba, "BrtSheetProtection", write_BrtSheetProtection(ws['!protect']));
  22233. }
  22234. function write_ws_bin(idx, opts, wb, rels) {
  22235. var ba = buf_array();
  22236. var s = wb.SheetNames[idx], ws = wb.Sheets[s] || {};
  22237. var c = s; try { if(wb && wb.Workbook) c = wb.Workbook.Sheets[idx].CodeName || c; } catch(e) {}
  22238. var r = safe_decode_range(ws['!ref'] || "A1");
  22239. if(r.e.c > 0x3FFF || r.e.r > 0xFFFFF) {
  22240. if(opts.WTF) throw new Error("Range " + (ws['!ref'] || "A1") + " exceeds format limit A1:XFD1048576");
  22241. r.e.c = Math.min(r.e.c, 0x3FFF);
  22242. r.e.r = Math.min(r.e.c, 0xFFFFF);
  22243. }
  22244. ws['!links'] = [];
  22245. /* passed back to write_zip and removed there */
  22246. ws['!comments'] = [];
  22247. write_record(ba, "BrtBeginSheet");
  22248. if(wb.vbaraw) write_record(ba, "BrtWsProp", write_BrtWsProp(c));
  22249. write_record(ba, "BrtWsDim", write_BrtWsDim(r));
  22250. write_WSVIEWS2(ba, ws, wb.Workbook);
  22251. write_WSFMTINFO(ba, ws);
  22252. write_COLINFOS(ba, ws, idx, opts, wb);
  22253. write_CELLTABLE(ba, ws, idx, opts, wb);
  22254. /* [BrtSheetCalcProp] */
  22255. write_SHEETPROTECT(ba, ws);
  22256. /* *([BrtRangeProtectionIso] BrtRangeProtection) */
  22257. /* [SCENMAN] */
  22258. write_AUTOFILTER(ba, ws);
  22259. /* [SORTSTATE] */
  22260. /* [DCON] */
  22261. /* [USERSHVIEWS] */
  22262. write_MERGECELLS(ba, ws);
  22263. /* [BrtPhoneticInfo] */
  22264. /* *CONDITIONALFORMATTING */
  22265. /* [DVALS] */
  22266. write_HLINKS(ba, ws, rels);
  22267. /* [BrtPrintOptions] */
  22268. if(ws['!margins']) write_record(ba, "BrtMargins", write_BrtMargins(ws['!margins']));
  22269. /* [BrtPageSetup] */
  22270. /* [HEADERFOOTER] */
  22271. /* [RWBRK] */
  22272. /* [COLBRK] */
  22273. /* *BrtBigName */
  22274. /* [CELLWATCHES] */
  22275. if(!opts || opts.ignoreEC || (opts.ignoreEC == (void 0))) write_IGNOREECS(ba, ws);
  22276. /* [SMARTTAGS] */
  22277. /* [BrtDrawing] */
  22278. write_LEGACYDRAWING(ba, ws, idx, rels);
  22279. /* [BrtLegacyDrawingHF] */
  22280. /* [BrtBkHim] */
  22281. /* [OLEOBJECTS] */
  22282. /* [ACTIVEXCONTROLS] */
  22283. /* [WEBPUBITEMS] */
  22284. /* [LISTPARTS] */
  22285. /* FRTWORKSHEET */
  22286. write_record(ba, "BrtEndSheet");
  22287. return ba.end();
  22288. }
  22289. function parse_numCache(data) {
  22290. var col = [];
  22291. /* 21.2.2.150 pt CT_NumVal */
  22292. (data.match(/<c:pt idx="(\d*)">(.*?)<\/c:pt>/mg)||[]).forEach(function(pt) {
  22293. var q = pt.match(/<c:pt idx="(\d*?)"><c:v>(.*)<\/c:v><\/c:pt>/);
  22294. if(!q) return;
  22295. col[+q[1]] = +q[2];
  22296. });
  22297. /* 21.2.2.71 formatCode CT_Xstring */
  22298. var nf = unescapexml((data.match(/<c:formatCode>([\s\S]*?)<\/c:formatCode>/) || ["","General"])[1]);
  22299. return [col, nf];
  22300. }
  22301. /* 21.2 DrawingML - Charts */
  22302. function parse_chart(data, name, opts, rels, wb, csheet) {
  22303. var cs = ((csheet || {"!type":"chart"}));
  22304. if(!data) return csheet;
  22305. /* 21.2.2.27 chart CT_Chart */
  22306. var C = 0, R = 0, col = "A";
  22307. var refguess = {s: {r:2000000, c:2000000}, e: {r:0, c:0} };
  22308. /* 21.2.2.120 numCache CT_NumData */
  22309. (data.match(/<c:numCache>[\s\S]*?<\/c:numCache>/gm)||[]).forEach(function(nc) {
  22310. var cache = parse_numCache(nc);
  22311. refguess.s.r = refguess.s.c = 0;
  22312. refguess.e.c = C;
  22313. col = encode_col(C);
  22314. cache[0].forEach(function(n,i) {
  22315. cs[col + encode_row(i)] = {t:'n', v:n, z:cache[1] };
  22316. R = i;
  22317. });
  22318. if(refguess.e.r < R) refguess.e.r = R;
  22319. ++C;
  22320. });
  22321. if(C > 0) cs["!ref"] = encode_range(refguess);
  22322. return cs;
  22323. }
  22324. RELS.CS = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/chartsheet";
  22325. var CS_XML_ROOT = writextag('chartsheet', null, {
  22326. 'xmlns': XMLNS.main[0],
  22327. 'xmlns:r': XMLNS.r
  22328. });
  22329. /* 18.3 Worksheets also covers Chartsheets */
  22330. function parse_cs_xml(data, opts, idx, rels, wb) {
  22331. if(!data) return data;
  22332. /* 18.3.1.12 chartsheet CT_ChartSheet */
  22333. if(!rels) rels = {'!id':{}};
  22334. var s = {'!type':"chart", '!chart':null, '!rel':""};
  22335. var m;
  22336. /* 18.3.1.83 sheetPr CT_ChartsheetPr */
  22337. var sheetPr = data.match(sheetprregex);
  22338. if(sheetPr) parse_ws_xml_sheetpr(sheetPr[0], s, wb, idx);
  22339. /* 18.3.1.36 drawing CT_Drawing */
  22340. if((m = data.match(/drawing r:id="(.*?)"/))) s['!rel'] = m[1];
  22341. if(rels['!id'][s['!rel']]) s['!chart'] = rels['!id'][s['!rel']];
  22342. return s;
  22343. }
  22344. function write_cs_xml(idx, opts, wb, rels) {
  22345. var o = [XML_HEADER, CS_XML_ROOT];
  22346. o[o.length] = writextag("drawing", null, {"r:id": "rId1"});
  22347. add_rels(rels, -1, "../drawings/drawing" + (idx+1) + ".xml", RELS.DRAW);
  22348. if(o.length>2) { o[o.length] = ('</chartsheet>'); o[1]=o[1].replace("/>",">"); }
  22349. return o.join("");
  22350. }
  22351. /* [MS-XLSB] 2.4.331 BrtCsProp */
  22352. function parse_BrtCsProp(data, length) {
  22353. data.l += 10;
  22354. var name = parse_XLWideString(data, length - 10);
  22355. return { name: name };
  22356. }
  22357. /* [MS-XLSB] 2.1.7.7 Chart Sheet */
  22358. function parse_cs_bin(data, opts, idx, rels, wb) {
  22359. if(!data) return data;
  22360. if(!rels) rels = {'!id':{}};
  22361. var s = {'!type':"chart", '!chart':null, '!rel':""};
  22362. var state = [];
  22363. var pass = false;
  22364. recordhopper(data, function cs_parse(val, R_n, RT) {
  22365. switch(RT) {
  22366. case 0x0226: /* 'BrtDrawing' */
  22367. s['!rel'] = val; break;
  22368. case 0x028B: /* 'BrtCsProp' */
  22369. if(!wb.Sheets[idx]) wb.Sheets[idx] = {};
  22370. if(val.name) wb.Sheets[idx].CodeName = val.name;
  22371. break;
  22372. case 0x0232: /* 'BrtBkHim' */
  22373. case 0x028C: /* 'BrtCsPageSetup' */
  22374. case 0x029D: /* 'BrtCsProtection' */
  22375. case 0x02A7: /* 'BrtCsProtectionIso' */
  22376. case 0x0227: /* 'BrtLegacyDrawing' */
  22377. case 0x0228: /* 'BrtLegacyDrawingHF' */
  22378. case 0x01DC: /* 'BrtMargins' */
  22379. case 0x0C00: /* 'BrtUid' */
  22380. break;
  22381. case 0x0023: /* 'BrtFRTBegin' */
  22382. pass = true; break;
  22383. case 0x0024: /* 'BrtFRTEnd' */
  22384. pass = false; break;
  22385. case 0x0025: /* 'BrtACBegin' */
  22386. state.push(R_n); break;
  22387. case 0x0026: /* 'BrtACEnd' */
  22388. state.pop(); break;
  22389. default:
  22390. if((R_n||"").indexOf("Begin") > 0) state.push(R_n);
  22391. else if((R_n||"").indexOf("End") > 0) state.pop();
  22392. else if(!pass || opts.WTF) throw new Error("Unexpected record " + RT + " " + R_n);
  22393. }
  22394. }, opts);
  22395. if(rels['!id'][s['!rel']]) s['!chart'] = rels['!id'][s['!rel']];
  22396. return s;
  22397. }
  22398. function write_cs_bin() {
  22399. var ba = buf_array();
  22400. write_record(ba, "BrtBeginSheet");
  22401. /* [BrtCsProp] */
  22402. /* CSVIEWS */
  22403. /* [[BrtCsProtectionIso] BrtCsProtection] */
  22404. /* [USERCSVIEWS] */
  22405. /* [BrtMargins] */
  22406. /* [BrtCsPageSetup] */
  22407. /* [HEADERFOOTER] */
  22408. /* BrtDrawing */
  22409. /* [BrtLegacyDrawing] */
  22410. /* [BrtLegacyDrawingHF] */
  22411. /* [BrtBkHim] */
  22412. /* [WEBPUBITEMS] */
  22413. /* FRTCHARTSHEET */
  22414. write_record(ba, "BrtEndSheet");
  22415. return ba.end();
  22416. }
  22417. /* 18.2.28 (CT_WorkbookProtection) Defaults */
  22418. var WBPropsDef = [
  22419. ['allowRefreshQuery', false, "bool"],
  22420. ['autoCompressPictures', true, "bool"],
  22421. ['backupFile', false, "bool"],
  22422. ['checkCompatibility', false, "bool"],
  22423. ['CodeName', ''],
  22424. ['date1904', false, "bool"],
  22425. ['defaultThemeVersion', 0, "int"],
  22426. ['filterPrivacy', false, "bool"],
  22427. ['hidePivotFieldList', false, "bool"],
  22428. ['promptedSolutions', false, "bool"],
  22429. ['publishItems', false, "bool"],
  22430. ['refreshAllConnections', false, "bool"],
  22431. ['saveExternalLinkValues', true, "bool"],
  22432. ['showBorderUnselectedTables', true, "bool"],
  22433. ['showInkAnnotation', true, "bool"],
  22434. ['showObjects', 'all'],
  22435. ['showPivotChartFilter', false, "bool"],
  22436. ['updateLinks', 'userSet']
  22437. ];
  22438. /* 18.2.30 (CT_BookView) Defaults */
  22439. var WBViewDef = [
  22440. ['activeTab', 0, "int"],
  22441. ['autoFilterDateGrouping', true, "bool"],
  22442. ['firstSheet', 0, "int"],
  22443. ['minimized', false, "bool"],
  22444. ['showHorizontalScroll', true, "bool"],
  22445. ['showSheetTabs', true, "bool"],
  22446. ['showVerticalScroll', true, "bool"],
  22447. ['tabRatio', 600, "int"],
  22448. ['visibility', 'visible']
  22449. //window{Height,Width}, {x,y}Window
  22450. ];
  22451. /* 18.2.19 (CT_Sheet) Defaults */
  22452. var SheetDef = [
  22453. //['state', 'visible']
  22454. ];
  22455. /* 18.2.2 (CT_CalcPr) Defaults */
  22456. var CalcPrDef = [
  22457. ['calcCompleted', 'true'],
  22458. ['calcMode', 'auto'],
  22459. ['calcOnSave', 'true'],
  22460. ['concurrentCalc', 'true'],
  22461. ['fullCalcOnLoad', 'false'],
  22462. ['fullPrecision', 'true'],
  22463. ['iterate', 'false'],
  22464. ['iterateCount', '100'],
  22465. ['iterateDelta', '0.001'],
  22466. ['refMode', 'A1']
  22467. ];
  22468. /* 18.2.3 (CT_CustomWorkbookView) Defaults */
  22469. /*var CustomWBViewDef = [
  22470. ['autoUpdate', 'false'],
  22471. ['changesSavedWin', 'false'],
  22472. ['includeHiddenRowCol', 'true'],
  22473. ['includePrintSettings', 'true'],
  22474. ['maximized', 'false'],
  22475. ['minimized', 'false'],
  22476. ['onlySync', 'false'],
  22477. ['personalView', 'false'],
  22478. ['showComments', 'commIndicator'],
  22479. ['showFormulaBar', 'true'],
  22480. ['showHorizontalScroll', 'true'],
  22481. ['showObjects', 'all'],
  22482. ['showSheetTabs', 'true'],
  22483. ['showStatusbar', 'true'],
  22484. ['showVerticalScroll', 'true'],
  22485. ['tabRatio', '600'],
  22486. ['xWindow', '0'],
  22487. ['yWindow', '0']
  22488. ];*/
  22489. function push_defaults_array(target, defaults) {
  22490. for(var j = 0; j != target.length; ++j) { var w = target[j];
  22491. for(var i=0; i != defaults.length; ++i) { var z = defaults[i];
  22492. if(w[z[0]] == null) w[z[0]] = z[1];
  22493. else switch(z[2]) {
  22494. case "bool": if(typeof w[z[0]] == "string") w[z[0]] = parsexmlbool(w[z[0]]); break;
  22495. case "int": if(typeof w[z[0]] == "string") w[z[0]] = parseInt(w[z[0]], 10); break;
  22496. }
  22497. }
  22498. }
  22499. }
  22500. function push_defaults(target, defaults) {
  22501. for(var i = 0; i != defaults.length; ++i) { var z = defaults[i];
  22502. if(target[z[0]] == null) target[z[0]] = z[1];
  22503. else switch(z[2]) {
  22504. case "bool": if(typeof target[z[0]] == "string") target[z[0]] = parsexmlbool(target[z[0]]); break;
  22505. case "int": if(typeof target[z[0]] == "string") target[z[0]] = parseInt(target[z[0]], 10); break;
  22506. }
  22507. }
  22508. }
  22509. function parse_wb_defaults(wb) {
  22510. push_defaults(wb.WBProps, WBPropsDef);
  22511. push_defaults(wb.CalcPr, CalcPrDef);
  22512. push_defaults_array(wb.WBView, WBViewDef);
  22513. push_defaults_array(wb.Sheets, SheetDef);
  22514. _ssfopts.date1904 = parsexmlbool(wb.WBProps.date1904);
  22515. }
  22516. function safe1904(wb) {
  22517. /* TODO: store date1904 somewhere else */
  22518. if(!wb.Workbook) return "false";
  22519. if(!wb.Workbook.WBProps) return "false";
  22520. return parsexmlbool(wb.Workbook.WBProps.date1904) ? "true" : "false";
  22521. }
  22522. var badchars = "][*?\/\\".split("");
  22523. function check_ws_name(n, safe) {
  22524. if(n.length > 31) { if(safe) return false; throw new Error("Sheet names cannot exceed 31 chars"); }
  22525. var _good = true;
  22526. badchars.forEach(function(c) {
  22527. if(n.indexOf(c) == -1) return;
  22528. if(!safe) throw new Error("Sheet name cannot contain : \\ / ? * [ ]");
  22529. _good = false;
  22530. });
  22531. return _good;
  22532. }
  22533. function check_wb_names(N, S, codes) {
  22534. N.forEach(function(n,i) {
  22535. check_ws_name(n);
  22536. for(var j = 0; j < i; ++j) if(n == N[j]) throw new Error("Duplicate Sheet Name: " + n);
  22537. if(codes) {
  22538. var cn = (S && S[i] && S[i].CodeName) || n;
  22539. if(cn.charCodeAt(0) == 95 && cn.length > 22) throw new Error("Bad Code Name: Worksheet" + cn);
  22540. }
  22541. });
  22542. }
  22543. function check_wb(wb) {
  22544. if(!wb || !wb.SheetNames || !wb.Sheets) throw new Error("Invalid Workbook");
  22545. if(!wb.SheetNames.length) throw new Error("Workbook is empty");
  22546. var Sheets = (wb.Workbook && wb.Workbook.Sheets) || [];
  22547. check_wb_names(wb.SheetNames, Sheets, !!wb.vbaraw);
  22548. for(var i = 0; i < wb.SheetNames.length; ++i) check_ws(wb.Sheets[wb.SheetNames[i]], wb.SheetNames[i], i);
  22549. /* TODO: validate workbook */
  22550. }
  22551. /* 18.2 Workbook */
  22552. var wbnsregex = /<\w+:workbook/;
  22553. function parse_wb_xml(data, opts) {
  22554. if(!data) throw new Error("Could not find file");
  22555. var wb = { AppVersion:{}, WBProps:{}, WBView:[], Sheets:[], CalcPr:{}, Names:[], xmlns: "" };
  22556. var pass = false, xmlns = "xmlns";
  22557. var dname = {}, dnstart = 0;
  22558. data.replace(tagregex, function xml_wb(x, idx) {
  22559. var y = parsexmltag(x);
  22560. switch(strip_ns(y[0])) {
  22561. case '<?xml': break;
  22562. /* 18.2.27 workbook CT_Workbook 1 */
  22563. case '<workbook':
  22564. if(x.match(wbnsregex)) xmlns = "xmlns" + x.match(/<(\w+):/)[1];
  22565. wb.xmlns = y[xmlns];
  22566. break;
  22567. case '</workbook>': break;
  22568. /* 18.2.13 fileVersion CT_FileVersion ? */
  22569. case '<fileVersion': delete y[0]; wb.AppVersion = y; break;
  22570. case '<fileVersion/>': case '</fileVersion>': break;
  22571. /* 18.2.12 fileSharing CT_FileSharing ? */
  22572. case '<fileSharing': case '<fileSharing/>': break;
  22573. /* 18.2.28 workbookPr CT_WorkbookPr ? */
  22574. case '<workbookPr':
  22575. case '<workbookPr/>':
  22576. WBPropsDef.forEach(function(w) {
  22577. if(y[w[0]] == null) return;
  22578. switch(w[2]) {
  22579. case "bool": wb.WBProps[w[0]] = parsexmlbool(y[w[0]]); break;
  22580. case "int": wb.WBProps[w[0]] = parseInt(y[w[0]], 10); break;
  22581. default: wb.WBProps[w[0]] = y[w[0]];
  22582. }
  22583. });
  22584. if(y.codeName) wb.WBProps.CodeName = y.codeName;
  22585. break;
  22586. case '</workbookPr>': break;
  22587. /* 18.2.29 workbookProtection CT_WorkbookProtection ? */
  22588. case '<workbookProtection': break;
  22589. case '<workbookProtection/>': break;
  22590. /* 18.2.1 bookViews CT_BookViews ? */
  22591. case '<bookViews': case '<bookViews>': case '</bookViews>': break;
  22592. /* 18.2.30 workbookView CT_BookView + */
  22593. case '<workbookView': case '<workbookView/>': delete y[0]; wb.WBView.push(y); break;
  22594. case '</workbookView>': break;
  22595. /* 18.2.20 sheets CT_Sheets 1 */
  22596. case '<sheets': case '<sheets>': case '</sheets>': break; // aggregate sheet
  22597. /* 18.2.19 sheet CT_Sheet + */
  22598. case '<sheet':
  22599. switch(y.state) {
  22600. case "hidden": y.Hidden = 1; break;
  22601. case "veryHidden": y.Hidden = 2; break;
  22602. default: y.Hidden = 0;
  22603. }
  22604. delete y.state;
  22605. y.name = unescapexml(utf8read(y.name));
  22606. delete y[0]; wb.Sheets.push(y); break;
  22607. case '</sheet>': break;
  22608. /* 18.2.15 functionGroups CT_FunctionGroups ? */
  22609. case '<functionGroups': case '<functionGroups/>': break;
  22610. /* 18.2.14 functionGroup CT_FunctionGroup + */
  22611. case '<functionGroup': break;
  22612. /* 18.2.9 externalReferences CT_ExternalReferences ? */
  22613. case '<externalReferences': case '</externalReferences>': case '<externalReferences>': break;
  22614. /* 18.2.8 externalReference CT_ExternalReference + */
  22615. case '<externalReference': break;
  22616. /* 18.2.6 definedNames CT_DefinedNames ? */
  22617. case '<definedNames/>': break;
  22618. case '<definedNames>': case '<definedNames': pass=true; break;
  22619. case '</definedNames>': pass=false; break;
  22620. /* 18.2.5 definedName CT_DefinedName + */
  22621. case '<definedName': {
  22622. dname = {};
  22623. dname.Name = utf8read(y.name);
  22624. if(y.comment) dname.Comment = y.comment;
  22625. if(y.localSheetId) dname.Sheet = +y.localSheetId;
  22626. if(parsexmlbool(y.hidden||"0")) dname.Hidden = true;
  22627. dnstart = idx + x.length;
  22628. } break;
  22629. case '</definedName>': {
  22630. dname.Ref = unescapexml(utf8read(data.slice(dnstart, idx)));
  22631. wb.Names.push(dname);
  22632. } break;
  22633. case '<definedName/>': break;
  22634. /* 18.2.2 calcPr CT_CalcPr ? */
  22635. case '<calcPr': delete y[0]; wb.CalcPr = y; break;
  22636. case '<calcPr/>': delete y[0]; wb.CalcPr = y; break;
  22637. case '</calcPr>': break;
  22638. /* 18.2.16 oleSize CT_OleSize ? (ref required) */
  22639. case '<oleSize': break;
  22640. /* 18.2.4 customWorkbookViews CT_CustomWorkbookViews ? */
  22641. case '<customWorkbookViews>': case '</customWorkbookViews>': case '<customWorkbookViews': break;
  22642. /* 18.2.3 customWorkbookView CT_CustomWorkbookView + */
  22643. case '<customWorkbookView': case '</customWorkbookView>': break;
  22644. /* 18.2.18 pivotCaches CT_PivotCaches ? */
  22645. case '<pivotCaches>': case '</pivotCaches>': case '<pivotCaches': break;
  22646. /* 18.2.17 pivotCache CT_PivotCache ? */
  22647. case '<pivotCache': break;
  22648. /* 18.2.21 smartTagPr CT_SmartTagPr ? */
  22649. case '<smartTagPr': case '<smartTagPr/>': break;
  22650. /* 18.2.23 smartTagTypes CT_SmartTagTypes ? */
  22651. case '<smartTagTypes': case '<smartTagTypes>': case '</smartTagTypes>': break;
  22652. /* 18.2.22 smartTagType CT_SmartTagType ? */
  22653. case '<smartTagType': break;
  22654. /* 18.2.24 webPublishing CT_WebPublishing ? */
  22655. case '<webPublishing': case '<webPublishing/>': break;
  22656. /* 18.2.11 fileRecoveryPr CT_FileRecoveryPr ? */
  22657. case '<fileRecoveryPr': case '<fileRecoveryPr/>': break;
  22658. /* 18.2.26 webPublishObjects CT_WebPublishObjects ? */
  22659. case '<webPublishObjects>': case '<webPublishObjects': case '</webPublishObjects>': break;
  22660. /* 18.2.25 webPublishObject CT_WebPublishObject ? */
  22661. case '<webPublishObject': break;
  22662. /* 18.2.10 extLst CT_ExtensionList ? */
  22663. case '<extLst': case '<extLst>': case '</extLst>': case '<extLst/>': break;
  22664. /* 18.2.7 ext CT_Extension + */
  22665. case '<ext': pass=true; break; //TODO: check with versions of excel
  22666. case '</ext>': pass=false; break;
  22667. /* Others */
  22668. case '<ArchID': break;
  22669. case '<AlternateContent':
  22670. case '<AlternateContent>': pass=true; break;
  22671. case '</AlternateContent>': pass=false; break;
  22672. /* TODO */
  22673. case '<revisionPtr': break;
  22674. default: if(!pass && opts.WTF) throw new Error('unrecognized ' + y[0] + ' in workbook');
  22675. }
  22676. return x;
  22677. });
  22678. if(XMLNS.main.indexOf(wb.xmlns) === -1) throw new Error("Unknown Namespace: " + wb.xmlns);
  22679. parse_wb_defaults(wb);
  22680. return wb;
  22681. }
  22682. var WB_XML_ROOT = writextag('workbook', null, {
  22683. 'xmlns': XMLNS.main[0],
  22684. //'xmlns:mx': XMLNS.mx,
  22685. //'xmlns:s': XMLNS.main[0],
  22686. 'xmlns:r': XMLNS.r
  22687. });
  22688. function write_wb_xml(wb) {
  22689. var o = [XML_HEADER];
  22690. o[o.length] = WB_XML_ROOT;
  22691. var write_names = (wb.Workbook && (wb.Workbook.Names||[]).length > 0);
  22692. /* fileVersion */
  22693. /* fileSharing */
  22694. var workbookPr = ({codeName:"ThisWorkbook"});
  22695. if(wb.Workbook && wb.Workbook.WBProps) {
  22696. WBPropsDef.forEach(function(x) {
  22697. if((wb.Workbook.WBProps[x[0]]) == null) return;
  22698. if((wb.Workbook.WBProps[x[0]]) == x[1]) return;
  22699. workbookPr[x[0]] = (wb.Workbook.WBProps[x[0]]);
  22700. });
  22701. if(wb.Workbook.WBProps.CodeName) { workbookPr.codeName = wb.Workbook.WBProps.CodeName; delete workbookPr.CodeName; }
  22702. }
  22703. o[o.length] = (writextag('workbookPr', null, workbookPr));
  22704. /* workbookProtection */
  22705. var sheets = wb.Workbook && wb.Workbook.Sheets || [];
  22706. var i = 0;
  22707. /* bookViews */
  22708. o[o.length] = "<sheets>";
  22709. for(i = 0; i != wb.SheetNames.length; ++i) {
  22710. var sht = ({name:escapexml(wb.SheetNames[i].slice(0,31))});
  22711. sht.sheetId = ""+(i+1);
  22712. sht["r:id"] = "rId"+(i+1);
  22713. if(sheets[i]) switch(sheets[i].Hidden) {
  22714. case 1: sht.state = "hidden"; break;
  22715. case 2: sht.state = "veryHidden"; break;
  22716. }
  22717. o[o.length] = (writextag('sheet',null,sht));
  22718. }
  22719. o[o.length] = "</sheets>";
  22720. /* functionGroups */
  22721. /* externalReferences */
  22722. if(write_names) {
  22723. o[o.length] = "<definedNames>";
  22724. if(wb.Workbook && wb.Workbook.Names) wb.Workbook.Names.forEach(function(n) {
  22725. var d = {name:n.Name};
  22726. if(n.Comment) d.comment = n.Comment;
  22727. if(n.Sheet != null) d.localSheetId = ""+n.Sheet;
  22728. if(n.Hidden) d.hidden = "1";
  22729. if(!n.Ref) return;
  22730. o[o.length] = writextag('definedName', String(n.Ref).replace(/</g, "&lt;").replace(/>/g, "&gt;"), d);
  22731. });
  22732. o[o.length] = "</definedNames>";
  22733. }
  22734. /* calcPr */
  22735. /* oleSize */
  22736. /* customWorkbookViews */
  22737. /* pivotCaches */
  22738. /* smartTagPr */
  22739. /* smartTagTypes */
  22740. /* webPublishing */
  22741. /* fileRecoveryPr */
  22742. /* webPublishObjects */
  22743. /* extLst */
  22744. if(o.length>2){ o[o.length] = '</workbook>'; o[1]=o[1].replace("/>",">"); }
  22745. return o.join("");
  22746. }
  22747. /* [MS-XLSB] 2.4.304 BrtBundleSh */
  22748. function parse_BrtBundleSh(data, length) {
  22749. var z = {};
  22750. z.Hidden = data.read_shift(4); //hsState ST_SheetState
  22751. z.iTabID = data.read_shift(4);
  22752. z.strRelID = parse_RelID(data,length-8);
  22753. z.name = parse_XLWideString(data);
  22754. return z;
  22755. }
  22756. function write_BrtBundleSh(data, o) {
  22757. if(!o) o = new_buf(127);
  22758. o.write_shift(4, data.Hidden);
  22759. o.write_shift(4, data.iTabID);
  22760. write_RelID(data.strRelID, o);
  22761. write_XLWideString(data.name.slice(0,31), o);
  22762. return o.length > o.l ? o.slice(0, o.l) : o;
  22763. }
  22764. /* [MS-XLSB] 2.4.815 BrtWbProp */
  22765. function parse_BrtWbProp(data, length) {
  22766. var o = ({});
  22767. var flags = data.read_shift(4);
  22768. o.defaultThemeVersion = data.read_shift(4);
  22769. var strName = (length > 8) ? parse_XLWideString(data) : "";
  22770. if(strName.length > 0) o.CodeName = strName;
  22771. o.autoCompressPictures = !!(flags & 0x10000);
  22772. o.backupFile = !!(flags & 0x40);
  22773. o.checkCompatibility = !!(flags & 0x1000);
  22774. o.date1904 = !!(flags & 0x01);
  22775. o.filterPrivacy = !!(flags & 0x08);
  22776. o.hidePivotFieldList = !!(flags & 0x400);
  22777. o.promptedSolutions = !!(flags & 0x10);
  22778. o.publishItems = !!(flags & 0x800);
  22779. o.refreshAllConnections = !!(flags & 0x40000);
  22780. o.saveExternalLinkValues = !!(flags & 0x80);
  22781. o.showBorderUnselectedTables = !!(flags & 0x04);
  22782. o.showInkAnnotation = !!(flags & 0x20);
  22783. o.showObjects = ["all", "placeholders", "none"][(flags >> 13) & 0x03];
  22784. o.showPivotChartFilter = !!(flags & 0x8000);
  22785. o.updateLinks = ["userSet", "never", "always"][(flags >> 8) & 0x03];
  22786. return o;
  22787. }
  22788. function write_BrtWbProp(data, o) {
  22789. if(!o) o = new_buf(72);
  22790. var flags = 0;
  22791. if(data) {
  22792. /* TODO: mirror parse_BrtWbProp fields */
  22793. if(data.filterPrivacy) flags |= 0x08;
  22794. }
  22795. o.write_shift(4, flags);
  22796. o.write_shift(4, 0);
  22797. write_XLSBCodeName(data && data.CodeName || "ThisWorkbook", o);
  22798. return o.slice(0, o.l);
  22799. }
  22800. function parse_BrtFRTArchID$(data, length) {
  22801. var o = {};
  22802. data.read_shift(4);
  22803. o.ArchID = data.read_shift(4);
  22804. data.l += length - 8;
  22805. return o;
  22806. }
  22807. /* [MS-XLSB] 2.4.687 BrtName */
  22808. function parse_BrtName(data, length, opts) {
  22809. var end = data.l + length;
  22810. data.l += 4; //var flags = data.read_shift(4);
  22811. data.l += 1; //var chKey = data.read_shift(1);
  22812. var itab = data.read_shift(4);
  22813. var name = parse_XLNameWideString(data);
  22814. var formula = parse_XLSBNameParsedFormula(data, 0, opts);
  22815. var comment = parse_XLNullableWideString(data);
  22816. //if(0 /* fProc */) {
  22817. // unusedstring1: XLNullableWideString
  22818. // description: XLNullableWideString
  22819. // helpTopic: XLNullableWideString
  22820. // unusedstring2: XLNullableWideString
  22821. //}
  22822. data.l = end;
  22823. var out = ({Name:name, Ptg:formula});
  22824. if(itab < 0xFFFFFFF) out.Sheet = itab;
  22825. if(comment) out.Comment = comment;
  22826. return out;
  22827. }
  22828. /* [MS-XLSB] 2.1.7.61 Workbook */
  22829. function parse_wb_bin(data, opts) {
  22830. var wb = { AppVersion:{}, WBProps:{}, WBView:[], Sheets:[], CalcPr:{}, xmlns: "" };
  22831. var state = [];
  22832. var pass = false;
  22833. if(!opts) opts = {};
  22834. opts.biff = 12;
  22835. var Names = [];
  22836. var supbooks = ([[]]);
  22837. supbooks.SheetNames = [];
  22838. supbooks.XTI = [];
  22839. recordhopper(data, function hopper_wb(val, R_n, RT) {
  22840. switch(RT) {
  22841. case 0x009C: /* 'BrtBundleSh' */
  22842. supbooks.SheetNames.push(val.name);
  22843. wb.Sheets.push(val); break;
  22844. case 0x0099: /* 'BrtWbProp' */
  22845. wb.WBProps = val; break;
  22846. case 0x0027: /* 'BrtName' */
  22847. if(val.Sheet != null) opts.SID = val.Sheet;
  22848. val.Ref = stringify_formula(val.Ptg, null, null, supbooks, opts);
  22849. delete opts.SID;
  22850. delete val.Ptg;
  22851. Names.push(val);
  22852. break;
  22853. case 0x040C: /* 'BrtNameExt' */ break;
  22854. case 0x0165: /* 'BrtSupSelf' */
  22855. case 0x0166: /* 'BrtSupSame' */
  22856. case 0x0163: /* 'BrtSupBookSrc' */
  22857. case 0x029B: /* 'BrtSupAddin' */
  22858. if(!supbooks[0].length) supbooks[0] = [RT, val];
  22859. else supbooks.push([RT, val]);
  22860. supbooks[supbooks.length - 1].XTI = [];
  22861. break;
  22862. case 0x016A: /* 'BrtExternSheet' */
  22863. if(supbooks.length === 0) { supbooks[0] = []; supbooks[0].XTI = []; }
  22864. supbooks[supbooks.length - 1].XTI = supbooks[supbooks.length - 1].XTI.concat(val);
  22865. supbooks.XTI = supbooks.XTI.concat(val);
  22866. break;
  22867. case 0x0169: /* 'BrtPlaceholderName' */
  22868. break;
  22869. /* case 'BrtModelTimeGroupingCalcCol' */
  22870. case 0x0C00: /* 'BrtUid' */
  22871. case 0x0C01: /* 'BrtRevisionPtr' */
  22872. case 0x0817: /* 'BrtAbsPath15' */
  22873. case 0x0216: /* 'BrtBookProtection' */
  22874. case 0x02A5: /* 'BrtBookProtectionIso' */
  22875. case 0x009E: /* 'BrtBookView' */
  22876. case 0x009D: /* 'BrtCalcProp' */
  22877. case 0x0262: /* 'BrtCrashRecErr' */
  22878. case 0x0802: /* 'BrtDecoupledPivotCacheID' */
  22879. case 0x009B: /* 'BrtFileRecover' */
  22880. case 0x0224: /* 'BrtFileSharing' */
  22881. case 0x02A4: /* 'BrtFileSharingIso' */
  22882. case 0x0080: /* 'BrtFileVersion' */
  22883. case 0x0299: /* 'BrtFnGroup' */
  22884. case 0x0850: /* 'BrtModelRelationship' */
  22885. case 0x084D: /* 'BrtModelTable' */
  22886. case 0x0225: /* 'BrtOleSize' */
  22887. case 0x0805: /* 'BrtPivotTableRef' */
  22888. case 0x0254: /* 'BrtSmartTagType' */
  22889. case 0x081C: /* 'BrtTableSlicerCacheID' */
  22890. case 0x081B: /* 'BrtTableSlicerCacheIDs' */
  22891. case 0x0822: /* 'BrtTimelineCachePivotCacheID' */
  22892. case 0x018D: /* 'BrtUserBookView' */
  22893. case 0x009A: /* 'BrtWbFactoid' */
  22894. case 0x045D: /* 'BrtWbProp14' */
  22895. case 0x0229: /* 'BrtWebOpt' */
  22896. case 0x082B: /* 'BrtWorkBookPr15' */
  22897. break;
  22898. case 0x0023: /* 'BrtFRTBegin' */
  22899. state.push(R_n); pass = true; break;
  22900. case 0x0024: /* 'BrtFRTEnd' */
  22901. state.pop(); pass = false; break;
  22902. case 0x0025: /* 'BrtACBegin' */
  22903. state.push(R_n); pass = true; break;
  22904. case 0x0026: /* 'BrtACEnd' */
  22905. state.pop(); pass = false; break;
  22906. case 0x0010: /* 'BrtFRTArchID$' */ break;
  22907. default:
  22908. if((R_n||"").indexOf("Begin") > 0){/* empty */}
  22909. else if((R_n||"").indexOf("End") > 0){/* empty */}
  22910. else if(!pass || (opts.WTF && state[state.length-1] != "BrtACBegin" && state[state.length-1] != "BrtFRTBegin")) throw new Error("Unexpected record " + RT + " " + R_n);
  22911. }
  22912. }, opts);
  22913. parse_wb_defaults(wb);
  22914. // $FlowIgnore
  22915. wb.Names = Names;
  22916. (wb).supbooks = supbooks;
  22917. return wb;
  22918. }
  22919. function write_BUNDLESHS(ba, wb) {
  22920. write_record(ba, "BrtBeginBundleShs");
  22921. for(var idx = 0; idx != wb.SheetNames.length; ++idx) {
  22922. var viz = wb.Workbook && wb.Workbook.Sheets && wb.Workbook.Sheets[idx] && wb.Workbook.Sheets[idx].Hidden || 0;
  22923. var d = { Hidden: viz, iTabID: idx+1, strRelID: 'rId' + (idx+1), name: wb.SheetNames[idx] };
  22924. write_record(ba, "BrtBundleSh", write_BrtBundleSh(d));
  22925. }
  22926. write_record(ba, "BrtEndBundleShs");
  22927. }
  22928. /* [MS-XLSB] 2.4.649 BrtFileVersion */
  22929. function write_BrtFileVersion(data, o) {
  22930. if(!o) o = new_buf(127);
  22931. for(var i = 0; i != 4; ++i) o.write_shift(4, 0);
  22932. write_XLWideString("SheetJS", o);
  22933. write_XLWideString(XLSX.version, o);
  22934. write_XLWideString(XLSX.version, o);
  22935. write_XLWideString("7262", o);
  22936. o.length = o.l;
  22937. return o.length > o.l ? o.slice(0, o.l) : o;
  22938. }
  22939. /* [MS-XLSB] 2.4.301 BrtBookView */
  22940. function write_BrtBookView(idx, o) {
  22941. if(!o) o = new_buf(29);
  22942. o.write_shift(-4, 0);
  22943. o.write_shift(-4, 460);
  22944. o.write_shift(4, 28800);
  22945. o.write_shift(4, 17600);
  22946. o.write_shift(4, 500);
  22947. o.write_shift(4, idx);
  22948. o.write_shift(4, idx);
  22949. var flags = 0x78;
  22950. o.write_shift(1, flags);
  22951. return o.length > o.l ? o.slice(0, o.l) : o;
  22952. }
  22953. function write_BOOKVIEWS(ba, wb) {
  22954. /* required if hidden tab appears before visible tab */
  22955. if(!wb.Workbook || !wb.Workbook.Sheets) return;
  22956. var sheets = wb.Workbook.Sheets;
  22957. var i = 0, vistab = -1, hidden = -1;
  22958. for(; i < sheets.length; ++i) {
  22959. if(!sheets[i] || !sheets[i].Hidden && vistab == -1) vistab = i;
  22960. else if(sheets[i].Hidden == 1 && hidden == -1) hidden = i;
  22961. }
  22962. if(hidden > vistab) return;
  22963. write_record(ba, "BrtBeginBookViews");
  22964. write_record(ba, "BrtBookView", write_BrtBookView(vistab));
  22965. /* 1*(BrtBookView *FRT) */
  22966. write_record(ba, "BrtEndBookViews");
  22967. }
  22968. /* [MS-XLSB] 2.4.305 BrtCalcProp */
  22969. /*function write_BrtCalcProp(data, o) {
  22970. if(!o) o = new_buf(26);
  22971. o.write_shift(4,0); // force recalc
  22972. o.write_shift(4,1);
  22973. o.write_shift(4,0);
  22974. write_Xnum(0, o);
  22975. o.write_shift(-4, 1023);
  22976. o.write_shift(1, 0x33);
  22977. o.write_shift(1, 0x00);
  22978. return o;
  22979. }*/
  22980. /* [MS-XLSB] 2.4.646 BrtFileRecover */
  22981. /*function write_BrtFileRecover(data, o) {
  22982. if(!o) o = new_buf(1);
  22983. o.write_shift(1,0);
  22984. return o;
  22985. }*/
  22986. /* [MS-XLSB] 2.1.7.61 Workbook */
  22987. function write_wb_bin(wb, opts) {
  22988. var ba = buf_array();
  22989. write_record(ba, "BrtBeginBook");
  22990. write_record(ba, "BrtFileVersion", write_BrtFileVersion());
  22991. /* [[BrtFileSharingIso] BrtFileSharing] */
  22992. write_record(ba, "BrtWbProp", write_BrtWbProp(wb.Workbook && wb.Workbook.WBProps || null));
  22993. /* [ACABSPATH] */
  22994. /* [[BrtBookProtectionIso] BrtBookProtection] */
  22995. write_BOOKVIEWS(ba, wb, opts);
  22996. write_BUNDLESHS(ba, wb, opts);
  22997. /* [FNGROUP] */
  22998. /* [EXTERNALS] */
  22999. /* *BrtName */
  23000. /* write_record(ba, "BrtCalcProp", write_BrtCalcProp()); */
  23001. /* [BrtOleSize] */
  23002. /* *(BrtUserBookView *FRT) */
  23003. /* [PIVOTCACHEIDS] */
  23004. /* [BrtWbFactoid] */
  23005. /* [SMARTTAGTYPES] */
  23006. /* [BrtWebOpt] */
  23007. /* write_record(ba, "BrtFileRecover", write_BrtFileRecover()); */
  23008. /* [WEBPUBITEMS] */
  23009. /* [CRERRS] */
  23010. /* FRTWORKBOOK */
  23011. write_record(ba, "BrtEndBook");
  23012. return ba.end();
  23013. }
  23014. function parse_wb(data, name, opts) {
  23015. if(name.slice(-4)===".bin") return parse_wb_bin((data), opts);
  23016. return parse_wb_xml((data), opts);
  23017. }
  23018. function parse_ws(data, name, idx, opts, rels, wb, themes, styles) {
  23019. if(name.slice(-4)===".bin") return parse_ws_bin((data), opts, idx, rels, wb, themes, styles);
  23020. return parse_ws_xml((data), opts, idx, rels, wb, themes, styles);
  23021. }
  23022. function parse_cs(data, name, idx, opts, rels, wb, themes, styles) {
  23023. if(name.slice(-4)===".bin") return parse_cs_bin((data), opts, idx, rels, wb, themes, styles);
  23024. return parse_cs_xml((data), opts, idx, rels, wb, themes, styles);
  23025. }
  23026. function parse_ms(data, name, idx, opts, rels, wb, themes, styles) {
  23027. if(name.slice(-4)===".bin") return parse_ms_bin((data), opts, idx, rels, wb, themes, styles);
  23028. return parse_ms_xml((data), opts, idx, rels, wb, themes, styles);
  23029. }
  23030. function parse_ds(data, name, idx, opts, rels, wb, themes, styles) {
  23031. if(name.slice(-4)===".bin") return parse_ds_bin((data), opts, idx, rels, wb, themes, styles);
  23032. return parse_ds_xml((data), opts, idx, rels, wb, themes, styles);
  23033. }
  23034. function parse_sty(data, name, themes, opts) {
  23035. if(name.slice(-4)===".bin") return parse_sty_bin((data), themes, opts);
  23036. return parse_sty_xml((data), themes, opts);
  23037. }
  23038. function parse_theme(data, name, opts) {
  23039. return parse_theme_xml(data, opts);
  23040. }
  23041. function parse_sst(data, name, opts) {
  23042. if(name.slice(-4)===".bin") return parse_sst_bin((data), opts);
  23043. return parse_sst_xml((data), opts);
  23044. }
  23045. function parse_cmnt(data, name, opts) {
  23046. if(name.slice(-4)===".bin") return parse_comments_bin((data), opts);
  23047. return parse_comments_xml((data), opts);
  23048. }
  23049. function parse_cc(data, name, opts) {
  23050. if(name.slice(-4)===".bin") return parse_cc_bin((data), name, opts);
  23051. return parse_cc_xml((data), name, opts);
  23052. }
  23053. function parse_xlink(data, name, opts) {
  23054. if(name.slice(-4)===".bin") return parse_xlink_bin((data), name, opts);
  23055. return parse_xlink_xml((data), name, opts);
  23056. }
  23057. function write_wb(wb, name, opts) {
  23058. return (name.slice(-4)===".bin" ? write_wb_bin : write_wb_xml)(wb, opts);
  23059. }
  23060. function write_ws(data, name, opts, wb, rels) {
  23061. return (name.slice(-4)===".bin" ? write_ws_bin : write_ws_xml)(data, opts, wb, rels);
  23062. }
  23063. // eslint-disable-next-line no-unused-vars
  23064. function write_cs(data, name, opts, wb, rels) {
  23065. return (name.slice(-4)===".bin" ? write_cs_bin : write_cs_xml)(data, opts, wb, rels);
  23066. }
  23067. function write_sty(data, name, opts) {
  23068. return (name.slice(-4)===".bin" ? write_sty_bin : write_sty_xml)(data, opts);
  23069. }
  23070. function write_sst(data, name, opts) {
  23071. return (name.slice(-4)===".bin" ? write_sst_bin : write_sst_xml)(data, opts);
  23072. }
  23073. function write_cmnt(data, name, opts) {
  23074. return (name.slice(-4)===".bin" ? write_comments_bin : write_comments_xml)(data, opts);
  23075. }
  23076. /*
  23077. function write_cc(data, name:string, opts) {
  23078. return (name.slice(-4)===".bin" ? write_cc_bin : write_cc_xml)(data, opts);
  23079. }
  23080. */
  23081. var attregexg2=/([\w:]+)=((?:")([^"]*)(?:")|(?:')([^']*)(?:'))/g;
  23082. var attregex2=/([\w:]+)=((?:")(?:[^"]*)(?:")|(?:')(?:[^']*)(?:'))/;
  23083. var _chr = function(c) { return String.fromCharCode(c); };
  23084. function xlml_parsexmltag(tag, skip_root) {
  23085. var words = tag.split(/\s+/);
  23086. var z = ([]); if(!skip_root) z[0] = words[0];
  23087. if(words.length === 1) return z;
  23088. var m = tag.match(attregexg2), y, j, w, i;
  23089. if(m) for(i = 0; i != m.length; ++i) {
  23090. y = m[i].match(attregex2);
  23091. if((j=y[1].indexOf(":")) === -1) z[y[1]] = y[2].slice(1,y[2].length-1);
  23092. else {
  23093. if(y[1].slice(0,6) === "xmlns:") w = "xmlns"+y[1].slice(6);
  23094. else w = y[1].slice(j+1);
  23095. z[w] = y[2].slice(1,y[2].length-1);
  23096. }
  23097. }
  23098. return z;
  23099. }
  23100. function xlml_parsexmltagobj(tag) {
  23101. var words = tag.split(/\s+/);
  23102. var z = {};
  23103. if(words.length === 1) return z;
  23104. var m = tag.match(attregexg2), y, j, w, i;
  23105. if(m) for(i = 0; i != m.length; ++i) {
  23106. y = m[i].match(attregex2);
  23107. if((j=y[1].indexOf(":")) === -1) z[y[1]] = y[2].slice(1,y[2].length-1);
  23108. else {
  23109. if(y[1].slice(0,6) === "xmlns:") w = "xmlns"+y[1].slice(6);
  23110. else w = y[1].slice(j+1);
  23111. z[w] = y[2].slice(1,y[2].length-1);
  23112. }
  23113. }
  23114. return z;
  23115. }
  23116. // ----
  23117. function xlml_format(format, value) {
  23118. var fmt = XLMLFormatMap[format] || unescapexml(format);
  23119. if(fmt === "General") return SSF._general(value);
  23120. return SSF.format(fmt, value);
  23121. }
  23122. function xlml_set_custprop(Custprops, key, cp, val) {
  23123. var oval = val;
  23124. switch((cp[0].match(/dt:dt="([\w.]+)"/)||["",""])[1]) {
  23125. case "boolean": oval = parsexmlbool(val); break;
  23126. case "i2": case "int": oval = parseInt(val, 10); break;
  23127. case "r4": case "float": oval = parseFloat(val); break;
  23128. case "date": case "dateTime.tz": oval = parseDate(val); break;
  23129. case "i8": case "string": case "fixed": case "uuid": case "bin.base64": break;
  23130. default: throw new Error("bad custprop:" + cp[0]);
  23131. }
  23132. Custprops[unescapexml(key)] = oval;
  23133. }
  23134. function safe_format_xlml(cell, nf, o) {
  23135. if(cell.t === 'z') return;
  23136. if(!o || o.cellText !== false) try {
  23137. if(cell.t === 'e') { cell.w = cell.w || BErr[cell.v]; }
  23138. else if(nf === "General") {
  23139. if(cell.t === 'n') {
  23140. if((cell.v|0) === cell.v) cell.w = SSF._general_int(cell.v);
  23141. else cell.w = SSF._general_num(cell.v);
  23142. }
  23143. else cell.w = SSF._general(cell.v);
  23144. }
  23145. else cell.w = xlml_format(nf||"General", cell.v);
  23146. } catch(e) { if(o.WTF) throw e; }
  23147. try {
  23148. var z = XLMLFormatMap[nf]||nf||"General";
  23149. if(o.cellNF) cell.z = z;
  23150. if(o.cellDates && cell.t == 'n' && SSF.is_date(z)) {
  23151. var _d = SSF.parse_date_code(cell.v); if(_d) { cell.t = 'd'; cell.v = new Date(_d.y, _d.m-1,_d.d,_d.H,_d.M,_d.S,_d.u); }
  23152. }
  23153. } catch(e) { if(o.WTF) throw e; }
  23154. }
  23155. function process_style_xlml(styles, stag, opts) {
  23156. if(opts.cellStyles) {
  23157. if(stag.Interior) {
  23158. var I = stag.Interior;
  23159. if(I.Pattern) I.patternType = XLMLPatternTypeMap[I.Pattern] || I.Pattern;
  23160. }
  23161. }
  23162. styles[stag.ID] = stag;
  23163. }
  23164. /* TODO: there must exist some form of OSP-blessed spec */
  23165. function parse_xlml_data(xml, ss, data, cell, base, styles, csty, row, arrayf, o) {
  23166. var nf = "General", sid = cell.StyleID, S = {}; o = o || {};
  23167. var interiors = [];
  23168. var i = 0;
  23169. if(sid === undefined && row) sid = row.StyleID;
  23170. if(sid === undefined && csty) sid = csty.StyleID;
  23171. while(styles[sid] !== undefined) {
  23172. if(styles[sid].nf) nf = styles[sid].nf;
  23173. if(styles[sid].Interior) interiors.push(styles[sid].Interior);
  23174. if(!styles[sid].Parent) break;
  23175. sid = styles[sid].Parent;
  23176. }
  23177. switch(data.Type) {
  23178. case 'Boolean':
  23179. cell.t = 'b';
  23180. cell.v = parsexmlbool(xml);
  23181. break;
  23182. case 'String':
  23183. cell.t = 's'; cell.r = xlml_fixstr(unescapexml(xml));
  23184. cell.v = xml.indexOf("<") > -1 ? unescapexml(ss) : cell.r;
  23185. break;
  23186. case 'DateTime':
  23187. if(xml.slice(-1) != "Z") xml += "Z";
  23188. cell.v = (parseDate(xml) - new Date(Date.UTC(1899, 11, 30))) / (24 * 60 * 60 * 1000);
  23189. if(cell.v !== cell.v) cell.v = unescapexml(xml);
  23190. else if(cell.v<60) cell.v = cell.v -1;
  23191. if(!nf || nf == "General") nf = "yyyy-mm-dd";
  23192. /* falls through */
  23193. case 'Number':
  23194. if(cell.v === undefined) cell.v=+xml;
  23195. if(!cell.t) cell.t = 'n';
  23196. break;
  23197. case 'Error': cell.t = 'e'; cell.v = RBErr[xml]; if(o.cellText !== false) cell.w = xml; break;
  23198. default: cell.t = 's'; cell.v = xlml_fixstr(ss||xml); break;
  23199. }
  23200. safe_format_xlml(cell, nf, o);
  23201. if(o.cellFormula !== false) {
  23202. if(cell.Formula) {
  23203. var fstr = unescapexml(cell.Formula);
  23204. /* strictly speaking, the leading = is required but some writers omit */
  23205. if(fstr.charCodeAt(0) == 61 /* = */) fstr = fstr.slice(1);
  23206. cell.f = rc_to_a1(fstr, base);
  23207. delete cell.Formula;
  23208. if(cell.ArrayRange == "RC") cell.F = rc_to_a1("RC:RC", base);
  23209. else if(cell.ArrayRange) {
  23210. cell.F = rc_to_a1(cell.ArrayRange, base);
  23211. arrayf.push([safe_decode_range(cell.F), cell.F]);
  23212. }
  23213. } else {
  23214. for(i = 0; i < arrayf.length; ++i)
  23215. if(base.r >= arrayf[i][0].s.r && base.r <= arrayf[i][0].e.r)
  23216. if(base.c >= arrayf[i][0].s.c && base.c <= arrayf[i][0].e.c)
  23217. cell.F = arrayf[i][1];
  23218. }
  23219. }
  23220. if(o.cellStyles) {
  23221. interiors.forEach(function(x) {
  23222. if(!S.patternType && x.patternType) S.patternType = x.patternType;
  23223. });
  23224. cell.s = S;
  23225. }
  23226. if(cell.StyleID !== undefined) cell.ixfe = cell.StyleID;
  23227. }
  23228. function xlml_clean_comment(comment) {
  23229. comment.t = comment.v || "";
  23230. comment.t = comment.t.replace(/\r\n/g,"\n").replace(/\r/g,"\n");
  23231. comment.v = comment.w = comment.ixfe = undefined;
  23232. }
  23233. function xlml_normalize(d) {
  23234. if(has_buf && Buffer.isBuffer(d)) return d.toString('utf8');
  23235. if(typeof d === 'string') return d;
  23236. /* duktape */
  23237. if(typeof Uint8Array !== 'undefined' && d instanceof Uint8Array) return utf8read(a2s(ab2a(d)));
  23238. throw new Error("Bad input format: expected Buffer or string");
  23239. }
  23240. /* TODO: Everything */
  23241. /* UOS uses CJK in tags */
  23242. var xlmlregex = /<(\/?)([^\s?>!\/:]*:|)([^\s?>:\/]+)[^>]*>/mg;
  23243. //var xlmlregex = /<(\/?)([a-z0-9]*:|)(\w+)[^>]*>/mg;
  23244. function parse_xlml_xml(d, _opts) {
  23245. var opts = _opts || {};
  23246. make_ssf(SSF);
  23247. var str = debom(xlml_normalize(d));
  23248. if(opts.type == 'binary' || opts.type == 'array' || opts.type == 'base64') {
  23249. if(typeof cptable !== 'undefined') str = cptable.utils.decode(65001, char_codes(str));
  23250. else str = utf8read(str);
  23251. }
  23252. var opening = str.slice(0, 1024).toLowerCase(), ishtml = false;
  23253. if(opening.indexOf("<?xml") == -1) ["html", "table", "head", "meta", "script", "style", "div"].forEach(function(tag) { if(opening.indexOf("<" + tag) >= 0) ishtml = true; });
  23254. if(ishtml) return HTML_.to_workbook(str, opts);
  23255. var Rn;
  23256. var state = [], tmp;
  23257. if(DENSE != null && opts.dense == null) opts.dense = DENSE;
  23258. var sheets = {}, sheetnames = [], cursheet = (opts.dense ? [] : {}), sheetname = "";
  23259. var table = {}, cell = ({}), row = {};// eslint-disable-line no-unused-vars
  23260. var dtag = xlml_parsexmltag('<Data ss:Type="String">'), didx = 0;
  23261. var c = 0, r = 0;
  23262. var refguess = {s: {r:2000000, c:2000000}, e: {r:0, c:0} };
  23263. var styles = {}, stag = {};
  23264. var ss = "", fidx = 0;
  23265. var merges = [];
  23266. var Props = {}, Custprops = {}, pidx = 0, cp = [];
  23267. var comments = [], comment = ({});
  23268. var cstys = [], csty, seencol = false;
  23269. var arrayf = [];
  23270. var rowinfo = [], rowobj = {}, cc = 0, rr = 0;
  23271. var Workbook = ({ Sheets:[], WBProps:{date1904:false} }), wsprops = {};
  23272. xlmlregex.lastIndex = 0;
  23273. str = str.replace(/<!--([\s\S]*?)-->/mg,"");
  23274. while((Rn = xlmlregex.exec(str))) switch(Rn[3]) {
  23275. case 'Data':
  23276. if(state[state.length-1][1]) break;
  23277. if(Rn[1]==='/') parse_xlml_data(str.slice(didx, Rn.index), ss, dtag, state[state.length-1][0]=="Comment"?comment:cell, {c:c,r:r}, styles, cstys[c], row, arrayf, opts);
  23278. else { ss = ""; dtag = xlml_parsexmltag(Rn[0]); didx = Rn.index + Rn[0].length; }
  23279. break;
  23280. case 'Cell':
  23281. if(Rn[1]==='/'){
  23282. if(comments.length > 0) cell.c = comments;
  23283. if((!opts.sheetRows || opts.sheetRows > r) && cell.v !== undefined) {
  23284. if(opts.dense) {
  23285. if(!cursheet[r]) cursheet[r] = [];
  23286. cursheet[r][c] = cell;
  23287. } else cursheet[encode_col(c) + encode_row(r)] = cell;
  23288. }
  23289. if(cell.HRef) {
  23290. cell.l = ({Target:cell.HRef});
  23291. if(cell.HRefScreenTip) cell.l.Tooltip = cell.HRefScreenTip;
  23292. delete cell.HRef; delete cell.HRefScreenTip;
  23293. }
  23294. if(cell.MergeAcross || cell.MergeDown) {
  23295. cc = c + (parseInt(cell.MergeAcross,10)|0);
  23296. rr = r + (parseInt(cell.MergeDown,10)|0);
  23297. merges.push({s:{c:c,r:r},e:{c:cc,r:rr}});
  23298. }
  23299. if(!opts.sheetStubs) { if(cell.MergeAcross) c = cc + 1; else ++c; }
  23300. else if(cell.MergeAcross || cell.MergeDown) {
  23301. for(var cma = c; cma <= cc; ++cma) {
  23302. for(var cmd = r; cmd <= rr; ++cmd) {
  23303. if(cma > c || cmd > r) {
  23304. if(opts.dense) {
  23305. if(!cursheet[cmd]) cursheet[cmd] = [];
  23306. cursheet[cmd][cma] = {t:'z'};
  23307. } else cursheet[encode_col(cma) + encode_row(cmd)] = {t:'z'};
  23308. }
  23309. }
  23310. }
  23311. c = cc + 1;
  23312. }
  23313. else ++c;
  23314. } else {
  23315. cell = xlml_parsexmltagobj(Rn[0]);
  23316. if(cell.Index) c = +cell.Index - 1;
  23317. if(c < refguess.s.c) refguess.s.c = c;
  23318. if(c > refguess.e.c) refguess.e.c = c;
  23319. if(Rn[0].slice(-2) === "/>") ++c;
  23320. comments = [];
  23321. }
  23322. break;
  23323. case 'Row':
  23324. if(Rn[1]==='/' || Rn[0].slice(-2) === "/>") {
  23325. if(r < refguess.s.r) refguess.s.r = r;
  23326. if(r > refguess.e.r) refguess.e.r = r;
  23327. if(Rn[0].slice(-2) === "/>") {
  23328. row = xlml_parsexmltag(Rn[0]);
  23329. if(row.Index) r = +row.Index - 1;
  23330. }
  23331. c = 0; ++r;
  23332. } else {
  23333. row = xlml_parsexmltag(Rn[0]);
  23334. if(row.Index) r = +row.Index - 1;
  23335. rowobj = {};
  23336. if(row.AutoFitHeight == "0" || row.Height) {
  23337. rowobj.hpx = parseInt(row.Height, 10); rowobj.hpt = px2pt(rowobj.hpx);
  23338. rowinfo[r] = rowobj;
  23339. }
  23340. if(row.Hidden == "1") { rowobj.hidden = true; rowinfo[r] = rowobj; }
  23341. }
  23342. break;
  23343. case 'Worksheet': /* TODO: read range from FullRows/FullColumns */
  23344. if(Rn[1]==='/'){
  23345. if((tmp=state.pop())[0]!==Rn[3]) throw new Error("Bad state: "+tmp.join("|"));
  23346. sheetnames.push(sheetname);
  23347. if(refguess.s.r <= refguess.e.r && refguess.s.c <= refguess.e.c) {
  23348. cursheet["!ref"] = encode_range(refguess);
  23349. if(opts.sheetRows && opts.sheetRows <= refguess.e.r) {
  23350. cursheet["!fullref"] = cursheet["!ref"];
  23351. refguess.e.r = opts.sheetRows - 1;
  23352. cursheet["!ref"] = encode_range(refguess);
  23353. }
  23354. }
  23355. if(merges.length) cursheet["!merges"] = merges;
  23356. if(cstys.length > 0) cursheet["!cols"] = cstys;
  23357. if(rowinfo.length > 0) cursheet["!rows"] = rowinfo;
  23358. sheets[sheetname] = cursheet;
  23359. } else {
  23360. refguess = {s: {r:2000000, c:2000000}, e: {r:0, c:0} };
  23361. r = c = 0;
  23362. state.push([Rn[3], false]);
  23363. tmp = xlml_parsexmltag(Rn[0]);
  23364. sheetname = unescapexml(tmp.Name);
  23365. cursheet = (opts.dense ? [] : {});
  23366. merges = [];
  23367. arrayf = [];
  23368. rowinfo = [];
  23369. wsprops = {name:sheetname, Hidden:0};
  23370. Workbook.Sheets.push(wsprops);
  23371. }
  23372. break;
  23373. case 'Table':
  23374. if(Rn[1]==='/'){if((tmp=state.pop())[0]!==Rn[3]) throw new Error("Bad state: "+tmp.join("|"));}
  23375. else if(Rn[0].slice(-2) == "/>") break;
  23376. else {
  23377. table = xlml_parsexmltag(Rn[0]);
  23378. state.push([Rn[3], false]);
  23379. cstys = []; seencol = false;
  23380. }
  23381. break;
  23382. case 'Style':
  23383. if(Rn[1]==='/') process_style_xlml(styles, stag, opts);
  23384. else stag = xlml_parsexmltag(Rn[0]);
  23385. break;
  23386. case 'NumberFormat':
  23387. stag.nf = unescapexml(xlml_parsexmltag(Rn[0]).Format || "General");
  23388. if(XLMLFormatMap[stag.nf]) stag.nf = XLMLFormatMap[stag.nf];
  23389. for(var ssfidx = 0; ssfidx != 0x188; ++ssfidx) if(SSF._table[ssfidx] == stag.nf) break;
  23390. if(ssfidx == 0x188) for(ssfidx = 0x39; ssfidx != 0x188; ++ssfidx) if(SSF._table[ssfidx] == null) { SSF.load(stag.nf, ssfidx); break; }
  23391. break;
  23392. case 'Column':
  23393. if(state[state.length-1][0] !== 'Table') break;
  23394. csty = xlml_parsexmltag(Rn[0]);
  23395. if(csty.Hidden) { csty.hidden = true; delete csty.Hidden; }
  23396. if(csty.Width) csty.wpx = parseInt(csty.Width, 10);
  23397. if(!seencol && csty.wpx > 10) {
  23398. seencol = true; MDW = DEF_MDW; //find_mdw_wpx(csty.wpx);
  23399. for(var _col = 0; _col < cstys.length; ++_col) if(cstys[_col]) process_col(cstys[_col]);
  23400. }
  23401. if(seencol) process_col(csty);
  23402. cstys[(csty.Index-1||cstys.length)] = csty;
  23403. for(var i = 0; i < +csty.Span; ++i) cstys[cstys.length] = dup(csty);
  23404. break;
  23405. case 'NamedRange':
  23406. if(!Workbook.Names) Workbook.Names = [];
  23407. var _NamedRange = parsexmltag(Rn[0]);
  23408. var _DefinedName = ({
  23409. Name: _NamedRange.Name,
  23410. Ref: rc_to_a1(_NamedRange.RefersTo.slice(1), {r:0, c:0})
  23411. });
  23412. if(Workbook.Sheets.length>0) _DefinedName.Sheet=Workbook.Sheets.length-1;
  23413. Workbook.Names.push(_DefinedName);
  23414. break;
  23415. case 'NamedCell': break;
  23416. case 'B': break;
  23417. case 'I': break;
  23418. case 'U': break;
  23419. case 'S': break;
  23420. case 'Sub': break;
  23421. case 'Sup': break;
  23422. case 'Span': break;
  23423. case 'Border': break;
  23424. case 'Alignment': break;
  23425. case 'Borders': break;
  23426. case 'Font':
  23427. if(Rn[0].slice(-2) === "/>") break;
  23428. else if(Rn[1]==="/") ss += str.slice(fidx, Rn.index);
  23429. else fidx = Rn.index + Rn[0].length;
  23430. break;
  23431. case 'Interior':
  23432. if(!opts.cellStyles) break;
  23433. stag.Interior = xlml_parsexmltag(Rn[0]);
  23434. break;
  23435. case 'Protection': break;
  23436. case 'Author':
  23437. case 'Title':
  23438. case 'Description':
  23439. case 'Created':
  23440. case 'Keywords':
  23441. case 'Subject':
  23442. case 'Category':
  23443. case 'Company':
  23444. case 'LastAuthor':
  23445. case 'LastSaved':
  23446. case 'LastPrinted':
  23447. case 'Version':
  23448. case 'Revision':
  23449. case 'TotalTime':
  23450. case 'HyperlinkBase':
  23451. case 'Manager':
  23452. case 'ContentStatus':
  23453. case 'Identifier':
  23454. case 'Language':
  23455. case 'AppName':
  23456. if(Rn[0].slice(-2) === "/>") break;
  23457. else if(Rn[1]==="/") xlml_set_prop(Props, Rn[3], str.slice(pidx, Rn.index));
  23458. else pidx = Rn.index + Rn[0].length;
  23459. break;
  23460. case 'Paragraphs': break;
  23461. case 'Styles':
  23462. case 'Workbook':
  23463. if(Rn[1]==='/'){if((tmp=state.pop())[0]!==Rn[3]) throw new Error("Bad state: "+tmp.join("|"));}
  23464. else state.push([Rn[3], false]);
  23465. break;
  23466. case 'Comment':
  23467. if(Rn[1]==='/'){
  23468. if((tmp=state.pop())[0]!==Rn[3]) throw new Error("Bad state: "+tmp.join("|"));
  23469. xlml_clean_comment(comment);
  23470. comments.push(comment);
  23471. } else {
  23472. state.push([Rn[3], false]);
  23473. tmp = xlml_parsexmltag(Rn[0]);
  23474. comment = ({a:tmp.Author});
  23475. }
  23476. break;
  23477. case 'AutoFilter':
  23478. if(Rn[1]==='/'){if((tmp=state.pop())[0]!==Rn[3]) throw new Error("Bad state: "+tmp.join("|"));}
  23479. else if(Rn[0].charAt(Rn[0].length-2) !== '/') {
  23480. var AutoFilter = xlml_parsexmltag(Rn[0]);
  23481. cursheet['!autofilter'] = { ref:rc_to_a1(AutoFilter.Range).replace(/\$/g,"") };
  23482. state.push([Rn[3], true]);
  23483. }
  23484. break;
  23485. case 'Name': break;
  23486. case 'ComponentOptions':
  23487. case 'DocumentProperties':
  23488. case 'CustomDocumentProperties':
  23489. case 'OfficeDocumentSettings':
  23490. case 'PivotTable':
  23491. case 'PivotCache':
  23492. case 'Names':
  23493. case 'MapInfo':
  23494. case 'PageBreaks':
  23495. case 'QueryTable':
  23496. case 'DataValidation':
  23497. case 'Sorting':
  23498. case 'Schema':
  23499. case 'data':
  23500. case 'ConditionalFormatting':
  23501. case 'SmartTagType':
  23502. case 'SmartTags':
  23503. case 'ExcelWorkbook':
  23504. case 'WorkbookOptions':
  23505. case 'WorksheetOptions':
  23506. if(Rn[1]==='/'){if((tmp=state.pop())[0]!==Rn[3]) throw new Error("Bad state: "+tmp.join("|"));}
  23507. else if(Rn[0].charAt(Rn[0].length-2) !== '/') state.push([Rn[3], true]);
  23508. break;
  23509. default:
  23510. /* FODS file root is <office:document> */
  23511. if(state.length == 0 && Rn[3] == "document") return parse_fods(str, opts);
  23512. /* UOS file root is <uof:UOF> */
  23513. if(state.length == 0 && Rn[3] == "UOF") return parse_fods(str, opts);
  23514. var seen = true;
  23515. switch(state[state.length-1][0]) {
  23516. /* OfficeDocumentSettings */
  23517. case 'OfficeDocumentSettings': switch(Rn[3]) {
  23518. case 'AllowPNG': break;
  23519. case 'RemovePersonalInformation': break;
  23520. case 'DownloadComponents': break;
  23521. case 'LocationOfComponents': break;
  23522. case 'Colors': break;
  23523. case 'Color': break;
  23524. case 'Index': break;
  23525. case 'RGB': break;
  23526. case 'PixelsPerInch': break; // TODO: set PPI
  23527. case 'TargetScreenSize': break;
  23528. case 'ReadOnlyRecommended': break;
  23529. default: seen = false;
  23530. } break;
  23531. /* ComponentOptions */
  23532. case 'ComponentOptions': switch(Rn[3]) {
  23533. case 'Toolbar': break;
  23534. case 'HideOfficeLogo': break;
  23535. case 'SpreadsheetAutoFit': break;
  23536. case 'Label': break;
  23537. case 'Caption': break;
  23538. case 'MaxHeight': break;
  23539. case 'MaxWidth': break;
  23540. case 'NextSheetNumber': break;
  23541. default: seen = false;
  23542. } break;
  23543. /* ExcelWorkbook */
  23544. case 'ExcelWorkbook': switch(Rn[3]) {
  23545. case 'Date1904':
  23546. Workbook.WBProps.date1904 = true;
  23547. break;
  23548. case 'WindowHeight': break;
  23549. case 'WindowWidth': break;
  23550. case 'WindowTopX': break;
  23551. case 'WindowTopY': break;
  23552. case 'TabRatio': break;
  23553. case 'ProtectStructure': break;
  23554. case 'ProtectWindows': break;
  23555. case 'ActiveSheet': break;
  23556. case 'DisplayInkNotes': break;
  23557. case 'FirstVisibleSheet': break;
  23558. case 'SupBook': break;
  23559. case 'SheetName': break;
  23560. case 'SheetIndex': break;
  23561. case 'SheetIndexFirst': break;
  23562. case 'SheetIndexLast': break;
  23563. case 'Dll': break;
  23564. case 'AcceptLabelsInFormulas': break;
  23565. case 'DoNotSaveLinkValues': break;
  23566. case 'Iteration': break;
  23567. case 'MaxIterations': break;
  23568. case 'MaxChange': break;
  23569. case 'Path': break;
  23570. case 'Xct': break;
  23571. case 'Count': break;
  23572. case 'SelectedSheets': break;
  23573. case 'Calculation': break;
  23574. case 'Uncalced': break;
  23575. case 'StartupPrompt': break;
  23576. case 'Crn': break;
  23577. case 'ExternName': break;
  23578. case 'Formula': break;
  23579. case 'ColFirst': break;
  23580. case 'ColLast': break;
  23581. case 'WantAdvise': break;
  23582. case 'Boolean': break;
  23583. case 'Error': break;
  23584. case 'Text': break;
  23585. case 'OLE': break;
  23586. case 'NoAutoRecover': break;
  23587. case 'PublishObjects': break;
  23588. case 'DoNotCalculateBeforeSave': break;
  23589. case 'Number': break;
  23590. case 'RefModeR1C1': break;
  23591. case 'EmbedSaveSmartTags': break;
  23592. default: seen = false;
  23593. } break;
  23594. /* WorkbookOptions */
  23595. case 'WorkbookOptions': switch(Rn[3]) {
  23596. case 'OWCVersion': break;
  23597. case 'Height': break;
  23598. case 'Width': break;
  23599. default: seen = false;
  23600. } break;
  23601. /* WorksheetOptions */
  23602. case 'WorksheetOptions': switch(Rn[3]) {
  23603. case 'Visible':
  23604. if(Rn[0].slice(-2) === "/>"){/* empty */}
  23605. else if(Rn[1]==="/") switch(str.slice(pidx, Rn.index)) {
  23606. case "SheetHidden": wsprops.Hidden = 1; break;
  23607. case "SheetVeryHidden": wsprops.Hidden = 2; break;
  23608. }
  23609. else pidx = Rn.index + Rn[0].length;
  23610. break;
  23611. case 'Header':
  23612. if(!cursheet['!margins']) default_margins(cursheet['!margins']={}, 'xlml');
  23613. cursheet['!margins'].header = parsexmltag(Rn[0]).Margin;
  23614. break;
  23615. case 'Footer':
  23616. if(!cursheet['!margins']) default_margins(cursheet['!margins']={}, 'xlml');
  23617. cursheet['!margins'].footer = parsexmltag(Rn[0]).Margin;
  23618. break;
  23619. case 'PageMargins':
  23620. var pagemargins = parsexmltag(Rn[0]);
  23621. if(!cursheet['!margins']) default_margins(cursheet['!margins']={},'xlml');
  23622. if(pagemargins.Top) cursheet['!margins'].top = pagemargins.Top;
  23623. if(pagemargins.Left) cursheet['!margins'].left = pagemargins.Left;
  23624. if(pagemargins.Right) cursheet['!margins'].right = pagemargins.Right;
  23625. if(pagemargins.Bottom) cursheet['!margins'].bottom = pagemargins.Bottom;
  23626. break;
  23627. case 'DisplayRightToLeft':
  23628. if(!Workbook.Views) Workbook.Views = [];
  23629. if(!Workbook.Views[0]) Workbook.Views[0] = {};
  23630. Workbook.Views[0].RTL = true;
  23631. break;
  23632. case 'Unsynced': break;
  23633. case 'Print': break;
  23634. case 'Panes': break;
  23635. case 'Scale': break;
  23636. case 'Pane': break;
  23637. case 'Number': break;
  23638. case 'Layout': break;
  23639. case 'PageSetup': break;
  23640. case 'Selected': break;
  23641. case 'ProtectObjects': break;
  23642. case 'EnableSelection': break;
  23643. case 'ProtectScenarios': break;
  23644. case 'ValidPrinterInfo': break;
  23645. case 'HorizontalResolution': break;
  23646. case 'VerticalResolution': break;
  23647. case 'NumberofCopies': break;
  23648. case 'ActiveRow': break;
  23649. case 'ActiveCol': break;
  23650. case 'ActivePane': break;
  23651. case 'TopRowVisible': break;
  23652. case 'TopRowBottomPane': break;
  23653. case 'LeftColumnVisible': break;
  23654. case 'LeftColumnRightPane': break;
  23655. case 'FitToPage': break;
  23656. case 'RangeSelection': break;
  23657. case 'PaperSizeIndex': break;
  23658. case 'PageLayoutZoom': break;
  23659. case 'PageBreakZoom': break;
  23660. case 'FilterOn': break;
  23661. case 'DoNotDisplayGridlines': break;
  23662. case 'SplitHorizontal': break;
  23663. case 'SplitVertical': break;
  23664. case 'FreezePanes': break;
  23665. case 'FrozenNoSplit': break;
  23666. case 'FitWidth': break;
  23667. case 'FitHeight': break;
  23668. case 'CommentsLayout': break;
  23669. case 'Zoom': break;
  23670. case 'LeftToRight': break;
  23671. case 'Gridlines': break;
  23672. case 'AllowSort': break;
  23673. case 'AllowFilter': break;
  23674. case 'AllowInsertRows': break;
  23675. case 'AllowDeleteRows': break;
  23676. case 'AllowInsertCols': break;
  23677. case 'AllowDeleteCols': break;
  23678. case 'AllowInsertHyperlinks': break;
  23679. case 'AllowFormatCells': break;
  23680. case 'AllowSizeCols': break;
  23681. case 'AllowSizeRows': break;
  23682. case 'NoSummaryRowsBelowDetail': break;
  23683. case 'TabColorIndex': break;
  23684. case 'DoNotDisplayHeadings': break;
  23685. case 'ShowPageLayoutZoom': break;
  23686. case 'NoSummaryColumnsRightDetail': break;
  23687. case 'BlackAndWhite': break;
  23688. case 'DoNotDisplayZeros': break;
  23689. case 'DisplayPageBreak': break;
  23690. case 'RowColHeadings': break;
  23691. case 'DoNotDisplayOutline': break;
  23692. case 'NoOrientation': break;
  23693. case 'AllowUsePivotTables': break;
  23694. case 'ZeroHeight': break;
  23695. case 'ViewableRange': break;
  23696. case 'Selection': break;
  23697. case 'ProtectContents': break;
  23698. default: seen = false;
  23699. } break;
  23700. /* PivotTable */
  23701. case 'PivotTable': case 'PivotCache': switch(Rn[3]) {
  23702. case 'ImmediateItemsOnDrop': break;
  23703. case 'ShowPageMultipleItemLabel': break;
  23704. case 'CompactRowIndent': break;
  23705. case 'Location': break;
  23706. case 'PivotField': break;
  23707. case 'Orientation': break;
  23708. case 'LayoutForm': break;
  23709. case 'LayoutSubtotalLocation': break;
  23710. case 'LayoutCompactRow': break;
  23711. case 'Position': break;
  23712. case 'PivotItem': break;
  23713. case 'DataType': break;
  23714. case 'DataField': break;
  23715. case 'SourceName': break;
  23716. case 'ParentField': break;
  23717. case 'PTLineItems': break;
  23718. case 'PTLineItem': break;
  23719. case 'CountOfSameItems': break;
  23720. case 'Item': break;
  23721. case 'ItemType': break;
  23722. case 'PTSource': break;
  23723. case 'CacheIndex': break;
  23724. case 'ConsolidationReference': break;
  23725. case 'FileName': break;
  23726. case 'Reference': break;
  23727. case 'NoColumnGrand': break;
  23728. case 'NoRowGrand': break;
  23729. case 'BlankLineAfterItems': break;
  23730. case 'Hidden': break;
  23731. case 'Subtotal': break;
  23732. case 'BaseField': break;
  23733. case 'MapChildItems': break;
  23734. case 'Function': break;
  23735. case 'RefreshOnFileOpen': break;
  23736. case 'PrintSetTitles': break;
  23737. case 'MergeLabels': break;
  23738. case 'DefaultVersion': break;
  23739. case 'RefreshName': break;
  23740. case 'RefreshDate': break;
  23741. case 'RefreshDateCopy': break;
  23742. case 'VersionLastRefresh': break;
  23743. case 'VersionLastUpdate': break;
  23744. case 'VersionUpdateableMin': break;
  23745. case 'VersionRefreshableMin': break;
  23746. case 'Calculation': break;
  23747. default: seen = false;
  23748. } break;
  23749. /* PageBreaks */
  23750. case 'PageBreaks': switch(Rn[3]) {
  23751. case 'ColBreaks': break;
  23752. case 'ColBreak': break;
  23753. case 'RowBreaks': break;
  23754. case 'RowBreak': break;
  23755. case 'ColStart': break;
  23756. case 'ColEnd': break;
  23757. case 'RowEnd': break;
  23758. default: seen = false;
  23759. } break;
  23760. /* AutoFilter */
  23761. case 'AutoFilter': switch(Rn[3]) {
  23762. case 'AutoFilterColumn': break;
  23763. case 'AutoFilterCondition': break;
  23764. case 'AutoFilterAnd': break;
  23765. case 'AutoFilterOr': break;
  23766. default: seen = false;
  23767. } break;
  23768. /* QueryTable */
  23769. case 'QueryTable': switch(Rn[3]) {
  23770. case 'Id': break;
  23771. case 'AutoFormatFont': break;
  23772. case 'AutoFormatPattern': break;
  23773. case 'QuerySource': break;
  23774. case 'QueryType': break;
  23775. case 'EnableRedirections': break;
  23776. case 'RefreshedInXl9': break;
  23777. case 'URLString': break;
  23778. case 'HTMLTables': break;
  23779. case 'Connection': break;
  23780. case 'CommandText': break;
  23781. case 'RefreshInfo': break;
  23782. case 'NoTitles': break;
  23783. case 'NextId': break;
  23784. case 'ColumnInfo': break;
  23785. case 'OverwriteCells': break;
  23786. case 'DoNotPromptForFile': break;
  23787. case 'TextWizardSettings': break;
  23788. case 'Source': break;
  23789. case 'Number': break;
  23790. case 'Decimal': break;
  23791. case 'ThousandSeparator': break;
  23792. case 'TrailingMinusNumbers': break;
  23793. case 'FormatSettings': break;
  23794. case 'FieldType': break;
  23795. case 'Delimiters': break;
  23796. case 'Tab': break;
  23797. case 'Comma': break;
  23798. case 'AutoFormatName': break;
  23799. case 'VersionLastEdit': break;
  23800. case 'VersionLastRefresh': break;
  23801. default: seen = false;
  23802. } break;
  23803. case 'Sorting':
  23804. case 'ConditionalFormatting':
  23805. case 'DataValidation':
  23806. switch(Rn[3]) {
  23807. case 'Range': break;
  23808. case 'Type': break;
  23809. case 'Min': break;
  23810. case 'Max': break;
  23811. case 'Sort': break;
  23812. case 'Descending': break;
  23813. case 'Orders': break;
  23814. case 'CaseSensitive': break;
  23815. case 'Value': break;
  23816. case 'ErrorStyle': break;
  23817. case 'ErrorMessage': break;
  23818. case 'ErrorTitle': break;
  23819. case 'CellRangeList': break;
  23820. case 'InputMessage': break;
  23821. case 'InputTitle': break;
  23822. case 'ComboHide': break;
  23823. case 'InputHide': break;
  23824. case 'Condition': break;
  23825. case 'Qualifier': break;
  23826. case 'UseBlank': break;
  23827. case 'Value1': break;
  23828. case 'Value2': break;
  23829. case 'Format': break;
  23830. default: seen = false;
  23831. } break;
  23832. /* MapInfo (schema) */
  23833. case 'MapInfo': case 'Schema': case 'data': switch(Rn[3]) {
  23834. case 'Map': break;
  23835. case 'Entry': break;
  23836. case 'Range': break;
  23837. case 'XPath': break;
  23838. case 'Field': break;
  23839. case 'XSDType': break;
  23840. case 'FilterOn': break;
  23841. case 'Aggregate': break;
  23842. case 'ElementType': break;
  23843. case 'AttributeType': break;
  23844. /* These are from xsd (XML Schema Definition) */
  23845. case 'schema':
  23846. case 'element':
  23847. case 'complexType':
  23848. case 'datatype':
  23849. case 'all':
  23850. case 'attribute':
  23851. case 'extends': break;
  23852. case 'row': break;
  23853. default: seen = false;
  23854. } break;
  23855. /* SmartTags (can be anything) */
  23856. case 'SmartTags': break;
  23857. default: seen = false; break;
  23858. }
  23859. if(seen) break;
  23860. /* CustomDocumentProperties */
  23861. if(!state[state.length-1][1]) throw 'Unrecognized tag: ' + Rn[3] + "|" + state.join("|");
  23862. if(state[state.length-1][0]==='CustomDocumentProperties') {
  23863. if(Rn[0].slice(-2) === "/>") break;
  23864. else if(Rn[1]==="/") xlml_set_custprop(Custprops, Rn[3], cp, str.slice(pidx, Rn.index));
  23865. else { cp = Rn; pidx = Rn.index + Rn[0].length; }
  23866. break;
  23867. }
  23868. if(opts.WTF) throw 'Unrecognized tag: ' + Rn[3] + "|" + state.join("|");
  23869. }
  23870. var out = ({});
  23871. if(!opts.bookSheets && !opts.bookProps) out.Sheets = sheets;
  23872. out.SheetNames = sheetnames;
  23873. out.Workbook = Workbook;
  23874. out.SSF = SSF.get_table();
  23875. out.Props = Props;
  23876. out.Custprops = Custprops;
  23877. return out;
  23878. }
  23879. function parse_xlml(data, opts) {
  23880. fix_read_opts(opts=opts||{});
  23881. switch(opts.type||"base64") {
  23882. case "base64": return parse_xlml_xml(Base64.decode(data), opts);
  23883. case "binary": case "buffer": case "file": return parse_xlml_xml(data, opts);
  23884. case "array": return parse_xlml_xml(a2s(data), opts);
  23885. }
  23886. }
  23887. /* TODO */
  23888. function write_props_xlml(wb, opts) {
  23889. var o = [];
  23890. /* DocumentProperties */
  23891. if(wb.Props) o.push(xlml_write_docprops(wb.Props, opts));
  23892. /* CustomDocumentProperties */
  23893. if(wb.Custprops) o.push(xlml_write_custprops(wb.Props, wb.Custprops, opts));
  23894. return o.join("");
  23895. }
  23896. /* TODO */
  23897. function write_wb_xlml() {
  23898. /* OfficeDocumentSettings */
  23899. /* ExcelWorkbook */
  23900. return "";
  23901. }
  23902. /* TODO */
  23903. function write_sty_xlml(wb, opts) {
  23904. /* Styles */
  23905. var styles = ['<Style ss:ID="Default" ss:Name="Normal"><NumberFormat/></Style>'];
  23906. opts.cellXfs.forEach(function(xf, id) {
  23907. var payload = [];
  23908. payload.push(writextag('NumberFormat', null, {"ss:Format": escapexml(SSF._table[xf.numFmtId])}));
  23909. styles.push(writextag('Style', payload.join(""), {"ss:ID": "s" + (21+id)}));
  23910. });
  23911. return writextag("Styles", styles.join(""));
  23912. }
  23913. function write_name_xlml(n) { return writextag("NamedRange", null, {"ss:Name": n.Name, "ss:RefersTo":"=" + a1_to_rc(n.Ref, {r:0,c:0})}); }
  23914. function write_names_xlml(wb) {
  23915. if(!((wb||{}).Workbook||{}).Names) return "";
  23916. var names = wb.Workbook.Names;
  23917. var out = [];
  23918. for(var i = 0; i < names.length; ++i) {
  23919. var n = names[i];
  23920. if(n.Sheet != null) continue;
  23921. if(n.Name.match(/^_xlfn\./)) continue;
  23922. out.push(write_name_xlml(n));
  23923. }
  23924. return writextag("Names", out.join(""));
  23925. }
  23926. function write_ws_xlml_names(ws, opts, idx, wb) {
  23927. if(!ws) return "";
  23928. if(!((wb||{}).Workbook||{}).Names) return "";
  23929. var names = wb.Workbook.Names;
  23930. var out = [];
  23931. for(var i = 0; i < names.length; ++i) {
  23932. var n = names[i];
  23933. if(n.Sheet != idx) continue;
  23934. /*switch(n.Name) {
  23935. case "_": continue;
  23936. }*/
  23937. if(n.Name.match(/^_xlfn\./)) continue;
  23938. out.push(write_name_xlml(n));
  23939. }
  23940. return out.join("");
  23941. }
  23942. /* WorksheetOptions */
  23943. function write_ws_xlml_wsopts(ws, opts, idx, wb) {
  23944. if(!ws) return "";
  23945. var o = [];
  23946. /* NOTE: spec technically allows any order, but stick with implied order */
  23947. /* FitToPage */
  23948. /* DoNotDisplayColHeaders */
  23949. /* DoNotDisplayRowHeaders */
  23950. /* ViewableRange */
  23951. /* Selection */
  23952. /* GridlineColor */
  23953. /* Name */
  23954. /* ExcelWorksheetType */
  23955. /* IntlMacro */
  23956. /* Unsynced */
  23957. /* Selected */
  23958. /* CodeName */
  23959. if(ws['!margins']) {
  23960. o.push("<PageSetup>");
  23961. if(ws['!margins'].header) o.push(writextag("Header", null, {'x:Margin':ws['!margins'].header}));
  23962. if(ws['!margins'].footer) o.push(writextag("Footer", null, {'x:Margin':ws['!margins'].footer}));
  23963. o.push(writextag("PageMargins", null, {
  23964. 'x:Bottom': ws['!margins'].bottom || "0.75",
  23965. 'x:Left': ws['!margins'].left || "0.7",
  23966. 'x:Right': ws['!margins'].right || "0.7",
  23967. 'x:Top': ws['!margins'].top || "0.75"
  23968. }));
  23969. o.push("</PageSetup>");
  23970. }
  23971. /* PageSetup */
  23972. /* DisplayPageBreak */
  23973. /* TransitionExpressionEvaluation */
  23974. /* TransitionFormulaEntry */
  23975. /* Print */
  23976. /* Zoom */
  23977. /* PageLayoutZoom */
  23978. /* PageBreakZoom */
  23979. /* ShowPageBreakZoom */
  23980. /* DefaultRowHeight */
  23981. /* DefaultColumnWidth */
  23982. /* StandardWidth */
  23983. if(wb && wb.Workbook && wb.Workbook.Sheets && wb.Workbook.Sheets[idx]) {
  23984. /* Visible */
  23985. if(wb.Workbook.Sheets[idx].Hidden) o.push(writextag("Visible", (wb.Workbook.Sheets[idx].Hidden == 1 ? "SheetHidden" : "SheetVeryHidden"), {}));
  23986. else {
  23987. /* Selected */
  23988. for(var i = 0; i < idx; ++i) if(wb.Workbook.Sheets[i] && !wb.Workbook.Sheets[i].Hidden) break;
  23989. if(i == idx) o.push("<Selected/>");
  23990. }
  23991. }
  23992. /* LeftColumnVisible */
  23993. if(((((wb||{}).Workbook||{}).Views||[])[0]||{}).RTL) o.push("<DisplayRightToLeft/>");
  23994. /* GridlineColorIndex */
  23995. /* DisplayFormulas */
  23996. /* DoNotDisplayGridlines */
  23997. /* DoNotDisplayHeadings */
  23998. /* DoNotDisplayOutline */
  23999. /* ApplyAutomaticOutlineStyles */
  24000. /* NoSummaryRowsBelowDetail */
  24001. /* NoSummaryColumnsRightDetail */
  24002. /* DoNotDisplayZeros */
  24003. /* ActiveRow */
  24004. /* ActiveColumn */
  24005. /* FilterOn */
  24006. /* RangeSelection */
  24007. /* TopRowVisible */
  24008. /* TopRowBottomPane */
  24009. /* LeftColumnRightPane */
  24010. /* ActivePane */
  24011. /* SplitHorizontal */
  24012. /* SplitVertical */
  24013. /* FreezePanes */
  24014. /* FrozenNoSplit */
  24015. /* TabColorIndex */
  24016. /* Panes */
  24017. /* NOTE: Password not supported in XLML Format */
  24018. if(ws['!protect']) {
  24019. o.push(writetag("ProtectContents", "True"));
  24020. if(ws['!protect'].objects) o.push(writetag("ProtectObjects", "True"));
  24021. if(ws['!protect'].scenarios) o.push(writetag("ProtectScenarios", "True"));
  24022. if(ws['!protect'].selectLockedCells != null && !ws['!protect'].selectLockedCells) o.push(writetag("EnableSelection", "NoSelection"));
  24023. else if(ws['!protect'].selectUnlockedCells != null && !ws['!protect'].selectUnlockedCells) o.push(writetag("EnableSelection", "UnlockedCells"));
  24024. [
  24025. [ "formatCells", "AllowFormatCells" ],
  24026. [ "formatColumns", "AllowSizeCols" ],
  24027. [ "formatRows", "AllowSizeRows" ],
  24028. [ "insertColumns", "AllowInsertCols" ],
  24029. [ "insertRows", "AllowInsertRows" ],
  24030. [ "insertHyperlinks", "AllowInsertHyperlinks" ],
  24031. [ "deleteColumns", "AllowDeleteCols" ],
  24032. [ "deleteRows", "AllowDeleteRows" ],
  24033. [ "sort", "AllowSort" ],
  24034. [ "autoFilter", "AllowFilter" ],
  24035. [ "pivotTables", "AllowUsePivotTables" ]
  24036. ].forEach(function(x) { if(ws['!protect'][x[0]]) o.push("<"+x[1]+"/>"); });
  24037. }
  24038. if(o.length == 0) return "";
  24039. return writextag("WorksheetOptions", o.join(""), {xmlns:XLMLNS.x});
  24040. }
  24041. function write_ws_xlml_comment(comments) {
  24042. return comments.map(function(c) {
  24043. // TODO: formatted text
  24044. var t = xlml_unfixstr(c.t||"");
  24045. var d =writextag("ss:Data", t, {"xmlns":"http://www.w3.org/TR/REC-html40"});
  24046. return writextag("Comment", d, {"ss:Author":c.a});
  24047. }).join("");
  24048. }
  24049. function write_ws_xlml_cell(cell, ref, ws, opts, idx, wb, addr){
  24050. if(!cell || (cell.v == undefined && cell.f == undefined)) return "";
  24051. var attr = {};
  24052. if(cell.f) attr["ss:Formula"] = "=" + escapexml(a1_to_rc(cell.f, addr));
  24053. if(cell.F && cell.F.slice(0, ref.length) == ref) {
  24054. var end = decode_cell(cell.F.slice(ref.length + 1));
  24055. attr["ss:ArrayRange"] = "RC:R" + (end.r == addr.r ? "" : "[" + (end.r - addr.r) + "]") + "C" + (end.c == addr.c ? "" : "[" + (end.c - addr.c) + "]");
  24056. }
  24057. if(cell.l && cell.l.Target) {
  24058. attr["ss:HRef"] = escapexml(cell.l.Target);
  24059. if(cell.l.Tooltip) attr["x:HRefScreenTip"] = escapexml(cell.l.Tooltip);
  24060. }
  24061. if(ws['!merges']) {
  24062. var marr = ws['!merges'];
  24063. for(var mi = 0; mi != marr.length; ++mi) {
  24064. if(marr[mi].s.c != addr.c || marr[mi].s.r != addr.r) continue;
  24065. if(marr[mi].e.c > marr[mi].s.c) attr['ss:MergeAcross'] = marr[mi].e.c - marr[mi].s.c;
  24066. if(marr[mi].e.r > marr[mi].s.r) attr['ss:MergeDown'] = marr[mi].e.r - marr[mi].s.r;
  24067. }
  24068. }
  24069. var t = "", p = "";
  24070. switch(cell.t) {
  24071. case 'z': return "";
  24072. case 'n': t = 'Number'; p = String(cell.v); break;
  24073. case 'b': t = 'Boolean'; p = (cell.v ? "1" : "0"); break;
  24074. case 'e': t = 'Error'; p = BErr[cell.v]; break;
  24075. case 'd': t = 'DateTime'; p = new Date(cell.v).toISOString(); if(cell.z == null) cell.z = cell.z || SSF._table[14]; break;
  24076. case 's': t = 'String'; p = escapexlml(cell.v||""); break;
  24077. }
  24078. /* TODO: cell style */
  24079. var os = get_cell_style(opts.cellXfs, cell, opts);
  24080. attr["ss:StyleID"] = "s" + (21+os);
  24081. attr["ss:Index"] = addr.c + 1;
  24082. var _v = (cell.v != null ? p : "");
  24083. var m = '<Data ss:Type="' + t + '">' + _v + '</Data>';
  24084. if((cell.c||[]).length > 0) m += write_ws_xlml_comment(cell.c);
  24085. return writextag("Cell", m, attr);
  24086. }
  24087. function write_ws_xlml_row(R, row) {
  24088. var o = '<Row ss:Index="' + (R+1) + '"';
  24089. if(row) {
  24090. if(row.hpt && !row.hpx) row.hpx = pt2px(row.hpt);
  24091. if(row.hpx) o += ' ss:AutoFitHeight="0" ss:Height="' + row.hpx + '"';
  24092. if(row.hidden) o += ' ss:Hidden="1"';
  24093. }
  24094. return o + '>';
  24095. }
  24096. /* TODO */
  24097. function write_ws_xlml_table(ws, opts, idx, wb) {
  24098. if(!ws['!ref']) return "";
  24099. var range = safe_decode_range(ws['!ref']);
  24100. var marr = ws['!merges'] || [], mi = 0;
  24101. var o = [];
  24102. if(ws['!cols']) ws['!cols'].forEach(function(n, i) {
  24103. process_col(n);
  24104. var w = !!n.width;
  24105. var p = col_obj_w(i, n);
  24106. var k = {"ss:Index":i+1};
  24107. if(w) k['ss:Width'] = width2px(p.width);
  24108. if(n.hidden) k['ss:Hidden']="1";
  24109. o.push(writextag("Column",null,k));
  24110. });
  24111. var dense = Array.isArray(ws);
  24112. for(var R = range.s.r; R <= range.e.r; ++R) {
  24113. var row = [write_ws_xlml_row(R, (ws['!rows']||[])[R])];
  24114. for(var C = range.s.c; C <= range.e.c; ++C) {
  24115. var skip = false;
  24116. for(mi = 0; mi != marr.length; ++mi) {
  24117. if(marr[mi].s.c > C) continue;
  24118. if(marr[mi].s.r > R) continue;
  24119. if(marr[mi].e.c < C) continue;
  24120. if(marr[mi].e.r < R) continue;
  24121. if(marr[mi].s.c != C || marr[mi].s.r != R) skip = true;
  24122. break;
  24123. }
  24124. if(skip) continue;
  24125. var addr = {r:R,c:C};
  24126. var ref = encode_cell(addr), cell = dense ? (ws[R]||[])[C] : ws[ref];
  24127. row.push(write_ws_xlml_cell(cell, ref, ws, opts, idx, wb, addr));
  24128. }
  24129. row.push("</Row>");
  24130. if(row.length > 2) o.push(row.join(""));
  24131. }
  24132. return o.join("");
  24133. }
  24134. function write_ws_xlml(idx, opts, wb) {
  24135. var o = [];
  24136. var s = wb.SheetNames[idx];
  24137. var ws = wb.Sheets[s];
  24138. var t = ws ? write_ws_xlml_names(ws, opts, idx, wb) : "";
  24139. if(t.length > 0) o.push("<Names>" + t + "</Names>");
  24140. /* Table */
  24141. t = ws ? write_ws_xlml_table(ws, opts, idx, wb) : "";
  24142. if(t.length > 0) o.push("<Table>" + t + "</Table>");
  24143. /* WorksheetOptions */
  24144. o.push(write_ws_xlml_wsopts(ws, opts, idx, wb));
  24145. return o.join("");
  24146. }
  24147. function write_xlml(wb, opts) {
  24148. if(!opts) opts = {};
  24149. if(!wb.SSF) wb.SSF = SSF.get_table();
  24150. if(wb.SSF) {
  24151. make_ssf(SSF); SSF.load_table(wb.SSF);
  24152. // $FlowIgnore
  24153. opts.revssf = evert_num(wb.SSF); opts.revssf[wb.SSF[65535]] = 0;
  24154. opts.ssf = wb.SSF;
  24155. opts.cellXfs = [];
  24156. get_cell_style(opts.cellXfs, {}, {revssf:{"General":0}});
  24157. }
  24158. var d = [];
  24159. d.push(write_props_xlml(wb, opts));
  24160. d.push(write_wb_xlml(wb, opts));
  24161. d.push("");
  24162. d.push("");
  24163. for(var i = 0; i < wb.SheetNames.length; ++i)
  24164. d.push(writextag("Worksheet", write_ws_xlml(i, opts, wb), {"ss:Name":escapexml(wb.SheetNames[i])}));
  24165. d[2] = write_sty_xlml(wb, opts);
  24166. d[3] = write_names_xlml(wb, opts);
  24167. return XML_HEADER + writextag("Workbook", d.join(""), {
  24168. 'xmlns': XLMLNS.ss,
  24169. 'xmlns:o': XLMLNS.o,
  24170. 'xmlns:x': XLMLNS.x,
  24171. 'xmlns:ss': XLMLNS.ss,
  24172. 'xmlns:dt': XLMLNS.dt,
  24173. 'xmlns:html': XLMLNS.html
  24174. });
  24175. }
  24176. /* [MS-OLEDS] 2.3.8 CompObjStream */
  24177. function parse_compobj(obj) {
  24178. var v = {};
  24179. var o = obj.content;
  24180. /* [MS-OLEDS] 2.3.7 CompObjHeader -- All fields MUST be ignored */
  24181. o.l = 28;
  24182. v.AnsiUserType = o.read_shift(0, "lpstr-ansi");
  24183. v.AnsiClipboardFormat = parse_ClipboardFormatOrAnsiString(o);
  24184. if(o.length - o.l <= 4) return v;
  24185. var m = o.read_shift(4);
  24186. if(m == 0 || m > 40) return v;
  24187. o.l-=4; v.Reserved1 = o.read_shift(0, "lpstr-ansi");
  24188. if(o.length - o.l <= 4) return v;
  24189. m = o.read_shift(4);
  24190. if(m !== 0x71b239f4) return v;
  24191. v.UnicodeClipboardFormat = parse_ClipboardFormatOrUnicodeString(o);
  24192. m = o.read_shift(4);
  24193. if(m == 0 || m > 40) return v;
  24194. o.l-=4; v.Reserved2 = o.read_shift(0, "lpwstr");
  24195. }
  24196. /*
  24197. Continue logic for:
  24198. - 2.4.58 Continue
  24199. - 2.4.59 ContinueBigName
  24200. - 2.4.60 ContinueFrt
  24201. - 2.4.61 ContinueFrt11
  24202. - 2.4.62 ContinueFrt12
  24203. */
  24204. function slurp(R, blob, length, opts) {
  24205. var l = length;
  24206. var bufs = [];
  24207. var d = blob.slice(blob.l,blob.l+l);
  24208. if(opts && opts.enc && opts.enc.insitu) switch(R.n) {
  24209. case 'BOF': case 'FilePass': case 'FileLock': case 'InterfaceHdr': case 'RRDInfo': case 'RRDHead': case 'UsrExcl': break;
  24210. default:
  24211. if(d.length === 0) break;
  24212. opts.enc.insitu(d);
  24213. }
  24214. bufs.push(d);
  24215. blob.l += l;
  24216. var next = (XLSRecordEnum[__readUInt16LE(blob,blob.l)]);
  24217. var start = 0;
  24218. while(next != null && next.n.slice(0,8) === 'Continue') {
  24219. l = __readUInt16LE(blob,blob.l+2);
  24220. start = blob.l + 4;
  24221. if(next.n == 'ContinueFrt') start += 4;
  24222. else if(next.n.slice(0,11) == 'ContinueFrt') start += 12;
  24223. bufs.push(blob.slice(start,blob.l+4+l));
  24224. blob.l += 4+l;
  24225. next = (XLSRecordEnum[__readUInt16LE(blob, blob.l)]);
  24226. }
  24227. var b = (bconcat(bufs));
  24228. prep_blob(b, 0);
  24229. var ll = 0; b.lens = [];
  24230. for(var j = 0; j < bufs.length; ++j) { b.lens.push(ll); ll += bufs[j].length; }
  24231. return R.f(b, b.length, opts);
  24232. }
  24233. function safe_format_xf(p, opts, date1904) {
  24234. if(p.t === 'z') return;
  24235. if(!p.XF) return;
  24236. var fmtid = 0;
  24237. try {
  24238. fmtid = p.z || p.XF.numFmtId || 0;
  24239. if(opts.cellNF) p.z = SSF._table[fmtid];
  24240. } catch(e) { if(opts.WTF) throw e; }
  24241. if(!opts || opts.cellText !== false) try {
  24242. if(p.t === 'e') { p.w = p.w || BErr[p.v]; }
  24243. else if(fmtid === 0 || fmtid == "General") {
  24244. if(p.t === 'n') {
  24245. if((p.v|0) === p.v) p.w = SSF._general_int(p.v);
  24246. else p.w = SSF._general_num(p.v);
  24247. }
  24248. else p.w = SSF._general(p.v);
  24249. }
  24250. else p.w = SSF.format(fmtid,p.v, {date1904:!!date1904});
  24251. } catch(e) { if(opts.WTF) throw e; }
  24252. if(opts.cellDates && fmtid && p.t == 'n' && SSF.is_date(SSF._table[fmtid] || String(fmtid))) {
  24253. var _d = SSF.parse_date_code(p.v); if(_d) { p.t = 'd'; p.v = new Date(_d.y, _d.m-1,_d.d,_d.H,_d.M,_d.S,_d.u); }
  24254. }
  24255. }
  24256. function make_cell(val, ixfe, t) {
  24257. return ({v:val, ixfe:ixfe, t:t});
  24258. }
  24259. // 2.3.2
  24260. function parse_workbook(blob, options) {
  24261. var wb = ({opts:{}});
  24262. var Sheets = {};
  24263. if(DENSE != null && options.dense == null) options.dense = DENSE;
  24264. var out = ((options.dense ? [] : {}));
  24265. var Directory = {};
  24266. var range = ({});
  24267. var last_formula = null;
  24268. var sst = ([]);
  24269. var cur_sheet = "";
  24270. var Preamble = {};
  24271. var lastcell, last_cell = "", cc, cmnt, rngC, rngR;
  24272. var sharedf = {};
  24273. var arrayf = [];
  24274. var temp_val;
  24275. var country;
  24276. var cell_valid = true;
  24277. var XFs = []; /* XF records */
  24278. var palette = [];
  24279. var Workbook = ({ Sheets:[], WBProps:{date1904:false}, Views:[{}] }), wsprops = {};
  24280. var get_rgb = function getrgb(icv) {
  24281. if(icv < 8) return XLSIcv[icv];
  24282. if(icv < 64) return palette[icv-8] || XLSIcv[icv];
  24283. return XLSIcv[icv];
  24284. };
  24285. var process_cell_style = function pcs(cell, line, options) {
  24286. var xfd = line.XF.data;
  24287. if(!xfd || !xfd.patternType || !options || !options.cellStyles) return;
  24288. line.s = ({});
  24289. line.s.patternType = xfd.patternType;
  24290. var t;
  24291. if((t = rgb2Hex(get_rgb(xfd.icvFore)))) { line.s.fgColor = {rgb:t}; }
  24292. if((t = rgb2Hex(get_rgb(xfd.icvBack)))) { line.s.bgColor = {rgb:t}; }
  24293. };
  24294. var addcell = function addcell(cell, line, options) {
  24295. if(file_depth > 1) return;
  24296. if(options.sheetRows && cell.r >= options.sheetRows) cell_valid = false;
  24297. if(!cell_valid) return;
  24298. if(options.cellStyles && line.XF && line.XF.data) process_cell_style(cell, line, options);
  24299. delete line.ixfe; delete line.XF;
  24300. lastcell = cell;
  24301. last_cell = encode_cell(cell);
  24302. if(!range || !range.s || !range.e) range = {s:{r:0,c:0},e:{r:0,c:0}};
  24303. if(cell.r < range.s.r) range.s.r = cell.r;
  24304. if(cell.c < range.s.c) range.s.c = cell.c;
  24305. if(cell.r + 1 > range.e.r) range.e.r = cell.r + 1;
  24306. if(cell.c + 1 > range.e.c) range.e.c = cell.c + 1;
  24307. if(options.cellFormula && line.f) {
  24308. for(var afi = 0; afi < arrayf.length; ++afi) {
  24309. if(arrayf[afi][0].s.c > cell.c || arrayf[afi][0].s.r > cell.r) continue;
  24310. if(arrayf[afi][0].e.c < cell.c || arrayf[afi][0].e.r < cell.r) continue;
  24311. line.F = encode_range(arrayf[afi][0]);
  24312. if(arrayf[afi][0].s.c != cell.c || arrayf[afi][0].s.r != cell.r) delete line.f;
  24313. if(line.f) line.f = "" + stringify_formula(arrayf[afi][1], range, cell, supbooks, opts);
  24314. break;
  24315. }
  24316. }
  24317. {
  24318. if(options.dense) {
  24319. if(!out[cell.r]) out[cell.r] = [];
  24320. out[cell.r][cell.c] = line;
  24321. } else out[last_cell] = line;
  24322. }
  24323. };
  24324. var opts = ({
  24325. enc: false, // encrypted
  24326. sbcch: 0, // cch in the preceding SupBook
  24327. snames: [], // sheetnames
  24328. sharedf: sharedf, // shared formulae by address
  24329. arrayf: arrayf, // array formulae array
  24330. rrtabid: [], // RRTabId
  24331. lastuser: "", // Last User from WriteAccess
  24332. biff: 8, // BIFF version
  24333. codepage: 0, // CP from CodePage record
  24334. winlocked: 0, // fLockWn from WinProtect
  24335. cellStyles: !!options && !!options.cellStyles,
  24336. WTF: !!options && !!options.wtf
  24337. });
  24338. if(options.password) opts.password = options.password;
  24339. var themes;
  24340. var merges = [];
  24341. var objects = [];
  24342. var colinfo = [], rowinfo = [];
  24343. // eslint-disable-next-line no-unused-vars
  24344. var defwidth = 0, defheight = 0; // twips / MDW respectively
  24345. var seencol = false;
  24346. var supbooks = ([]); // 1-indexed, will hold extern names
  24347. supbooks.SheetNames = opts.snames;
  24348. supbooks.sharedf = opts.sharedf;
  24349. supbooks.arrayf = opts.arrayf;
  24350. supbooks.names = [];
  24351. supbooks.XTI = [];
  24352. var last_Rn = '';
  24353. var file_depth = 0; /* TODO: make a real stack */
  24354. var BIFF2Fmt = 0, BIFF2FmtTable = [];
  24355. var FilterDatabases = []; /* TODO: sort out supbooks and process elsewhere */
  24356. var last_lbl;
  24357. /* explicit override for some broken writers */
  24358. opts.codepage = 1200;
  24359. set_cp(1200);
  24360. var seen_codepage = false;
  24361. while(blob.l < blob.length - 1) {
  24362. var s = blob.l;
  24363. var RecordType = blob.read_shift(2);
  24364. if(RecordType === 0 && last_Rn === 'EOF') break;
  24365. var length = (blob.l === blob.length ? 0 : blob.read_shift(2));
  24366. var R = XLSRecordEnum[RecordType];
  24367. //console.log(RecordType.toString(16), RecordType, R, blob.l, length, blob.length);
  24368. //if(!R) console.log(blob.slice(blob.l, blob.l + length));
  24369. if(R && R.f) {
  24370. if(options.bookSheets) {
  24371. if(last_Rn === 'BoundSheet8' && R.n !== 'BoundSheet8') break;
  24372. }
  24373. last_Rn = R.n;
  24374. if(R.r === 2 || R.r == 12) {
  24375. var rt = blob.read_shift(2); length -= 2;
  24376. if(!opts.enc && rt !== RecordType && (((rt&0xFF)<<8)|(rt>>8)) !== RecordType) throw new Error("rt mismatch: " + rt + "!=" + RecordType);
  24377. if(R.r == 12){ blob.l += 10; length -= 10; } // skip FRT
  24378. }
  24379. //console.error(R,blob.l,length,blob.length);
  24380. var val;
  24381. if(R.n === 'EOF') val = R.f(blob, length, opts);
  24382. else val = slurp(R, blob, length, opts);
  24383. var Rn = R.n;
  24384. if(file_depth == 0 && Rn != 'BOF') continue;
  24385. /* nested switch statements to workaround V8 128 limit */
  24386. switch(Rn) {
  24387. /* Workbook Options */
  24388. case 'Date1904':
  24389. wb.opts.Date1904 = Workbook.WBProps.date1904 = val; break;
  24390. case 'WriteProtect': wb.opts.WriteProtect = true; break;
  24391. case 'FilePass':
  24392. if(!opts.enc) blob.l = 0;
  24393. opts.enc = val;
  24394. if(!options.password) throw new Error("File is password-protected");
  24395. if(val.valid == null) throw new Error("Encryption scheme unsupported");
  24396. if(!val.valid) throw new Error("Password is incorrect");
  24397. break;
  24398. case 'WriteAccess': opts.lastuser = val; break;
  24399. case 'FileSharing': break; //TODO
  24400. case 'CodePage':
  24401. /* overrides based on test cases */
  24402. switch(val) {
  24403. case 0x5212: val = 1200; break;
  24404. case 0x8000: val = 10000; break;
  24405. case 0x8001: val = 1252; break;
  24406. }
  24407. set_cp(opts.codepage = val);
  24408. seen_codepage = true;
  24409. break;
  24410. case 'RRTabId': opts.rrtabid = val; break;
  24411. case 'WinProtect': opts.winlocked = val; break;
  24412. case 'Template': break; // TODO
  24413. case 'BookBool': break; // TODO
  24414. case 'UsesELFs': break;
  24415. case 'MTRSettings': break;
  24416. case 'RefreshAll':
  24417. case 'CalcCount':
  24418. case 'CalcDelta':
  24419. case 'CalcIter':
  24420. case 'CalcMode':
  24421. case 'CalcPrecision':
  24422. case 'CalcSaveRecalc':
  24423. wb.opts[Rn] = val; break;
  24424. case 'CalcRefMode': opts.CalcRefMode = val; break; // TODO: implement R1C1
  24425. case 'Uncalced': break;
  24426. case 'ForceFullCalculation': wb.opts.FullCalc = val; break;
  24427. case 'WsBool':
  24428. if(val.fDialog) out["!type"] = "dialog";
  24429. break; // TODO
  24430. case 'XF':
  24431. XFs.push(val); break;
  24432. case 'ExtSST': break; // TODO
  24433. case 'BookExt': break; // TODO
  24434. case 'RichTextStream': break;
  24435. case 'BkHim': break;
  24436. case 'SupBook':
  24437. supbooks.push([val]);
  24438. supbooks[supbooks.length-1].XTI = [];
  24439. break;
  24440. case 'ExternName':
  24441. supbooks[supbooks.length-1].push(val);
  24442. break;
  24443. case 'Index': break; // TODO
  24444. case 'Lbl':
  24445. last_lbl = ({
  24446. Name: val.Name,
  24447. Ref: stringify_formula(val.rgce,range,null,supbooks,opts)
  24448. });
  24449. if(val.itab > 0) last_lbl.Sheet = val.itab - 1;
  24450. supbooks.names.push(last_lbl);
  24451. if(!supbooks[0]) { supbooks[0] = []; supbooks[0].XTI = []; }
  24452. supbooks[supbooks.length-1].push(val);
  24453. if(val.Name == "_xlnm._FilterDatabase" && val.itab > 0)
  24454. if(val.rgce && val.rgce[0] && val.rgce[0][0] && val.rgce[0][0][0] == 'PtgArea3d')
  24455. FilterDatabases[val.itab - 1] = { ref: encode_range(val.rgce[0][0][1][2]) };
  24456. break;
  24457. case 'ExternCount': opts.ExternCount = val; break;
  24458. case 'ExternSheet':
  24459. if(supbooks.length == 0) { supbooks[0] = []; supbooks[0].XTI = []; }
  24460. supbooks[supbooks.length - 1].XTI = supbooks[supbooks.length - 1].XTI.concat(val); supbooks.XTI = supbooks.XTI.concat(val); break;
  24461. case 'NameCmt':
  24462. /* TODO: search for correct name */
  24463. if(opts.biff < 8) break;
  24464. if(last_lbl != null) last_lbl.Comment = val[1];
  24465. break;
  24466. case 'Protect': out["!protect"] = val; break; /* for sheet or book */
  24467. case 'Password': if(val !== 0 && opts.WTF) console.error("Password verifier: " + val); break;
  24468. case 'Prot4Rev': case 'Prot4RevPass': break; /*TODO: Revision Control*/
  24469. case 'BoundSheet8': {
  24470. Directory[val.pos] = val;
  24471. opts.snames.push(val.name);
  24472. } break;
  24473. case 'EOF': {
  24474. if(--file_depth) break;
  24475. if(range.e) {
  24476. if(range.e.r > 0 && range.e.c > 0) {
  24477. range.e.r--; range.e.c--;
  24478. out["!ref"] = encode_range(range);
  24479. if(options.sheetRows && options.sheetRows <= range.e.r) {
  24480. var tmpri = range.e.r;
  24481. range.e.r = options.sheetRows - 1;
  24482. out["!fullref"] = out["!ref"];
  24483. out["!ref"] = encode_range(range);
  24484. range.e.r = tmpri;
  24485. }
  24486. range.e.r++; range.e.c++;
  24487. }
  24488. if(merges.length > 0) out["!merges"] = merges;
  24489. if(objects.length > 0) out["!objects"] = objects;
  24490. if(colinfo.length > 0) out["!cols"] = colinfo;
  24491. if(rowinfo.length > 0) out["!rows"] = rowinfo;
  24492. Workbook.Sheets.push(wsprops);
  24493. }
  24494. if(cur_sheet === "") Preamble = out; else Sheets[cur_sheet] = out;
  24495. out = ((options.dense ? [] : {}));
  24496. } break;
  24497. case 'BOF': {
  24498. if(opts.biff === 8) opts.biff = {
  24499. 0x0009:2,
  24500. 0x0209:3,
  24501. 0x0409:4
  24502. }[RecordType] || {
  24503. 0x0200:2,
  24504. 0x0300:3,
  24505. 0x0400:4,
  24506. 0x0500:5,
  24507. 0x0600:8,
  24508. 0x0002:2,
  24509. 0x0007:2
  24510. }[val.BIFFVer] || 8;
  24511. if(opts.biff == 8 && val.BIFFVer == 0 && val.dt == 16) opts.biff = 2;
  24512. if(file_depth++) break;
  24513. cell_valid = true;
  24514. out = ((options.dense ? [] : {}));
  24515. if(opts.biff < 8 && !seen_codepage) { seen_codepage = true; set_cp(opts.codepage = options.codepage || 1252); }
  24516. if(opts.biff < 5) {
  24517. if(cur_sheet === "") cur_sheet = "Sheet1";
  24518. range = {s:{r:0,c:0},e:{r:0,c:0}};
  24519. /* fake BoundSheet8 */
  24520. var fakebs8 = {pos: blob.l - length, name:cur_sheet};
  24521. Directory[fakebs8.pos] = fakebs8;
  24522. opts.snames.push(cur_sheet);
  24523. }
  24524. else cur_sheet = (Directory[s] || {name:""}).name;
  24525. if(val.dt == 0x20) out["!type"] = "chart";
  24526. if(val.dt == 0x40) out["!type"] = "macro";
  24527. merges = [];
  24528. objects = [];
  24529. opts.arrayf = arrayf = [];
  24530. colinfo = []; rowinfo = [];
  24531. defwidth = defheight = 0;
  24532. seencol = false;
  24533. wsprops = {Hidden:(Directory[s]||{hs:0}).hs, name:cur_sheet };
  24534. } break;
  24535. case 'Number': case 'BIFF2NUM': case 'BIFF2INT': {
  24536. if(out["!type"] == "chart") if(options.dense ? (out[val.r]||[])[val.c]: out[encode_cell({c:val.c, r:val.r})]) ++val.c;
  24537. temp_val = ({ixfe: val.ixfe, XF: XFs[val.ixfe]||{}, v:val.val, t:'n'});
  24538. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24539. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24540. addcell({c:val.c, r:val.r}, temp_val, options);
  24541. } break;
  24542. case 'BoolErr': {
  24543. temp_val = ({ixfe: val.ixfe, XF: XFs[val.ixfe], v:val.val, t:val.t});
  24544. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24545. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24546. addcell({c:val.c, r:val.r}, temp_val, options);
  24547. } break;
  24548. case 'RK': {
  24549. temp_val = ({ixfe: val.ixfe, XF: XFs[val.ixfe], v:val.rknum, t:'n'});
  24550. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24551. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24552. addcell({c:val.c, r:val.r}, temp_val, options);
  24553. } break;
  24554. case 'MulRk': {
  24555. for(var j = val.c; j <= val.C; ++j) {
  24556. var ixfe = val.rkrec[j-val.c][0];
  24557. temp_val= ({ixfe:ixfe, XF:XFs[ixfe], v:val.rkrec[j-val.c][1], t:'n'});
  24558. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24559. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24560. addcell({c:j, r:val.r}, temp_val, options);
  24561. }
  24562. } break;
  24563. case 'Formula': {
  24564. if(val.val == 'String') { last_formula = val; break; }
  24565. temp_val = make_cell(val.val, val.cell.ixfe, val.tt);
  24566. temp_val.XF = XFs[temp_val.ixfe];
  24567. if(options.cellFormula) {
  24568. var _f = val.formula;
  24569. if(_f && _f[0] && _f[0][0] && _f[0][0][0] == 'PtgExp') {
  24570. var _fr = _f[0][0][1][0], _fc = _f[0][0][1][1];
  24571. var _fe = encode_cell({r:_fr, c:_fc});
  24572. if(sharedf[_fe]) temp_val.f = ""+stringify_formula(val.formula,range,val.cell,supbooks, opts);
  24573. else temp_val.F = ((options.dense ? (out[_fr]||[])[_fc]: out[_fe]) || {}).F;
  24574. } else temp_val.f = ""+stringify_formula(val.formula,range,val.cell,supbooks, opts);
  24575. }
  24576. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24577. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24578. addcell(val.cell, temp_val, options);
  24579. last_formula = val;
  24580. } break;
  24581. case 'String': {
  24582. if(last_formula) { /* technically always true */
  24583. last_formula.val = val;
  24584. temp_val = make_cell(val, last_formula.cell.ixfe, 's');
  24585. temp_val.XF = XFs[temp_val.ixfe];
  24586. if(options.cellFormula) {
  24587. temp_val.f = ""+stringify_formula(last_formula.formula, range, last_formula.cell, supbooks, opts);
  24588. }
  24589. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24590. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24591. addcell(last_formula.cell, temp_val, options);
  24592. last_formula = null;
  24593. } else throw new Error("String record expects Formula");
  24594. } break;
  24595. case 'Array': {
  24596. arrayf.push(val);
  24597. var _arraystart = encode_cell(val[0].s);
  24598. cc = options.dense ? (out[val[0].s.r]||[])[val[0].s.c] : out[_arraystart];
  24599. if(options.cellFormula && cc) {
  24600. if(!last_formula) break; /* technically unreachable */
  24601. if(!_arraystart || !cc) break;
  24602. cc.f = ""+stringify_formula(val[1], range, val[0], supbooks, opts);
  24603. cc.F = encode_range(val[0]);
  24604. }
  24605. } break;
  24606. case 'ShrFmla': {
  24607. if(!cell_valid) break;
  24608. if(!options.cellFormula) break;
  24609. if(last_cell) {
  24610. /* TODO: capture range */
  24611. if(!last_formula) break; /* technically unreachable */
  24612. sharedf[encode_cell(last_formula.cell)]= val[0];
  24613. cc = options.dense ? (out[last_formula.cell.r]||[])[last_formula.cell.c] : out[encode_cell(last_formula.cell)];
  24614. (cc||{}).f = ""+stringify_formula(val[0], range, lastcell, supbooks, opts);
  24615. }
  24616. } break;
  24617. case 'LabelSst':
  24618. temp_val=make_cell(sst[val.isst].t, val.ixfe, 's');
  24619. temp_val.XF = XFs[temp_val.ixfe];
  24620. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24621. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24622. addcell({c:val.c, r:val.r}, temp_val, options);
  24623. break;
  24624. case 'Blank': if(options.sheetStubs) {
  24625. temp_val = ({ixfe: val.ixfe, XF: XFs[val.ixfe], t:'z'});
  24626. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24627. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24628. addcell({c:val.c, r:val.r}, temp_val, options);
  24629. } break;
  24630. case 'MulBlank': if(options.sheetStubs) {
  24631. for(var _j = val.c; _j <= val.C; ++_j) {
  24632. var _ixfe = val.ixfe[_j-val.c];
  24633. temp_val= ({ixfe:_ixfe, XF:XFs[_ixfe], t:'z'});
  24634. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24635. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24636. addcell({c:_j, r:val.r}, temp_val, options);
  24637. }
  24638. } break;
  24639. case 'RString':
  24640. case 'Label': case 'BIFF2STR':
  24641. temp_val=make_cell(val.val, val.ixfe, 's');
  24642. temp_val.XF = XFs[temp_val.ixfe];
  24643. if(BIFF2Fmt > 0) temp_val.z = BIFF2FmtTable[(temp_val.ixfe>>8) & 0x1F];
  24644. safe_format_xf(temp_val, options, wb.opts.Date1904);
  24645. addcell({c:val.c, r:val.r}, temp_val, options);
  24646. break;
  24647. case 'Dimensions': {
  24648. if(file_depth === 1) range = val; /* TODO: stack */
  24649. } break;
  24650. case 'SST': {
  24651. sst = val;
  24652. } break;
  24653. case 'Format': { /* val = [id, fmt] */
  24654. if(opts.biff == 4) {
  24655. BIFF2FmtTable[BIFF2Fmt++] = val[1];
  24656. for(var b4idx = 0; b4idx < BIFF2Fmt + 163; ++b4idx) if(SSF._table[b4idx] == val[1]) break;
  24657. if(b4idx >= 163) SSF.load(val[1], BIFF2Fmt + 163);
  24658. }
  24659. else SSF.load(val[1], val[0]);
  24660. } break;
  24661. case 'BIFF2FORMAT': {
  24662. BIFF2FmtTable[BIFF2Fmt++] = val;
  24663. for(var b2idx = 0; b2idx < BIFF2Fmt + 163; ++b2idx) if(SSF._table[b2idx] == val) break;
  24664. if(b2idx >= 163) SSF.load(val, BIFF2Fmt + 163);
  24665. } break;
  24666. case 'MergeCells': merges = merges.concat(val); break;
  24667. case 'Obj': objects[val.cmo[0]] = opts.lastobj = val; break;
  24668. case 'TxO': opts.lastobj.TxO = val; break;
  24669. case 'ImData': opts.lastobj.ImData = val; break;
  24670. case 'HLink': {
  24671. for(rngR = val[0].s.r; rngR <= val[0].e.r; ++rngR)
  24672. for(rngC = val[0].s.c; rngC <= val[0].e.c; ++rngC) {
  24673. cc = options.dense ? (out[rngR]||[])[rngC] : out[encode_cell({c:rngC,r:rngR})];
  24674. if(cc) cc.l = val[1];
  24675. }
  24676. } break;
  24677. case 'HLinkTooltip': {
  24678. for(rngR = val[0].s.r; rngR <= val[0].e.r; ++rngR)
  24679. for(rngC = val[0].s.c; rngC <= val[0].e.c; ++rngC) {
  24680. cc = options.dense ? (out[rngR]||[])[rngC] : out[encode_cell({c:rngC,r:rngR})];
  24681. if(cc && cc.l) cc.l.Tooltip = val[1];
  24682. }
  24683. } break;
  24684. /* Comments */
  24685. case 'Note': {
  24686. if(opts.biff <= 5 && opts.biff >= 2) break; /* TODO: BIFF5 */
  24687. cc = options.dense ? (out[val[0].r]||[])[val[0].c] : out[encode_cell(val[0])];
  24688. var noteobj = objects[val[2]];
  24689. if(!cc) break;
  24690. if(!cc.c) cc.c = [];
  24691. cmnt = {a:val[1],t:noteobj.TxO.t};
  24692. cc.c.push(cmnt);
  24693. } break;
  24694. default: switch(R.n) { /* nested */
  24695. case 'ClrtClient': break;
  24696. case 'XFExt': update_xfext(XFs[val.ixfe], val.ext); break;
  24697. case 'DefColWidth': defwidth = val; break;
  24698. case 'DefaultRowHeight': defheight = val[1]; break; // TODO: flags
  24699. case 'ColInfo': {
  24700. if(!opts.cellStyles) break;
  24701. while(val.e >= val.s) {
  24702. colinfo[val.e--] = { width: val.w/256 };
  24703. if(!seencol) { seencol = true; find_mdw_colw(val.w/256); }
  24704. process_col(colinfo[val.e+1]);
  24705. }
  24706. } break;
  24707. case 'Row': {
  24708. var rowobj = {};
  24709. if(val.level != null) { rowinfo[val.r] = rowobj; rowobj.level = val.level; }
  24710. if(val.hidden) { rowinfo[val.r] = rowobj; rowobj.hidden = true; }
  24711. if(val.hpt) {
  24712. rowinfo[val.r] = rowobj;
  24713. rowobj.hpt = val.hpt; rowobj.hpx = pt2px(val.hpt);
  24714. }
  24715. } break;
  24716. case 'LeftMargin':
  24717. case 'RightMargin':
  24718. case 'TopMargin':
  24719. case 'BottomMargin':
  24720. if(!out['!margins']) default_margins(out['!margins'] = {});
  24721. out['!margins'][Rn.slice(0,-6).toLowerCase()] = val;
  24722. break;
  24723. case 'Setup': // TODO
  24724. if(!out['!margins']) default_margins(out['!margins'] = {});
  24725. out['!margins'].header = val.header;
  24726. out['!margins'].footer = val.footer;
  24727. break;
  24728. case 'Window2': // TODO
  24729. // $FlowIgnore
  24730. if(val.RTL) Workbook.Views[0].RTL = true;
  24731. break;
  24732. case 'Header': break; // TODO
  24733. case 'Footer': break; // TODO
  24734. case 'HCenter': break; // TODO
  24735. case 'VCenter': break; // TODO
  24736. case 'Pls': break; // TODO
  24737. case 'GCW': break;
  24738. case 'LHRecord': break;
  24739. case 'DBCell': break; // TODO
  24740. case 'EntExU2': break; // TODO
  24741. case 'SxView': break; // TODO
  24742. case 'Sxvd': break; // TODO
  24743. case 'SXVI': break; // TODO
  24744. case 'SXVDEx': break; // TODO
  24745. case 'SxIvd': break; // TODO
  24746. case 'SXString': break; // TODO
  24747. case 'Sync': break;
  24748. case 'Addin': break;
  24749. case 'SXDI': break; // TODO
  24750. case 'SXLI': break; // TODO
  24751. case 'SXEx': break; // TODO
  24752. case 'QsiSXTag': break; // TODO
  24753. case 'Selection': break;
  24754. case 'Feat': break;
  24755. case 'FeatHdr': case 'FeatHdr11': break;
  24756. case 'Feature11': case 'Feature12': case 'List12': break;
  24757. case 'Country': country = val; break;
  24758. case 'RecalcId': break;
  24759. case 'DxGCol': break; // TODO: htmlify
  24760. case 'Fbi': case 'Fbi2': case 'GelFrame': break;
  24761. case 'Font': break; // TODO
  24762. case 'XFCRC': break; // TODO
  24763. case 'Style': break; // TODO
  24764. case 'StyleExt': break; // TODO
  24765. case 'Palette': palette = val; break;
  24766. case 'Theme': themes = val; break;
  24767. /* Protection */
  24768. case 'ScenarioProtect': break;
  24769. case 'ObjProtect': break;
  24770. /* Conditional Formatting */
  24771. case 'CondFmt12': break;
  24772. /* Table */
  24773. case 'Table': break; // TODO
  24774. case 'TableStyles': break; // TODO
  24775. case 'TableStyle': break; // TODO
  24776. case 'TableStyleElement': break; // TODO
  24777. /* PivotTable */
  24778. case 'SXStreamID': break; // TODO
  24779. case 'SXVS': break; // TODO
  24780. case 'DConRef': break; // TODO
  24781. case 'SXAddl': break; // TODO
  24782. case 'DConBin': break; // TODO
  24783. case 'DConName': break; // TODO
  24784. case 'SXPI': break; // TODO
  24785. case 'SxFormat': break; // TODO
  24786. case 'SxSelect': break; // TODO
  24787. case 'SxRule': break; // TODO
  24788. case 'SxFilt': break; // TODO
  24789. case 'SxItm': break; // TODO
  24790. case 'SxDXF': break; // TODO
  24791. /* Scenario Manager */
  24792. case 'ScenMan': break;
  24793. /* Data Consolidation */
  24794. case 'DCon': break;
  24795. /* Watched Cell */
  24796. case 'CellWatch': break;
  24797. /* Print Settings */
  24798. case 'PrintRowCol': break;
  24799. case 'PrintGrid': break;
  24800. case 'PrintSize': break;
  24801. case 'XCT': break;
  24802. case 'CRN': break;
  24803. case 'Scl': {
  24804. //console.log("Zoom Level:", val[0]/val[1],val);
  24805. } break;
  24806. case 'SheetExt': {
  24807. /* empty */
  24808. } break;
  24809. case 'SheetExtOptional': {
  24810. /* empty */
  24811. } break;
  24812. /* VBA */
  24813. case 'ObNoMacros': {
  24814. /* empty */
  24815. } break;
  24816. case 'ObProj': {
  24817. /* empty */
  24818. } break;
  24819. case 'CodeName': {
  24820. if(!cur_sheet) Workbook.WBProps.CodeName = val || "ThisWorkbook";
  24821. else wsprops.CodeName = val || wsprops.name;
  24822. } break;
  24823. case 'GUIDTypeLib': {
  24824. /* empty */
  24825. } break;
  24826. case 'WOpt': break; // TODO: WTF?
  24827. case 'PhoneticInfo': break;
  24828. case 'OleObjectSize': break;
  24829. /* Differential Formatting */
  24830. case 'DXF': case 'DXFN': case 'DXFN12': case 'DXFN12List': case 'DXFN12NoCB': break;
  24831. /* Data Validation */
  24832. case 'Dv': case 'DVal': break;
  24833. /* Data Series */
  24834. case 'BRAI': case 'Series': case 'SeriesText': break;
  24835. /* Data Connection */
  24836. case 'DConn': break;
  24837. case 'DbOrParamQry': break;
  24838. case 'DBQueryExt': break;
  24839. case 'OleDbConn': break;
  24840. case 'ExtString': break;
  24841. /* Formatting */
  24842. case 'IFmtRecord': break;
  24843. case 'CondFmt': case 'CF': case 'CF12': case 'CFEx': break;
  24844. /* Explicitly Ignored */
  24845. case 'Excel9File': break;
  24846. case 'Units': break;
  24847. case 'InterfaceHdr': case 'Mms': case 'InterfaceEnd': case 'DSF': break;
  24848. case 'BuiltInFnGroupCount': /* 2.4.30 0x0E or 0x10 but excel 2011 generates 0x11? */ break;
  24849. /* View Stuff */
  24850. case 'Window1': case 'HideObj': case 'GridSet': case 'Guts':
  24851. case 'UserBView': case 'UserSViewBegin': case 'UserSViewEnd':
  24852. case 'Pane': break;
  24853. default: switch(R.n) { /* nested */
  24854. /* Chart */
  24855. case 'Dat':
  24856. case 'Begin': case 'End':
  24857. case 'StartBlock': case 'EndBlock':
  24858. case 'Frame': case 'Area':
  24859. case 'Axis': case 'AxisLine': case 'Tick': break;
  24860. case 'AxesUsed':
  24861. case 'CrtLayout12': case 'CrtLayout12A': case 'CrtLink': case 'CrtLine': case 'CrtMlFrt': case 'CrtMlFrtContinue': break;
  24862. case 'LineFormat': case 'AreaFormat':
  24863. case 'Chart': case 'Chart3d': case 'Chart3DBarShape': case 'ChartFormat': case 'ChartFrtInfo': break;
  24864. case 'PlotArea': case 'PlotGrowth': break;
  24865. case 'SeriesList': case 'SerParent': case 'SerAuxTrend': break;
  24866. case 'DataFormat': case 'SerToCrt': case 'FontX': break;
  24867. case 'CatSerRange': case 'AxcExt': case 'SerFmt': break;
  24868. case 'ShtProps': break;
  24869. case 'DefaultText': case 'Text': case 'CatLab': break;
  24870. case 'DataLabExtContents': break;
  24871. case 'Legend': case 'LegendException': break;
  24872. case 'Pie': case 'Scatter': break;
  24873. case 'PieFormat': case 'MarkerFormat': break;
  24874. case 'StartObject': case 'EndObject': break;
  24875. case 'AlRuns': case 'ObjectLink': break;
  24876. case 'SIIndex': break;
  24877. case 'AttachedLabel': case 'YMult': break;
  24878. /* Chart Group */
  24879. case 'Line': case 'Bar': break;
  24880. case 'Surf': break;
  24881. /* Axis Group */
  24882. case 'AxisParent': break;
  24883. case 'Pos': break;
  24884. case 'ValueRange': break;
  24885. /* Pivot Chart */
  24886. case 'SXViewEx9': break; // TODO
  24887. case 'SXViewLink': break;
  24888. case 'PivotChartBits': break;
  24889. case 'SBaseRef': break;
  24890. case 'TextPropsStream': break;
  24891. /* Chart Misc */
  24892. case 'LnExt': break;
  24893. case 'MkrExt': break;
  24894. case 'CrtCoopt': break;
  24895. /* Query Table */
  24896. case 'Qsi': case 'Qsif': case 'Qsir': case 'QsiSXTag': break;
  24897. case 'TxtQry': break;
  24898. /* Filter */
  24899. case 'FilterMode': break;
  24900. case 'AutoFilter': case 'AutoFilterInfo': break;
  24901. case 'AutoFilter12': break;
  24902. case 'DropDownObjIds': break;
  24903. case 'Sort': break;
  24904. case 'SortData': break;
  24905. /* Drawing */
  24906. case 'ShapePropsStream': break;
  24907. case 'MsoDrawing': case 'MsoDrawingGroup': case 'MsoDrawingSelection': break;
  24908. /* Pub Stuff */
  24909. case 'WebPub': case 'AutoWebPub': break;
  24910. /* Print Stuff */
  24911. case 'HeaderFooter': case 'HFPicture': case 'PLV':
  24912. case 'HorizontalPageBreaks': case 'VerticalPageBreaks': break;
  24913. /* Behavioral */
  24914. case 'Backup': case 'CompressPictures': case 'Compat12': break;
  24915. /* Should not Happen */
  24916. case 'Continue': case 'ContinueFrt12': break;
  24917. /* Future Records */
  24918. case 'FrtFontList': case 'FrtWrapper': break;
  24919. default: switch(R.n) { /* nested */
  24920. /* BIFF5 records */
  24921. case 'TabIdConf': case 'Radar': case 'RadarArea': case 'DropBar': case 'Intl': case 'CoordList': case 'SerAuxErrBar': break;
  24922. /* BIFF2-4 records */
  24923. case 'BIFF2FONTCLR': case 'BIFF2FMTCNT': case 'BIFF2FONTXTRA': break;
  24924. case 'BIFF2XF': case 'BIFF3XF': case 'BIFF4XF': break;
  24925. case 'BIFF4FMTCNT': case 'BIFF2ROW': case 'BIFF2WINDOW2': break;
  24926. /* Miscellaneous */
  24927. case 'SCENARIO': case 'DConBin': case 'PicF': case 'DataLabExt':
  24928. case 'Lel': case 'BopPop': case 'BopPopCustom': case 'RealTimeData':
  24929. case 'Name': break;
  24930. case 'LHNGraph': case 'FnGroupName': case 'AddMenu': case 'LPr': break;
  24931. case 'ListObj': case 'ListField': break;
  24932. case 'RRSort': break;
  24933. case 'BigName': break;
  24934. case 'ToolbarHdr': case 'ToolbarEnd': break;
  24935. case 'DDEObjName': break;
  24936. case 'FRTArchId$': break;
  24937. default: if(options.WTF) throw 'Unrecognized Record ' + R.n;
  24938. }}}}
  24939. } else blob.l += length;
  24940. }
  24941. wb.SheetNames=keys(Directory).sort(function(a,b) { return Number(a) - Number(b); }).map(function(x){return Directory[x].name;});
  24942. if(!options.bookSheets) wb.Sheets=Sheets;
  24943. if(wb.Sheets) FilterDatabases.forEach(function(r,i) { wb.Sheets[wb.SheetNames[i]]['!autofilter'] = r; });
  24944. wb.Preamble=Preamble;
  24945. wb.Strings = sst;
  24946. wb.SSF = SSF.get_table();
  24947. if(opts.enc) wb.Encryption = opts.enc;
  24948. if(themes) wb.Themes = themes;
  24949. wb.Metadata = {};
  24950. if(country !== undefined) wb.Metadata.Country = country;
  24951. if(supbooks.names.length > 0) Workbook.Names = supbooks.names;
  24952. wb.Workbook = Workbook;
  24953. return wb;
  24954. }
  24955. /* TODO: split props*/
  24956. var PSCLSID = {
  24957. SI: "e0859ff2f94f6810ab9108002b27b3d9",
  24958. DSI: "02d5cdd59c2e1b10939708002b2cf9ae",
  24959. UDI: "05d5cdd59c2e1b10939708002b2cf9ae"
  24960. };
  24961. function parse_xls_props(cfb, props, o) {
  24962. /* [MS-OSHARED] 2.3.3.2.2 Document Summary Information Property Set */
  24963. var DSI = CFB.find(cfb, '!DocumentSummaryInformation');
  24964. if(DSI && DSI.size > 0) try {
  24965. var DocSummary = parse_PropertySetStream(DSI, DocSummaryPIDDSI, PSCLSID.DSI);
  24966. for(var d in DocSummary) props[d] = DocSummary[d];
  24967. } catch(e) {if(o.WTF) throw e;/* empty */}
  24968. /* [MS-OSHARED] 2.3.3.2.1 Summary Information Property Set*/
  24969. var SI = CFB.find(cfb, '!SummaryInformation');
  24970. if(SI && SI.size > 0) try {
  24971. var Summary = parse_PropertySetStream(SI, SummaryPIDSI, PSCLSID.SI);
  24972. for(var s in Summary) if(props[s] == null) props[s] = Summary[s];
  24973. } catch(e) {if(o.WTF) throw e;/* empty */}
  24974. if(props.HeadingPairs && props.TitlesOfParts) {
  24975. load_props_pairs(props.HeadingPairs, props.TitlesOfParts, props, o);
  24976. delete props.HeadingPairs; delete props.TitlesOfParts;
  24977. }
  24978. }
  24979. function write_xls_props(wb, cfb) {
  24980. var DSEntries = [], SEntries = [], CEntries = [];
  24981. var i = 0, Keys;
  24982. if(wb.Props) {
  24983. Keys = keys(wb.Props);
  24984. // $FlowIgnore
  24985. for(i = 0; i < Keys.length; ++i) (DocSummaryRE.hasOwnProperty(Keys[i]) ? DSEntries : SummaryRE.hasOwnProperty(Keys[i]) ? SEntries : CEntries).push([Keys[i], wb.Props[Keys[i]]]);
  24986. }
  24987. if(wb.Custprops) {
  24988. Keys = keys(wb.Custprops);
  24989. // $FlowIgnore
  24990. for(i = 0; i < Keys.length; ++i) if(!(wb.Props||{}).hasOwnProperty(Keys[i])) (DocSummaryRE.hasOwnProperty(Keys[i]) ? DSEntries : SummaryRE.hasOwnProperty(Keys[i]) ? SEntries : CEntries).push([Keys[i], wb.Custprops[Keys[i]]]);
  24991. }
  24992. var CEntries2 = [];
  24993. for(i = 0; i < CEntries.length; ++i) {
  24994. if(XLSPSSkip.indexOf(CEntries[i][0]) > -1) continue;
  24995. if(CEntries[i][1] == null) continue;
  24996. CEntries2.push(CEntries[i]);
  24997. }
  24998. if(SEntries.length) CFB.utils.cfb_add(cfb, "/\u0005SummaryInformation", write_PropertySetStream(SEntries, PSCLSID.SI, SummaryRE, SummaryPIDSI));
  24999. if(DSEntries.length || CEntries2.length) CFB.utils.cfb_add(cfb, "/\u0005DocumentSummaryInformation", write_PropertySetStream(DSEntries, PSCLSID.DSI, DocSummaryRE, DocSummaryPIDDSI, CEntries2.length ? CEntries2 : null, PSCLSID.UDI));
  25000. }
  25001. function parse_xlscfb(cfb, options) {
  25002. if(!options) options = {};
  25003. fix_read_opts(options);
  25004. reset_cp();
  25005. if(options.codepage) set_ansi(options.codepage);
  25006. var CompObj, WB;
  25007. if(cfb.FullPaths) {
  25008. if(CFB.find(cfb, '/encryption')) throw new Error("File is password-protected");
  25009. CompObj = CFB.find(cfb, '!CompObj');
  25010. WB = CFB.find(cfb, '/Workbook') || CFB.find(cfb, '/Book');
  25011. } else {
  25012. switch(options.type) {
  25013. case 'base64': cfb = s2a(Base64.decode(cfb)); break;
  25014. case 'binary': cfb = s2a(cfb); break;
  25015. case 'buffer': break;
  25016. case 'array': if(!Array.isArray(cfb)) cfb = Array.prototype.slice.call(cfb); break;
  25017. }
  25018. prep_blob(cfb, 0);
  25019. WB = ({content: cfb});
  25020. }
  25021. var WorkbookP;
  25022. var _data;
  25023. if(CompObj) parse_compobj(CompObj);
  25024. if(options.bookProps && !options.bookSheets) WorkbookP = ({});
  25025. else {
  25026. var T = has_buf ? 'buffer' : 'array';
  25027. if(WB && WB.content) WorkbookP = parse_workbook(WB.content, options);
  25028. /* Quattro Pro 7-8 */
  25029. else if((_data=CFB.find(cfb, 'PerfectOffice_MAIN')) && _data.content) WorkbookP = WK_.to_workbook(_data.content, (options.type = T, options));
  25030. /* Quattro Pro 9 */
  25031. else if((_data=CFB.find(cfb, 'NativeContent_MAIN')) && _data.content) WorkbookP = WK_.to_workbook(_data.content, (options.type = T, options));
  25032. else throw new Error("Cannot find Workbook stream");
  25033. if(options.bookVBA && cfb.FullPaths && CFB.find(cfb, '/_VBA_PROJECT_CUR/VBA/dir')) WorkbookP.vbaraw = make_vba_xls(cfb);
  25034. }
  25035. var props = {};
  25036. if(cfb.FullPaths) parse_xls_props(cfb, props, options);
  25037. WorkbookP.Props = WorkbookP.Custprops = props; /* TODO: split up properties */
  25038. if(options.bookFiles) WorkbookP.cfb = cfb;
  25039. /*WorkbookP.CompObjP = CompObjP; // TODO: storage? */
  25040. return WorkbookP;
  25041. }
  25042. function write_xlscfb(wb, opts) {
  25043. var o = opts || {};
  25044. var cfb = CFB.utils.cfb_new({root:"R"});
  25045. var wbpath = "/Workbook";
  25046. switch(o.bookType || "xls") {
  25047. case "xls": o.bookType = "biff8";
  25048. /* falls through */
  25049. case "xla": if(!o.bookType) o.bookType = "xla";
  25050. /* falls through */
  25051. case "biff8": wbpath = "/Workbook"; o.biff = 8; break;
  25052. case "biff5": wbpath = "/Book"; o.biff = 5; break;
  25053. default: throw new Error("invalid type " + o.bookType + " for XLS CFB");
  25054. }
  25055. CFB.utils.cfb_add(cfb, wbpath, write_biff_buf(wb, o));
  25056. if(o.biff == 8 && (wb.Props || wb.Custprops)) write_xls_props(wb, cfb);
  25057. // TODO: SI, DSI, CO
  25058. if(o.biff == 8 && wb.vbaraw) fill_vba_xls(cfb, CFB.read(wb.vbaraw, {type: typeof wb.vbaraw == "string" ? "binary" : "buffer"}));
  25059. return cfb;
  25060. }
  25061. /* [MS-XLSB] 2.3 Record Enumeration */
  25062. var XLSBRecordEnum = {
  25063. 0x0000: { n:"BrtRowHdr", f:parse_BrtRowHdr },
  25064. 0x0001: { n:"BrtCellBlank", f:parse_BrtCellBlank },
  25065. 0x0002: { n:"BrtCellRk", f:parse_BrtCellRk },
  25066. 0x0003: { n:"BrtCellError", f:parse_BrtCellError },
  25067. 0x0004: { n:"BrtCellBool", f:parse_BrtCellBool },
  25068. 0x0005: { n:"BrtCellReal", f:parse_BrtCellReal },
  25069. 0x0006: { n:"BrtCellSt", f:parse_BrtCellSt },
  25070. 0x0007: { n:"BrtCellIsst", f:parse_BrtCellIsst },
  25071. 0x0008: { n:"BrtFmlaString", f:parse_BrtFmlaString },
  25072. 0x0009: { n:"BrtFmlaNum", f:parse_BrtFmlaNum },
  25073. 0x000A: { n:"BrtFmlaBool", f:parse_BrtFmlaBool },
  25074. 0x000B: { n:"BrtFmlaError", f:parse_BrtFmlaError },
  25075. 0x0010: { n:"BrtFRTArchID$", f:parse_BrtFRTArchID$ },
  25076. 0x0013: { n:"BrtSSTItem", f:parse_RichStr },
  25077. 0x0014: { n:"BrtPCDIMissing" },
  25078. 0x0015: { n:"BrtPCDINumber" },
  25079. 0x0016: { n:"BrtPCDIBoolean" },
  25080. 0x0017: { n:"BrtPCDIError" },
  25081. 0x0018: { n:"BrtPCDIString" },
  25082. 0x0019: { n:"BrtPCDIDatetime" },
  25083. 0x001A: { n:"BrtPCDIIndex" },
  25084. 0x001B: { n:"BrtPCDIAMissing" },
  25085. 0x001C: { n:"BrtPCDIANumber" },
  25086. 0x001D: { n:"BrtPCDIABoolean" },
  25087. 0x001E: { n:"BrtPCDIAError" },
  25088. 0x001F: { n:"BrtPCDIAString" },
  25089. 0x0020: { n:"BrtPCDIADatetime" },
  25090. 0x0021: { n:"BrtPCRRecord" },
  25091. 0x0022: { n:"BrtPCRRecordDt" },
  25092. 0x0023: { n:"BrtFRTBegin" },
  25093. 0x0024: { n:"BrtFRTEnd" },
  25094. 0x0025: { n:"BrtACBegin" },
  25095. 0x0026: { n:"BrtACEnd" },
  25096. 0x0027: { n:"BrtName", f:parse_BrtName },
  25097. 0x0028: { n:"BrtIndexRowBlock" },
  25098. 0x002A: { n:"BrtIndexBlock" },
  25099. 0x002B: { n:"BrtFont", f:parse_BrtFont },
  25100. 0x002C: { n:"BrtFmt", f:parse_BrtFmt },
  25101. 0x002D: { n:"BrtFill", f:parse_BrtFill },
  25102. 0x002E: { n:"BrtBorder", f:parse_BrtBorder },
  25103. 0x002F: { n:"BrtXF", f:parse_BrtXF },
  25104. 0x0030: { n:"BrtStyle" },
  25105. 0x0031: { n:"BrtCellMeta" },
  25106. 0x0032: { n:"BrtValueMeta" },
  25107. 0x0033: { n:"BrtMdb" },
  25108. 0x0034: { n:"BrtBeginFmd" },
  25109. 0x0035: { n:"BrtEndFmd" },
  25110. 0x0036: { n:"BrtBeginMdx" },
  25111. 0x0037: { n:"BrtEndMdx" },
  25112. 0x0038: { n:"BrtBeginMdxTuple" },
  25113. 0x0039: { n:"BrtEndMdxTuple" },
  25114. 0x003A: { n:"BrtMdxMbrIstr" },
  25115. 0x003B: { n:"BrtStr" },
  25116. 0x003C: { n:"BrtColInfo", f:parse_ColInfo },
  25117. 0x003E: { n:"BrtCellRString" },
  25118. 0x003F: { n:"BrtCalcChainItem$", f:parse_BrtCalcChainItem$ },
  25119. 0x0040: { n:"BrtDVal" },
  25120. 0x0041: { n:"BrtSxvcellNum" },
  25121. 0x0042: { n:"BrtSxvcellStr" },
  25122. 0x0043: { n:"BrtSxvcellBool" },
  25123. 0x0044: { n:"BrtSxvcellErr" },
  25124. 0x0045: { n:"BrtSxvcellDate" },
  25125. 0x0046: { n:"BrtSxvcellNil" },
  25126. 0x0080: { n:"BrtFileVersion" },
  25127. 0x0081: { n:"BrtBeginSheet" },
  25128. 0x0082: { n:"BrtEndSheet" },
  25129. 0x0083: { n:"BrtBeginBook", f:parsenoop, p:0 },
  25130. 0x0084: { n:"BrtEndBook" },
  25131. 0x0085: { n:"BrtBeginWsViews" },
  25132. 0x0086: { n:"BrtEndWsViews" },
  25133. 0x0087: { n:"BrtBeginBookViews" },
  25134. 0x0088: { n:"BrtEndBookViews" },
  25135. 0x0089: { n:"BrtBeginWsView", f:parse_BrtBeginWsView },
  25136. 0x008A: { n:"BrtEndWsView" },
  25137. 0x008B: { n:"BrtBeginCsViews" },
  25138. 0x008C: { n:"BrtEndCsViews" },
  25139. 0x008D: { n:"BrtBeginCsView" },
  25140. 0x008E: { n:"BrtEndCsView" },
  25141. 0x008F: { n:"BrtBeginBundleShs" },
  25142. 0x0090: { n:"BrtEndBundleShs" },
  25143. 0x0091: { n:"BrtBeginSheetData" },
  25144. 0x0092: { n:"BrtEndSheetData" },
  25145. 0x0093: { n:"BrtWsProp", f:parse_BrtWsProp },
  25146. 0x0094: { n:"BrtWsDim", f:parse_BrtWsDim, p:16 },
  25147. 0x0097: { n:"BrtPane" },
  25148. 0x0098: { n:"BrtSel" },
  25149. 0x0099: { n:"BrtWbProp", f:parse_BrtWbProp },
  25150. 0x009A: { n:"BrtWbFactoid" },
  25151. 0x009B: { n:"BrtFileRecover" },
  25152. 0x009C: { n:"BrtBundleSh", f:parse_BrtBundleSh },
  25153. 0x009D: { n:"BrtCalcProp" },
  25154. 0x009E: { n:"BrtBookView" },
  25155. 0x009F: { n:"BrtBeginSst", f:parse_BrtBeginSst },
  25156. 0x00A0: { n:"BrtEndSst" },
  25157. 0x00A1: { n:"BrtBeginAFilter", f:parse_UncheckedRfX },
  25158. 0x00A2: { n:"BrtEndAFilter" },
  25159. 0x00A3: { n:"BrtBeginFilterColumn" },
  25160. 0x00A4: { n:"BrtEndFilterColumn" },
  25161. 0x00A5: { n:"BrtBeginFilters" },
  25162. 0x00A6: { n:"BrtEndFilters" },
  25163. 0x00A7: { n:"BrtFilter" },
  25164. 0x00A8: { n:"BrtColorFilter" },
  25165. 0x00A9: { n:"BrtIconFilter" },
  25166. 0x00AA: { n:"BrtTop10Filter" },
  25167. 0x00AB: { n:"BrtDynamicFilter" },
  25168. 0x00AC: { n:"BrtBeginCustomFilters" },
  25169. 0x00AD: { n:"BrtEndCustomFilters" },
  25170. 0x00AE: { n:"BrtCustomFilter" },
  25171. 0x00AF: { n:"BrtAFilterDateGroupItem" },
  25172. 0x00B0: { n:"BrtMergeCell", f:parse_BrtMergeCell },
  25173. 0x00B1: { n:"BrtBeginMergeCells" },
  25174. 0x00B2: { n:"BrtEndMergeCells" },
  25175. 0x00B3: { n:"BrtBeginPivotCacheDef" },
  25176. 0x00B4: { n:"BrtEndPivotCacheDef" },
  25177. 0x00B5: { n:"BrtBeginPCDFields" },
  25178. 0x00B6: { n:"BrtEndPCDFields" },
  25179. 0x00B7: { n:"BrtBeginPCDField" },
  25180. 0x00B8: { n:"BrtEndPCDField" },
  25181. 0x00B9: { n:"BrtBeginPCDSource" },
  25182. 0x00BA: { n:"BrtEndPCDSource" },
  25183. 0x00BB: { n:"BrtBeginPCDSRange" },
  25184. 0x00BC: { n:"BrtEndPCDSRange" },
  25185. 0x00BD: { n:"BrtBeginPCDFAtbl" },
  25186. 0x00BE: { n:"BrtEndPCDFAtbl" },
  25187. 0x00BF: { n:"BrtBeginPCDIRun" },
  25188. 0x00C0: { n:"BrtEndPCDIRun" },
  25189. 0x00C1: { n:"BrtBeginPivotCacheRecords" },
  25190. 0x00C2: { n:"BrtEndPivotCacheRecords" },
  25191. 0x00C3: { n:"BrtBeginPCDHierarchies" },
  25192. 0x00C4: { n:"BrtEndPCDHierarchies" },
  25193. 0x00C5: { n:"BrtBeginPCDHierarchy" },
  25194. 0x00C6: { n:"BrtEndPCDHierarchy" },
  25195. 0x00C7: { n:"BrtBeginPCDHFieldsUsage" },
  25196. 0x00C8: { n:"BrtEndPCDHFieldsUsage" },
  25197. 0x00C9: { n:"BrtBeginExtConnection" },
  25198. 0x00CA: { n:"BrtEndExtConnection" },
  25199. 0x00CB: { n:"BrtBeginECDbProps" },
  25200. 0x00CC: { n:"BrtEndECDbProps" },
  25201. 0x00CD: { n:"BrtBeginECOlapProps" },
  25202. 0x00CE: { n:"BrtEndECOlapProps" },
  25203. 0x00CF: { n:"BrtBeginPCDSConsol" },
  25204. 0x00D0: { n:"BrtEndPCDSConsol" },
  25205. 0x00D1: { n:"BrtBeginPCDSCPages" },
  25206. 0x00D2: { n:"BrtEndPCDSCPages" },
  25207. 0x00D3: { n:"BrtBeginPCDSCPage" },
  25208. 0x00D4: { n:"BrtEndPCDSCPage" },
  25209. 0x00D5: { n:"BrtBeginPCDSCPItem" },
  25210. 0x00D6: { n:"BrtEndPCDSCPItem" },
  25211. 0x00D7: { n:"BrtBeginPCDSCSets" },
  25212. 0x00D8: { n:"BrtEndPCDSCSets" },
  25213. 0x00D9: { n:"BrtBeginPCDSCSet" },
  25214. 0x00DA: { n:"BrtEndPCDSCSet" },
  25215. 0x00DB: { n:"BrtBeginPCDFGroup" },
  25216. 0x00DC: { n:"BrtEndPCDFGroup" },
  25217. 0x00DD: { n:"BrtBeginPCDFGItems" },
  25218. 0x00DE: { n:"BrtEndPCDFGItems" },
  25219. 0x00DF: { n:"BrtBeginPCDFGRange" },
  25220. 0x00E0: { n:"BrtEndPCDFGRange" },
  25221. 0x00E1: { n:"BrtBeginPCDFGDiscrete" },
  25222. 0x00E2: { n:"BrtEndPCDFGDiscrete" },
  25223. 0x00E3: { n:"BrtBeginPCDSDTupleCache" },
  25224. 0x00E4: { n:"BrtEndPCDSDTupleCache" },
  25225. 0x00E5: { n:"BrtBeginPCDSDTCEntries" },
  25226. 0x00E6: { n:"BrtEndPCDSDTCEntries" },
  25227. 0x00E7: { n:"BrtBeginPCDSDTCEMembers" },
  25228. 0x00E8: { n:"BrtEndPCDSDTCEMembers" },
  25229. 0x00E9: { n:"BrtBeginPCDSDTCEMember" },
  25230. 0x00EA: { n:"BrtEndPCDSDTCEMember" },
  25231. 0x00EB: { n:"BrtBeginPCDSDTCQueries" },
  25232. 0x00EC: { n:"BrtEndPCDSDTCQueries" },
  25233. 0x00ED: { n:"BrtBeginPCDSDTCQuery" },
  25234. 0x00EE: { n:"BrtEndPCDSDTCQuery" },
  25235. 0x00EF: { n:"BrtBeginPCDSDTCSets" },
  25236. 0x00F0: { n:"BrtEndPCDSDTCSets" },
  25237. 0x00F1: { n:"BrtBeginPCDSDTCSet" },
  25238. 0x00F2: { n:"BrtEndPCDSDTCSet" },
  25239. 0x00F3: { n:"BrtBeginPCDCalcItems" },
  25240. 0x00F4: { n:"BrtEndPCDCalcItems" },
  25241. 0x00F5: { n:"BrtBeginPCDCalcItem" },
  25242. 0x00F6: { n:"BrtEndPCDCalcItem" },
  25243. 0x00F7: { n:"BrtBeginPRule" },
  25244. 0x00F8: { n:"BrtEndPRule" },
  25245. 0x00F9: { n:"BrtBeginPRFilters" },
  25246. 0x00FA: { n:"BrtEndPRFilters" },
  25247. 0x00FB: { n:"BrtBeginPRFilter" },
  25248. 0x00FC: { n:"BrtEndPRFilter" },
  25249. 0x00FD: { n:"BrtBeginPNames" },
  25250. 0x00FE: { n:"BrtEndPNames" },
  25251. 0x00FF: { n:"BrtBeginPName" },
  25252. 0x0100: { n:"BrtEndPName" },
  25253. 0x0101: { n:"BrtBeginPNPairs" },
  25254. 0x0102: { n:"BrtEndPNPairs" },
  25255. 0x0103: { n:"BrtBeginPNPair" },
  25256. 0x0104: { n:"BrtEndPNPair" },
  25257. 0x0105: { n:"BrtBeginECWebProps" },
  25258. 0x0106: { n:"BrtEndECWebProps" },
  25259. 0x0107: { n:"BrtBeginEcWpTables" },
  25260. 0x0108: { n:"BrtEndECWPTables" },
  25261. 0x0109: { n:"BrtBeginECParams" },
  25262. 0x010A: { n:"BrtEndECParams" },
  25263. 0x010B: { n:"BrtBeginECParam" },
  25264. 0x010C: { n:"BrtEndECParam" },
  25265. 0x010D: { n:"BrtBeginPCDKPIs" },
  25266. 0x010E: { n:"BrtEndPCDKPIs" },
  25267. 0x010F: { n:"BrtBeginPCDKPI" },
  25268. 0x0110: { n:"BrtEndPCDKPI" },
  25269. 0x0111: { n:"BrtBeginDims" },
  25270. 0x0112: { n:"BrtEndDims" },
  25271. 0x0113: { n:"BrtBeginDim" },
  25272. 0x0114: { n:"BrtEndDim" },
  25273. 0x0115: { n:"BrtIndexPartEnd" },
  25274. 0x0116: { n:"BrtBeginStyleSheet" },
  25275. 0x0117: { n:"BrtEndStyleSheet" },
  25276. 0x0118: { n:"BrtBeginSXView" },
  25277. 0x0119: { n:"BrtEndSXVI" },
  25278. 0x011A: { n:"BrtBeginSXVI" },
  25279. 0x011B: { n:"BrtBeginSXVIs" },
  25280. 0x011C: { n:"BrtEndSXVIs" },
  25281. 0x011D: { n:"BrtBeginSXVD" },
  25282. 0x011E: { n:"BrtEndSXVD" },
  25283. 0x011F: { n:"BrtBeginSXVDs" },
  25284. 0x0120: { n:"BrtEndSXVDs" },
  25285. 0x0121: { n:"BrtBeginSXPI" },
  25286. 0x0122: { n:"BrtEndSXPI" },
  25287. 0x0123: { n:"BrtBeginSXPIs" },
  25288. 0x0124: { n:"BrtEndSXPIs" },
  25289. 0x0125: { n:"BrtBeginSXDI" },
  25290. 0x0126: { n:"BrtEndSXDI" },
  25291. 0x0127: { n:"BrtBeginSXDIs" },
  25292. 0x0128: { n:"BrtEndSXDIs" },
  25293. 0x0129: { n:"BrtBeginSXLI" },
  25294. 0x012A: { n:"BrtEndSXLI" },
  25295. 0x012B: { n:"BrtBeginSXLIRws" },
  25296. 0x012C: { n:"BrtEndSXLIRws" },
  25297. 0x012D: { n:"BrtBeginSXLICols" },
  25298. 0x012E: { n:"BrtEndSXLICols" },
  25299. 0x012F: { n:"BrtBeginSXFormat" },
  25300. 0x0130: { n:"BrtEndSXFormat" },
  25301. 0x0131: { n:"BrtBeginSXFormats" },
  25302. 0x0132: { n:"BrtEndSxFormats" },
  25303. 0x0133: { n:"BrtBeginSxSelect" },
  25304. 0x0134: { n:"BrtEndSxSelect" },
  25305. 0x0135: { n:"BrtBeginISXVDRws" },
  25306. 0x0136: { n:"BrtEndISXVDRws" },
  25307. 0x0137: { n:"BrtBeginISXVDCols" },
  25308. 0x0138: { n:"BrtEndISXVDCols" },
  25309. 0x0139: { n:"BrtEndSXLocation" },
  25310. 0x013A: { n:"BrtBeginSXLocation" },
  25311. 0x013B: { n:"BrtEndSXView" },
  25312. 0x013C: { n:"BrtBeginSXTHs" },
  25313. 0x013D: { n:"BrtEndSXTHs" },
  25314. 0x013E: { n:"BrtBeginSXTH" },
  25315. 0x013F: { n:"BrtEndSXTH" },
  25316. 0x0140: { n:"BrtBeginISXTHRws" },
  25317. 0x0141: { n:"BrtEndISXTHRws" },
  25318. 0x0142: { n:"BrtBeginISXTHCols" },
  25319. 0x0143: { n:"BrtEndISXTHCols" },
  25320. 0x0144: { n:"BrtBeginSXTDMPS" },
  25321. 0x0145: { n:"BrtEndSXTDMPs" },
  25322. 0x0146: { n:"BrtBeginSXTDMP" },
  25323. 0x0147: { n:"BrtEndSXTDMP" },
  25324. 0x0148: { n:"BrtBeginSXTHItems" },
  25325. 0x0149: { n:"BrtEndSXTHItems" },
  25326. 0x014A: { n:"BrtBeginSXTHItem" },
  25327. 0x014B: { n:"BrtEndSXTHItem" },
  25328. 0x014C: { n:"BrtBeginMetadata" },
  25329. 0x014D: { n:"BrtEndMetadata" },
  25330. 0x014E: { n:"BrtBeginEsmdtinfo" },
  25331. 0x014F: { n:"BrtMdtinfo" },
  25332. 0x0150: { n:"BrtEndEsmdtinfo" },
  25333. 0x0151: { n:"BrtBeginEsmdb" },
  25334. 0x0152: { n:"BrtEndEsmdb" },
  25335. 0x0153: { n:"BrtBeginEsfmd" },
  25336. 0x0154: { n:"BrtEndEsfmd" },
  25337. 0x0155: { n:"BrtBeginSingleCells" },
  25338. 0x0156: { n:"BrtEndSingleCells" },
  25339. 0x0157: { n:"BrtBeginList" },
  25340. 0x0158: { n:"BrtEndList" },
  25341. 0x0159: { n:"BrtBeginListCols" },
  25342. 0x015A: { n:"BrtEndListCols" },
  25343. 0x015B: { n:"BrtBeginListCol" },
  25344. 0x015C: { n:"BrtEndListCol" },
  25345. 0x015D: { n:"BrtBeginListXmlCPr" },
  25346. 0x015E: { n:"BrtEndListXmlCPr" },
  25347. 0x015F: { n:"BrtListCCFmla" },
  25348. 0x0160: { n:"BrtListTrFmla" },
  25349. 0x0161: { n:"BrtBeginExternals" },
  25350. 0x0162: { n:"BrtEndExternals" },
  25351. 0x0163: { n:"BrtSupBookSrc", f:parse_RelID},
  25352. 0x0165: { n:"BrtSupSelf" },
  25353. 0x0166: { n:"BrtSupSame" },
  25354. 0x0167: { n:"BrtSupTabs" },
  25355. 0x0168: { n:"BrtBeginSupBook" },
  25356. 0x0169: { n:"BrtPlaceholderName" },
  25357. 0x016A: { n:"BrtExternSheet", f:parse_ExternSheet },
  25358. 0x016B: { n:"BrtExternTableStart" },
  25359. 0x016C: { n:"BrtExternTableEnd" },
  25360. 0x016E: { n:"BrtExternRowHdr" },
  25361. 0x016F: { n:"BrtExternCellBlank" },
  25362. 0x0170: { n:"BrtExternCellReal" },
  25363. 0x0171: { n:"BrtExternCellBool" },
  25364. 0x0172: { n:"BrtExternCellError" },
  25365. 0x0173: { n:"BrtExternCellString" },
  25366. 0x0174: { n:"BrtBeginEsmdx" },
  25367. 0x0175: { n:"BrtEndEsmdx" },
  25368. 0x0176: { n:"BrtBeginMdxSet" },
  25369. 0x0177: { n:"BrtEndMdxSet" },
  25370. 0x0178: { n:"BrtBeginMdxMbrProp" },
  25371. 0x0179: { n:"BrtEndMdxMbrProp" },
  25372. 0x017A: { n:"BrtBeginMdxKPI" },
  25373. 0x017B: { n:"BrtEndMdxKPI" },
  25374. 0x017C: { n:"BrtBeginEsstr" },
  25375. 0x017D: { n:"BrtEndEsstr" },
  25376. 0x017E: { n:"BrtBeginPRFItem" },
  25377. 0x017F: { n:"BrtEndPRFItem" },
  25378. 0x0180: { n:"BrtBeginPivotCacheIDs" },
  25379. 0x0181: { n:"BrtEndPivotCacheIDs" },
  25380. 0x0182: { n:"BrtBeginPivotCacheID" },
  25381. 0x0183: { n:"BrtEndPivotCacheID" },
  25382. 0x0184: { n:"BrtBeginISXVIs" },
  25383. 0x0185: { n:"BrtEndISXVIs" },
  25384. 0x0186: { n:"BrtBeginColInfos" },
  25385. 0x0187: { n:"BrtEndColInfos" },
  25386. 0x0188: { n:"BrtBeginRwBrk" },
  25387. 0x0189: { n:"BrtEndRwBrk" },
  25388. 0x018A: { n:"BrtBeginColBrk" },
  25389. 0x018B: { n:"BrtEndColBrk" },
  25390. 0x018C: { n:"BrtBrk" },
  25391. 0x018D: { n:"BrtUserBookView" },
  25392. 0x018E: { n:"BrtInfo" },
  25393. 0x018F: { n:"BrtCUsr" },
  25394. 0x0190: { n:"BrtUsr" },
  25395. 0x0191: { n:"BrtBeginUsers" },
  25396. 0x0193: { n:"BrtEOF" },
  25397. 0x0194: { n:"BrtUCR" },
  25398. 0x0195: { n:"BrtRRInsDel" },
  25399. 0x0196: { n:"BrtRREndInsDel" },
  25400. 0x0197: { n:"BrtRRMove" },
  25401. 0x0198: { n:"BrtRREndMove" },
  25402. 0x0199: { n:"BrtRRChgCell" },
  25403. 0x019A: { n:"BrtRREndChgCell" },
  25404. 0x019B: { n:"BrtRRHeader" },
  25405. 0x019C: { n:"BrtRRUserView" },
  25406. 0x019D: { n:"BrtRRRenSheet" },
  25407. 0x019E: { n:"BrtRRInsertSh" },
  25408. 0x019F: { n:"BrtRRDefName" },
  25409. 0x01A0: { n:"BrtRRNote" },
  25410. 0x01A1: { n:"BrtRRConflict" },
  25411. 0x01A2: { n:"BrtRRTQSIF" },
  25412. 0x01A3: { n:"BrtRRFormat" },
  25413. 0x01A4: { n:"BrtRREndFormat" },
  25414. 0x01A5: { n:"BrtRRAutoFmt" },
  25415. 0x01A6: { n:"BrtBeginUserShViews" },
  25416. 0x01A7: { n:"BrtBeginUserShView" },
  25417. 0x01A8: { n:"BrtEndUserShView" },
  25418. 0x01A9: { n:"BrtEndUserShViews" },
  25419. 0x01AA: { n:"BrtArrFmla", f:parse_BrtArrFmla },
  25420. 0x01AB: { n:"BrtShrFmla", f:parse_BrtShrFmla },
  25421. 0x01AC: { n:"BrtTable" },
  25422. 0x01AD: { n:"BrtBeginExtConnections" },
  25423. 0x01AE: { n:"BrtEndExtConnections" },
  25424. 0x01AF: { n:"BrtBeginPCDCalcMems" },
  25425. 0x01B0: { n:"BrtEndPCDCalcMems" },
  25426. 0x01B1: { n:"BrtBeginPCDCalcMem" },
  25427. 0x01B2: { n:"BrtEndPCDCalcMem" },
  25428. 0x01B3: { n:"BrtBeginPCDHGLevels" },
  25429. 0x01B4: { n:"BrtEndPCDHGLevels" },
  25430. 0x01B5: { n:"BrtBeginPCDHGLevel" },
  25431. 0x01B6: { n:"BrtEndPCDHGLevel" },
  25432. 0x01B7: { n:"BrtBeginPCDHGLGroups" },
  25433. 0x01B8: { n:"BrtEndPCDHGLGroups" },
  25434. 0x01B9: { n:"BrtBeginPCDHGLGroup" },
  25435. 0x01BA: { n:"BrtEndPCDHGLGroup" },
  25436. 0x01BB: { n:"BrtBeginPCDHGLGMembers" },
  25437. 0x01BC: { n:"BrtEndPCDHGLGMembers" },
  25438. 0x01BD: { n:"BrtBeginPCDHGLGMember" },
  25439. 0x01BE: { n:"BrtEndPCDHGLGMember" },
  25440. 0x01BF: { n:"BrtBeginQSI" },
  25441. 0x01C0: { n:"BrtEndQSI" },
  25442. 0x01C1: { n:"BrtBeginQSIR" },
  25443. 0x01C2: { n:"BrtEndQSIR" },
  25444. 0x01C3: { n:"BrtBeginDeletedNames" },
  25445. 0x01C4: { n:"BrtEndDeletedNames" },
  25446. 0x01C5: { n:"BrtBeginDeletedName" },
  25447. 0x01C6: { n:"BrtEndDeletedName" },
  25448. 0x01C7: { n:"BrtBeginQSIFs" },
  25449. 0x01C8: { n:"BrtEndQSIFs" },
  25450. 0x01C9: { n:"BrtBeginQSIF" },
  25451. 0x01CA: { n:"BrtEndQSIF" },
  25452. 0x01CB: { n:"BrtBeginAutoSortScope" },
  25453. 0x01CC: { n:"BrtEndAutoSortScope" },
  25454. 0x01CD: { n:"BrtBeginConditionalFormatting" },
  25455. 0x01CE: { n:"BrtEndConditionalFormatting" },
  25456. 0x01CF: { n:"BrtBeginCFRule" },
  25457. 0x01D0: { n:"BrtEndCFRule" },
  25458. 0x01D1: { n:"BrtBeginIconSet" },
  25459. 0x01D2: { n:"BrtEndIconSet" },
  25460. 0x01D3: { n:"BrtBeginDatabar" },
  25461. 0x01D4: { n:"BrtEndDatabar" },
  25462. 0x01D5: { n:"BrtBeginColorScale" },
  25463. 0x01D6: { n:"BrtEndColorScale" },
  25464. 0x01D7: { n:"BrtCFVO" },
  25465. 0x01D8: { n:"BrtExternValueMeta" },
  25466. 0x01D9: { n:"BrtBeginColorPalette" },
  25467. 0x01DA: { n:"BrtEndColorPalette" },
  25468. 0x01DB: { n:"BrtIndexedColor" },
  25469. 0x01DC: { n:"BrtMargins", f:parse_BrtMargins },
  25470. 0x01DD: { n:"BrtPrintOptions" },
  25471. 0x01DE: { n:"BrtPageSetup" },
  25472. 0x01DF: { n:"BrtBeginHeaderFooter" },
  25473. 0x01E0: { n:"BrtEndHeaderFooter" },
  25474. 0x01E1: { n:"BrtBeginSXCrtFormat" },
  25475. 0x01E2: { n:"BrtEndSXCrtFormat" },
  25476. 0x01E3: { n:"BrtBeginSXCrtFormats" },
  25477. 0x01E4: { n:"BrtEndSXCrtFormats" },
  25478. 0x01E5: { n:"BrtWsFmtInfo", f:parse_BrtWsFmtInfo },
  25479. 0x01E6: { n:"BrtBeginMgs" },
  25480. 0x01E7: { n:"BrtEndMGs" },
  25481. 0x01E8: { n:"BrtBeginMGMaps" },
  25482. 0x01E9: { n:"BrtEndMGMaps" },
  25483. 0x01EA: { n:"BrtBeginMG" },
  25484. 0x01EB: { n:"BrtEndMG" },
  25485. 0x01EC: { n:"BrtBeginMap" },
  25486. 0x01ED: { n:"BrtEndMap" },
  25487. 0x01EE: { n:"BrtHLink", f:parse_BrtHLink },
  25488. 0x01EF: { n:"BrtBeginDCon" },
  25489. 0x01F0: { n:"BrtEndDCon" },
  25490. 0x01F1: { n:"BrtBeginDRefs" },
  25491. 0x01F2: { n:"BrtEndDRefs" },
  25492. 0x01F3: { n:"BrtDRef" },
  25493. 0x01F4: { n:"BrtBeginScenMan" },
  25494. 0x01F5: { n:"BrtEndScenMan" },
  25495. 0x01F6: { n:"BrtBeginSct" },
  25496. 0x01F7: { n:"BrtEndSct" },
  25497. 0x01F8: { n:"BrtSlc" },
  25498. 0x01F9: { n:"BrtBeginDXFs" },
  25499. 0x01FA: { n:"BrtEndDXFs" },
  25500. 0x01FB: { n:"BrtDXF" },
  25501. 0x01FC: { n:"BrtBeginTableStyles" },
  25502. 0x01FD: { n:"BrtEndTableStyles" },
  25503. 0x01FE: { n:"BrtBeginTableStyle" },
  25504. 0x01FF: { n:"BrtEndTableStyle" },
  25505. 0x0200: { n:"BrtTableStyleElement" },
  25506. 0x0201: { n:"BrtTableStyleClient" },
  25507. 0x0202: { n:"BrtBeginVolDeps" },
  25508. 0x0203: { n:"BrtEndVolDeps" },
  25509. 0x0204: { n:"BrtBeginVolType" },
  25510. 0x0205: { n:"BrtEndVolType" },
  25511. 0x0206: { n:"BrtBeginVolMain" },
  25512. 0x0207: { n:"BrtEndVolMain" },
  25513. 0x0208: { n:"BrtBeginVolTopic" },
  25514. 0x0209: { n:"BrtEndVolTopic" },
  25515. 0x020A: { n:"BrtVolSubtopic" },
  25516. 0x020B: { n:"BrtVolRef" },
  25517. 0x020C: { n:"BrtVolNum" },
  25518. 0x020D: { n:"BrtVolErr" },
  25519. 0x020E: { n:"BrtVolStr" },
  25520. 0x020F: { n:"BrtVolBool" },
  25521. 0x0210: { n:"BrtBeginCalcChain$" },
  25522. 0x0211: { n:"BrtEndCalcChain$" },
  25523. 0x0212: { n:"BrtBeginSortState" },
  25524. 0x0213: { n:"BrtEndSortState" },
  25525. 0x0214: { n:"BrtBeginSortCond" },
  25526. 0x0215: { n:"BrtEndSortCond" },
  25527. 0x0216: { n:"BrtBookProtection" },
  25528. 0x0217: { n:"BrtSheetProtection" },
  25529. 0x0218: { n:"BrtRangeProtection" },
  25530. 0x0219: { n:"BrtPhoneticInfo" },
  25531. 0x021A: { n:"BrtBeginECTxtWiz" },
  25532. 0x021B: { n:"BrtEndECTxtWiz" },
  25533. 0x021C: { n:"BrtBeginECTWFldInfoLst" },
  25534. 0x021D: { n:"BrtEndECTWFldInfoLst" },
  25535. 0x021E: { n:"BrtBeginECTwFldInfo" },
  25536. 0x0224: { n:"BrtFileSharing" },
  25537. 0x0225: { n:"BrtOleSize" },
  25538. 0x0226: { n:"BrtDrawing", f:parse_RelID },
  25539. 0x0227: { n:"BrtLegacyDrawing" },
  25540. 0x0228: { n:"BrtLegacyDrawingHF" },
  25541. 0x0229: { n:"BrtWebOpt" },
  25542. 0x022A: { n:"BrtBeginWebPubItems" },
  25543. 0x022B: { n:"BrtEndWebPubItems" },
  25544. 0x022C: { n:"BrtBeginWebPubItem" },
  25545. 0x022D: { n:"BrtEndWebPubItem" },
  25546. 0x022E: { n:"BrtBeginSXCondFmt" },
  25547. 0x022F: { n:"BrtEndSXCondFmt" },
  25548. 0x0230: { n:"BrtBeginSXCondFmts" },
  25549. 0x0231: { n:"BrtEndSXCondFmts" },
  25550. 0x0232: { n:"BrtBkHim" },
  25551. 0x0234: { n:"BrtColor" },
  25552. 0x0235: { n:"BrtBeginIndexedColors" },
  25553. 0x0236: { n:"BrtEndIndexedColors" },
  25554. 0x0239: { n:"BrtBeginMRUColors" },
  25555. 0x023A: { n:"BrtEndMRUColors" },
  25556. 0x023C: { n:"BrtMRUColor" },
  25557. 0x023D: { n:"BrtBeginDVals" },
  25558. 0x023E: { n:"BrtEndDVals" },
  25559. 0x0241: { n:"BrtSupNameStart" },
  25560. 0x0242: { n:"BrtSupNameValueStart" },
  25561. 0x0243: { n:"BrtSupNameValueEnd" },
  25562. 0x0244: { n:"BrtSupNameNum" },
  25563. 0x0245: { n:"BrtSupNameErr" },
  25564. 0x0246: { n:"BrtSupNameSt" },
  25565. 0x0247: { n:"BrtSupNameNil" },
  25566. 0x0248: { n:"BrtSupNameBool" },
  25567. 0x0249: { n:"BrtSupNameFmla" },
  25568. 0x024A: { n:"BrtSupNameBits" },
  25569. 0x024B: { n:"BrtSupNameEnd" },
  25570. 0x024C: { n:"BrtEndSupBook" },
  25571. 0x024D: { n:"BrtCellSmartTagProperty" },
  25572. 0x024E: { n:"BrtBeginCellSmartTag" },
  25573. 0x024F: { n:"BrtEndCellSmartTag" },
  25574. 0x0250: { n:"BrtBeginCellSmartTags" },
  25575. 0x0251: { n:"BrtEndCellSmartTags" },
  25576. 0x0252: { n:"BrtBeginSmartTags" },
  25577. 0x0253: { n:"BrtEndSmartTags" },
  25578. 0x0254: { n:"BrtSmartTagType" },
  25579. 0x0255: { n:"BrtBeginSmartTagTypes" },
  25580. 0x0256: { n:"BrtEndSmartTagTypes" },
  25581. 0x0257: { n:"BrtBeginSXFilters" },
  25582. 0x0258: { n:"BrtEndSXFilters" },
  25583. 0x0259: { n:"BrtBeginSXFILTER" },
  25584. 0x025A: { n:"BrtEndSXFilter" },
  25585. 0x025B: { n:"BrtBeginFills" },
  25586. 0x025C: { n:"BrtEndFills" },
  25587. 0x025D: { n:"BrtBeginCellWatches" },
  25588. 0x025E: { n:"BrtEndCellWatches" },
  25589. 0x025F: { n:"BrtCellWatch" },
  25590. 0x0260: { n:"BrtBeginCRErrs" },
  25591. 0x0261: { n:"BrtEndCRErrs" },
  25592. 0x0262: { n:"BrtCrashRecErr" },
  25593. 0x0263: { n:"BrtBeginFonts" },
  25594. 0x0264: { n:"BrtEndFonts" },
  25595. 0x0265: { n:"BrtBeginBorders" },
  25596. 0x0266: { n:"BrtEndBorders" },
  25597. 0x0267: { n:"BrtBeginFmts" },
  25598. 0x0268: { n:"BrtEndFmts" },
  25599. 0x0269: { n:"BrtBeginCellXFs" },
  25600. 0x026A: { n:"BrtEndCellXFs" },
  25601. 0x026B: { n:"BrtBeginStyles" },
  25602. 0x026C: { n:"BrtEndStyles" },
  25603. 0x0271: { n:"BrtBigName" },
  25604. 0x0272: { n:"BrtBeginCellStyleXFs" },
  25605. 0x0273: { n:"BrtEndCellStyleXFs" },
  25606. 0x0274: { n:"BrtBeginComments" },
  25607. 0x0275: { n:"BrtEndComments" },
  25608. 0x0276: { n:"BrtBeginCommentAuthors" },
  25609. 0x0277: { n:"BrtEndCommentAuthors" },
  25610. 0x0278: { n:"BrtCommentAuthor", f:parse_BrtCommentAuthor },
  25611. 0x0279: { n:"BrtBeginCommentList" },
  25612. 0x027A: { n:"BrtEndCommentList" },
  25613. 0x027B: { n:"BrtBeginComment", f:parse_BrtBeginComment},
  25614. 0x027C: { n:"BrtEndComment" },
  25615. 0x027D: { n:"BrtCommentText", f:parse_BrtCommentText },
  25616. 0x027E: { n:"BrtBeginOleObjects" },
  25617. 0x027F: { n:"BrtOleObject" },
  25618. 0x0280: { n:"BrtEndOleObjects" },
  25619. 0x0281: { n:"BrtBeginSxrules" },
  25620. 0x0282: { n:"BrtEndSxRules" },
  25621. 0x0283: { n:"BrtBeginActiveXControls" },
  25622. 0x0284: { n:"BrtActiveX" },
  25623. 0x0285: { n:"BrtEndActiveXControls" },
  25624. 0x0286: { n:"BrtBeginPCDSDTCEMembersSortBy" },
  25625. 0x0288: { n:"BrtBeginCellIgnoreECs" },
  25626. 0x0289: { n:"BrtCellIgnoreEC" },
  25627. 0x028A: { n:"BrtEndCellIgnoreECs" },
  25628. 0x028B: { n:"BrtCsProp", f:parse_BrtCsProp },
  25629. 0x028C: { n:"BrtCsPageSetup" },
  25630. 0x028D: { n:"BrtBeginUserCsViews" },
  25631. 0x028E: { n:"BrtEndUserCsViews" },
  25632. 0x028F: { n:"BrtBeginUserCsView" },
  25633. 0x0290: { n:"BrtEndUserCsView" },
  25634. 0x0291: { n:"BrtBeginPcdSFCIEntries" },
  25635. 0x0292: { n:"BrtEndPCDSFCIEntries" },
  25636. 0x0293: { n:"BrtPCDSFCIEntry" },
  25637. 0x0294: { n:"BrtBeginListParts" },
  25638. 0x0295: { n:"BrtListPart" },
  25639. 0x0296: { n:"BrtEndListParts" },
  25640. 0x0297: { n:"BrtSheetCalcProp" },
  25641. 0x0298: { n:"BrtBeginFnGroup" },
  25642. 0x0299: { n:"BrtFnGroup" },
  25643. 0x029A: { n:"BrtEndFnGroup" },
  25644. 0x029B: { n:"BrtSupAddin" },
  25645. 0x029C: { n:"BrtSXTDMPOrder" },
  25646. 0x029D: { n:"BrtCsProtection" },
  25647. 0x029F: { n:"BrtBeginWsSortMap" },
  25648. 0x02A0: { n:"BrtEndWsSortMap" },
  25649. 0x02A1: { n:"BrtBeginRRSort" },
  25650. 0x02A2: { n:"BrtEndRRSort" },
  25651. 0x02A3: { n:"BrtRRSortItem" },
  25652. 0x02A4: { n:"BrtFileSharingIso" },
  25653. 0x02A5: { n:"BrtBookProtectionIso" },
  25654. 0x02A6: { n:"BrtSheetProtectionIso" },
  25655. 0x02A7: { n:"BrtCsProtectionIso" },
  25656. 0x02A8: { n:"BrtRangeProtectionIso" },
  25657. 0x0400: { n:"BrtRwDescent" },
  25658. 0x0401: { n:"BrtKnownFonts" },
  25659. 0x0402: { n:"BrtBeginSXTupleSet" },
  25660. 0x0403: { n:"BrtEndSXTupleSet" },
  25661. 0x0404: { n:"BrtBeginSXTupleSetHeader" },
  25662. 0x0405: { n:"BrtEndSXTupleSetHeader" },
  25663. 0x0406: { n:"BrtSXTupleSetHeaderItem" },
  25664. 0x0407: { n:"BrtBeginSXTupleSetData" },
  25665. 0x0408: { n:"BrtEndSXTupleSetData" },
  25666. 0x0409: { n:"BrtBeginSXTupleSetRow" },
  25667. 0x040A: { n:"BrtEndSXTupleSetRow" },
  25668. 0x040B: { n:"BrtSXTupleSetRowItem" },
  25669. 0x040C: { n:"BrtNameExt" },
  25670. 0x040D: { n:"BrtPCDH14" },
  25671. 0x040E: { n:"BrtBeginPCDCalcMem14" },
  25672. 0x040F: { n:"BrtEndPCDCalcMem14" },
  25673. 0x0410: { n:"BrtSXTH14" },
  25674. 0x0411: { n:"BrtBeginSparklineGroup" },
  25675. 0x0412: { n:"BrtEndSparklineGroup" },
  25676. 0x0413: { n:"BrtSparkline" },
  25677. 0x0414: { n:"BrtSXDI14" },
  25678. 0x0415: { n:"BrtWsFmtInfoEx14" },
  25679. 0x0416: { n:"BrtBeginConditionalFormatting14" },
  25680. 0x0417: { n:"BrtEndConditionalFormatting14" },
  25681. 0x0418: { n:"BrtBeginCFRule14" },
  25682. 0x0419: { n:"BrtEndCFRule14" },
  25683. 0x041A: { n:"BrtCFVO14" },
  25684. 0x041B: { n:"BrtBeginDatabar14" },
  25685. 0x041C: { n:"BrtBeginIconSet14" },
  25686. 0x041D: { n:"BrtDVal14" },
  25687. 0x041E: { n:"BrtBeginDVals14" },
  25688. 0x041F: { n:"BrtColor14" },
  25689. 0x0420: { n:"BrtBeginSparklines" },
  25690. 0x0421: { n:"BrtEndSparklines" },
  25691. 0x0422: { n:"BrtBeginSparklineGroups" },
  25692. 0x0423: { n:"BrtEndSparklineGroups" },
  25693. 0x0425: { n:"BrtSXVD14" },
  25694. 0x0426: { n:"BrtBeginSXView14" },
  25695. 0x0427: { n:"BrtEndSXView14" },
  25696. 0x0428: { n:"BrtBeginSXView16" },
  25697. 0x0429: { n:"BrtEndSXView16" },
  25698. 0x042A: { n:"BrtBeginPCD14" },
  25699. 0x042B: { n:"BrtEndPCD14" },
  25700. 0x042C: { n:"BrtBeginExtConn14" },
  25701. 0x042D: { n:"BrtEndExtConn14" },
  25702. 0x042E: { n:"BrtBeginSlicerCacheIDs" },
  25703. 0x042F: { n:"BrtEndSlicerCacheIDs" },
  25704. 0x0430: { n:"BrtBeginSlicerCacheID" },
  25705. 0x0431: { n:"BrtEndSlicerCacheID" },
  25706. 0x0433: { n:"BrtBeginSlicerCache" },
  25707. 0x0434: { n:"BrtEndSlicerCache" },
  25708. 0x0435: { n:"BrtBeginSlicerCacheDef" },
  25709. 0x0436: { n:"BrtEndSlicerCacheDef" },
  25710. 0x0437: { n:"BrtBeginSlicersEx" },
  25711. 0x0438: { n:"BrtEndSlicersEx" },
  25712. 0x0439: { n:"BrtBeginSlicerEx" },
  25713. 0x043A: { n:"BrtEndSlicerEx" },
  25714. 0x043B: { n:"BrtBeginSlicer" },
  25715. 0x043C: { n:"BrtEndSlicer" },
  25716. 0x043D: { n:"BrtSlicerCachePivotTables" },
  25717. 0x043E: { n:"BrtBeginSlicerCacheOlapImpl" },
  25718. 0x043F: { n:"BrtEndSlicerCacheOlapImpl" },
  25719. 0x0440: { n:"BrtBeginSlicerCacheLevelsData" },
  25720. 0x0441: { n:"BrtEndSlicerCacheLevelsData" },
  25721. 0x0442: { n:"BrtBeginSlicerCacheLevelData" },
  25722. 0x0443: { n:"BrtEndSlicerCacheLevelData" },
  25723. 0x0444: { n:"BrtBeginSlicerCacheSiRanges" },
  25724. 0x0445: { n:"BrtEndSlicerCacheSiRanges" },
  25725. 0x0446: { n:"BrtBeginSlicerCacheSiRange" },
  25726. 0x0447: { n:"BrtEndSlicerCacheSiRange" },
  25727. 0x0448: { n:"BrtSlicerCacheOlapItem" },
  25728. 0x0449: { n:"BrtBeginSlicerCacheSelections" },
  25729. 0x044A: { n:"BrtSlicerCacheSelection" },
  25730. 0x044B: { n:"BrtEndSlicerCacheSelections" },
  25731. 0x044C: { n:"BrtBeginSlicerCacheNative" },
  25732. 0x044D: { n:"BrtEndSlicerCacheNative" },
  25733. 0x044E: { n:"BrtSlicerCacheNativeItem" },
  25734. 0x044F: { n:"BrtRangeProtection14" },
  25735. 0x0450: { n:"BrtRangeProtectionIso14" },
  25736. 0x0451: { n:"BrtCellIgnoreEC14" },
  25737. 0x0457: { n:"BrtList14" },
  25738. 0x0458: { n:"BrtCFIcon" },
  25739. 0x0459: { n:"BrtBeginSlicerCachesPivotCacheIDs" },
  25740. 0x045A: { n:"BrtEndSlicerCachesPivotCacheIDs" },
  25741. 0x045B: { n:"BrtBeginSlicers" },
  25742. 0x045C: { n:"BrtEndSlicers" },
  25743. 0x045D: { n:"BrtWbProp14" },
  25744. 0x045E: { n:"BrtBeginSXEdit" },
  25745. 0x045F: { n:"BrtEndSXEdit" },
  25746. 0x0460: { n:"BrtBeginSXEdits" },
  25747. 0x0461: { n:"BrtEndSXEdits" },
  25748. 0x0462: { n:"BrtBeginSXChange" },
  25749. 0x0463: { n:"BrtEndSXChange" },
  25750. 0x0464: { n:"BrtBeginSXChanges" },
  25751. 0x0465: { n:"BrtEndSXChanges" },
  25752. 0x0466: { n:"BrtSXTupleItems" },
  25753. 0x0468: { n:"BrtBeginSlicerStyle" },
  25754. 0x0469: { n:"BrtEndSlicerStyle" },
  25755. 0x046A: { n:"BrtSlicerStyleElement" },
  25756. 0x046B: { n:"BrtBeginStyleSheetExt14" },
  25757. 0x046C: { n:"BrtEndStyleSheetExt14" },
  25758. 0x046D: { n:"BrtBeginSlicerCachesPivotCacheID" },
  25759. 0x046E: { n:"BrtEndSlicerCachesPivotCacheID" },
  25760. 0x046F: { n:"BrtBeginConditionalFormattings" },
  25761. 0x0470: { n:"BrtEndConditionalFormattings" },
  25762. 0x0471: { n:"BrtBeginPCDCalcMemExt" },
  25763. 0x0472: { n:"BrtEndPCDCalcMemExt" },
  25764. 0x0473: { n:"BrtBeginPCDCalcMemsExt" },
  25765. 0x0474: { n:"BrtEndPCDCalcMemsExt" },
  25766. 0x0475: { n:"BrtPCDField14" },
  25767. 0x0476: { n:"BrtBeginSlicerStyles" },
  25768. 0x0477: { n:"BrtEndSlicerStyles" },
  25769. 0x0478: { n:"BrtBeginSlicerStyleElements" },
  25770. 0x0479: { n:"BrtEndSlicerStyleElements" },
  25771. 0x047A: { n:"BrtCFRuleExt" },
  25772. 0x047B: { n:"BrtBeginSXCondFmt14" },
  25773. 0x047C: { n:"BrtEndSXCondFmt14" },
  25774. 0x047D: { n:"BrtBeginSXCondFmts14" },
  25775. 0x047E: { n:"BrtEndSXCondFmts14" },
  25776. 0x0480: { n:"BrtBeginSortCond14" },
  25777. 0x0481: { n:"BrtEndSortCond14" },
  25778. 0x0482: { n:"BrtEndDVals14" },
  25779. 0x0483: { n:"BrtEndIconSet14" },
  25780. 0x0484: { n:"BrtEndDatabar14" },
  25781. 0x0485: { n:"BrtBeginColorScale14" },
  25782. 0x0486: { n:"BrtEndColorScale14" },
  25783. 0x0487: { n:"BrtBeginSxrules14" },
  25784. 0x0488: { n:"BrtEndSxrules14" },
  25785. 0x0489: { n:"BrtBeginPRule14" },
  25786. 0x048A: { n:"BrtEndPRule14" },
  25787. 0x048B: { n:"BrtBeginPRFilters14" },
  25788. 0x048C: { n:"BrtEndPRFilters14" },
  25789. 0x048D: { n:"BrtBeginPRFilter14" },
  25790. 0x048E: { n:"BrtEndPRFilter14" },
  25791. 0x048F: { n:"BrtBeginPRFItem14" },
  25792. 0x0490: { n:"BrtEndPRFItem14" },
  25793. 0x0491: { n:"BrtBeginCellIgnoreECs14" },
  25794. 0x0492: { n:"BrtEndCellIgnoreECs14" },
  25795. 0x0493: { n:"BrtDxf14" },
  25796. 0x0494: { n:"BrtBeginDxF14s" },
  25797. 0x0495: { n:"BrtEndDxf14s" },
  25798. 0x0499: { n:"BrtFilter14" },
  25799. 0x049A: { n:"BrtBeginCustomFilters14" },
  25800. 0x049C: { n:"BrtCustomFilter14" },
  25801. 0x049D: { n:"BrtIconFilter14" },
  25802. 0x049E: { n:"BrtPivotCacheConnectionName" },
  25803. 0x0800: { n:"BrtBeginDecoupledPivotCacheIDs" },
  25804. 0x0801: { n:"BrtEndDecoupledPivotCacheIDs" },
  25805. 0x0802: { n:"BrtDecoupledPivotCacheID" },
  25806. 0x0803: { n:"BrtBeginPivotTableRefs" },
  25807. 0x0804: { n:"BrtEndPivotTableRefs" },
  25808. 0x0805: { n:"BrtPivotTableRef" },
  25809. 0x0806: { n:"BrtSlicerCacheBookPivotTables" },
  25810. 0x0807: { n:"BrtBeginSxvcells" },
  25811. 0x0808: { n:"BrtEndSxvcells" },
  25812. 0x0809: { n:"BrtBeginSxRow" },
  25813. 0x080A: { n:"BrtEndSxRow" },
  25814. 0x080C: { n:"BrtPcdCalcMem15" },
  25815. 0x0813: { n:"BrtQsi15" },
  25816. 0x0814: { n:"BrtBeginWebExtensions" },
  25817. 0x0815: { n:"BrtEndWebExtensions" },
  25818. 0x0816: { n:"BrtWebExtension" },
  25819. 0x0817: { n:"BrtAbsPath15" },
  25820. 0x0818: { n:"BrtBeginPivotTableUISettings" },
  25821. 0x0819: { n:"BrtEndPivotTableUISettings" },
  25822. 0x081B: { n:"BrtTableSlicerCacheIDs" },
  25823. 0x081C: { n:"BrtTableSlicerCacheID" },
  25824. 0x081D: { n:"BrtBeginTableSlicerCache" },
  25825. 0x081E: { n:"BrtEndTableSlicerCache" },
  25826. 0x081F: { n:"BrtSxFilter15" },
  25827. 0x0820: { n:"BrtBeginTimelineCachePivotCacheIDs" },
  25828. 0x0821: { n:"BrtEndTimelineCachePivotCacheIDs" },
  25829. 0x0822: { n:"BrtTimelineCachePivotCacheID" },
  25830. 0x0823: { n:"BrtBeginTimelineCacheIDs" },
  25831. 0x0824: { n:"BrtEndTimelineCacheIDs" },
  25832. 0x0825: { n:"BrtBeginTimelineCacheID" },
  25833. 0x0826: { n:"BrtEndTimelineCacheID" },
  25834. 0x0827: { n:"BrtBeginTimelinesEx" },
  25835. 0x0828: { n:"BrtEndTimelinesEx" },
  25836. 0x0829: { n:"BrtBeginTimelineEx" },
  25837. 0x082A: { n:"BrtEndTimelineEx" },
  25838. 0x082B: { n:"BrtWorkBookPr15" },
  25839. 0x082C: { n:"BrtPCDH15" },
  25840. 0x082D: { n:"BrtBeginTimelineStyle" },
  25841. 0x082E: { n:"BrtEndTimelineStyle" },
  25842. 0x082F: { n:"BrtTimelineStyleElement" },
  25843. 0x0830: { n:"BrtBeginTimelineStylesheetExt15" },
  25844. 0x0831: { n:"BrtEndTimelineStylesheetExt15" },
  25845. 0x0832: { n:"BrtBeginTimelineStyles" },
  25846. 0x0833: { n:"BrtEndTimelineStyles" },
  25847. 0x0834: { n:"BrtBeginTimelineStyleElements" },
  25848. 0x0835: { n:"BrtEndTimelineStyleElements" },
  25849. 0x0836: { n:"BrtDxf15" },
  25850. 0x0837: { n:"BrtBeginDxfs15" },
  25851. 0x0838: { n:"brtEndDxfs15" },
  25852. 0x0839: { n:"BrtSlicerCacheHideItemsWithNoData" },
  25853. 0x083A: { n:"BrtBeginItemUniqueNames" },
  25854. 0x083B: { n:"BrtEndItemUniqueNames" },
  25855. 0x083C: { n:"BrtItemUniqueName" },
  25856. 0x083D: { n:"BrtBeginExtConn15" },
  25857. 0x083E: { n:"BrtEndExtConn15" },
  25858. 0x083F: { n:"BrtBeginOledbPr15" },
  25859. 0x0840: { n:"BrtEndOledbPr15" },
  25860. 0x0841: { n:"BrtBeginDataFeedPr15" },
  25861. 0x0842: { n:"BrtEndDataFeedPr15" },
  25862. 0x0843: { n:"BrtTextPr15" },
  25863. 0x0844: { n:"BrtRangePr15" },
  25864. 0x0845: { n:"BrtDbCommand15" },
  25865. 0x0846: { n:"BrtBeginDbTables15" },
  25866. 0x0847: { n:"BrtEndDbTables15" },
  25867. 0x0848: { n:"BrtDbTable15" },
  25868. 0x0849: { n:"BrtBeginDataModel" },
  25869. 0x084A: { n:"BrtEndDataModel" },
  25870. 0x084B: { n:"BrtBeginModelTables" },
  25871. 0x084C: { n:"BrtEndModelTables" },
  25872. 0x084D: { n:"BrtModelTable" },
  25873. 0x084E: { n:"BrtBeginModelRelationships" },
  25874. 0x084F: { n:"BrtEndModelRelationships" },
  25875. 0x0850: { n:"BrtModelRelationship" },
  25876. 0x0851: { n:"BrtBeginECTxtWiz15" },
  25877. 0x0852: { n:"BrtEndECTxtWiz15" },
  25878. 0x0853: { n:"BrtBeginECTWFldInfoLst15" },
  25879. 0x0854: { n:"BrtEndECTWFldInfoLst15" },
  25880. 0x0855: { n:"BrtBeginECTWFldInfo15" },
  25881. 0x0856: { n:"BrtFieldListActiveItem" },
  25882. 0x0857: { n:"BrtPivotCacheIdVersion" },
  25883. 0x0858: { n:"BrtSXDI15" },
  25884. 0x0859: { n:"BrtBeginModelTimeGroupings" },
  25885. 0x085A: { n:"BrtEndModelTimeGroupings" },
  25886. 0x085B: { n:"BrtBeginModelTimeGrouping" },
  25887. 0x085C: { n:"BrtEndModelTimeGrouping" },
  25888. 0x085D: { n:"BrtModelTimeGroupingCalcCol" },
  25889. 0x0C00: { n:"BrtUid" },
  25890. 0x0C01: { n:"BrtRevisionPtr" },
  25891. 0x13e7: { n:"BrtBeginCalcFeatures" },
  25892. 0x13e8: { n:"BrtEndCalcFeatures" },
  25893. 0x13e9: { n:"BrtCalcFeature" },
  25894. 0xFFFF: { n:"" }
  25895. };
  25896. var XLSBRE = evert_key(XLSBRecordEnum, 'n');
  25897. /* [MS-XLS] 2.3 Record Enumeration */
  25898. var XLSRecordEnum = {
  25899. 0x0003: { n:"BIFF2NUM", f:parse_BIFF2NUM },
  25900. 0x0004: { n:"BIFF2STR", f:parse_BIFF2STR },
  25901. 0x0006: { n:"Formula", f:parse_Formula },
  25902. 0x0009: { n:'BOF', f:parse_BOF },
  25903. 0x000a: { n:'EOF', f:parsenoop2 },
  25904. 0x000c: { n:"CalcCount", f:parseuint16 },
  25905. 0x000d: { n:"CalcMode", f:parseuint16 },
  25906. 0x000e: { n:"CalcPrecision", f:parsebool },
  25907. 0x000f: { n:"CalcRefMode", f:parsebool },
  25908. 0x0010: { n:"CalcDelta", f:parse_Xnum },
  25909. 0x0011: { n:"CalcIter", f:parsebool },
  25910. 0x0012: { n:"Protect", f:parsebool },
  25911. 0x0013: { n:"Password", f:parseuint16 },
  25912. 0x0014: { n:"Header", f:parse_XLHeaderFooter },
  25913. 0x0015: { n:"Footer", f:parse_XLHeaderFooter },
  25914. 0x0017: { n:"ExternSheet", f:parse_ExternSheet },
  25915. 0x0018: { n:"Lbl", f:parse_Lbl },
  25916. 0x0019: { n:"WinProtect", f:parsebool },
  25917. 0x001a: { n:"VerticalPageBreaks" },
  25918. 0x001b: { n:"HorizontalPageBreaks" },
  25919. 0x001c: { n:"Note", f:parse_Note },
  25920. 0x001d: { n:"Selection" },
  25921. 0x0022: { n:"Date1904", f:parsebool },
  25922. 0x0023: { n:"ExternName", f:parse_ExternName },
  25923. 0x0026: { n:"LeftMargin", f:parse_Xnum },
  25924. 0x0027: { n:"RightMargin", f:parse_Xnum },
  25925. 0x0028: { n:"TopMargin", f:parse_Xnum },
  25926. 0x0029: { n:"BottomMargin", f:parse_Xnum },
  25927. 0x002a: { n:"PrintRowCol", f:parsebool },
  25928. 0x002b: { n:"PrintGrid", f:parsebool },
  25929. 0x002f: { n:"FilePass", f:parse_FilePass },
  25930. 0x0031: { n:"Font", f:parse_Font },
  25931. 0x0033: { n:"PrintSize", f:parseuint16 },
  25932. 0x003c: { n:"Continue" },
  25933. 0x003d: { n:"Window1", f:parse_Window1 },
  25934. 0x0040: { n:"Backup", f:parsebool },
  25935. 0x0041: { n:"Pane" },
  25936. 0x0042: { n:'CodePage', f:parseuint16 },
  25937. 0x004d: { n:"Pls" },
  25938. 0x0050: { n:"DCon" },
  25939. 0x0051: { n:"DConRef" },
  25940. 0x0052: { n:"DConName" },
  25941. 0x0055: { n:"DefColWidth", f:parseuint16 },
  25942. 0x0059: { n:"XCT" },
  25943. 0x005a: { n:"CRN" },
  25944. 0x005b: { n:"FileSharing" },
  25945. 0x005c: { n:'WriteAccess', f:parse_WriteAccess },
  25946. 0x005d: { n:"Obj", f:parse_Obj },
  25947. 0x005e: { n:"Uncalced" },
  25948. 0x005f: { n:"CalcSaveRecalc", f:parsebool },
  25949. 0x0060: { n:"Template" },
  25950. 0x0061: { n:"Intl" },
  25951. 0x0063: { n:"ObjProtect", f:parsebool },
  25952. 0x007d: { n:"ColInfo", f:parse_ColInfo },
  25953. 0x0080: { n:"Guts", f:parse_Guts },
  25954. 0x0081: { n:"WsBool", f:parse_WsBool },
  25955. 0x0082: { n:"GridSet", f:parseuint16 },
  25956. 0x0083: { n:"HCenter", f:parsebool },
  25957. 0x0084: { n:"VCenter", f:parsebool },
  25958. 0x0085: { n:'BoundSheet8', f:parse_BoundSheet8 },
  25959. 0x0086: { n:"WriteProtect" },
  25960. 0x008c: { n:"Country", f:parse_Country },
  25961. 0x008d: { n:"HideObj", f:parseuint16 },
  25962. 0x0090: { n:"Sort" },
  25963. 0x0092: { n:"Palette", f:parse_Palette },
  25964. 0x0097: { n:"Sync" },
  25965. 0x0098: { n:"LPr" },
  25966. 0x0099: { n:"DxGCol" },
  25967. 0x009a: { n:"FnGroupName" },
  25968. 0x009b: { n:"FilterMode" },
  25969. 0x009c: { n:"BuiltInFnGroupCount", f:parseuint16 },
  25970. 0x009d: { n:"AutoFilterInfo" },
  25971. 0x009e: { n:"AutoFilter" },
  25972. 0x00a0: { n:"Scl", f:parse_Scl },
  25973. 0x00a1: { n:"Setup", f:parse_Setup },
  25974. 0x00ae: { n:"ScenMan" },
  25975. 0x00af: { n:"SCENARIO" },
  25976. 0x00b0: { n:"SxView" },
  25977. 0x00b1: { n:"Sxvd" },
  25978. 0x00b2: { n:"SXVI" },
  25979. 0x00b4: { n:"SxIvd" },
  25980. 0x00b5: { n:"SXLI" },
  25981. 0x00b6: { n:"SXPI" },
  25982. 0x00b8: { n:"DocRoute" },
  25983. 0x00b9: { n:"RecipName" },
  25984. 0x00bd: { n:"MulRk", f:parse_MulRk },
  25985. 0x00be: { n:"MulBlank", f:parse_MulBlank },
  25986. 0x00c1: { n:'Mms', f:parsenoop2 },
  25987. 0x00c5: { n:"SXDI" },
  25988. 0x00c6: { n:"SXDB" },
  25989. 0x00c7: { n:"SXFDB" },
  25990. 0x00c8: { n:"SXDBB" },
  25991. 0x00c9: { n:"SXNum" },
  25992. 0x00ca: { n:"SxBool", f:parsebool },
  25993. 0x00cb: { n:"SxErr" },
  25994. 0x00cc: { n:"SXInt" },
  25995. 0x00cd: { n:"SXString" },
  25996. 0x00ce: { n:"SXDtr" },
  25997. 0x00cf: { n:"SxNil" },
  25998. 0x00d0: { n:"SXTbl" },
  25999. 0x00d1: { n:"SXTBRGIITM" },
  26000. 0x00d2: { n:"SxTbpg" },
  26001. 0x00d3: { n:"ObProj" },
  26002. 0x00d5: { n:"SXStreamID" },
  26003. 0x00d7: { n:"DBCell" },
  26004. 0x00d8: { n:"SXRng" },
  26005. 0x00d9: { n:"SxIsxoper" },
  26006. 0x00da: { n:"BookBool", f:parseuint16 },
  26007. 0x00dc: { n:"DbOrParamQry" },
  26008. 0x00dd: { n:"ScenarioProtect", f:parsebool },
  26009. 0x00de: { n:"OleObjectSize" },
  26010. 0x00e0: { n:"XF", f:parse_XF },
  26011. 0x00e1: { n:'InterfaceHdr', f:parse_InterfaceHdr },
  26012. 0x00e2: { n:'InterfaceEnd', f:parsenoop2 },
  26013. 0x00e3: { n:"SXVS" },
  26014. 0x00e5: { n:"MergeCells", f:parse_MergeCells },
  26015. 0x00e9: { n:"BkHim" },
  26016. 0x00eb: { n:"MsoDrawingGroup" },
  26017. 0x00ec: { n:"MsoDrawing" },
  26018. 0x00ed: { n:"MsoDrawingSelection" },
  26019. 0x00ef: { n:"PhoneticInfo" },
  26020. 0x00f0: { n:"SxRule" },
  26021. 0x00f1: { n:"SXEx" },
  26022. 0x00f2: { n:"SxFilt" },
  26023. 0x00f4: { n:"SxDXF" },
  26024. 0x00f5: { n:"SxItm" },
  26025. 0x00f6: { n:"SxName" },
  26026. 0x00f7: { n:"SxSelect" },
  26027. 0x00f8: { n:"SXPair" },
  26028. 0x00f9: { n:"SxFmla" },
  26029. 0x00fb: { n:"SxFormat" },
  26030. 0x00fc: { n:"SST", f:parse_SST },
  26031. 0x00fd: { n:"LabelSst", f:parse_LabelSst },
  26032. 0x00ff: { n:"ExtSST", f:parse_ExtSST },
  26033. 0x0100: { n:"SXVDEx" },
  26034. 0x0103: { n:"SXFormula" },
  26035. 0x0122: { n:"SXDBEx" },
  26036. 0x0137: { n:"RRDInsDel" },
  26037. 0x0138: { n:"RRDHead" },
  26038. 0x013b: { n:"RRDChgCell" },
  26039. 0x013d: { n:"RRTabId", f:parseuint16a },
  26040. 0x013e: { n:"RRDRenSheet" },
  26041. 0x013f: { n:"RRSort" },
  26042. 0x0140: { n:"RRDMove" },
  26043. 0x014a: { n:"RRFormat" },
  26044. 0x014b: { n:"RRAutoFmt" },
  26045. 0x014d: { n:"RRInsertSh" },
  26046. 0x014e: { n:"RRDMoveBegin" },
  26047. 0x014f: { n:"RRDMoveEnd" },
  26048. 0x0150: { n:"RRDInsDelBegin" },
  26049. 0x0151: { n:"RRDInsDelEnd" },
  26050. 0x0152: { n:"RRDConflict" },
  26051. 0x0153: { n:"RRDDefName" },
  26052. 0x0154: { n:"RRDRstEtxp" },
  26053. 0x015f: { n:"LRng" },
  26054. 0x0160: { n:"UsesELFs", f:parsebool },
  26055. 0x0161: { n:"DSF", f:parsenoop2 },
  26056. 0x0191: { n:"CUsr" },
  26057. 0x0192: { n:"CbUsr" },
  26058. 0x0193: { n:"UsrInfo" },
  26059. 0x0194: { n:"UsrExcl" },
  26060. 0x0195: { n:"FileLock" },
  26061. 0x0196: { n:"RRDInfo" },
  26062. 0x0197: { n:"BCUsrs" },
  26063. 0x0198: { n:"UsrChk" },
  26064. 0x01a9: { n:"UserBView" },
  26065. 0x01aa: { n:"UserSViewBegin" },
  26066. 0x01ab: { n:"UserSViewEnd" },
  26067. 0x01ac: { n:"RRDUserView" },
  26068. 0x01ad: { n:"Qsi" },
  26069. 0x01ae: { n:"SupBook", f:parse_SupBook },
  26070. 0x01af: { n:"Prot4Rev", f:parsebool },
  26071. 0x01b0: { n:"CondFmt" },
  26072. 0x01b1: { n:"CF" },
  26073. 0x01b2: { n:"DVal" },
  26074. 0x01b5: { n:"DConBin" },
  26075. 0x01b6: { n:"TxO", f:parse_TxO },
  26076. 0x01b7: { n:"RefreshAll", f:parsebool },
  26077. 0x01b8: { n:"HLink", f:parse_HLink },
  26078. 0x01b9: { n:"Lel" },
  26079. 0x01ba: { n:"CodeName", f:parse_XLUnicodeString },
  26080. 0x01bb: { n:"SXFDBType" },
  26081. 0x01bc: { n:"Prot4RevPass", f:parseuint16 },
  26082. 0x01bd: { n:"ObNoMacros" },
  26083. 0x01be: { n:"Dv" },
  26084. 0x01c0: { n:"Excel9File", f:parsenoop2 },
  26085. 0x01c1: { n:"RecalcId", f:parse_RecalcId, r:2},
  26086. 0x01c2: { n:"EntExU2", f:parsenoop2 },
  26087. 0x0200: { n:"Dimensions", f:parse_Dimensions },
  26088. 0x0201: { n:"Blank", f:parse_Blank },
  26089. 0x0203: { n:"Number", f:parse_Number },
  26090. 0x0204: { n:"Label", f:parse_Label },
  26091. 0x0205: { n:"BoolErr", f:parse_BoolErr },
  26092. 0x0206: { n:"Formula", f:parse_Formula },
  26093. 0x0207: { n:"String", f:parse_String },
  26094. 0x0208: { n:'Row', f:parse_Row },
  26095. 0x020b: { n:"Index" },
  26096. 0x0221: { n:"Array", f:parse_Array },
  26097. 0x0225: { n:"DefaultRowHeight", f:parse_DefaultRowHeight },
  26098. 0x0236: { n:"Table" },
  26099. 0x023e: { n:"Window2", f:parse_Window2 },
  26100. 0x027e: { n:"RK", f:parse_RK },
  26101. 0x0293: { n:"Style" },
  26102. 0x0406: { n:"Formula", f:parse_Formula },
  26103. 0x0418: { n:"BigName" },
  26104. 0x041e: { n:"Format", f:parse_Format },
  26105. 0x043c: { n:"ContinueBigName" },
  26106. 0x04bc: { n:"ShrFmla", f:parse_ShrFmla },
  26107. 0x0800: { n:"HLinkTooltip", f:parse_HLinkTooltip },
  26108. 0x0801: { n:"WebPub" },
  26109. 0x0802: { n:"QsiSXTag" },
  26110. 0x0803: { n:"DBQueryExt" },
  26111. 0x0804: { n:"ExtString" },
  26112. 0x0805: { n:"TxtQry" },
  26113. 0x0806: { n:"Qsir" },
  26114. 0x0807: { n:"Qsif" },
  26115. 0x0808: { n:"RRDTQSIF" },
  26116. 0x0809: { n:'BOF', f:parse_BOF },
  26117. 0x080a: { n:"OleDbConn" },
  26118. 0x080b: { n:"WOpt" },
  26119. 0x080c: { n:"SXViewEx" },
  26120. 0x080d: { n:"SXTH" },
  26121. 0x080e: { n:"SXPIEx" },
  26122. 0x080f: { n:"SXVDTEx" },
  26123. 0x0810: { n:"SXViewEx9" },
  26124. 0x0812: { n:"ContinueFrt" },
  26125. 0x0813: { n:"RealTimeData" },
  26126. 0x0850: { n:"ChartFrtInfo" },
  26127. 0x0851: { n:"FrtWrapper" },
  26128. 0x0852: { n:"StartBlock" },
  26129. 0x0853: { n:"EndBlock" },
  26130. 0x0854: { n:"StartObject" },
  26131. 0x0855: { n:"EndObject" },
  26132. 0x0856: { n:"CatLab" },
  26133. 0x0857: { n:"YMult" },
  26134. 0x0858: { n:"SXViewLink" },
  26135. 0x0859: { n:"PivotChartBits" },
  26136. 0x085a: { n:"FrtFontList" },
  26137. 0x0862: { n:"SheetExt" },
  26138. 0x0863: { n:"BookExt", r:12},
  26139. 0x0864: { n:"SXAddl" },
  26140. 0x0865: { n:"CrErr" },
  26141. 0x0866: { n:"HFPicture" },
  26142. 0x0867: { n:'FeatHdr', f:parsenoop2 },
  26143. 0x0868: { n:"Feat" },
  26144. 0x086a: { n:"DataLabExt" },
  26145. 0x086b: { n:"DataLabExtContents" },
  26146. 0x086c: { n:"CellWatch" },
  26147. 0x0871: { n:"FeatHdr11" },
  26148. 0x0872: { n:"Feature11" },
  26149. 0x0874: { n:"DropDownObjIds" },
  26150. 0x0875: { n:"ContinueFrt11" },
  26151. 0x0876: { n:"DConn" },
  26152. 0x0877: { n:"List12" },
  26153. 0x0878: { n:"Feature12" },
  26154. 0x0879: { n:"CondFmt12" },
  26155. 0x087a: { n:"CF12" },
  26156. 0x087b: { n:"CFEx" },
  26157. 0x087c: { n:"XFCRC", f:parse_XFCRC, r:12 },
  26158. 0x087d: { n:"XFExt", f:parse_XFExt, r:12 },
  26159. 0x087e: { n:"AutoFilter12" },
  26160. 0x087f: { n:"ContinueFrt12" },
  26161. 0x0884: { n:"MDTInfo" },
  26162. 0x0885: { n:"MDXStr" },
  26163. 0x0886: { n:"MDXTuple" },
  26164. 0x0887: { n:"MDXSet" },
  26165. 0x0888: { n:"MDXProp" },
  26166. 0x0889: { n:"MDXKPI" },
  26167. 0x088a: { n:"MDB" },
  26168. 0x088b: { n:"PLV" },
  26169. 0x088c: { n:"Compat12", f:parsebool, r:12 },
  26170. 0x088d: { n:"DXF" },
  26171. 0x088e: { n:"TableStyles", r:12 },
  26172. 0x088f: { n:"TableStyle" },
  26173. 0x0890: { n:"TableStyleElement" },
  26174. 0x0892: { n:"StyleExt" },
  26175. 0x0893: { n:"NamePublish" },
  26176. 0x0894: { n:"NameCmt", f:parse_NameCmt, r:12 },
  26177. 0x0895: { n:"SortData" },
  26178. 0x0896: { n:"Theme", f:parse_Theme, r:12 },
  26179. 0x0897: { n:"GUIDTypeLib" },
  26180. 0x0898: { n:"FnGrp12" },
  26181. 0x0899: { n:"NameFnGrp12" },
  26182. 0x089a: { n:"MTRSettings", f:parse_MTRSettings, r:12 },
  26183. 0x089b: { n:"CompressPictures", f:parsenoop2 },
  26184. 0x089c: { n:"HeaderFooter" },
  26185. 0x089d: { n:"CrtLayout12" },
  26186. 0x089e: { n:"CrtMlFrt" },
  26187. 0x089f: { n:"CrtMlFrtContinue" },
  26188. 0x08a3: { n:"ForceFullCalculation", f:parse_ForceFullCalculation },
  26189. 0x08a4: { n:"ShapePropsStream" },
  26190. 0x08a5: { n:"TextPropsStream" },
  26191. 0x08a6: { n:"RichTextStream" },
  26192. 0x08a7: { n:"CrtLayout12A" },
  26193. 0x1001: { n:"Units" },
  26194. 0x1002: { n:"Chart" },
  26195. 0x1003: { n:"Series" },
  26196. 0x1006: { n:"DataFormat" },
  26197. 0x1007: { n:"LineFormat" },
  26198. 0x1009: { n:"MarkerFormat" },
  26199. 0x100a: { n:"AreaFormat" },
  26200. 0x100b: { n:"PieFormat" },
  26201. 0x100c: { n:"AttachedLabel" },
  26202. 0x100d: { n:"SeriesText" },
  26203. 0x1014: { n:"ChartFormat" },
  26204. 0x1015: { n:"Legend" },
  26205. 0x1016: { n:"SeriesList" },
  26206. 0x1017: { n:"Bar" },
  26207. 0x1018: { n:"Line" },
  26208. 0x1019: { n:"Pie" },
  26209. 0x101a: { n:"Area" },
  26210. 0x101b: { n:"Scatter" },
  26211. 0x101c: { n:"CrtLine" },
  26212. 0x101d: { n:"Axis" },
  26213. 0x101e: { n:"Tick" },
  26214. 0x101f: { n:"ValueRange" },
  26215. 0x1020: { n:"CatSerRange" },
  26216. 0x1021: { n:"AxisLine" },
  26217. 0x1022: { n:"CrtLink" },
  26218. 0x1024: { n:"DefaultText" },
  26219. 0x1025: { n:"Text" },
  26220. 0x1026: { n:"FontX", f:parseuint16 },
  26221. 0x1027: { n:"ObjectLink" },
  26222. 0x1032: { n:"Frame" },
  26223. 0x1033: { n:"Begin" },
  26224. 0x1034: { n:"End" },
  26225. 0x1035: { n:"PlotArea" },
  26226. 0x103a: { n:"Chart3d" },
  26227. 0x103c: { n:"PicF" },
  26228. 0x103d: { n:"DropBar" },
  26229. 0x103e: { n:"Radar" },
  26230. 0x103f: { n:"Surf" },
  26231. 0x1040: { n:"RadarArea" },
  26232. 0x1041: { n:"AxisParent" },
  26233. 0x1043: { n:"LegendException" },
  26234. 0x1044: { n:"ShtProps", f:parse_ShtProps },
  26235. 0x1045: { n:"SerToCrt" },
  26236. 0x1046: { n:"AxesUsed" },
  26237. 0x1048: { n:"SBaseRef" },
  26238. 0x104a: { n:"SerParent" },
  26239. 0x104b: { n:"SerAuxTrend" },
  26240. 0x104e: { n:"IFmtRecord" },
  26241. 0x104f: { n:"Pos" },
  26242. 0x1050: { n:"AlRuns" },
  26243. 0x1051: { n:"BRAI" },
  26244. 0x105b: { n:"SerAuxErrBar" },
  26245. 0x105c: { n:"ClrtClient", f:parse_ClrtClient },
  26246. 0x105d: { n:"SerFmt" },
  26247. 0x105f: { n:"Chart3DBarShape" },
  26248. 0x1060: { n:"Fbi" },
  26249. 0x1061: { n:"BopPop" },
  26250. 0x1062: { n:"AxcExt" },
  26251. 0x1063: { n:"Dat" },
  26252. 0x1064: { n:"PlotGrowth" },
  26253. 0x1065: { n:"SIIndex" },
  26254. 0x1066: { n:"GelFrame" },
  26255. 0x1067: { n:"BopPopCustom" },
  26256. 0x1068: { n:"Fbi2" },
  26257. 0x0000: { n:"Dimensions", f:parse_Dimensions },
  26258. 0x0002: { n:"BIFF2INT", f:parse_BIFF2INT },
  26259. 0x0005: { n:"BoolErr", f:parse_BoolErr },
  26260. 0x0007: { n:"String", f:parse_BIFF2STRING },
  26261. 0x0008: { n:"BIFF2ROW" },
  26262. 0x000b: { n:"Index" },
  26263. 0x0016: { n:"ExternCount", f:parseuint16 },
  26264. 0x001e: { n:"BIFF2FORMAT", f:parse_BIFF2Format },
  26265. 0x001f: { n:"BIFF2FMTCNT" }, /* 16-bit cnt of BIFF2FORMAT records */
  26266. 0x0020: { n:"BIFF2COLINFO" },
  26267. 0x0021: { n:"Array", f:parse_Array },
  26268. 0x0025: { n:"DefaultRowHeight", f:parse_DefaultRowHeight },
  26269. 0x0032: { n:"BIFF2FONTXTRA", f:parse_BIFF2FONTXTRA },
  26270. 0x0034: { n:"DDEObjName" },
  26271. 0x003e: { n:"BIFF2WINDOW2" },
  26272. 0x0043: { n:"BIFF2XF" },
  26273. 0x0045: { n:"BIFF2FONTCLR" },
  26274. 0x0056: { n:"BIFF4FMTCNT" }, /* 16-bit cnt, similar to BIFF2 */
  26275. 0x007e: { n:"RK" }, /* Not necessarily same as 0x027e */
  26276. 0x007f: { n:"ImData", f:parse_ImData },
  26277. 0x0087: { n:"Addin" },
  26278. 0x0088: { n:"Edg" },
  26279. 0x0089: { n:"Pub" },
  26280. 0x0091: { n:"Sub" },
  26281. 0x0094: { n:"LHRecord" },
  26282. 0x0095: { n:"LHNGraph" },
  26283. 0x0096: { n:"Sound" },
  26284. 0x00a9: { n:"CoordList" },
  26285. 0x00ab: { n:"GCW" },
  26286. 0x00bc: { n:"ShrFmla" }, /* Not necessarily same as 0x04bc */
  26287. 0x00bf: { n:"ToolbarHdr" },
  26288. 0x00c0: { n:"ToolbarEnd" },
  26289. 0x00c2: { n:"AddMenu" },
  26290. 0x00c3: { n:"DelMenu" },
  26291. 0x00d6: { n:"RString", f:parse_RString },
  26292. 0x00df: { n:"UDDesc" },
  26293. 0x00ea: { n:"TabIdConf" },
  26294. 0x0162: { n:"XL5Modify" },
  26295. 0x01a5: { n:"FileSharing2" },
  26296. 0x0209: { n:'BOF', f:parse_BOF },
  26297. 0x0218: { n:"Lbl", f:parse_Lbl },
  26298. 0x0223: { n:"ExternName", f:parse_ExternName },
  26299. 0x0231: { n:"Font" },
  26300. 0x0243: { n:"BIFF3XF" },
  26301. 0x0409: { n:'BOF', f:parse_BOF },
  26302. 0x0443: { n:"BIFF4XF" },
  26303. 0x086d: { n:"FeatInfo" },
  26304. 0x0873: { n:"FeatInfo11" },
  26305. 0x0881: { n:"SXAddl12" },
  26306. 0x08c0: { n:"AutoWebPub" },
  26307. 0x08c1: { n:"ListObj" },
  26308. 0x08c2: { n:"ListField" },
  26309. 0x08c3: { n:"ListDV" },
  26310. 0x08c4: { n:"ListCondFmt" },
  26311. 0x08c5: { n:"ListCF" },
  26312. 0x08c6: { n:"FMQry" },
  26313. 0x08c7: { n:"FMSQry" },
  26314. 0x08c8: { n:"PLV" },
  26315. 0x08c9: { n:"LnExt" },
  26316. 0x08ca: { n:"MkrExt" },
  26317. 0x08cb: { n:"CrtCoopt" },
  26318. 0x08d6: { n:"FRTArchId$", r:12 },
  26319. 0x7262: {}
  26320. };
  26321. var XLSRE = evert_key(XLSRecordEnum, 'n');
  26322. function write_biff_rec(ba, type, payload, length) {
  26323. var t = +type || +XLSRE[type];
  26324. if(isNaN(t)) return;
  26325. var len = length || (payload||[]).length || 0;
  26326. var o = ba.next(4);
  26327. o.write_shift(2, t);
  26328. o.write_shift(2, len);
  26329. if(len > 0 && is_buf(payload)) ba.push(payload);
  26330. }
  26331. function write_BIFF2Cell(out, r, c) {
  26332. if(!out) out = new_buf(7);
  26333. out.write_shift(2, r);
  26334. out.write_shift(2, c);
  26335. out.write_shift(2, 0);
  26336. out.write_shift(1, 0);
  26337. return out;
  26338. }
  26339. function write_BIFF2BERR(r, c, val, t) {
  26340. var out = new_buf(9);
  26341. write_BIFF2Cell(out, r, c);
  26342. if(t == 'e') { out.write_shift(1, val); out.write_shift(1, 1); }
  26343. else { out.write_shift(1, val?1:0); out.write_shift(1, 0); }
  26344. return out;
  26345. }
  26346. /* TODO: codepage, large strings */
  26347. function write_BIFF2LABEL(r, c, val) {
  26348. var out = new_buf(8 + 2*val.length);
  26349. write_BIFF2Cell(out, r, c);
  26350. out.write_shift(1, val.length);
  26351. out.write_shift(val.length, val, 'sbcs');
  26352. return out.l < out.length ? out.slice(0, out.l) : out;
  26353. }
  26354. function write_ws_biff2_cell(ba, cell, R, C) {
  26355. if(cell.v != null) switch(cell.t) {
  26356. case 'd': case 'n':
  26357. var v = cell.t == 'd' ? datenum(parseDate(cell.v)) : cell.v;
  26358. if((v == (v|0)) && (v >= 0) && (v < 65536))
  26359. write_biff_rec(ba, 0x0002, write_BIFF2INT(R, C, v));
  26360. else
  26361. write_biff_rec(ba, 0x0003, write_BIFF2NUM(R,C, v));
  26362. return;
  26363. case 'b': case 'e': write_biff_rec(ba, 0x0005, write_BIFF2BERR(R, C, cell.v, cell.t)); return;
  26364. /* TODO: codepage, sst */
  26365. case 's': case 'str':
  26366. write_biff_rec(ba, 0x0004, write_BIFF2LABEL(R, C, cell.v));
  26367. return;
  26368. }
  26369. write_biff_rec(ba, 0x0001, write_BIFF2Cell(null, R, C));
  26370. }
  26371. function write_ws_biff2(ba, ws, idx, opts) {
  26372. var dense = Array.isArray(ws);
  26373. var range = safe_decode_range(ws['!ref'] || "A1"), ref, rr = "", cols = [];
  26374. if(range.e.c > 0xFF || range.e.r > 0x3FFF) {
  26375. if(opts.WTF) throw new Error("Range " + (ws['!ref'] || "A1") + " exceeds format limit A1:IV16384");
  26376. range.e.c = Math.min(range.e.c, 0xFF);
  26377. range.e.r = Math.min(range.e.c, 0x3FFF);
  26378. ref = encode_range(range);
  26379. }
  26380. for(var R = range.s.r; R <= range.e.r; ++R) {
  26381. rr = encode_row(R);
  26382. for(var C = range.s.c; C <= range.e.c; ++C) {
  26383. if(R === range.s.r) cols[C] = encode_col(C);
  26384. ref = cols[C] + rr;
  26385. var cell = dense ? (ws[R]||[])[C] : ws[ref];
  26386. if(!cell) continue;
  26387. /* write cell */
  26388. write_ws_biff2_cell(ba, cell, R, C, opts);
  26389. }
  26390. }
  26391. }
  26392. /* Based on test files */
  26393. function write_biff2_buf(wb, opts) {
  26394. var o = opts || {};
  26395. if(DENSE != null && o.dense == null) o.dense = DENSE;
  26396. var ba = buf_array();
  26397. var idx = 0;
  26398. for(var i=0;i<wb.SheetNames.length;++i) if(wb.SheetNames[i] == o.sheet) idx=i;
  26399. if(idx == 0 && !!o.sheet && wb.SheetNames[0] != o.sheet) throw new Error("Sheet not found: " + o.sheet);
  26400. write_biff_rec(ba, 0x0009, write_BOF(wb, 0x10, o));
  26401. /* ... */
  26402. write_ws_biff2(ba, wb.Sheets[wb.SheetNames[idx]], idx, o, wb);
  26403. /* ... */
  26404. write_biff_rec(ba, 0x000A);
  26405. return ba.end();
  26406. }
  26407. function write_FONTS_biff8(ba, data, opts) {
  26408. write_biff_rec(ba, "Font", write_Font({
  26409. sz:12,
  26410. color: {theme:1},
  26411. name: "Arial",
  26412. family: 2,
  26413. scheme: "minor"
  26414. }, opts));
  26415. }
  26416. function write_FMTS_biff8(ba, NF, opts) {
  26417. if(!NF) return;
  26418. [[5,8],[23,26],[41,44],[/*63*/50,/*66],[164,*/392]].forEach(function(r) {
  26419. for(var i = r[0]; i <= r[1]; ++i) if(NF[i] != null) write_biff_rec(ba, "Format", write_Format(i, NF[i], opts));
  26420. });
  26421. }
  26422. function write_FEAT(ba, ws) {
  26423. /* [MS-XLS] 2.4.112 */
  26424. var o = new_buf(19);
  26425. o.write_shift(4, 0x867); o.write_shift(4, 0); o.write_shift(4, 0);
  26426. o.write_shift(2, 3); o.write_shift(1, 1); o.write_shift(4, 0);
  26427. write_biff_rec(ba, "FeatHdr", o);
  26428. /* [MS-XLS] 2.4.111 */
  26429. o = new_buf(39);
  26430. o.write_shift(4, 0x868); o.write_shift(4, 0); o.write_shift(4, 0);
  26431. o.write_shift(2, 3); o.write_shift(1, 0); o.write_shift(4, 0);
  26432. o.write_shift(2, 1); o.write_shift(4, 4); o.write_shift(2, 0);
  26433. write_Ref8U(safe_decode_range(ws['!ref']||"A1"), o);
  26434. o.write_shift(4, 4);
  26435. write_biff_rec(ba, "Feat", o);
  26436. }
  26437. function write_CELLXFS_biff8(ba, opts) {
  26438. for(var i = 0; i < 16; ++i) write_biff_rec(ba, "XF", write_XF({numFmtId:0, style:true}, 0, opts));
  26439. opts.cellXfs.forEach(function(c) {
  26440. write_biff_rec(ba, "XF", write_XF(c, 0, opts));
  26441. });
  26442. }
  26443. function write_ws_biff8_hlinks(ba, ws) {
  26444. for(var R=0; R<ws['!links'].length; ++R) {
  26445. var HL = ws['!links'][R];
  26446. write_biff_rec(ba, "HLink", write_HLink(HL));
  26447. if(HL[1].Tooltip) write_biff_rec(ba, "HLinkTooltip", write_HLinkTooltip(HL));
  26448. }
  26449. delete ws['!links'];
  26450. }
  26451. function write_ws_biff8_cell(ba, cell, R, C, opts) {
  26452. var os = 16 + get_cell_style(opts.cellXfs, cell, opts);
  26453. if(cell.v != null) switch(cell.t) {
  26454. case 'd': case 'n':
  26455. var v = cell.t == 'd' ? datenum(parseDate(cell.v)) : cell.v;
  26456. /* TODO: emit RK as appropriate */
  26457. write_biff_rec(ba, "Number", write_Number(R, C, v, os, opts));
  26458. return;
  26459. case 'b': case 'e': write_biff_rec(ba, 0x0205, write_BoolErr(R, C, cell.v, os, opts, cell.t)); return;
  26460. /* TODO: codepage, sst */
  26461. case 's': case 'str':
  26462. write_biff_rec(ba, "Label", write_Label(R, C, cell.v, os, opts));
  26463. return;
  26464. }
  26465. write_biff_rec(ba, "Blank", write_XLSCell(R, C, os));
  26466. }
  26467. /* [MS-XLS] 2.1.7.20.5 */
  26468. function write_ws_biff8(idx, opts, wb) {
  26469. var ba = buf_array();
  26470. var s = wb.SheetNames[idx], ws = wb.Sheets[s] || {};
  26471. var _WB = ((wb||{}).Workbook||{});
  26472. var _sheet = ((_WB.Sheets||[])[idx]||{});
  26473. var dense = Array.isArray(ws);
  26474. var b8 = opts.biff == 8;
  26475. var ref, rr = "", cols = [];
  26476. var range = safe_decode_range(ws['!ref'] || "A1");
  26477. var MAX_ROWS = b8 ? 65536 : 16384;
  26478. if(range.e.c > 0xFF || range.e.r >= MAX_ROWS) {
  26479. if(opts.WTF) throw new Error("Range " + (ws['!ref'] || "A1") + " exceeds format limit A1:IV16384");
  26480. range.e.c = Math.min(range.e.c, 0xFF);
  26481. range.e.r = Math.min(range.e.c, MAX_ROWS-1);
  26482. }
  26483. write_biff_rec(ba, 0x0809, write_BOF(wb, 0x10, opts));
  26484. /* ... */
  26485. write_biff_rec(ba, "CalcMode", writeuint16(1));
  26486. write_biff_rec(ba, "CalcCount", writeuint16(100));
  26487. write_biff_rec(ba, "CalcRefMode", writebool(true));
  26488. write_biff_rec(ba, "CalcIter", writebool(false));
  26489. write_biff_rec(ba, "CalcDelta", write_Xnum(0.001));
  26490. write_biff_rec(ba, "CalcSaveRecalc", writebool(true));
  26491. write_biff_rec(ba, "PrintRowCol", writebool(false));
  26492. write_biff_rec(ba, "PrintGrid", writebool(false));
  26493. write_biff_rec(ba, "GridSet", writeuint16(1));
  26494. write_biff_rec(ba, "Guts", write_Guts([0,0]));
  26495. /* ... */
  26496. write_biff_rec(ba, "HCenter", writebool(false));
  26497. write_biff_rec(ba, "VCenter", writebool(false));
  26498. /* ... */
  26499. write_biff_rec(ba, 0x200, write_Dimensions(range, opts));
  26500. /* ... */
  26501. if(b8) ws['!links'] = [];
  26502. for(var R = range.s.r; R <= range.e.r; ++R) {
  26503. rr = encode_row(R);
  26504. for(var C = range.s.c; C <= range.e.c; ++C) {
  26505. if(R === range.s.r) cols[C] = encode_col(C);
  26506. ref = cols[C] + rr;
  26507. var cell = dense ? (ws[R]||[])[C] : ws[ref];
  26508. if(!cell) continue;
  26509. /* write cell */
  26510. write_ws_biff8_cell(ba, cell, R, C, opts);
  26511. if(b8 && cell.l) ws['!links'].push([ref, cell.l]);
  26512. }
  26513. }
  26514. var cname = _sheet.CodeName || _sheet.name || s;
  26515. /* ... */
  26516. if(b8 && _WB.Views) write_biff_rec(ba, "Window2", write_Window2(_WB.Views[0]));
  26517. /* ... */
  26518. if(b8 && (ws['!merges']||[]).length) write_biff_rec(ba, "MergeCells", write_MergeCells(ws['!merges']));
  26519. /* ... */
  26520. if(b8) write_ws_biff8_hlinks(ba, ws);
  26521. /* ... */
  26522. write_biff_rec(ba, "CodeName", write_XLUnicodeString(cname, opts));
  26523. /* ... */
  26524. if(b8) write_FEAT(ba, ws);
  26525. /* ... */
  26526. write_biff_rec(ba, "EOF");
  26527. return ba.end();
  26528. }
  26529. /* [MS-XLS] 2.1.7.20.3 */
  26530. function write_biff8_global(wb, bufs, opts) {
  26531. var A = buf_array();
  26532. var _WB = ((wb||{}).Workbook||{});
  26533. var _sheets = (_WB.Sheets||[]);
  26534. var _wb = _WB.WBProps||{};
  26535. var b8 = opts.biff == 8, b5 = opts.biff == 5;
  26536. write_biff_rec(A, 0x0809, write_BOF(wb, 0x05, opts));
  26537. if(opts.bookType == "xla") write_biff_rec(A, "Addin");
  26538. write_biff_rec(A, "InterfaceHdr", b8 ? writeuint16(0x04b0) : null);
  26539. write_biff_rec(A, "Mms", writezeroes(2));
  26540. if(b5) write_biff_rec(A, "ToolbarHdr");
  26541. if(b5) write_biff_rec(A, "ToolbarEnd");
  26542. write_biff_rec(A, "InterfaceEnd");
  26543. write_biff_rec(A, "WriteAccess", write_WriteAccess("SheetJS", opts));
  26544. write_biff_rec(A, "CodePage", writeuint16(b8 ? 0x04b0 : 0x04E4));
  26545. if(b8) write_biff_rec(A, "DSF", writeuint16(0));
  26546. if(b8) write_biff_rec(A, "Excel9File");
  26547. write_biff_rec(A, "RRTabId", write_RRTabId(wb.SheetNames.length));
  26548. if(b8 && wb.vbaraw) {
  26549. write_biff_rec(A, "ObProj");
  26550. var cname = _wb.CodeName || "ThisWorkbook";
  26551. write_biff_rec(A, "CodeName", write_XLUnicodeString(cname, opts));
  26552. }
  26553. write_biff_rec(A, "BuiltInFnGroupCount", writeuint16(0x11));
  26554. write_biff_rec(A, "WinProtect", writebool(false));
  26555. write_biff_rec(A, "Protect", writebool(false));
  26556. write_biff_rec(A, "Password", writeuint16(0));
  26557. if(b8) write_biff_rec(A, "Prot4Rev", writebool(false));
  26558. if(b8) write_biff_rec(A, "Prot4RevPass", writeuint16(0));
  26559. write_biff_rec(A, "Window1", write_Window1(opts));
  26560. write_biff_rec(A, "Backup", writebool(false));
  26561. write_biff_rec(A, "HideObj", writeuint16(0));
  26562. write_biff_rec(A, "Date1904", writebool(safe1904(wb)=="true"));
  26563. write_biff_rec(A, "CalcPrecision", writebool(true));
  26564. if(b8) write_biff_rec(A, "RefreshAll", writebool(false));
  26565. write_biff_rec(A, "BookBool", writeuint16(0));
  26566. /* ... */
  26567. write_FONTS_biff8(A, wb, opts);
  26568. write_FMTS_biff8(A, wb.SSF, opts);
  26569. write_CELLXFS_biff8(A, opts);
  26570. /* ... */
  26571. if(b8) write_biff_rec(A, "UsesELFs", writebool(false));
  26572. var a = A.end();
  26573. var C = buf_array();
  26574. if(b8) write_biff_rec(C, "Country", write_Country());
  26575. /* BIFF8: [SST *Continue] ExtSST */
  26576. write_biff_rec(C, "EOF");
  26577. var c = C.end();
  26578. var B = buf_array();
  26579. var blen = 0, j = 0;
  26580. for(j = 0; j < wb.SheetNames.length; ++j) blen += (b8 ? 12 : 11) + (b8 ? 2 : 1) * wb.SheetNames[j].length;
  26581. var start = a.length + blen + c.length;
  26582. for(j = 0; j < wb.SheetNames.length; ++j) {
  26583. var _sheet = _sheets[j] || ({});
  26584. write_biff_rec(B, "BoundSheet8", write_BoundSheet8({pos:start, hs:_sheet.Hidden||0, dt:0, name:wb.SheetNames[j]}, opts));
  26585. start += bufs[j].length;
  26586. }
  26587. /* 1*BoundSheet8 */
  26588. var b = B.end();
  26589. if(blen != b.length) throw new Error("BS8 " + blen + " != " + b.length);
  26590. var out = [];
  26591. if(a.length) out.push(a);
  26592. if(b.length) out.push(b);
  26593. if(c.length) out.push(c);
  26594. return __toBuffer([out]);
  26595. }
  26596. /* [MS-XLS] 2.1.7.20 Workbook Stream */
  26597. function write_biff8_buf(wb, opts) {
  26598. var o = opts || {};
  26599. var bufs = [];
  26600. if(wb && !wb.SSF) {
  26601. wb.SSF = SSF.get_table();
  26602. }
  26603. if(wb && wb.SSF) {
  26604. make_ssf(SSF); SSF.load_table(wb.SSF);
  26605. // $FlowIgnore
  26606. o.revssf = evert_num(wb.SSF); o.revssf[wb.SSF[65535]] = 0;
  26607. o.ssf = wb.SSF;
  26608. }
  26609. o.cellXfs = [];
  26610. o.Strings = []; o.Strings.Count = 0; o.Strings.Unique = 0;
  26611. get_cell_style(o.cellXfs, {}, {revssf:{"General":0}});
  26612. for(var i = 0; i < wb.SheetNames.length; ++i) bufs[bufs.length] = write_ws_biff8(i, o, wb);
  26613. bufs.unshift(write_biff8_global(wb, bufs, o));
  26614. return __toBuffer([bufs]);
  26615. }
  26616. function write_biff_buf(wb, opts) {
  26617. var o = opts || {};
  26618. switch(o.biff || 2) {
  26619. case 8: case 5: return write_biff8_buf(wb, opts);
  26620. case 4: case 3: case 2: return write_biff2_buf(wb, opts);
  26621. }
  26622. throw new Error("invalid type " + o.bookType + " for BIFF");
  26623. }
  26624. /* note: browser DOM element cannot see mso- style attrs, must parse */
  26625. var HTML_ = (function() {
  26626. function html_to_sheet(str, _opts) {
  26627. var opts = _opts || {};
  26628. if(DENSE != null && opts.dense == null) opts.dense = DENSE;
  26629. var ws = opts.dense ? ([]) : ({});
  26630. var mtch = str.match(/<table/i);
  26631. if(!mtch) throw new Error("Invalid HTML: could not find <table>");
  26632. var mtch2 = str.match(/<\/table/i);
  26633. var i = mtch.index, j = mtch2 && mtch2.index || str.length;
  26634. var rows = split_regex(str.slice(i, j), /(:?<tr[^>]*>)/i, "<tr>");
  26635. var R = -1, C = 0, RS = 0, CS = 0;
  26636. var range = {s:{r:10000000, c:10000000},e:{r:0,c:0}};
  26637. var merges = [];
  26638. for(i = 0; i < rows.length; ++i) {
  26639. var row = rows[i].trim();
  26640. var hd = row.slice(0,3).toLowerCase();
  26641. if(hd == "<tr") { ++R; if(opts.sheetRows && opts.sheetRows <= R) { --R; break; } C = 0; continue; }
  26642. if(hd != "<td" && hd != "<th") continue;
  26643. var cells = row.split(/<\/t[dh]>/i);
  26644. for(j = 0; j < cells.length; ++j) {
  26645. var cell = cells[j].trim();
  26646. if(!cell.match(/<t[dh]/i)) continue;
  26647. var m = cell, cc = 0;
  26648. /* TODO: parse styles etc */
  26649. while(m.charAt(0) == "<" && (cc = m.indexOf(">")) > -1) m = m.slice(cc+1);
  26650. var tag = parsexmltag(cell.slice(0, cell.indexOf(">")));
  26651. CS = tag.colspan ? +tag.colspan : 1;
  26652. if((RS = +tag.rowspan)>1 || CS>1) merges.push({s:{r:R,c:C},e:{r:R + (RS||1) - 1, c:C + CS - 1}});
  26653. var _t = tag.t || "";
  26654. /* TODO: generate stub cells */
  26655. if(!m.length) { C += CS; continue; }
  26656. m = htmldecode(m);
  26657. if(range.s.r > R) range.s.r = R; if(range.e.r < R) range.e.r = R;
  26658. if(range.s.c > C) range.s.c = C; if(range.e.c < C) range.e.c = C;
  26659. if(!m.length) continue;
  26660. var o = {t:'s', v:m};
  26661. if(opts.raw || !m.trim().length || _t == 's'){}
  26662. else if(m === 'TRUE') o = {t:'b', v:true};
  26663. else if(m === 'FALSE') o = {t:'b', v:false};
  26664. else if(!isNaN(fuzzynum(m))) o = {t:'n', v:fuzzynum(m)};
  26665. else if(!isNaN(fuzzydate(m).getDate())) {
  26666. o = ({t:'d', v:parseDate(m)});
  26667. if(!opts.cellDates) o = ({t:'n', v:datenum(o.v)});
  26668. o.z = opts.dateNF || SSF._table[14];
  26669. }
  26670. if(opts.dense) { if(!ws[R]) ws[R] = []; ws[R][C] = o; }
  26671. else ws[encode_cell({r:R, c:C})] = o;
  26672. C += CS;
  26673. }
  26674. }
  26675. ws['!ref'] = encode_range(range);
  26676. return ws;
  26677. }
  26678. function html_to_book(str, opts) {
  26679. return sheet_to_workbook(html_to_sheet(str, opts), opts);
  26680. }
  26681. function make_html_row(ws, r, R, o) {
  26682. var M = (ws['!merges'] ||[]);
  26683. var oo = [];
  26684. for(var C = r.s.c; C <= r.e.c; ++C) {
  26685. var RS = 0, CS = 0;
  26686. for(var j = 0; j < M.length; ++j) {
  26687. if(M[j].s.r > R || M[j].s.c > C) continue;
  26688. if(M[j].e.r < R || M[j].e.c < C) continue;
  26689. if(M[j].s.r < R || M[j].s.c < C) { RS = -1; break; }
  26690. RS = M[j].e.r - M[j].s.r + 1; CS = M[j].e.c - M[j].s.c + 1; break;
  26691. }
  26692. if(RS < 0) continue;
  26693. var coord = encode_cell({r:R,c:C});
  26694. var cell = o.dense ? (ws[R]||[])[C] : ws[coord];
  26695. var sp = {};
  26696. if(RS > 1) sp.rowspan = RS;
  26697. if(CS > 1) sp.colspan = CS;
  26698. /* TODO: html entities */
  26699. var w = (cell && cell.v != null) && (cell.h || escapehtml(cell.w || (format_cell(cell), cell.w) || "")) || "";
  26700. sp.t = cell && cell.t || 'z';
  26701. if(o.editable) w = '<span contenteditable="true">' + w + '</span>';
  26702. sp.id = "sjs-" + coord;
  26703. oo.push(writextag('td', w, sp));
  26704. }
  26705. var preamble = "<tr>";
  26706. return preamble + oo.join("") + "</tr>";
  26707. }
  26708. function make_html_preamble(ws, R, o) {
  26709. var out = [];
  26710. return out.join("") + '<table' + (o && o.id ? ' id="' + o.id + '"' : "") + '>';
  26711. }
  26712. var _BEGIN = '<html><head><meta charset="utf-8"/><title>SheetJS Table Export</title></head><body>';
  26713. var _END = '</body></html>';
  26714. function sheet_to_html(ws, opts/*, wb:?Workbook*/) {
  26715. var o = opts || {};
  26716. var header = o.header != null ? o.header : _BEGIN;
  26717. var footer = o.footer != null ? o.footer : _END;
  26718. var out = [header];
  26719. var r = decode_range(ws['!ref']);
  26720. o.dense = Array.isArray(ws);
  26721. out.push(make_html_preamble(ws, r, o));
  26722. for(var R = r.s.r; R <= r.e.r; ++R) out.push(make_html_row(ws, r, R, o));
  26723. out.push("</table>" + footer);
  26724. return out.join("");
  26725. }
  26726. return {
  26727. to_workbook: html_to_book,
  26728. to_sheet: html_to_sheet,
  26729. _row: make_html_row,
  26730. BEGIN: _BEGIN,
  26731. END: _END,
  26732. _preamble: make_html_preamble,
  26733. from_sheet: sheet_to_html
  26734. };
  26735. })();
  26736. function parse_dom_table(table, _opts) {
  26737. var opts = _opts || {};
  26738. if(DENSE != null) opts.dense = DENSE;
  26739. var ws = opts.dense ? ([]) : ({});
  26740. var rows = table.getElementsByTagName('tr');
  26741. var sheetRows = opts.sheetRows || 10000000;
  26742. var range = {s:{r:0,c:0},e:{r:0,c:0}};
  26743. var merges = [], midx = 0;
  26744. var rowinfo = [];
  26745. var _R = 0, R = 0, _C, C, RS, CS;
  26746. for(; _R < rows.length && R < sheetRows; ++_R) {
  26747. var row = rows[_R];
  26748. if (is_dom_element_hidden(row)) {
  26749. if (opts.display) continue;
  26750. rowinfo[R] = {hidden: true};
  26751. }
  26752. var elts = (row.children);
  26753. for(_C = C = 0; _C < elts.length; ++_C) {
  26754. var elt = elts[_C];
  26755. if (opts.display && is_dom_element_hidden(elt)) continue;
  26756. var v = htmldecode(elt.innerHTML);
  26757. for(midx = 0; midx < merges.length; ++midx) {
  26758. var m = merges[midx];
  26759. if(m.s.c == C && m.s.r <= R && R <= m.e.r) { C = m.e.c+1; midx = -1; }
  26760. }
  26761. /* TODO: figure out how to extract nonstandard mso- style */
  26762. CS = +elt.getAttribute("colspan") || 1;
  26763. if((RS = +elt.getAttribute("rowspan"))>0 || CS>1) merges.push({s:{r:R,c:C},e:{r:R + (RS||1) - 1, c:C + CS - 1}});
  26764. var o = {t:'s', v:v};
  26765. var _t = elt.getAttribute("t") || "";
  26766. if(v != null) {
  26767. if(v.length == 0) o.t = _t || 'z';
  26768. else if(opts.raw || v.trim().length == 0 || _t == "s"){}
  26769. else if(v === 'TRUE') o = {t:'b', v:true};
  26770. else if(v === 'FALSE') o = {t:'b', v:false};
  26771. else if(!isNaN(fuzzynum(v))) o = {t:'n', v:fuzzynum(v)};
  26772. else if(!isNaN(fuzzydate(v).getDate())) {
  26773. o = ({t:'d', v:parseDate(v)});
  26774. if(!opts.cellDates) o = ({t:'n', v:datenum(o.v)});
  26775. o.z = opts.dateNF || SSF._table[14];
  26776. }
  26777. }
  26778. if(opts.dense) { if(!ws[R]) ws[R] = []; ws[R][C] = o; }
  26779. else ws[encode_cell({c:C, r:R})] = o;
  26780. if(range.e.c < C) range.e.c = C;
  26781. C += CS;
  26782. }
  26783. ++R;
  26784. }
  26785. if(merges.length) ws['!merges'] = merges;
  26786. if(rowinfo.length) ws['!rows'] = rowinfo;
  26787. range.e.r = R - 1;
  26788. ws['!ref'] = encode_range(range);
  26789. if(R >= sheetRows) ws['!fullref'] = encode_range((range.e.r = rows.length-_R+R-1,range)); // We can count the real number of rows to parse but we don't to improve the performance
  26790. return ws;
  26791. }
  26792. function table_to_book(table, opts) {
  26793. return sheet_to_workbook(parse_dom_table(table, opts), opts);
  26794. }
  26795. function is_dom_element_hidden(element) {
  26796. var display = '';
  26797. var get_computed_style = get_get_computed_style_function(element);
  26798. if(get_computed_style) display = get_computed_style(element).getPropertyValue('display');
  26799. if(!display) display = element.style.display; // Fallback for cases when getComputedStyle is not available (e.g. an old browser or some Node.js environments) or doesn't work (e.g. if the element is not inserted to a document)
  26800. return display === 'none';
  26801. }
  26802. /* global getComputedStyle */
  26803. function get_get_computed_style_function(element) {
  26804. // The proper getComputedStyle implementation is the one defined in the element window
  26805. if(element.ownerDocument.defaultView && typeof element.ownerDocument.defaultView.getComputedStyle === 'function') return element.ownerDocument.defaultView.getComputedStyle;
  26806. // If it is not available, try to get one from the global namespace
  26807. if(typeof getComputedStyle === 'function') return getComputedStyle;
  26808. return null;
  26809. }
  26810. /* OpenDocument */
  26811. var parse_content_xml = (function() {
  26812. var parse_text_p = function(text) {
  26813. /* 6.1.2 White Space Characters */
  26814. var fixed = text
  26815. .replace(/[\t\r\n]/g, " ").trim().replace(/ +/g, " ")
  26816. .replace(/<text:s\/>/g," ")
  26817. .replace(/<text:s text:c="(\d+)"\/>/g, function($$,$1) { return Array(parseInt($1,10)+1).join(" "); })
  26818. .replace(/<text:tab[^>]*\/>/g,"\t")
  26819. .replace(/<text:line-break\/>/g,"\n");
  26820. var v = unescapexml(fixed.replace(/<[^>]*>/g,""));
  26821. return [v];
  26822. };
  26823. var number_formats = {
  26824. /* ods name: [short ssf fmt, long ssf fmt] */
  26825. day: ["d", "dd"],
  26826. month: ["m", "mm"],
  26827. year: ["y", "yy"],
  26828. hours: ["h", "hh"],
  26829. minutes: ["m", "mm"],
  26830. seconds: ["s", "ss"],
  26831. "am-pm": ["A/P", "AM/PM"],
  26832. "day-of-week": ["ddd", "dddd"],
  26833. era: ["e", "ee"],
  26834. /* there is no native representation of LO "Q" format */
  26835. quarter: ["\\Qm", "m\\\"th quarter\""]
  26836. };
  26837. return function pcx(d, _opts) {
  26838. var opts = _opts || {};
  26839. if(DENSE != null && opts.dense == null) opts.dense = DENSE;
  26840. var str = xlml_normalize(d);
  26841. var state = [], tmp;
  26842. var tag;
  26843. var NFtag = {name:""}, NF = "", pidx = 0;
  26844. var sheetag;
  26845. var rowtag;
  26846. var Sheets = {}, SheetNames = [];
  26847. var ws = opts.dense ? ([]) : ({});
  26848. var Rn, q;
  26849. var ctag = ({value:""});
  26850. var textp = "", textpidx = 0, textptag;
  26851. var textR = [];
  26852. var R = -1, C = -1, range = {s: {r:1000000,c:10000000}, e: {r:0, c:0}};
  26853. var row_ol = 0;
  26854. var number_format_map = {};
  26855. var merges = [], mrange = {}, mR = 0, mC = 0;
  26856. var rowinfo = [], rowpeat = 1, colpeat = 1;
  26857. var arrayf = [];
  26858. var WB = {Names:[]};
  26859. var atag = ({});
  26860. var _Ref = ["", ""];
  26861. var comments = [], comment = ({});
  26862. var creator = "", creatoridx = 0;
  26863. var isstub = false, intable = false;
  26864. var i = 0;
  26865. xlmlregex.lastIndex = 0;
  26866. str = str.replace(/<!--([\s\S]*?)-->/mg,"").replace(/<!DOCTYPE[^\[]*\[[^\]]*\]>/gm,"");
  26867. while((Rn = xlmlregex.exec(str))) switch((Rn[3]=Rn[3].replace(/_.*$/,""))) {
  26868. case 'table': case '工作表': // 9.1.2 <table:table>
  26869. if(Rn[1]==='/') {
  26870. if(range.e.c >= range.s.c && range.e.r >= range.s.r) ws['!ref'] = encode_range(range);
  26871. if(opts.sheetRows > 0 && opts.sheetRows <= range.e.r) {
  26872. ws['!fullref'] = ws['!ref'];
  26873. range.e.r = opts.sheetRows - 1;
  26874. ws['!ref'] = encode_range(range);
  26875. }
  26876. if(merges.length) ws['!merges'] = merges;
  26877. if(rowinfo.length) ws["!rows"] = rowinfo;
  26878. sheetag.name = sheetag['名称'] || sheetag.name;
  26879. if(typeof JSON !== 'undefined') JSON.stringify(sheetag);
  26880. SheetNames.push(sheetag.name);
  26881. Sheets[sheetag.name] = ws;
  26882. intable = false;
  26883. }
  26884. else if(Rn[0].charAt(Rn[0].length-2) !== '/') {
  26885. sheetag = parsexmltag(Rn[0], false);
  26886. R = C = -1;
  26887. range.s.r = range.s.c = 10000000; range.e.r = range.e.c = 0;
  26888. ws = opts.dense ? ([]) : ({}); merges = [];
  26889. rowinfo = [];
  26890. intable = true;
  26891. }
  26892. break;
  26893. case 'table-row-group': // 9.1.9 <table:table-row-group>
  26894. if(Rn[1] === "/") --row_ol; else ++row_ol;
  26895. break;
  26896. case 'table-row': case '行': // 9.1.3 <table:table-row>
  26897. if(Rn[1] === '/') { R+=rowpeat; rowpeat = 1; break; }
  26898. rowtag = parsexmltag(Rn[0], false);
  26899. if(rowtag['行号']) R = rowtag['行号'] - 1; else if(R == -1) R = 0;
  26900. rowpeat = +rowtag['number-rows-repeated'] || 1;
  26901. /* TODO: remove magic */
  26902. if(rowpeat < 10) for(i = 0; i < rowpeat; ++i) if(row_ol > 0) rowinfo[R + i] = {level: row_ol};
  26903. C = -1; break;
  26904. case 'covered-table-cell': // 9.1.5 <table:covered-table-cell>
  26905. if(Rn[1] !== '/') ++C;
  26906. if(opts.sheetStubs) {
  26907. if(opts.dense) { if(!ws[R]) ws[R] = []; ws[R][C] = {t:'z'}; }
  26908. else ws[encode_cell({r:R,c:C})] = {t:'z'};
  26909. }
  26910. textp = ""; textR = [];
  26911. break; /* stub */
  26912. case 'table-cell': case '数据':
  26913. if(Rn[0].charAt(Rn[0].length-2) === '/') {
  26914. ++C;
  26915. ctag = parsexmltag(Rn[0], false);
  26916. colpeat = parseInt(ctag['number-columns-repeated']||"1", 10);
  26917. q = ({t:'z', v:null});
  26918. if(ctag.formula && opts.cellFormula != false) q.f = ods_to_csf_formula(unescapexml(ctag.formula));
  26919. if((ctag['数据类型'] || ctag['value-type']) == "string") {
  26920. q.t = "s"; q.v = unescapexml(ctag['string-value'] || "");
  26921. if(opts.dense) {
  26922. if(!ws[R]) ws[R] = [];
  26923. ws[R][C] = q;
  26924. } else {
  26925. ws[encode_cell({r:R,c:C})] = q;
  26926. }
  26927. }
  26928. C+= colpeat-1;
  26929. } else if(Rn[1]!=='/') {
  26930. ++C;
  26931. colpeat = 1;
  26932. var rptR = rowpeat ? R + rowpeat - 1 : R;
  26933. if(C > range.e.c) range.e.c = C;
  26934. if(C < range.s.c) range.s.c = C;
  26935. if(R < range.s.r) range.s.r = R;
  26936. if(rptR > range.e.r) range.e.r = rptR;
  26937. ctag = parsexmltag(Rn[0], false);
  26938. comments = []; comment = ({});
  26939. q = ({t:ctag['数据类型'] || ctag['value-type'], v:null});
  26940. if(opts.cellFormula) {
  26941. if(ctag.formula) ctag.formula = unescapexml(ctag.formula);
  26942. if(ctag['number-matrix-columns-spanned'] && ctag['number-matrix-rows-spanned']) {
  26943. mR = parseInt(ctag['number-matrix-rows-spanned'],10) || 0;
  26944. mC = parseInt(ctag['number-matrix-columns-spanned'],10) || 0;
  26945. mrange = {s: {r:R,c:C}, e:{r:R + mR-1,c:C + mC-1}};
  26946. q.F = encode_range(mrange);
  26947. arrayf.push([mrange, q.F]);
  26948. }
  26949. if(ctag.formula) q.f = ods_to_csf_formula(ctag.formula);
  26950. else for(i = 0; i < arrayf.length; ++i)
  26951. if(R >= arrayf[i][0].s.r && R <= arrayf[i][0].e.r)
  26952. if(C >= arrayf[i][0].s.c && C <= arrayf[i][0].e.c)
  26953. q.F = arrayf[i][1];
  26954. }
  26955. if(ctag['number-columns-spanned'] || ctag['number-rows-spanned']) {
  26956. mR = parseInt(ctag['number-rows-spanned'],10) || 0;
  26957. mC = parseInt(ctag['number-columns-spanned'],10) || 0;
  26958. mrange = {s: {r:R,c:C}, e:{r:R + mR-1,c:C + mC-1}};
  26959. merges.push(mrange);
  26960. }
  26961. /* 19.675.2 table:number-columns-repeated */
  26962. if(ctag['number-columns-repeated']) colpeat = parseInt(ctag['number-columns-repeated'], 10);
  26963. /* 19.385 office:value-type */
  26964. switch(q.t) {
  26965. case 'boolean': q.t = 'b'; q.v = parsexmlbool(ctag['boolean-value']); break;
  26966. case 'float': q.t = 'n'; q.v = parseFloat(ctag.value); break;
  26967. case 'percentage': q.t = 'n'; q.v = parseFloat(ctag.value); break;
  26968. case 'currency': q.t = 'n'; q.v = parseFloat(ctag.value); break;
  26969. case 'date': q.t = 'd'; q.v = parseDate(ctag['date-value']);
  26970. if(!opts.cellDates) { q.t = 'n'; q.v = datenum(q.v); }
  26971. q.z = 'm/d/yy'; break;
  26972. case 'time': q.t = 'n'; q.v = parse_isodur(ctag['time-value'])/86400; break;
  26973. case 'number': q.t = 'n'; q.v = parseFloat(ctag['数据数值']); break;
  26974. default:
  26975. if(q.t === 'string' || q.t === 'text' || !q.t) {
  26976. q.t = 's';
  26977. if(ctag['string-value'] != null) { textp = unescapexml(ctag['string-value']); textR = []; }
  26978. } else throw new Error('Unsupported value type ' + q.t);
  26979. }
  26980. } else {
  26981. isstub = false;
  26982. if(q.t === 's') {
  26983. q.v = textp || '';
  26984. if(textR.length) q.R = textR;
  26985. isstub = textpidx == 0;
  26986. }
  26987. if(atag.Target) q.l = atag;
  26988. if(comments.length > 0) { q.c = comments; comments = []; }
  26989. if(textp && opts.cellText !== false) q.w = textp;
  26990. if(!isstub || opts.sheetStubs) {
  26991. if(!(opts.sheetRows && opts.sheetRows <= R)) {
  26992. for(var rpt = 0; rpt < rowpeat; ++rpt) {
  26993. colpeat = parseInt(ctag['number-columns-repeated']||"1", 10);
  26994. if(opts.dense) {
  26995. if(!ws[R + rpt]) ws[R + rpt] = [];
  26996. ws[R + rpt][C] = rpt == 0 ? q : dup(q);
  26997. while(--colpeat > 0) ws[R + rpt][C + colpeat] = dup(q);
  26998. } else {
  26999. ws[encode_cell({r:R + rpt,c:C})] = q;
  27000. while(--colpeat > 0) ws[encode_cell({r:R + rpt,c:C + colpeat})] = dup(q);
  27001. }
  27002. if(range.e.c <= C) range.e.c = C;
  27003. }
  27004. }
  27005. }
  27006. colpeat = parseInt(ctag['number-columns-repeated']||"1", 10);
  27007. C += colpeat-1; colpeat = 0;
  27008. q = {};
  27009. textp = ""; textR = [];
  27010. }
  27011. atag = ({});
  27012. break; // 9.1.4 <table:table-cell>
  27013. /* pure state */
  27014. case 'document': // TODO: <office:document> is the root for FODS
  27015. case 'document-content': case '电子表格文档': // 3.1.3.2 <office:document-content>
  27016. case 'spreadsheet': case '主体': // 3.7 <office:spreadsheet>
  27017. case 'scripts': // 3.12 <office:scripts>
  27018. case 'styles': // TODO <office:styles>
  27019. case 'font-face-decls': // 3.14 <office:font-face-decls>
  27020. if(Rn[1]==='/'){if((tmp=state.pop())[0]!==Rn[3]) throw "Bad state: "+tmp;}
  27021. else if(Rn[0].charAt(Rn[0].length-2) !== '/') state.push([Rn[3], true]);
  27022. break;
  27023. case 'annotation': // 14.1 <office:annotation>
  27024. if(Rn[1]==='/'){
  27025. if((tmp=state.pop())[0]!==Rn[3]) throw "Bad state: "+tmp;
  27026. comment.t = textp;
  27027. if(textR.length) comment.R = textR;
  27028. comment.a = creator;
  27029. comments.push(comment);
  27030. }
  27031. else if(Rn[0].charAt(Rn[0].length-2) !== '/') {state.push([Rn[3], false]);}
  27032. creator = ""; creatoridx = 0;
  27033. textp = ""; textpidx = 0; textR = [];
  27034. break;
  27035. case 'creator': // 4.3.2.7 <dc:creator>
  27036. if(Rn[1]==='/') { creator = str.slice(creatoridx,Rn.index); }
  27037. else creatoridx = Rn.index + Rn[0].length;
  27038. break;
  27039. /* ignore state */
  27040. case 'meta': case '元数据': // TODO: <office:meta> <uof:元数据> FODS/UOF
  27041. case 'settings': // TODO: <office:settings>
  27042. case 'config-item-set': // TODO: <office:config-item-set>
  27043. case 'config-item-map-indexed': // TODO: <office:config-item-map-indexed>
  27044. case 'config-item-map-entry': // TODO: <office:config-item-map-entry>
  27045. case 'config-item-map-named': // TODO: <office:config-item-map-entry>
  27046. case 'shapes': // 9.2.8 <table:shapes>
  27047. case 'frame': // 10.4.2 <draw:frame>
  27048. case 'text-box': // 10.4.3 <draw:text-box>
  27049. case 'image': // 10.4.4 <draw:image>
  27050. case 'data-pilot-tables': // 9.6.2 <table:data-pilot-tables>
  27051. case 'list-style': // 16.30 <text:list-style>
  27052. case 'form': // 13.13 <form:form>
  27053. case 'dde-links': // 9.8 <table:dde-links>
  27054. case 'event-listeners': // TODO
  27055. case 'chart': // TODO
  27056. if(Rn[1]==='/'){if((tmp=state.pop())[0]!==Rn[3]) throw "Bad state: "+tmp;}
  27057. else if(Rn[0].charAt(Rn[0].length-2) !== '/') state.push([Rn[3], false]);
  27058. textp = ""; textpidx = 0; textR = [];
  27059. break;
  27060. case 'scientific-number': // TODO: <number:scientific-number>
  27061. break;
  27062. case 'currency-symbol': // TODO: <number:currency-symbol>
  27063. break;
  27064. case 'currency-style': // TODO: <number:currency-style>
  27065. break;
  27066. case 'number-style': // 16.27.2 <number:number-style>
  27067. case 'percentage-style': // 16.27.9 <number:percentage-style>
  27068. case 'date-style': // 16.27.10 <number:date-style>
  27069. case 'time-style': // 16.27.18 <number:time-style>
  27070. if(Rn[1]==='/'){
  27071. number_format_map[NFtag.name] = NF;
  27072. if((tmp=state.pop())[0]!==Rn[3]) throw "Bad state: "+tmp;
  27073. } else if(Rn[0].charAt(Rn[0].length-2) !== '/') {
  27074. NF = "";
  27075. NFtag = parsexmltag(Rn[0], false);
  27076. state.push([Rn[3], true]);
  27077. } break;
  27078. case 'script': break; // 3.13 <office:script>
  27079. case 'libraries': break; // TODO: <ooo:libraries>
  27080. case 'automatic-styles': break; // 3.15.3 <office:automatic-styles>
  27081. case 'master-styles': break; // TODO: <office:master-styles>
  27082. case 'default-style': // TODO: <style:default-style>
  27083. case 'page-layout': break; // TODO: <style:page-layout>
  27084. case 'style': // 16.2 <style:style>
  27085. break;
  27086. case 'map': break; // 16.3 <style:map>
  27087. case 'font-face': break; // 16.21 <style:font-face>
  27088. case 'paragraph-properties': break; // 17.6 <style:paragraph-properties>
  27089. case 'table-properties': break; // 17.15 <style:table-properties>
  27090. case 'table-column-properties': break; // 17.16 <style:table-column-properties>
  27091. case 'table-row-properties': break; // 17.17 <style:table-row-properties>
  27092. case 'table-cell-properties': break; // 17.18 <style:table-cell-properties>
  27093. case 'number': // 16.27.3 <number:number>
  27094. switch(state[state.length-1][0]) {
  27095. case 'time-style':
  27096. case 'date-style':
  27097. tag = parsexmltag(Rn[0], false);
  27098. NF += number_formats[Rn[3]][tag.style==='long'?1:0]; break;
  27099. } break;
  27100. case 'fraction': break; // TODO 16.27.6 <number:fraction>
  27101. case 'day': // 16.27.11 <number:day>
  27102. case 'month': // 16.27.12 <number:month>
  27103. case 'year': // 16.27.13 <number:year>
  27104. case 'era': // 16.27.14 <number:era>
  27105. case 'day-of-week': // 16.27.15 <number:day-of-week>
  27106. case 'week-of-year': // 16.27.16 <number:week-of-year>
  27107. case 'quarter': // 16.27.17 <number:quarter>
  27108. case 'hours': // 16.27.19 <number:hours>
  27109. case 'minutes': // 16.27.20 <number:minutes>
  27110. case 'seconds': // 16.27.21 <number:seconds>
  27111. case 'am-pm': // 16.27.22 <number:am-pm>
  27112. switch(state[state.length-1][0]) {
  27113. case 'time-style':
  27114. case 'date-style':
  27115. tag = parsexmltag(Rn[0], false);
  27116. NF += number_formats[Rn[3]][tag.style==='long'?1:0]; break;
  27117. } break;
  27118. case 'boolean-style': break; // 16.27.23 <number:boolean-style>
  27119. case 'boolean': break; // 16.27.24 <number:boolean>
  27120. case 'text-style': break; // 16.27.25 <number:text-style>
  27121. case 'text': // 16.27.26 <number:text>
  27122. if(Rn[0].slice(-2) === "/>") break;
  27123. else if(Rn[1]==="/") switch(state[state.length-1][0]) {
  27124. case 'number-style':
  27125. case 'date-style':
  27126. case 'time-style':
  27127. NF += str.slice(pidx, Rn.index);
  27128. break;
  27129. }
  27130. else pidx = Rn.index + Rn[0].length;
  27131. break;
  27132. case 'named-range': // 9.4.12 <table:named-range>
  27133. tag = parsexmltag(Rn[0], false);
  27134. _Ref = ods_to_csf_3D(tag['cell-range-address']);
  27135. var nrange = ({Name:tag.name, Ref:_Ref[0] + '!' + _Ref[1]});
  27136. if(intable) nrange.Sheet = SheetNames.length;
  27137. WB.Names.push(nrange);
  27138. break;
  27139. case 'text-content': break; // 16.27.27 <number:text-content>
  27140. case 'text-properties': break; // 16.27.27 <style:text-properties>
  27141. case 'embedded-text': break; // 16.27.4 <number:embedded-text>
  27142. case 'body': case '电子表格': break; // 3.3 16.9.6 19.726.3
  27143. case 'forms': break; // 12.25.2 13.2
  27144. case 'table-column': break; // 9.1.6 <table:table-column>
  27145. case 'table-header-rows': break; // 9.1.7 <table:table-header-rows>
  27146. case 'table-rows': break; // 9.1.12 <table:table-rows>
  27147. /* TODO: outline levels */
  27148. case 'table-column-group': break; // 9.1.10 <table:table-column-group>
  27149. case 'table-header-columns': break; // 9.1.11 <table:table-header-columns>
  27150. case 'table-columns': break; // 9.1.12 <table:table-columns>
  27151. case 'null-date': break; // 9.4.2 <table:null-date> TODO: date1904
  27152. case 'graphic-properties': break; // 17.21 <style:graphic-properties>
  27153. case 'calculation-settings': break; // 9.4.1 <table:calculation-settings>
  27154. case 'named-expressions': break; // 9.4.11 <table:named-expressions>
  27155. case 'label-range': break; // 9.4.9 <table:label-range>
  27156. case 'label-ranges': break; // 9.4.10 <table:label-ranges>
  27157. case 'named-expression': break; // 9.4.13 <table:named-expression>
  27158. case 'sort': break; // 9.4.19 <table:sort>
  27159. case 'sort-by': break; // 9.4.20 <table:sort-by>
  27160. case 'sort-groups': break; // 9.4.22 <table:sort-groups>
  27161. case 'tab': break; // 6.1.4 <text:tab>
  27162. case 'line-break': break; // 6.1.5 <text:line-break>
  27163. case 'span': break; // 6.1.7 <text:span>
  27164. case 'p': case '文本串': // 5.1.3 <text:p>
  27165. if(Rn[1]==='/' && (!ctag || !ctag['string-value'])) {
  27166. var ptp = parse_text_p(str.slice(textpidx,Rn.index), textptag);
  27167. textp = (textp.length > 0 ? textp + "\n" : "") + ptp[0];
  27168. } else { textptag = parsexmltag(Rn[0], false); textpidx = Rn.index + Rn[0].length; }
  27169. break; // <text:p>
  27170. case 's': break; // <text:s>
  27171. case 'database-range': // 9.4.15 <table:database-range>
  27172. if(Rn[1]==='/') break;
  27173. try {
  27174. _Ref = ods_to_csf_3D(parsexmltag(Rn[0])['target-range-address']);
  27175. Sheets[_Ref[0]]['!autofilter'] = { ref:_Ref[1] };
  27176. } catch(e) {/* empty */}
  27177. break;
  27178. case 'date': break; // <*:date>
  27179. case 'object': break; // 10.4.6.2 <draw:object>
  27180. case 'title': case '标题': break; // <*:title> OR <uof:标题>
  27181. case 'desc': break; // <*:desc>
  27182. case 'binary-data': break; // 10.4.5 TODO: b64 blob
  27183. /* 9.2 Advanced Tables */
  27184. case 'table-source': break; // 9.2.6
  27185. case 'scenario': break; // 9.2.6
  27186. case 'iteration': break; // 9.4.3 <table:iteration>
  27187. case 'content-validations': break; // 9.4.4 <table:
  27188. case 'content-validation': break; // 9.4.5 <table:
  27189. case 'help-message': break; // 9.4.6 <table:
  27190. case 'error-message': break; // 9.4.7 <table:
  27191. case 'database-ranges': break; // 9.4.14 <table:database-ranges>
  27192. case 'filter': break; // 9.5.2 <table:filter>
  27193. case 'filter-and': break; // 9.5.3 <table:filter-and>
  27194. case 'filter-or': break; // 9.5.4 <table:filter-or>
  27195. case 'filter-condition': break; // 9.5.5 <table:filter-condition>
  27196. case 'list-level-style-bullet': break; // 16.31 <text:
  27197. case 'list-level-style-number': break; // 16.32 <text:
  27198. case 'list-level-properties': break; // 17.19 <style:
  27199. /* 7.3 Document Fields */
  27200. case 'sender-firstname': // 7.3.6.2
  27201. case 'sender-lastname': // 7.3.6.3
  27202. case 'sender-initials': // 7.3.6.4
  27203. case 'sender-title': // 7.3.6.5
  27204. case 'sender-position': // 7.3.6.6
  27205. case 'sender-email': // 7.3.6.7
  27206. case 'sender-phone-private': // 7.3.6.8
  27207. case 'sender-fax': // 7.3.6.9
  27208. case 'sender-company': // 7.3.6.10
  27209. case 'sender-phone-work': // 7.3.6.11
  27210. case 'sender-street': // 7.3.6.12
  27211. case 'sender-city': // 7.3.6.13
  27212. case 'sender-postal-code': // 7.3.6.14
  27213. case 'sender-country': // 7.3.6.15
  27214. case 'sender-state-or-province': // 7.3.6.16
  27215. case 'author-name': // 7.3.7.1
  27216. case 'author-initials': // 7.3.7.2
  27217. case 'chapter': // 7.3.8
  27218. case 'file-name': // 7.3.9
  27219. case 'template-name': // 7.3.9
  27220. case 'sheet-name': // 7.3.9
  27221. break;
  27222. case 'event-listener':
  27223. break;
  27224. /* TODO: FODS Properties */
  27225. case 'initial-creator':
  27226. case 'creation-date':
  27227. case 'print-date':
  27228. case 'generator':
  27229. case 'document-statistic':
  27230. case 'user-defined':
  27231. case 'editing-duration':
  27232. case 'editing-cycles':
  27233. break;
  27234. /* TODO: FODS Config */
  27235. case 'config-item':
  27236. break;
  27237. /* TODO: style tokens */
  27238. case 'page-number': break; // TODO <text:page-number>
  27239. case 'page-count': break; // TODO <text:page-count>
  27240. case 'time': break; // TODO <text:time>
  27241. /* 9.3 Advanced Table Cells */
  27242. case 'cell-range-source': break; // 9.3.1 <table:
  27243. case 'detective': break; // 9.3.2 <table:
  27244. case 'operation': break; // 9.3.3 <table:
  27245. case 'highlighted-range': break; // 9.3.4 <table:
  27246. /* 9.6 Data Pilot Tables <table: */
  27247. case 'data-pilot-table': // 9.6.3
  27248. case 'source-cell-range': // 9.6.5
  27249. case 'source-service': // 9.6.6
  27250. case 'data-pilot-field': // 9.6.7
  27251. case 'data-pilot-level': // 9.6.8
  27252. case 'data-pilot-subtotals': // 9.6.9
  27253. case 'data-pilot-subtotal': // 9.6.10
  27254. case 'data-pilot-members': // 9.6.11
  27255. case 'data-pilot-member': // 9.6.12
  27256. case 'data-pilot-display-info': // 9.6.13
  27257. case 'data-pilot-sort-info': // 9.6.14
  27258. case 'data-pilot-layout-info': // 9.6.15
  27259. case 'data-pilot-field-reference': // 9.6.16
  27260. case 'data-pilot-groups': // 9.6.17
  27261. case 'data-pilot-group': // 9.6.18
  27262. case 'data-pilot-group-member': // 9.6.19
  27263. break;
  27264. /* 10.3 Drawing Shapes */
  27265. case 'rect': // 10.3.2
  27266. break;
  27267. /* 14.6 DDE Connections */
  27268. case 'dde-connection-decls': // 14.6.2 <text:
  27269. case 'dde-connection-decl': // 14.6.3 <text:
  27270. case 'dde-link': // 14.6.4 <table:
  27271. case 'dde-source': // 14.6.5 <office:
  27272. break;
  27273. case 'properties': break; // 13.7 <form:properties>
  27274. case 'property': break; // 13.8 <form:property>
  27275. case 'a': // 6.1.8 hyperlink
  27276. if(Rn[1]!== '/') {
  27277. atag = parsexmltag(Rn[0], false);
  27278. if(!atag.href) break;
  27279. atag.Target = atag.href; delete atag.href;
  27280. if(atag.Target.charAt(0) == "#" && atag.Target.indexOf(".") > -1) {
  27281. _Ref = ods_to_csf_3D(atag.Target.slice(1));
  27282. atag.Target = "#" + _Ref[0] + "!" + _Ref[1];
  27283. }
  27284. }
  27285. break;
  27286. /* non-standard */
  27287. case 'table-protection': break;
  27288. case 'data-pilot-grand-total': break; // <table:
  27289. case 'office-document-common-attrs': break; // bare
  27290. default: switch(Rn[2]) {
  27291. case 'dc:': // TODO: properties
  27292. case 'calcext:': // ignore undocumented extensions
  27293. case 'loext:': // ignore undocumented extensions
  27294. case 'ooo:': // ignore undocumented extensions
  27295. case 'chartooo:': // ignore undocumented extensions
  27296. case 'draw:': // TODO: drawing
  27297. case 'style:': // TODO: styles
  27298. case 'chart:': // TODO: charts
  27299. case 'form:': // TODO: forms
  27300. case 'uof:': // TODO: uof
  27301. case '表:': // TODO: uof
  27302. case '字:': // TODO: uof
  27303. break;
  27304. default: if(opts.WTF) throw new Error(Rn);
  27305. }
  27306. }
  27307. var out = ({
  27308. Sheets: Sheets,
  27309. SheetNames: SheetNames,
  27310. Workbook: WB
  27311. });
  27312. if(opts.bookSheets) delete out.Sheets;
  27313. return out;
  27314. };
  27315. })();
  27316. function parse_ods(zip, opts) {
  27317. opts = opts || ({});
  27318. var ods = !!safegetzipfile(zip, 'objectdata');
  27319. if(ods) parse_manifest(getzipdata(zip, 'META-INF/manifest.xml'), opts);
  27320. var content = getzipstr(zip, 'content.xml');
  27321. if(!content) throw new Error("Missing content.xml in " + (ods ? "ODS" : "UOF")+ " file");
  27322. var wb = parse_content_xml(ods ? content : utf8read(content), opts);
  27323. if(safegetzipfile(zip, 'meta.xml')) wb.Props = parse_core_props(getzipdata(zip, 'meta.xml'));
  27324. return wb;
  27325. }
  27326. function parse_fods(data, opts) {
  27327. return parse_content_xml(data, opts);
  27328. }
  27329. /* OpenDocument */
  27330. var write_styles_ods = (function() {
  27331. var payload = '<office:document-styles ' + wxt_helper({
  27332. 'xmlns:office': "urn:oasis:names:tc:opendocument:xmlns:office:1.0",
  27333. 'xmlns:table': "urn:oasis:names:tc:opendocument:xmlns:table:1.0",
  27334. 'xmlns:style': "urn:oasis:names:tc:opendocument:xmlns:style:1.0",
  27335. 'xmlns:text': "urn:oasis:names:tc:opendocument:xmlns:text:1.0",
  27336. 'xmlns:draw': "urn:oasis:names:tc:opendocument:xmlns:drawing:1.0",
  27337. 'xmlns:fo': "urn:oasis:names:tc:opendocument:xmlns:xsl-fo-compatible:1.0",
  27338. 'xmlns:xlink': "http://www.w3.org/1999/xlink",
  27339. 'xmlns:dc': "http://purl.org/dc/elements/1.1/",
  27340. 'xmlns:number': "urn:oasis:names:tc:opendocument:xmlns:datastyle:1.0",
  27341. 'xmlns:svg': "urn:oasis:names:tc:opendocument:xmlns:svg-compatible:1.0",
  27342. 'xmlns:of': "urn:oasis:names:tc:opendocument:xmlns:of:1.2",
  27343. 'office:version': "1.2"
  27344. }) + '></office:document-styles>';
  27345. return function wso() {
  27346. return XML_HEADER + payload;
  27347. };
  27348. })();
  27349. var write_content_ods = (function() {
  27350. /* 6.1.2 White Space Characters */
  27351. var write_text_p = function(text) {
  27352. return escapexml(text)
  27353. .replace(/ +/g, function($$){return '<text:s text:c="'+$$.length+'"/>';})
  27354. .replace(/\t/g, "<text:tab/>")
  27355. .replace(/\n/g, "<text:line-break/>")
  27356. .replace(/^ /, "<text:s/>").replace(/ $/, "<text:s/>");
  27357. };
  27358. var null_cell_xml = ' <table:table-cell />\n';
  27359. var covered_cell_xml = ' <table:covered-table-cell/>\n';
  27360. var write_ws = function(ws, wb, i) {
  27361. /* Section 9 Tables */
  27362. var o = [];
  27363. o.push(' <table:table table:name="' + escapexml(wb.SheetNames[i]) + '">\n');
  27364. var R=0,C=0, range = decode_range(ws['!ref']);
  27365. var marr = ws['!merges'] || [], mi = 0;
  27366. var dense = Array.isArray(ws);
  27367. for(R = 0; R < range.s.r; ++R) o.push(' <table:table-row></table:table-row>\n');
  27368. for(; R <= range.e.r; ++R) {
  27369. o.push(' <table:table-row>\n');
  27370. for(C=0; C < range.s.c; ++C) o.push(null_cell_xml);
  27371. for(; C <= range.e.c; ++C) {
  27372. var skip = false, ct = {}, textp = "";
  27373. for(mi = 0; mi != marr.length; ++mi) {
  27374. if(marr[mi].s.c > C) continue;
  27375. if(marr[mi].s.r > R) continue;
  27376. if(marr[mi].e.c < C) continue;
  27377. if(marr[mi].e.r < R) continue;
  27378. if(marr[mi].s.c != C || marr[mi].s.r != R) skip = true;
  27379. ct['table:number-columns-spanned'] = (marr[mi].e.c - marr[mi].s.c + 1);
  27380. ct['table:number-rows-spanned'] = (marr[mi].e.r - marr[mi].s.r + 1);
  27381. break;
  27382. }
  27383. if(skip) { o.push(covered_cell_xml); continue; }
  27384. var ref = encode_cell({r:R, c:C}), cell = dense ? (ws[R]||[])[C]: ws[ref];
  27385. if(cell && cell.f) {
  27386. ct['table:formula'] = escapexml(csf_to_ods_formula(cell.f));
  27387. if(cell.F) {
  27388. if(cell.F.slice(0, ref.length) == ref) {
  27389. var _Fref = decode_range(cell.F);
  27390. ct['table:number-matrix-columns-spanned'] = (_Fref.e.c - _Fref.s.c + 1);
  27391. ct['table:number-matrix-rows-spanned'] = (_Fref.e.r - _Fref.s.r + 1);
  27392. }
  27393. }
  27394. }
  27395. if(!cell) { o.push(null_cell_xml); continue; }
  27396. switch(cell.t) {
  27397. case 'b':
  27398. textp = (cell.v ? 'TRUE' : 'FALSE');
  27399. ct['office:value-type'] = "boolean";
  27400. ct['office:boolean-value'] = (cell.v ? 'true' : 'false');
  27401. break;
  27402. case 'n':
  27403. textp = (cell.w||String(cell.v||0));
  27404. ct['office:value-type'] = "float";
  27405. ct['office:value'] = (cell.v||0);
  27406. break;
  27407. case 's': case 'str':
  27408. textp = cell.v;
  27409. ct['office:value-type'] = "string";
  27410. break;
  27411. case 'd':
  27412. textp = (cell.w||(parseDate(cell.v).toISOString()));
  27413. ct['office:value-type'] = "date";
  27414. ct['office:date-value'] = (parseDate(cell.v).toISOString());
  27415. ct['table:style-name'] = "ce1";
  27416. break;
  27417. //case 'e':
  27418. default: o.push(null_cell_xml); continue;
  27419. }
  27420. var text_p = write_text_p(textp);
  27421. if(cell.l && cell.l.Target) {
  27422. var _tgt = cell.l.Target; _tgt = _tgt.charAt(0) == "#" ? "#" + csf_to_ods_3D(_tgt.slice(1)) : _tgt;
  27423. text_p = writextag('text:a', text_p, {'xlink:href': _tgt});
  27424. }
  27425. o.push(' ' + writextag('table:table-cell', writextag('text:p', text_p, {}), ct) + '\n');
  27426. }
  27427. o.push(' </table:table-row>\n');
  27428. }
  27429. o.push(' </table:table>\n');
  27430. return o.join("");
  27431. };
  27432. var write_automatic_styles_ods = function(o) {
  27433. o.push(' <office:automatic-styles>\n');
  27434. o.push(' <number:date-style style:name="N37" number:automatic-order="true">\n');
  27435. o.push(' <number:month number:style="long"/>\n');
  27436. o.push(' <number:text>/</number:text>\n');
  27437. o.push(' <number:day number:style="long"/>\n');
  27438. o.push(' <number:text>/</number:text>\n');
  27439. o.push(' <number:year/>\n');
  27440. o.push(' </number:date-style>\n');
  27441. o.push(' <style:style style:name="ce1" style:family="table-cell" style:parent-style-name="Default" style:data-style-name="N37"/>\n');
  27442. o.push(' </office:automatic-styles>\n');
  27443. };
  27444. return function wcx(wb, opts) {
  27445. var o = [XML_HEADER];
  27446. /* 3.1.3.2 */
  27447. var attr = wxt_helper({
  27448. 'xmlns:office': "urn:oasis:names:tc:opendocument:xmlns:office:1.0",
  27449. 'xmlns:table': "urn:oasis:names:tc:opendocument:xmlns:table:1.0",
  27450. 'xmlns:style': "urn:oasis:names:tc:opendocument:xmlns:style:1.0",
  27451. 'xmlns:text': "urn:oasis:names:tc:opendocument:xmlns:text:1.0",
  27452. 'xmlns:draw': "urn:oasis:names:tc:opendocument:xmlns:drawing:1.0",
  27453. 'xmlns:fo': "urn:oasis:names:tc:opendocument:xmlns:xsl-fo-compatible:1.0",
  27454. 'xmlns:xlink': "http://www.w3.org/1999/xlink",
  27455. 'xmlns:dc': "http://purl.org/dc/elements/1.1/",
  27456. 'xmlns:meta': "urn:oasis:names:tc:opendocument:xmlns:meta:1.0",
  27457. 'xmlns:number': "urn:oasis:names:tc:opendocument:xmlns:datastyle:1.0",
  27458. 'xmlns:presentation': "urn:oasis:names:tc:opendocument:xmlns:presentation:1.0",
  27459. 'xmlns:svg': "urn:oasis:names:tc:opendocument:xmlns:svg-compatible:1.0",
  27460. 'xmlns:chart': "urn:oasis:names:tc:opendocument:xmlns:chart:1.0",
  27461. 'xmlns:dr3d': "urn:oasis:names:tc:opendocument:xmlns:dr3d:1.0",
  27462. 'xmlns:math': "http://www.w3.org/1998/Math/MathML",
  27463. 'xmlns:form': "urn:oasis:names:tc:opendocument:xmlns:form:1.0",
  27464. 'xmlns:script': "urn:oasis:names:tc:opendocument:xmlns:script:1.0",
  27465. 'xmlns:ooo': "http://openoffice.org/2004/office",
  27466. 'xmlns:ooow': "http://openoffice.org/2004/writer",
  27467. 'xmlns:oooc': "http://openoffice.org/2004/calc",
  27468. 'xmlns:dom': "http://www.w3.org/2001/xml-events",
  27469. 'xmlns:xforms': "http://www.w3.org/2002/xforms",
  27470. 'xmlns:xsd': "http://www.w3.org/2001/XMLSchema",
  27471. 'xmlns:xsi': "http://www.w3.org/2001/XMLSchema-instance",
  27472. 'xmlns:sheet': "urn:oasis:names:tc:opendocument:sh33tjs:1.0",
  27473. 'xmlns:rpt': "http://openoffice.org/2005/report",
  27474. 'xmlns:of': "urn:oasis:names:tc:opendocument:xmlns:of:1.2",
  27475. 'xmlns:xhtml': "http://www.w3.org/1999/xhtml",
  27476. 'xmlns:grddl': "http://www.w3.org/2003/g/data-view#",
  27477. 'xmlns:tableooo': "http://openoffice.org/2009/table",
  27478. 'xmlns:drawooo': "http://openoffice.org/2010/draw",
  27479. 'xmlns:calcext': "urn:org:documentfoundation:names:experimental:calc:xmlns:calcext:1.0",
  27480. 'xmlns:loext': "urn:org:documentfoundation:names:experimental:office:xmlns:loext:1.0",
  27481. 'xmlns:field': "urn:openoffice:names:experimental:ooo-ms-interop:xmlns:field:1.0",
  27482. 'xmlns:formx': "urn:openoffice:names:experimental:ooxml-odf-interop:xmlns:form:1.0",
  27483. 'xmlns:css3t': "http://www.w3.org/TR/css3-text/",
  27484. 'office:version': "1.2"
  27485. });
  27486. var fods = wxt_helper({
  27487. 'xmlns:config': "urn:oasis:names:tc:opendocument:xmlns:config:1.0",
  27488. 'office:mimetype': "application/vnd.oasis.opendocument.spreadsheet"
  27489. });
  27490. if(opts.bookType == "fods") o.push('<office:document' + attr + fods + '>\n');
  27491. else o.push('<office:document-content' + attr + '>\n');
  27492. write_automatic_styles_ods(o);
  27493. o.push(' <office:body>\n');
  27494. o.push(' <office:spreadsheet>\n');
  27495. for(var i = 0; i != wb.SheetNames.length; ++i) o.push(write_ws(wb.Sheets[wb.SheetNames[i]], wb, i, opts));
  27496. o.push(' </office:spreadsheet>\n');
  27497. o.push(' </office:body>\n');
  27498. if(opts.bookType == "fods") o.push('</office:document>');
  27499. else o.push('</office:document-content>');
  27500. return o.join("");
  27501. };
  27502. })();
  27503. function write_ods(wb, opts) {
  27504. if(opts.bookType == "fods") return write_content_ods(wb, opts);
  27505. var zip = new jszip();
  27506. var f = "";
  27507. var manifest = [];
  27508. var rdf = [];
  27509. /* Part 3 Section 3.3 MIME Media Type */
  27510. f = "mimetype";
  27511. zip.file(f, "application/vnd.oasis.opendocument.spreadsheet");
  27512. /* Part 1 Section 2.2 Documents */
  27513. f = "content.xml";
  27514. zip.file(f, write_content_ods(wb, opts));
  27515. manifest.push([f, "text/xml"]);
  27516. rdf.push([f, "ContentFile"]);
  27517. /* TODO: these are hard-coded styles to satiate excel */
  27518. f = "styles.xml";
  27519. zip.file(f, write_styles_ods(wb, opts));
  27520. manifest.push([f, "text/xml"]);
  27521. rdf.push([f, "StylesFile"]);
  27522. /* TODO: this is hard-coded to satiate excel */
  27523. f = "meta.xml";
  27524. zip.file(f, write_meta_ods());
  27525. manifest.push([f, "text/xml"]);
  27526. rdf.push([f, "MetadataFile"]);
  27527. /* Part 3 Section 6 Metadata Manifest File */
  27528. f = "manifest.rdf";
  27529. zip.file(f, write_rdf(rdf/*, opts*/));
  27530. manifest.push([f, "application/rdf+xml"]);
  27531. /* Part 3 Section 4 Manifest File */
  27532. f = "META-INF/manifest.xml";
  27533. zip.file(f, write_manifest(manifest/*, opts*/));
  27534. return zip;
  27535. }
  27536. function write_sheet_index(wb, sheet) {
  27537. if(!sheet) return 0;
  27538. var idx = wb.SheetNames.indexOf(sheet);
  27539. if(idx == -1) throw new Error("Sheet not found: " + sheet);
  27540. return idx;
  27541. }
  27542. function write_obj_str(factory) {
  27543. return function write_str(wb, o) {
  27544. var idx = write_sheet_index(wb, o.sheet);
  27545. return factory.from_sheet(wb.Sheets[wb.SheetNames[idx]], o, wb);
  27546. };
  27547. }
  27548. var write_htm_str = write_obj_str(HTML_);
  27549. var write_csv_str = write_obj_str({from_sheet:sheet_to_csv});
  27550. var write_slk_str = write_obj_str(SYLK);
  27551. var write_dif_str = write_obj_str(DIF);
  27552. var write_prn_str = write_obj_str(PRN);
  27553. var write_rtf_str = write_obj_str(RTF);
  27554. var write_txt_str = write_obj_str({from_sheet:sheet_to_txt});
  27555. var write_dbf_buf = write_obj_str(DBF);
  27556. var write_eth_str = write_obj_str(ETH);
  27557. function fix_opts_func(defaults) {
  27558. return function fix_opts(opts) {
  27559. for(var i = 0; i != defaults.length; ++i) {
  27560. var d = defaults[i];
  27561. if(opts[d[0]] === undefined) opts[d[0]] = d[1];
  27562. if(d[2] === 'n') opts[d[0]] = Number(opts[d[0]]);
  27563. }
  27564. };
  27565. }
  27566. var fix_read_opts = fix_opts_func([
  27567. ['cellNF', false], /* emit cell number format string as .z */
  27568. ['cellHTML', true], /* emit html string as .h */
  27569. ['cellFormula', true], /* emit formulae as .f */
  27570. ['cellStyles', false], /* emits style/theme as .s */
  27571. ['cellText', true], /* emit formatted text as .w */
  27572. ['cellDates', false], /* emit date cells with type `d` */
  27573. ['sheetStubs', false], /* emit empty cells */
  27574. ['sheetRows', 0, 'n'], /* read n rows (0 = read all rows) */
  27575. ['bookDeps', false], /* parse calculation chains */
  27576. ['bookSheets', false], /* only try to get sheet names (no Sheets) */
  27577. ['bookProps', false], /* only try to get properties (no Sheets) */
  27578. ['bookFiles', false], /* include raw file structure (keys, files, cfb) */
  27579. ['bookVBA', false], /* include vba raw data (vbaraw) */
  27580. ['password',''], /* password */
  27581. ['WTF', false] /* WTF mode (throws errors) */
  27582. ]);
  27583. var fix_write_opts = fix_opts_func([
  27584. ['cellDates', false], /* write date cells with type `d` */
  27585. ['bookSST', false], /* Generate Shared String Table */
  27586. ['bookType', 'xlsx'], /* Type of workbook (xlsx/m/b) */
  27587. ['compression', false], /* Use file compression */
  27588. ['WTF', false] /* WTF mode (throws errors) */
  27589. ]);
  27590. function get_sheet_type(n) {
  27591. if(RELS.WS.indexOf(n) > -1) return "sheet";
  27592. if(RELS.CS && n == RELS.CS) return "chart";
  27593. if(RELS.DS && n == RELS.DS) return "dialog";
  27594. if(RELS.MS && n == RELS.MS) return "macro";
  27595. return (n && n.length) ? n : "sheet";
  27596. }
  27597. function safe_parse_wbrels(wbrels, sheets) {
  27598. if(!wbrels) return 0;
  27599. try {
  27600. wbrels = sheets.map(function pwbr(w) { if(!w.id) w.id = w.strRelID; return [w.name, wbrels['!id'][w.id].Target, get_sheet_type(wbrels['!id'][w.id].Type)]; });
  27601. } catch(e) { return null; }
  27602. return !wbrels || wbrels.length === 0 ? null : wbrels;
  27603. }
  27604. function safe_parse_sheet(zip, path, relsPath, sheet, idx, sheetRels, sheets, stype, opts, wb, themes, styles) {
  27605. try {
  27606. sheetRels[sheet]=parse_rels(getzipstr(zip, relsPath, true), path);
  27607. var data = getzipdata(zip, path);
  27608. var _ws;
  27609. switch(stype) {
  27610. case 'sheet': _ws = parse_ws(data, path, idx, opts, sheetRels[sheet], wb, themes, styles); break;
  27611. case 'chart': _ws = parse_cs(data, path, idx, opts, sheetRels[sheet], wb, themes, styles);
  27612. if(!_ws || !_ws['!chart']) break;
  27613. var dfile = resolve_path(_ws['!chart'].Target, path);
  27614. var drelsp = get_rels_path(dfile);
  27615. var draw = parse_drawing(getzipstr(zip, dfile, true), parse_rels(getzipstr(zip, drelsp, true), dfile));
  27616. var chartp = resolve_path(draw, dfile);
  27617. var crelsp = get_rels_path(chartp);
  27618. _ws = parse_chart(getzipstr(zip, chartp, true), chartp, opts, parse_rels(getzipstr(zip, crelsp, true), chartp), wb, _ws);
  27619. break;
  27620. case 'macro': _ws = parse_ms(data, path, idx, opts, sheetRels[sheet], wb, themes, styles); break;
  27621. case 'dialog': _ws = parse_ds(data, path, idx, opts, sheetRels[sheet], wb, themes, styles); break;
  27622. }
  27623. sheets[sheet] = _ws;
  27624. } catch(e) { if(opts.WTF) throw e; }
  27625. }
  27626. function strip_front_slash(x) { return x.charAt(0) == '/' ? x.slice(1) : x; }
  27627. function parse_zip(zip, opts) {
  27628. make_ssf(SSF);
  27629. opts = opts || {};
  27630. fix_read_opts(opts);
  27631. /* OpenDocument Part 3 Section 2.2.1 OpenDocument Package */
  27632. if(safegetzipfile(zip, 'META-INF/manifest.xml')) return parse_ods(zip, opts);
  27633. /* UOC */
  27634. if(safegetzipfile(zip, 'objectdata.xml')) return parse_ods(zip, opts);
  27635. /* Numbers */
  27636. if(safegetzipfile(zip, 'Index/Document.iwa')) throw new Error('Unsupported NUMBERS file');
  27637. var entries = zipentries(zip);
  27638. var dir = parse_ct((getzipstr(zip, '[Content_Types].xml')));
  27639. var xlsb = false;
  27640. var sheets, binname;
  27641. if(dir.workbooks.length === 0) {
  27642. binname = "xl/workbook.xml";
  27643. if(getzipdata(zip,binname, true)) dir.workbooks.push(binname);
  27644. }
  27645. if(dir.workbooks.length === 0) {
  27646. binname = "xl/workbook.bin";
  27647. if(!getzipdata(zip,binname,true)) throw new Error("Could not find workbook");
  27648. dir.workbooks.push(binname);
  27649. xlsb = true;
  27650. }
  27651. if(dir.workbooks[0].slice(-3) == "bin") xlsb = true;
  27652. var themes = ({});
  27653. var styles = ({});
  27654. if(!opts.bookSheets && !opts.bookProps) {
  27655. strs = [];
  27656. if(dir.sst) try { strs=parse_sst(getzipdata(zip, strip_front_slash(dir.sst)), dir.sst, opts); } catch(e) { if(opts.WTF) throw e; }
  27657. if(opts.cellStyles && dir.themes.length) themes = parse_theme(getzipstr(zip, dir.themes[0].replace(/^\//,''), true)||"",dir.themes[0], opts);
  27658. if(dir.style) styles = parse_sty(getzipdata(zip, strip_front_slash(dir.style)), dir.style, themes, opts);
  27659. }
  27660. /*var externbooks = */dir.links.map(function(link) {
  27661. return parse_xlink(getzipdata(zip, strip_front_slash(link)), link, opts);
  27662. });
  27663. var wb = parse_wb(getzipdata(zip, strip_front_slash(dir.workbooks[0])), dir.workbooks[0], opts);
  27664. var props = {}, propdata = "";
  27665. if(dir.coreprops.length) {
  27666. propdata = getzipdata(zip, strip_front_slash(dir.coreprops[0]), true);
  27667. if(propdata) props = parse_core_props(propdata);
  27668. if(dir.extprops.length !== 0) {
  27669. propdata = getzipdata(zip, strip_front_slash(dir.extprops[0]), true);
  27670. if(propdata) parse_ext_props(propdata, props, opts);
  27671. }
  27672. }
  27673. var custprops = {};
  27674. if(!opts.bookSheets || opts.bookProps) {
  27675. if (dir.custprops.length !== 0) {
  27676. propdata = getzipstr(zip, strip_front_slash(dir.custprops[0]), true);
  27677. if(propdata) custprops = parse_cust_props(propdata, opts);
  27678. }
  27679. }
  27680. var out = ({});
  27681. if(opts.bookSheets || opts.bookProps) {
  27682. if(wb.Sheets) sheets = wb.Sheets.map(function pluck(x){ return x.name; });
  27683. else if(props.Worksheets && props.SheetNames.length > 0) sheets=props.SheetNames;
  27684. if(opts.bookProps) { out.Props = props; out.Custprops = custprops; }
  27685. if(opts.bookSheets && typeof sheets !== 'undefined') out.SheetNames = sheets;
  27686. if(opts.bookSheets ? out.SheetNames : opts.bookProps) return out;
  27687. }
  27688. sheets = {};
  27689. var deps = {};
  27690. if(opts.bookDeps && dir.calcchain) deps=parse_cc(getzipdata(zip, strip_front_slash(dir.calcchain)),dir.calcchain,opts);
  27691. var i=0;
  27692. var sheetRels = ({});
  27693. var path, relsPath;
  27694. {
  27695. var wbsheets = wb.Sheets;
  27696. props.Worksheets = wbsheets.length;
  27697. props.SheetNames = [];
  27698. for(var j = 0; j != wbsheets.length; ++j) {
  27699. props.SheetNames[j] = wbsheets[j].name;
  27700. }
  27701. }
  27702. var wbext = xlsb ? "bin" : "xml";
  27703. var wbrelsi = dir.workbooks[0].lastIndexOf("/");
  27704. var wbrelsfile = (dir.workbooks[0].slice(0, wbrelsi+1) + "_rels/" + dir.workbooks[0].slice(wbrelsi+1) + ".rels").replace(/^\//,"");
  27705. if(!safegetzipfile(zip, wbrelsfile)) wbrelsfile = 'xl/_rels/workbook.' + wbext + '.rels';
  27706. var wbrels = parse_rels(getzipstr(zip, wbrelsfile, true), wbrelsfile);
  27707. if(wbrels) wbrels = safe_parse_wbrels(wbrels, wb.Sheets);
  27708. /* Numbers iOS hack */
  27709. var nmode = (getzipdata(zip,"xl/worksheets/sheet.xml",true))?1:0;
  27710. for(i = 0; i != props.Worksheets; ++i) {
  27711. var stype = "sheet";
  27712. if(wbrels && wbrels[i]) {
  27713. path = 'xl/' + (wbrels[i][1]).replace(/[\/]?xl\//, "");
  27714. if(!safegetzipfile(zip, path)) path = wbrels[i][1];
  27715. if(!safegetzipfile(zip, path)) path = wbrelsfile.replace(/_rels\/.*$/,"") + wbrels[i][1];
  27716. stype = wbrels[i][2];
  27717. } else {
  27718. path = 'xl/worksheets/sheet'+(i+1-nmode)+"." + wbext;
  27719. path = path.replace(/sheet0\./,"sheet.");
  27720. }
  27721. relsPath = path.replace(/^(.*)(\/)([^\/]*)$/, "$1/_rels/$3.rels");
  27722. safe_parse_sheet(zip, path, relsPath, props.SheetNames[i], i, sheetRels, sheets, stype, opts, wb, themes, styles);
  27723. }
  27724. if(dir.comments) parse_comments(zip, dir.comments, sheets, sheetRels, opts);
  27725. out = ({
  27726. Directory: dir,
  27727. Workbook: wb,
  27728. Props: props,
  27729. Custprops: custprops,
  27730. Deps: deps,
  27731. Sheets: sheets,
  27732. SheetNames: props.SheetNames,
  27733. Strings: strs,
  27734. Styles: styles,
  27735. Themes: themes,
  27736. SSF: SSF.get_table()
  27737. });
  27738. if(opts.bookFiles) {
  27739. out.keys = entries;
  27740. out.files = zip.files;
  27741. }
  27742. if(opts.bookVBA) {
  27743. if(dir.vba.length > 0) out.vbaraw = getzipdata(zip,strip_front_slash(dir.vba[0]),true);
  27744. else if(dir.defaults && dir.defaults.bin === CT_VBA) out.vbaraw = getzipdata(zip, 'xl/vbaProject.bin',true);
  27745. }
  27746. return out;
  27747. }
  27748. /* [MS-OFFCRYPTO] 2.1.1 */
  27749. function parse_xlsxcfb(cfb, _opts) {
  27750. var opts = _opts || {};
  27751. var f = 'Workbook', data = CFB.find(cfb, f);
  27752. try {
  27753. f = '/!DataSpaces/Version';
  27754. data = CFB.find(cfb, f); if(!data || !data.content) throw new Error("ECMA-376 Encrypted file missing " + f);
  27755. /*var version = */parse_DataSpaceVersionInfo(data.content);
  27756. /* 2.3.4.1 */
  27757. f = '/!DataSpaces/DataSpaceMap';
  27758. data = CFB.find(cfb, f); if(!data || !data.content) throw new Error("ECMA-376 Encrypted file missing " + f);
  27759. var dsm = parse_DataSpaceMap(data.content);
  27760. if(dsm.length !== 1 || dsm[0].comps.length !== 1 || dsm[0].comps[0].t !== 0 || dsm[0].name !== "StrongEncryptionDataSpace" || dsm[0].comps[0].v !== "EncryptedPackage")
  27761. throw new Error("ECMA-376 Encrypted file bad " + f);
  27762. /* 2.3.4.2 */
  27763. f = '/!DataSpaces/DataSpaceInfo/StrongEncryptionDataSpace';
  27764. data = CFB.find(cfb, f); if(!data || !data.content) throw new Error("ECMA-376 Encrypted file missing " + f);
  27765. var seds = parse_DataSpaceDefinition(data.content);
  27766. if(seds.length != 1 || seds[0] != "StrongEncryptionTransform")
  27767. throw new Error("ECMA-376 Encrypted file bad " + f);
  27768. /* 2.3.4.3 */
  27769. f = '/!DataSpaces/TransformInfo/StrongEncryptionTransform/!Primary';
  27770. data = CFB.find(cfb, f); if(!data || !data.content) throw new Error("ECMA-376 Encrypted file missing " + f);
  27771. /*var hdr = */parse_Primary(data.content);
  27772. } catch(e) {}
  27773. f = '/EncryptionInfo';
  27774. data = CFB.find(cfb, f); if(!data || !data.content) throw new Error("ECMA-376 Encrypted file missing " + f);
  27775. var einfo = parse_EncryptionInfo(data.content);
  27776. /* 2.3.4.4 */
  27777. f = '/EncryptedPackage';
  27778. data = CFB.find(cfb, f); if(!data || !data.content) throw new Error("ECMA-376 Encrypted file missing " + f);
  27779. /*global decrypt_agile */
  27780. if(einfo[0] == 0x04 && typeof decrypt_agile !== 'undefined') return decrypt_agile(einfo[1], data.content, opts.password || "", opts);
  27781. /*global decrypt_std76 */
  27782. if(einfo[0] == 0x02 && typeof decrypt_std76 !== 'undefined') return decrypt_std76(einfo[1], data.content, opts.password || "", opts);
  27783. throw new Error("File is password-protected");
  27784. }
  27785. function write_zip(wb, opts) {
  27786. _shapeid = 1024;
  27787. if(opts.bookType == "ods") return write_ods(wb, opts);
  27788. if(wb && !wb.SSF) {
  27789. wb.SSF = SSF.get_table();
  27790. }
  27791. if(wb && wb.SSF) {
  27792. make_ssf(SSF); SSF.load_table(wb.SSF);
  27793. // $FlowIgnore
  27794. opts.revssf = evert_num(wb.SSF); opts.revssf[wb.SSF[65535]] = 0;
  27795. opts.ssf = wb.SSF;
  27796. }
  27797. opts.rels = {}; opts.wbrels = {};
  27798. opts.Strings = []; opts.Strings.Count = 0; opts.Strings.Unique = 0;
  27799. if(browser_has_Map) opts.revStrings = new Map();
  27800. else { opts.revStrings = {}; opts.revStrings.foo = []; delete opts.revStrings.foo; }
  27801. var wbext = opts.bookType == "xlsb" ? "bin" : "xml";
  27802. var vbafmt = VBAFMTS.indexOf(opts.bookType) > -1;
  27803. var ct = new_ct();
  27804. fix_write_opts(opts = opts || {});
  27805. var zip = new jszip();
  27806. var f = "", rId = 0;
  27807. opts.cellXfs = [];
  27808. get_cell_style(opts.cellXfs, {}, {revssf:{"General":0}});
  27809. if(!wb.Props) wb.Props = {};
  27810. f = "docProps/core.xml";
  27811. zip.file(f, write_core_props(wb.Props, opts));
  27812. ct.coreprops.push(f);
  27813. add_rels(opts.rels, 2, f, RELS.CORE_PROPS);
  27814. f = "docProps/app.xml";
  27815. if(wb.Props && wb.Props.SheetNames){/* empty */}
  27816. else if(!wb.Workbook || !wb.Workbook.Sheets) wb.Props.SheetNames = wb.SheetNames;
  27817. else {
  27818. var _sn = [];
  27819. for(var _i = 0; _i < wb.SheetNames.length; ++_i)
  27820. if((wb.Workbook.Sheets[_i]||{}).Hidden != 2) _sn.push(wb.SheetNames[_i]);
  27821. wb.Props.SheetNames = _sn;
  27822. }
  27823. wb.Props.Worksheets = wb.Props.SheetNames.length;
  27824. zip.file(f, write_ext_props(wb.Props, opts));
  27825. ct.extprops.push(f);
  27826. add_rels(opts.rels, 3, f, RELS.EXT_PROPS);
  27827. if(wb.Custprops !== wb.Props && keys(wb.Custprops||{}).length > 0) {
  27828. f = "docProps/custom.xml";
  27829. zip.file(f, write_cust_props(wb.Custprops, opts));
  27830. ct.custprops.push(f);
  27831. add_rels(opts.rels, 4, f, RELS.CUST_PROPS);
  27832. }
  27833. for(rId=1;rId <= wb.SheetNames.length; ++rId) {
  27834. var wsrels = {'!id':{}};
  27835. var ws = wb.Sheets[wb.SheetNames[rId-1]];
  27836. var _type = (ws || {})["!type"] || "sheet";
  27837. switch(_type) {
  27838. case "chart": /*
  27839. f = "xl/chartsheets/sheet" + rId + "." + wbext;
  27840. zip.file(f, write_cs(rId-1, f, opts, wb, wsrels));
  27841. ct.charts.push(f);
  27842. add_rels(wsrels, -1, "chartsheets/sheet" + rId + "." + wbext, RELS.CS);
  27843. break; */
  27844. /* falls through */
  27845. default:
  27846. f = "xl/worksheets/sheet" + rId + "." + wbext;
  27847. zip.file(f, write_ws(rId-1, f, opts, wb, wsrels));
  27848. ct.sheets.push(f);
  27849. add_rels(opts.wbrels, -1, "worksheets/sheet" + rId + "." + wbext, RELS.WS[0]);
  27850. }
  27851. if(ws) {
  27852. var comments = ws['!comments'];
  27853. var need_vml = false;
  27854. if(comments && comments.length > 0) {
  27855. var cf = "xl/comments" + rId + "." + wbext;
  27856. zip.file(cf, write_cmnt(comments, cf, opts));
  27857. ct.comments.push(cf);
  27858. add_rels(wsrels, -1, "../comments" + rId + "." + wbext, RELS.CMNT);
  27859. need_vml = true;
  27860. }
  27861. if(ws['!legacy']) {
  27862. if(need_vml) zip.file("xl/drawings/vmlDrawing" + (rId) + ".vml", write_comments_vml(rId, ws['!comments']));
  27863. }
  27864. delete ws['!comments'];
  27865. delete ws['!legacy'];
  27866. }
  27867. if(wsrels['!id'].rId1) zip.file(get_rels_path(f), write_rels(wsrels));
  27868. }
  27869. if(opts.Strings != null && opts.Strings.length > 0) {
  27870. f = "xl/sharedStrings." + wbext;
  27871. zip.file(f, write_sst(opts.Strings, f, opts));
  27872. ct.strs.push(f);
  27873. add_rels(opts.wbrels, -1, "sharedStrings." + wbext, RELS.SST);
  27874. }
  27875. f = "xl/workbook." + wbext;
  27876. zip.file(f, write_wb(wb, f, opts));
  27877. ct.workbooks.push(f);
  27878. add_rels(opts.rels, 1, f, RELS.WB);
  27879. /* TODO: something more intelligent with themes */
  27880. f = "xl/theme/theme1.xml";
  27881. zip.file(f, write_theme(wb.Themes, opts));
  27882. ct.themes.push(f);
  27883. add_rels(opts.wbrels, -1, "theme/theme1.xml", RELS.THEME);
  27884. /* TODO: something more intelligent with styles */
  27885. f = "xl/styles." + wbext;
  27886. zip.file(f, write_sty(wb, f, opts));
  27887. ct.styles.push(f);
  27888. add_rels(opts.wbrels, -1, "styles." + wbext, RELS.STY);
  27889. if(wb.vbaraw && vbafmt) {
  27890. f = "xl/vbaProject.bin";
  27891. zip.file(f, wb.vbaraw);
  27892. ct.vba.push(f);
  27893. add_rels(opts.wbrels, -1, "vbaProject.bin", RELS.VBA);
  27894. }
  27895. zip.file("[Content_Types].xml", write_ct(ct, opts));
  27896. zip.file('_rels/.rels', write_rels(opts.rels));
  27897. zip.file('xl/_rels/workbook.' + wbext + '.rels', write_rels(opts.wbrels));
  27898. delete opts.revssf; delete opts.ssf;
  27899. return zip;
  27900. }
  27901. function firstbyte(f,o) {
  27902. var x = "";
  27903. switch((o||{}).type || "base64") {
  27904. case 'buffer': return [f[0], f[1], f[2], f[3]];
  27905. case 'base64': x = Base64.decode(f.slice(0,24)); break;
  27906. case 'binary': x = f; break;
  27907. case 'array': return [f[0], f[1], f[2], f[3]];
  27908. default: throw new Error("Unrecognized type " + (o && o.type || "undefined"));
  27909. }
  27910. return [x.charCodeAt(0), x.charCodeAt(1), x.charCodeAt(2), x.charCodeAt(3)];
  27911. }
  27912. function read_cfb(cfb, opts) {
  27913. if(CFB.find(cfb, "EncryptedPackage")) return parse_xlsxcfb(cfb, opts);
  27914. return parse_xlscfb(cfb, opts);
  27915. }
  27916. function read_zip(data, opts) {
  27917. var zip, d = data;
  27918. var o = opts||{};
  27919. if(!o.type) o.type = (has_buf && Buffer.isBuffer(data)) ? "buffer" : "base64";
  27920. switch(o.type) {
  27921. case "base64": zip = new jszip(d, { base64:true }); break;
  27922. case "binary": case "array": zip = new jszip(d, { base64:false }); break;
  27923. case "buffer": zip = new jszip(d); break;
  27924. default: throw new Error("Unrecognized type " + o.type);
  27925. }
  27926. return parse_zip(zip, o);
  27927. }
  27928. function read_plaintext(data, o) {
  27929. var i = 0;
  27930. main: while(i < data.length) switch(data.charCodeAt(i)) {
  27931. case 0x0A: case 0x0D: case 0x20: ++i; break;
  27932. case 0x3C: return parse_xlml(data.slice(i),o);
  27933. default: break main;
  27934. }
  27935. return PRN.to_workbook(data, o);
  27936. }
  27937. function read_plaintext_raw(data, o) {
  27938. var str = "", bytes = firstbyte(data, o);
  27939. switch(o.type) {
  27940. case 'base64': str = Base64.decode(data); break;
  27941. case 'binary': str = data; break;
  27942. case 'buffer': str = data.toString('binary'); break;
  27943. case 'array': str = cc2str(data); break;
  27944. default: throw new Error("Unrecognized type " + o.type);
  27945. }
  27946. if(bytes[0] == 0xEF && bytes[1] == 0xBB && bytes[2] == 0xBF) str = utf8read(str);
  27947. return read_plaintext(str, o);
  27948. }
  27949. function read_utf16(data, o) {
  27950. var d = data;
  27951. if(o.type == 'base64') d = Base64.decode(d);
  27952. d = cptable.utils.decode(1200, d.slice(2), 'str');
  27953. o.type = "binary";
  27954. return read_plaintext(d, o);
  27955. }
  27956. function bstrify(data) {
  27957. return !data.match(/[^\x00-\x7F]/) ? data : utf8write(data);
  27958. }
  27959. function read_prn(data, d, o, str) {
  27960. if(str) { o.type = "string"; return PRN.to_workbook(data, o); }
  27961. return PRN.to_workbook(d, o);
  27962. }
  27963. function readSync(data, opts) {
  27964. reset_cp();
  27965. if(typeof ArrayBuffer !== 'undefined' && data instanceof ArrayBuffer) return readSync(new Uint8Array(data), opts);
  27966. var d = data, n = [0,0,0,0], str = false;
  27967. var o = opts||{};
  27968. _ssfopts = {};
  27969. if(o.dateNF) _ssfopts.dateNF = o.dateNF;
  27970. if(!o.type) o.type = (has_buf && Buffer.isBuffer(data)) ? "buffer" : "base64";
  27971. if(o.type == "file") { o.type = has_buf ? "buffer" : "binary"; d = read_binary(data); }
  27972. if(o.type == "string") { str = true; o.type = "binary"; o.codepage = 65001; d = bstrify(data); }
  27973. if(o.type == 'array' && typeof Uint8Array !== 'undefined' && data instanceof Uint8Array && typeof ArrayBuffer !== 'undefined') {
  27974. // $FlowIgnore
  27975. var ab=new ArrayBuffer(3), vu=new Uint8Array(ab); vu.foo="bar";
  27976. // $FlowIgnore
  27977. if(!vu.foo) {o=dup(o); o.type='array'; return readSync(ab2a(d), o);}
  27978. }
  27979. switch((n = firstbyte(d, o))[0]) {
  27980. case 0xD0: return read_cfb(CFB.read(d, o), o);
  27981. case 0x09: return parse_xlscfb(d, o);
  27982. case 0x3C: return parse_xlml(d, o);
  27983. case 0x49: if(n[1] === 0x44) return read_wb_ID(d, o); break;
  27984. case 0x54: if(n[1] === 0x41 && n[2] === 0x42 && n[3] === 0x4C) return DIF.to_workbook(d, o); break;
  27985. case 0x50: return (n[1] === 0x4B && n[2] < 0x09 && n[3] < 0x09) ? read_zip(d, o) : read_prn(data, d, o, str);
  27986. case 0xEF: return n[3] === 0x3C ? parse_xlml(d, o) : read_prn(data, d, o, str);
  27987. case 0xFF: if(n[1] === 0xFE) { return read_utf16(d, o); } break;
  27988. case 0x00: if(n[1] === 0x00 && n[2] >= 0x02 && n[3] === 0x00) return WK_.to_workbook(d, o); break;
  27989. case 0x03: case 0x83: case 0x8B: case 0x8C: return DBF.to_workbook(d, o);
  27990. case 0x7B: if(n[1] === 0x5C && n[2] === 0x72 && n[3] === 0x74) return RTF.to_workbook(d, o); break;
  27991. case 0x0A: case 0x0D: case 0x20: return read_plaintext_raw(d, o);
  27992. }
  27993. if(n[2] <= 12 && n[3] <= 31) return DBF.to_workbook(d, o);
  27994. return read_prn(data, d, o, str);
  27995. }
  27996. function readFileSync(filename, opts) {
  27997. var o = opts||{}; o.type = 'file';
  27998. return readSync(filename, o);
  27999. }
  28000. function write_cfb_ctr(cfb, o) {
  28001. switch(o.type) {
  28002. case "base64": case "binary": break;
  28003. case "buffer": case "array": o.type = ""; break;
  28004. case "file": return write_dl(o.file, CFB.write(cfb, {type:has_buf ? 'buffer' : ""}));
  28005. case "string": throw new Error("'string' output type invalid for '" + o.bookType + "' files");
  28006. default: throw new Error("Unrecognized type " + o.type);
  28007. }
  28008. return CFB.write(cfb, o);
  28009. }
  28010. /*global encrypt_agile */
  28011. function write_zip_type(wb, opts) {
  28012. var o = opts||{};
  28013. style_builder = new StyleBuilder(opts);
  28014. var z = write_zip(wb, o);
  28015. var oopts = {};
  28016. if(o.compression) oopts.compression = 'DEFLATE';
  28017. if(o.password) oopts.type = has_buf ? "nodebuffer" : "string";
  28018. else switch(o.type) {
  28019. case "base64": oopts.type = "base64"; break;
  28020. case "binary": oopts.type = "string"; break;
  28021. case "string": throw new Error("'string' output type invalid for '" + o.bookType + "' files");
  28022. case "buffer":
  28023. case "file": oopts.type = has_buf ? "nodebuffer" : "string"; break;
  28024. default: throw new Error("Unrecognized type " + o.type);
  28025. }
  28026. var out = z.generate(oopts);
  28027. if(o.password && typeof encrypt_agile !== 'undefined') return write_cfb_ctr(encrypt_agile(out, o.password), o);
  28028. if(o.type === "file") return write_dl(o.file, out);
  28029. return o.type == "string" ? utf8read(out) : out;
  28030. }
  28031. function write_cfb_type(wb, opts) {
  28032. var o = opts||{};
  28033. var cfb = write_xlscfb(wb, o);
  28034. return write_cfb_ctr(cfb, o);
  28035. }
  28036. function write_string_type(out, opts, bom) {
  28037. if(!bom) bom = "";
  28038. var o = bom + out;
  28039. switch(opts.type) {
  28040. case "base64": return Base64.encode(utf8write(o));
  28041. case "binary": return utf8write(o);
  28042. case "string": return out;
  28043. case "file": return write_dl(opts.file, o, 'utf8');
  28044. case "buffer": {
  28045. // $FlowIgnore
  28046. if(has_buf) return Buffer_from(o, 'utf8');
  28047. else return write_string_type(o, {type:'binary'}).split("").map(function(c) { return c.charCodeAt(0); });
  28048. }
  28049. }
  28050. throw new Error("Unrecognized type " + opts.type);
  28051. }
  28052. function write_stxt_type(out, opts) {
  28053. switch(opts.type) {
  28054. case "base64": return Base64.encode(out);
  28055. case "binary": return out;
  28056. case "string": return out; /* override in sheet_to_txt */
  28057. case "file": return write_dl(opts.file, out, 'binary');
  28058. case "buffer": {
  28059. // $FlowIgnore
  28060. if(has_buf) return Buffer_from(out, 'binary');
  28061. else return out.split("").map(function(c) { return c.charCodeAt(0); });
  28062. }
  28063. }
  28064. throw new Error("Unrecognized type " + opts.type);
  28065. }
  28066. /* TODO: test consistency */
  28067. function write_binary_type(out, opts) {
  28068. switch(opts.type) {
  28069. case "string":
  28070. case "base64":
  28071. case "binary":
  28072. var bstr = "";
  28073. // $FlowIgnore
  28074. for(var i = 0; i < out.length; ++i) bstr += String.fromCharCode(out[i]);
  28075. return opts.type == 'base64' ? Base64.encode(bstr) : opts.type == 'string' ? utf8read(bstr) : bstr;
  28076. case "file": return write_dl(opts.file, out);
  28077. case "buffer": return out;
  28078. default: throw new Error("Unrecognized type " + opts.type);
  28079. }
  28080. }
  28081. function writeSync(wb, opts) {
  28082. check_wb(wb);
  28083. var o = opts||{};
  28084. if(o.type == "array") { o.type = "binary"; var out = (writeSync(wb, o)); o.type = "array"; return s2ab(out); }
  28085. switch(o.bookType || 'xlsb') {
  28086. case 'xml':
  28087. case 'xlml': return write_string_type(write_xlml(wb, o), o);
  28088. case 'slk':
  28089. case 'sylk': return write_string_type(write_slk_str(wb, o), o);
  28090. case 'htm':
  28091. case 'html': return write_string_type(write_htm_str(wb, o), o);
  28092. case 'txt': return write_stxt_type(write_txt_str(wb, o), o);
  28093. case 'csv': return write_string_type(write_csv_str(wb, o), o, "\ufeff");
  28094. case 'dif': return write_string_type(write_dif_str(wb, o), o);
  28095. case 'dbf': return write_binary_type(write_dbf_buf(wb, o), o);
  28096. case 'prn': return write_string_type(write_prn_str(wb, o), o);
  28097. case 'rtf': return write_string_type(write_rtf_str(wb, o), o);
  28098. case 'eth': return write_string_type(write_eth_str(wb, o), o);
  28099. case 'fods': return write_string_type(write_ods(wb, o), o);
  28100. case 'biff2': if(!o.biff) o.biff = 2; /* falls through */
  28101. case 'biff3': if(!o.biff) o.biff = 3; /* falls through */
  28102. case 'biff4': if(!o.biff) o.biff = 4; return write_binary_type(write_biff_buf(wb, o), o);
  28103. case 'biff5': if(!o.biff) o.biff = 5; /* falls through */
  28104. case 'biff8':
  28105. case 'xla':
  28106. case 'xls': if(!o.biff) o.biff = 8; return write_cfb_type(wb, o);
  28107. case 'xlsx':
  28108. case 'xlsm':
  28109. case 'xlam':
  28110. case 'xlsb':
  28111. case 'ods': return write_zip_type(wb, o);
  28112. default: throw new Error ("Unrecognized bookType |" + o.bookType + "|");
  28113. }
  28114. }
  28115. function resolve_book_type(o) {
  28116. if(o.bookType) return;
  28117. var _BT = {
  28118. "xls": "biff8",
  28119. "htm": "html",
  28120. "slk": "sylk",
  28121. "socialcalc": "eth",
  28122. "Sh33tJS": "WTF"
  28123. };
  28124. var ext = o.file.slice(o.file.lastIndexOf(".")).toLowerCase();
  28125. if(ext.match(/^\.[a-z]+$/)) o.bookType = ext.slice(1);
  28126. o.bookType = _BT[o.bookType] || o.bookType;
  28127. }
  28128. function writeFileSync(wb, filename, opts) {
  28129. var o = opts||{}; o.type = 'file';
  28130. o.file = filename;
  28131. resolve_book_type(o);
  28132. return writeSync(wb, o);
  28133. }
  28134. function writeFileAsync(filename, wb, opts, cb) {
  28135. var o = opts||{}; o.type = 'file';
  28136. o.file = filename;
  28137. resolve_book_type(o);
  28138. o.type = 'buffer';
  28139. var _cb = cb; if(!(_cb instanceof Function)) _cb = (opts);
  28140. return _fs.writeFile(filename, writeSync(wb, o), _cb);
  28141. }
  28142. function make_json_row(sheet, r, R, cols, header, hdr, dense, o) {
  28143. var rr = encode_row(R);
  28144. var defval = o.defval, raw = o.raw || !o.hasOwnProperty("raw");
  28145. var isempty = true;
  28146. var row = (header === 1) ? [] : {};
  28147. if(header !== 1) {
  28148. if(Object.defineProperty) try { Object.defineProperty(row, '__rowNum__', {value:R, enumerable:false}); } catch(e) { row.__rowNum__ = R; }
  28149. else row.__rowNum__ = R;
  28150. }
  28151. if(!dense || sheet[R]) for (var C = r.s.c; C <= r.e.c; ++C) {
  28152. var val = dense ? sheet[R][C] : sheet[cols[C] + rr];
  28153. if(val === undefined || val.t === undefined) {
  28154. if(defval === undefined) continue;
  28155. if(hdr[C] != null) { row[hdr[C]] = defval; }
  28156. continue;
  28157. }
  28158. var v = val.v;
  28159. switch(val.t){
  28160. case 'z': if(v == null) break; continue;
  28161. case 'e': v = void 0; break;
  28162. case 's': case 'd': case 'b': case 'n': break;
  28163. default: throw new Error('unrecognized type ' + val.t);
  28164. }
  28165. if(hdr[C] != null) {
  28166. if(v == null) {
  28167. if(defval !== undefined) row[hdr[C]] = defval;
  28168. else if(raw && v === null) row[hdr[C]] = null;
  28169. else continue;
  28170. } else {
  28171. row[hdr[C]] = raw ? v : format_cell(val,v,o);
  28172. }
  28173. if(v != null) isempty = false;
  28174. }
  28175. }
  28176. return { row: row, isempty: isempty };
  28177. }
  28178. function sheet_to_json(sheet, opts) {
  28179. if(sheet == null || sheet["!ref"] == null) return [];
  28180. var val = {t:'n',v:0}, header = 0, offset = 1, hdr = [], v=0, vv="";
  28181. var r = {s:{r:0,c:0},e:{r:0,c:0}};
  28182. var o = opts || {};
  28183. var range = o.range != null ? o.range : sheet["!ref"];
  28184. if(o.header === 1) header = 1;
  28185. else if(o.header === "A") header = 2;
  28186. else if(Array.isArray(o.header)) header = 3;
  28187. switch(typeof range) {
  28188. case 'string': r = safe_decode_range(range); break;
  28189. case 'number': r = safe_decode_range(sheet["!ref"]); r.s.r = range; break;
  28190. default: r = range;
  28191. }
  28192. if(header > 0) offset = 0;
  28193. var rr = encode_row(r.s.r);
  28194. var cols = [];
  28195. var out = [];
  28196. var outi = 0, counter = 0;
  28197. var dense = Array.isArray(sheet);
  28198. var R = r.s.r, C = 0, CC = 0;
  28199. if(dense && !sheet[R]) sheet[R] = [];
  28200. for(C = r.s.c; C <= r.e.c; ++C) {
  28201. cols[C] = encode_col(C);
  28202. val = dense ? sheet[R][C] : sheet[cols[C] + rr];
  28203. switch(header) {
  28204. case 1: hdr[C] = C - r.s.c; break;
  28205. case 2: hdr[C] = cols[C]; break;
  28206. case 3: hdr[C] = o.header[C - r.s.c]; break;
  28207. default:
  28208. if(val == null) val = {w: "__EMPTY", t: "s"};
  28209. vv = v = format_cell(val, null, o);
  28210. counter = 0;
  28211. for(CC = 0; CC < hdr.length; ++CC) if(hdr[CC] == vv) vv = v + "_" + (++counter);
  28212. hdr[C] = vv;
  28213. }
  28214. }
  28215. for (R = r.s.r + offset; R <= r.e.r; ++R) {
  28216. var row = make_json_row(sheet, r, R, cols, header, hdr, dense, o);
  28217. if((row.isempty === false) || (header === 1 ? o.blankrows !== false : !!o.blankrows)) out[outi++] = row.row;
  28218. }
  28219. out.length = outi;
  28220. return out;
  28221. }
  28222. var qreg = /"/g;
  28223. function make_csv_row(sheet, r, R, cols, fs, rs, FS, o) {
  28224. var isempty = true;
  28225. var row = [], txt = "", rr = encode_row(R);
  28226. for(var C = r.s.c; C <= r.e.c; ++C) {
  28227. if (!cols[C]) continue;
  28228. var val = o.dense ? (sheet[R]||[])[C]: sheet[cols[C] + rr];
  28229. if(val == null) txt = "";
  28230. else if(val.v != null) {
  28231. isempty = false;
  28232. txt = ''+format_cell(val, null, o);
  28233. for(var i = 0, cc = 0; i !== txt.length; ++i) if((cc = txt.charCodeAt(i)) === fs || cc === rs || cc === 34) {txt = "\"" + txt.replace(qreg, '""') + "\""; break; }
  28234. if(txt == "ID") txt = '"ID"';
  28235. } else if(val.f != null && !val.F) {
  28236. isempty = false;
  28237. txt = '=' + val.f; if(txt.indexOf(",") >= 0) txt = '"' + txt.replace(qreg, '""') + '"';
  28238. } else txt = "";
  28239. /* NOTE: Excel CSV does not support array formulae */
  28240. row.push(txt);
  28241. }
  28242. if(o.blankrows === false && isempty) return null;
  28243. return row.join(FS);
  28244. }
  28245. function sheet_to_csv(sheet, opts) {
  28246. var out = [];
  28247. var o = opts == null ? {} : opts;
  28248. if(sheet == null || sheet["!ref"] == null) return "";
  28249. var r = safe_decode_range(sheet["!ref"]);
  28250. var FS = o.FS !== undefined ? o.FS : ",", fs = FS.charCodeAt(0);
  28251. var RS = o.RS !== undefined ? o.RS : "\n", rs = RS.charCodeAt(0);
  28252. var endregex = new RegExp((FS=="|" ? "\\|" : FS)+"+$");
  28253. var row = "", cols = [];
  28254. o.dense = Array.isArray(sheet);
  28255. var colinfo = o.skipHidden && sheet["!cols"] || [];
  28256. var rowinfo = o.skipHidden && sheet["!rows"] || [];
  28257. for(var C = r.s.c; C <= r.e.c; ++C) if (!((colinfo[C]||{}).hidden)) cols[C] = encode_col(C);
  28258. for(var R = r.s.r; R <= r.e.r; ++R) {
  28259. if ((rowinfo[R]||{}).hidden) continue;
  28260. row = make_csv_row(sheet, r, R, cols, fs, rs, FS, o);
  28261. if(row == null) { continue; }
  28262. if(o.strip) row = row.replace(endregex,"");
  28263. out.push(row + RS);
  28264. }
  28265. delete o.dense;
  28266. return out.join("");
  28267. }
  28268. function sheet_to_txt(sheet, opts) {
  28269. if(!opts) opts = {}; opts.FS = "\t"; opts.RS = "\n";
  28270. var s = sheet_to_csv(sheet, opts);
  28271. if(typeof cptable == 'undefined' || opts.type == 'string') return s;
  28272. var o = cptable.utils.encode(1200, s, 'str');
  28273. return String.fromCharCode(255) + String.fromCharCode(254) + o;
  28274. }
  28275. function sheet_to_formulae(sheet) {
  28276. var y = "", x, val="";
  28277. if(sheet == null || sheet["!ref"] == null) return [];
  28278. var r = safe_decode_range(sheet['!ref']), rr = "", cols = [], C;
  28279. var cmds = [];
  28280. var dense = Array.isArray(sheet);
  28281. for(C = r.s.c; C <= r.e.c; ++C) cols[C] = encode_col(C);
  28282. for(var R = r.s.r; R <= r.e.r; ++R) {
  28283. rr = encode_row(R);
  28284. for(C = r.s.c; C <= r.e.c; ++C) {
  28285. y = cols[C] + rr;
  28286. x = dense ? (sheet[R]||[])[C] : sheet[y];
  28287. val = "";
  28288. if(x === undefined) continue;
  28289. else if(x.F != null) {
  28290. y = x.F;
  28291. if(!x.f) continue;
  28292. val = x.f;
  28293. if(y.indexOf(":") == -1) y = y + ":" + y;
  28294. }
  28295. if(x.f != null) val = x.f;
  28296. else if(x.t == 'z') continue;
  28297. else if(x.t == 'n' && x.v != null) val = "" + x.v;
  28298. else if(x.t == 'b') val = x.v ? "TRUE" : "FALSE";
  28299. else if(x.w !== undefined) val = "'" + x.w;
  28300. else if(x.v === undefined) continue;
  28301. else if(x.t == 's') val = "'" + x.v;
  28302. else val = ""+x.v;
  28303. cmds[cmds.length] = y + "=" + val;
  28304. }
  28305. }
  28306. return cmds;
  28307. }
  28308. function sheet_add_json(_ws, js, opts) {
  28309. var o = opts || {};
  28310. var offset = +!o.skipHeader;
  28311. var ws = _ws || ({});
  28312. var _R = 0, _C = 0;
  28313. if(ws && o.origin != null) {
  28314. if(typeof o.origin == 'number') _R = o.origin;
  28315. else {
  28316. var _origin = typeof o.origin == "string" ? decode_cell(o.origin) : o.origin;
  28317. _R = _origin.r; _C = _origin.c;
  28318. }
  28319. }
  28320. var cell;
  28321. var range = ({s: {c:0, r:0}, e: {c:_C, r:_R + js.length - 1 + offset}});
  28322. if(ws['!ref']) {
  28323. var _range = safe_decode_range(ws['!ref']);
  28324. range.e.c = Math.max(range.e.c, _range.e.c);
  28325. range.e.r = Math.max(range.e.r, _range.e.r);
  28326. if(_R == -1) { _R = range.e.r + 1; range.e.r = _R + js.length - 1 + offset; }
  28327. }
  28328. var hdr = o.header || [], C = 0;
  28329. js.forEach(function (JS, R) {
  28330. keys(JS).forEach(function(k) {
  28331. if((C=hdr.indexOf(k)) == -1) hdr[C=hdr.length] = k;
  28332. var v = JS[k];
  28333. var t = 'z';
  28334. var z = "";
  28335. if(v && typeof v === 'object' && !(v instanceof Date)){
  28336. ws[encode_cell({c:_C + C,r:_R + R + offset})] = v;
  28337. } else {
  28338. if(typeof v == 'number') t = 'n';
  28339. else if(typeof v == 'boolean') t = 'b';
  28340. else if(typeof v == 'string') t = 's';
  28341. else if(v instanceof Date) {
  28342. t = 'd';
  28343. if(!o.cellDates) { t = 'n'; v = datenum(v); }
  28344. z = o.dateNF || SSF._table[14];
  28345. }
  28346. ws[encode_cell({c:_C + C,r:_R + R + offset})] = cell = ({t:t, v:v});
  28347. if(z) cell.z = z;
  28348. }
  28349. });
  28350. });
  28351. range.e.c = Math.max(range.e.c, _C + hdr.length - 1);
  28352. var __R = encode_row(_R);
  28353. if(offset) for(C = 0; C < hdr.length; ++C) ws[encode_col(C + _C) + __R] = {t:'s', v:hdr[C]};
  28354. ws['!ref'] = encode_range(range);
  28355. return ws;
  28356. }
  28357. function json_to_sheet(js, opts) { return sheet_add_json(null, js, opts); }
  28358. var utils = {
  28359. encode_col: encode_col,
  28360. encode_row: encode_row,
  28361. encode_cell: encode_cell,
  28362. encode_range: encode_range,
  28363. decode_col: decode_col,
  28364. decode_row: decode_row,
  28365. split_cell: split_cell,
  28366. decode_cell: decode_cell,
  28367. decode_range: decode_range,
  28368. format_cell: format_cell,
  28369. get_formulae: sheet_to_formulae,
  28370. make_csv: sheet_to_csv,
  28371. make_json: sheet_to_json,
  28372. make_formulae: sheet_to_formulae,
  28373. sheet_add_aoa: sheet_add_aoa,
  28374. sheet_add_json: sheet_add_json,
  28375. aoa_to_sheet: aoa_to_sheet,
  28376. json_to_sheet: json_to_sheet,
  28377. table_to_sheet: parse_dom_table,
  28378. table_to_book: table_to_book,
  28379. sheet_to_csv: sheet_to_csv,
  28380. sheet_to_txt: sheet_to_txt,
  28381. sheet_to_json: sheet_to_json,
  28382. sheet_to_html: HTML_.from_sheet,
  28383. sheet_to_dif: DIF.from_sheet,
  28384. sheet_to_slk: SYLK.from_sheet,
  28385. sheet_to_eth: ETH.from_sheet,
  28386. sheet_to_formulae: sheet_to_formulae,
  28387. sheet_to_row_object_array: sheet_to_json
  28388. };
  28389. (function(utils) {
  28390. utils.consts = utils.consts || {};
  28391. function add_consts(R/*Array<any>*/) { R.forEach(function(a){ utils.consts[a[0]] = a[1]; }); }
  28392. function get_default(x, y, z) { return x[y] != null ? x[y] : (x[y] = z); }
  28393. /* get cell, creating a stub if necessary */
  28394. function ws_get_cell_stub(ws, R, C) {
  28395. /* A1 cell address */
  28396. if(typeof R == "string") return ws[R] || (ws[R] = {t:'z'});
  28397. /* cell address object */
  28398. if(typeof R != "number") return ws_get_cell_stub(ws, encode_cell(R));
  28399. /* R and C are 0-based indices */
  28400. return ws_get_cell_stub(ws, encode_cell({r:R,c:C||0}));
  28401. }
  28402. /* find sheet index for given name / validate index */
  28403. function wb_sheet_idx(wb, sh) {
  28404. if(typeof sh == "number") {
  28405. if(sh >= 0 && wb.SheetNames.length > sh) return sh;
  28406. throw new Error("Cannot find sheet # " + sh);
  28407. } else if(typeof sh == "string") {
  28408. var idx = wb.SheetNames.indexOf(sh);
  28409. if(idx > -1) return idx;
  28410. throw new Error("Cannot find sheet name |" + sh + "|");
  28411. } else throw new Error("Cannot find sheet |" + sh + "|");
  28412. }
  28413. /* simple blank workbook object */
  28414. utils.book_new = function() {
  28415. return { SheetNames: [], Sheets: {} };
  28416. };
  28417. /* add a worksheet to the end of a given workbook */
  28418. utils.book_append_sheet = function(wb, ws, name) {
  28419. if(!name) for(var i = 1; i <= 0xFFFF; ++i) if(wb.SheetNames.indexOf(name = "Sheet" + i) == -1) break;
  28420. if(!name) throw new Error("Too many worksheets");
  28421. check_ws_name(name);
  28422. if(wb.SheetNames.indexOf(name) >= 0) throw new Error("Worksheet with name |" + name + "| already exists!");
  28423. wb.SheetNames.push(name);
  28424. wb.Sheets[name] = ws;
  28425. };
  28426. /* set sheet visibility (visible/hidden/very hidden) */
  28427. utils.book_set_sheet_visibility = function(wb, sh, vis) {
  28428. get_default(wb,"Workbook",{});
  28429. get_default(wb.Workbook,"Sheets",[]);
  28430. var idx = wb_sheet_idx(wb, sh);
  28431. // $FlowIgnore
  28432. get_default(wb.Workbook.Sheets,idx, {});
  28433. switch(vis) {
  28434. case 0: case 1: case 2: break;
  28435. default: throw new Error("Bad sheet visibility setting " + vis);
  28436. }
  28437. // $FlowIgnore
  28438. wb.Workbook.Sheets[idx].Hidden = vis;
  28439. };
  28440. add_consts([
  28441. ["SHEET_VISIBLE", 0],
  28442. ["SHEET_HIDDEN", 1],
  28443. ["SHEET_VERY_HIDDEN", 2]
  28444. ]);
  28445. /* set number format */
  28446. utils.cell_set_number_format = function(cell, fmt) {
  28447. cell.z = fmt;
  28448. return cell;
  28449. };
  28450. /* set cell hyperlink */
  28451. utils.cell_set_hyperlink = function(cell, target, tooltip) {
  28452. if(!target) {
  28453. delete cell.l;
  28454. } else {
  28455. cell.l = ({ Target: target });
  28456. if(tooltip) cell.l.Tooltip = tooltip;
  28457. }
  28458. return cell;
  28459. };
  28460. utils.cell_set_internal_link = function(cell, range, tooltip) { return utils.cell_set_hyperlink(cell, "#" + range, tooltip); };
  28461. /* add to cell comments */
  28462. utils.cell_add_comment = function(cell, text, author) {
  28463. if(!cell.c) cell.c = [];
  28464. cell.c.push({t:text, a:author||"SheetJS"});
  28465. };
  28466. /* set array formula and flush related cells */
  28467. utils.sheet_set_array_formula = function(ws, range, formula) {
  28468. var rng = typeof range != "string" ? range : safe_decode_range(range);
  28469. var rngstr = typeof range == "string" ? range : encode_range(range);
  28470. for(var R = rng.s.r; R <= rng.e.r; ++R) for(var C = rng.s.c; C <= rng.e.c; ++C) {
  28471. var cell = ws_get_cell_stub(ws, R, C);
  28472. cell.t = 'n';
  28473. cell.F = rngstr;
  28474. delete cell.v;
  28475. if(R == rng.s.r && C == rng.s.c) cell.f = formula;
  28476. }
  28477. return ws;
  28478. };
  28479. return utils;
  28480. })(utils);
  28481. if(has_buf && typeof require != 'undefined') (function() {
  28482. var Readable = {}.Readable;
  28483. var write_csv_stream = function(sheet, opts) {
  28484. var stream = Readable();
  28485. var o = opts == null ? {} : opts;
  28486. if(sheet == null || sheet["!ref"] == null) { stream.push(null); return stream; }
  28487. var r = safe_decode_range(sheet["!ref"]);
  28488. var FS = o.FS !== undefined ? o.FS : ",", fs = FS.charCodeAt(0);
  28489. var RS = o.RS !== undefined ? o.RS : "\n", rs = RS.charCodeAt(0);
  28490. var endregex = new RegExp((FS=="|" ? "\\|" : FS)+"+$");
  28491. var row = "", cols = [];
  28492. o.dense = Array.isArray(sheet);
  28493. var colinfo = o.skipHidden && sheet["!cols"] || [];
  28494. var rowinfo = o.skipHidden && sheet["!rows"] || [];
  28495. for(var C = r.s.c; C <= r.e.c; ++C) if (!((colinfo[C]||{}).hidden)) cols[C] = encode_col(C);
  28496. var R = r.s.r;
  28497. var BOM = false;
  28498. stream._read = function() {
  28499. if(!BOM) { BOM = true; return stream.push("\uFEFF"); }
  28500. if(R > r.e.r) return stream.push(null);
  28501. while(R <= r.e.r) {
  28502. ++R;
  28503. if ((rowinfo[R-1]||{}).hidden) continue;
  28504. row = make_csv_row(sheet, r, R-1, cols, fs, rs, FS, o);
  28505. if(row != null) {
  28506. if(o.strip) row = row.replace(endregex,"");
  28507. stream.push(row + RS);
  28508. break;
  28509. }
  28510. }
  28511. };
  28512. return stream;
  28513. };
  28514. var write_html_stream = function(ws, opts) {
  28515. var stream = Readable();
  28516. var o = opts || {};
  28517. var header = o.header != null ? o.header : HTML_.BEGIN;
  28518. var footer = o.footer != null ? o.footer : HTML_.END;
  28519. stream.push(header);
  28520. var r = decode_range(ws['!ref']);
  28521. o.dense = Array.isArray(ws);
  28522. stream.push(HTML_._preamble(ws, r, o));
  28523. var R = r.s.r;
  28524. var end = false;
  28525. stream._read = function() {
  28526. if(R > r.e.r) {
  28527. if(!end) { end = true; stream.push("</table>" + footer); }
  28528. return stream.push(null);
  28529. }
  28530. while(R <= r.e.r) {
  28531. stream.push(HTML_._row(ws, r, R, o));
  28532. ++R;
  28533. break;
  28534. }
  28535. };
  28536. return stream;
  28537. };
  28538. var write_json_stream = function(sheet, opts) {
  28539. var stream = Readable({objectMode:true});
  28540. if(sheet == null || sheet["!ref"] == null) { stream.push(null); return stream; }
  28541. var val = {t:'n',v:0}, header = 0, offset = 1, hdr = [], v=0, vv="";
  28542. var r = {s:{r:0,c:0},e:{r:0,c:0}};
  28543. var o = opts || {};
  28544. var range = o.range != null ? o.range : sheet["!ref"];
  28545. if(o.header === 1) header = 1;
  28546. else if(o.header === "A") header = 2;
  28547. else if(Array.isArray(o.header)) header = 3;
  28548. switch(typeof range) {
  28549. case 'string': r = safe_decode_range(range); break;
  28550. case 'number': r = safe_decode_range(sheet["!ref"]); r.s.r = range; break;
  28551. default: r = range;
  28552. }
  28553. if(header > 0) offset = 0;
  28554. var rr = encode_row(r.s.r);
  28555. var cols = [];
  28556. var counter = 0;
  28557. var dense = Array.isArray(sheet);
  28558. var R = r.s.r, C = 0, CC = 0;
  28559. if(dense && !sheet[R]) sheet[R] = [];
  28560. for(C = r.s.c; C <= r.e.c; ++C) {
  28561. cols[C] = encode_col(C);
  28562. val = dense ? sheet[R][C] : sheet[cols[C] + rr];
  28563. switch(header) {
  28564. case 1: hdr[C] = C - r.s.c; break;
  28565. case 2: hdr[C] = cols[C]; break;
  28566. case 3: hdr[C] = o.header[C - r.s.c]; break;
  28567. default:
  28568. if(val == null) val = {w: "__EMPTY", t: "s"};
  28569. vv = v = format_cell(val, null, o);
  28570. counter = 0;
  28571. for(CC = 0; CC < hdr.length; ++CC) if(hdr[CC] == vv) vv = v + "_" + (++counter);
  28572. hdr[C] = vv;
  28573. }
  28574. }
  28575. R = r.s.r + offset;
  28576. stream._read = function() {
  28577. if(R > r.e.r) return stream.push(null);
  28578. while(R <= r.e.r) {
  28579. //if ((rowinfo[R-1]||{}).hidden) continue;
  28580. var row = make_json_row(sheet, r, R, cols, header, hdr, dense, o);
  28581. ++R;
  28582. if((row.isempty === false) || (header === 1 ? o.blankrows !== false : !!o.blankrows)) {
  28583. stream.push(row.row);
  28584. break;
  28585. }
  28586. }
  28587. };
  28588. return stream;
  28589. };
  28590. XLSX.stream = {
  28591. to_json: write_json_stream,
  28592. to_html: write_html_stream,
  28593. to_csv: write_csv_stream
  28594. };
  28595. })();
  28596. /**
  28597. * 样式魔改区 -- start
  28598. */
  28599. /////////////////////////////////////////////////////////////////////////////////////////////////////
  28600. var XmlNode = (function () {
  28601. function XmlNode(tagName, attributes, children) {
  28602. if (!(this instanceof XmlNode)) {
  28603. return new XmlNode(tagName, attributes, children);
  28604. }
  28605. this.tagName = tagName;
  28606. this._attributes = attributes || {};
  28607. this._children = children || [];
  28608. this._prefix = '';
  28609. return this;
  28610. }
  28611. XmlNode.prototype.createElement = function () {
  28612. return new XmlNode(arguments)
  28613. }
  28614. XmlNode.prototype.children = function() {
  28615. return this._children;
  28616. }
  28617. XmlNode.prototype.append = function (node) {
  28618. this._children.push(node);
  28619. return this;
  28620. }
  28621. XmlNode.prototype.prefix = function (prefix) {
  28622. if (arguments.length==0) { return this._prefix;}
  28623. this._prefix = prefix;
  28624. return this;
  28625. }
  28626. XmlNode.prototype.attr = function (attr, value) {
  28627. if (value == undefined) {
  28628. delete this._attributes[attr];
  28629. return this;
  28630. }
  28631. if (arguments.length == 0) {
  28632. return this._attributes;
  28633. }
  28634. else if (typeof attr == 'string' && arguments.length == 1) {
  28635. return this._attributes.attr[attr];
  28636. }
  28637. if (typeof attr == 'object' && arguments.length == 1) {
  28638. for (var key in attr) {
  28639. this._attributes[key] = attr[key];
  28640. }
  28641. }
  28642. else if (arguments.length == 2 && typeof attr == 'string') {
  28643. this._attributes[attr] = value;
  28644. }
  28645. return this;
  28646. }
  28647. var APOS = "'"; QUOTE = '"'
  28648. var ESCAPED_QUOTE = { }
  28649. ESCAPED_QUOTE[QUOTE] = '&quot;'
  28650. ESCAPED_QUOTE[APOS] = '&apos;'
  28651. XmlNode.prototype.escapeAttributeValue = function(att_value) {
  28652. return '"' + att_value.replace(/\"/g,'&quot;') + '"';// TODO Extend with four other codes
  28653. }
  28654. XmlNode.prototype.toXml = function (node) {
  28655. if (!node) node = this;
  28656. var xml = node._prefix;
  28657. xml += '<' + node.tagName;
  28658. if (node._attributes) {
  28659. for (var key in node._attributes) {
  28660. xml += ' ' + key + '=' + this.escapeAttributeValue(''+node._attributes[key]) + ''
  28661. }
  28662. }
  28663. if (node._children && node._children.length > 0) {
  28664. xml += ">";
  28665. for (var i = 0; i < node._children.length; i++) {
  28666. xml += this.toXml(node._children[i]);
  28667. }
  28668. xml += '</' + node.tagName + '>';
  28669. }
  28670. else {
  28671. xml += '/>';
  28672. }
  28673. return xml;
  28674. }
  28675. return XmlNode;
  28676. })();
  28677. /////////////////////////////////////////////////////////////////////////////////////////////////////
  28678. var StyleBuilder = function (options) {
  28679. var customNumFmtId = 164;
  28680. var table_fmt = {
  28681. 0: 'General',
  28682. 1: '0',
  28683. 2: '0.00',
  28684. 3: '#,##0',
  28685. 4: '#,##0.00',
  28686. 9: '0%',
  28687. 10: '0.00%',
  28688. 11: '0.00E+00',
  28689. 12: '# ?/?',
  28690. 13: '# ??/??',
  28691. 14: 'm/d/yy',
  28692. 15: 'd-mmm-yy',
  28693. 16: 'd-mmm',
  28694. 17: 'mmm-yy',
  28695. 18: 'h:mm AM/PM',
  28696. 19: 'h:mm:ss AM/PM',
  28697. 20: 'h:mm',
  28698. 21: 'h:mm:ss',
  28699. 22: 'm/d/yy h:mm',
  28700. 37: '#,##0 ;(#,##0)',
  28701. 38: '#,##0 ;[Red](#,##0)',
  28702. 39: '#,##0.00;(#,##0.00)',
  28703. 40: '#,##0.00;[Red](#,##0.00)',
  28704. 45: 'mm:ss',
  28705. 46: '[h]:mm:ss',
  28706. 47: 'mmss.0',
  28707. 48: '##0.0E+0',
  28708. 49: '@',
  28709. 56: '"上午/下午 "hh"時"mm"分"ss"秒 "' };
  28710. var fmt_table = {};
  28711. for (var idx in table_fmt) {
  28712. fmt_table[table_fmt[idx]] = idx;
  28713. }
  28714. // cache style specs to avoid excessive duplication
  28715. _hashIndex = {};
  28716. _listIndex = [];
  28717. return {
  28718. initialize: function (options) {
  28719. this.$fonts = XmlNode('fonts').attr('count',0).attr("x14ac:knownFonts","1");
  28720. this.$fills = XmlNode('fills').attr('count',0);
  28721. this.$borders = XmlNode('borders').attr('count',0);
  28722. this.$numFmts = XmlNode('numFmts').attr('count',0);
  28723. this.$cellStyleXfs = XmlNode('cellStyleXfs');
  28724. this.$xf = XmlNode('xf')
  28725. .attr('numFmtId', 0)
  28726. .attr('fontId', 0)
  28727. .attr('fillId', 0)
  28728. .attr('borderId', 0);
  28729. this.$cellXfs = XmlNode('cellXfs').attr('count',0);
  28730. this.$cellStyles = XmlNode('cellStyles')
  28731. .append(XmlNode('cellStyle')
  28732. .attr('name', 'Normal')
  28733. .attr('xfId',0)
  28734. .attr('builtinId',0)
  28735. );
  28736. this.$dxfs = XmlNode('dxfs').attr('count', "0");
  28737. this.$tableStyles = XmlNode('tableStyles')
  28738. .attr('count','0')
  28739. .attr('defaultTableStyle','TableStyleMedium9')
  28740. .attr('defaultPivotStyle','PivotStyleMedium4')
  28741. this.$styles = XmlNode('styleSheet')
  28742. .attr('xmlns:mc','http://schemas.openxmlformats.org/markup-compatibility/2006')
  28743. .attr('xmlns:x14ac','http://schemas.microsoft.com/office/spreadsheetml/2009/9/ac')
  28744. .attr('xmlns','http://schemas.openxmlformats.org/spreadsheetml/2006/main')
  28745. .attr('mc:Ignorable','x14ac')
  28746. .prefix('<?xml version="1.0" encoding="UTF-8" standalone="yes"?>')
  28747. .append(this.$numFmts)
  28748. .append(this.$fonts)
  28749. .append(this.$fills)
  28750. .append(this.$borders)
  28751. .append(this.$cellStyleXfs.append(this.$xf))
  28752. .append(this.$cellXfs)
  28753. .append(this.$cellStyles)
  28754. .append(this.$dxfs)
  28755. .append(this.$tableStyles);
  28756. // need to specify styles at index 0 and 1.
  28757. // the second style MUST be gray125 for some reason
  28758. var defaultStyle = options.defaultCellStyle || {};
  28759. if (!defaultStyle.font) defaultStyle.font = {name: 'Calibri', sz: '12'};
  28760. if (!defaultStyle.font.name) defaultStyle.font.name = 'Calibri';
  28761. if (!defaultStyle.font.sz) defaultStyle.font.sz = 11;
  28762. if (!defaultStyle.fill) defaultStyle.fill = { patternType: "none", fgColor: {}};
  28763. if (!defaultStyle.border) defaultStyle.border = {};
  28764. if (!defaultStyle.numFmt) defaultStyle.numFmt = 0;
  28765. this.defaultStyle = defaultStyle;
  28766. var gray125Style = JSON.parse(JSON.stringify(defaultStyle));
  28767. gray125Style.fill = {patternType: "gray125", fgColor: { }}
  28768. this.addStyles([defaultStyle, gray125Style]);
  28769. return this;
  28770. },
  28771. // create a style entry and returns an integer index that can be used in the cell .s property
  28772. // these format of this object follows the emerging Common Spreadsheet Format
  28773. addStyle: function (attributes) {
  28774. var hashKey = JSON.stringify(attributes);
  28775. var index = _hashIndex[hashKey];
  28776. if (index == undefined) {
  28777. index = this._addXf(attributes); //_listIndex.push(attributes) -1;
  28778. _hashIndex[hashKey] = index;
  28779. }
  28780. else {
  28781. index = _hashIndex[hashKey];
  28782. }
  28783. return index;
  28784. },
  28785. // create style entries and returns array of integer indexes that can be used in cell .s property
  28786. addStyles: function (styles) {
  28787. var self = this;
  28788. return styles.map(function (style) {
  28789. return self.addStyle(style);
  28790. })
  28791. },
  28792. _duckTypeStyle: function(attributes) {
  28793. if (typeof attributes == 'object' && (attributes.patternFill || attributes.fgColor)) {
  28794. return {fill: attributes }; // this must be read via XLSX.parseFile(...)
  28795. }
  28796. else if (attributes.font || attributes.numFmt || attributes.border || attributes.fill) {
  28797. return attributes;
  28798. }
  28799. else {
  28800. return this._getStyleCSS(attributes)
  28801. }
  28802. },
  28803. _getStyleCSS: function(css) {
  28804. return css; //TODO
  28805. },
  28806. // Create an <xf> record for the style as well as corresponding <font>, <fill>, <border>, <numfmts>
  28807. // Right now this is simple and creates a <font>, <fill>, <border>, <numfmts> for every <xf>
  28808. // We could perhaps get fancier and avoid duplicating auxiliary entries as Excel presumably intended, but bother.
  28809. _addXf: function (attributes) {
  28810. var fontId = this._addFont(attributes.font);
  28811. var fillId = this._addFill(attributes.fill);
  28812. var borderId = this._addBorder(attributes.border);
  28813. var numFmtId = this._addNumFmt(attributes.numFmt);
  28814. var $xf = XmlNode('xf')
  28815. .attr("numFmtId", numFmtId)
  28816. .attr("fontId", fontId)
  28817. .attr("fillId", fillId)
  28818. .attr("borderId", borderId)
  28819. .attr("xfId", "0");
  28820. if (fontId > 0) {
  28821. $xf.attr('applyFont', "1");
  28822. }
  28823. if (fillId > 0) {
  28824. $xf.attr('applyFill', "1");
  28825. }
  28826. if (borderId > 0) {
  28827. $xf.attr('applyBorder', "1");
  28828. }
  28829. if (numFmtId > 0) {
  28830. $xf.attr('applyNumberFormat', "1");
  28831. }
  28832. if (attributes.alignment) {
  28833. var $alignment = XmlNode('alignment');
  28834. if (attributes.alignment.horizontal) { $alignment.attr('horizontal', attributes.alignment.horizontal);}
  28835. if (attributes.alignment.vertical) { $alignment.attr('vertical', attributes.alignment.vertical);}
  28836. if (attributes.alignment.indent) { $alignment.attr('indent', attributes.alignment.indent);}
  28837. if (attributes.alignment.readingOrder) { $alignment.attr('readingOrder', attributes.alignment.readingOrder);}
  28838. if (attributes.alignment.wrapText) { $alignment.attr('wrapText', attributes.alignment.wrapText);}
  28839. if (attributes.alignment.textRotation!=undefined) { $alignment.attr('textRotation', attributes.alignment.textRotation);}
  28840. $xf.append($alignment).attr('applyAlignment',1)
  28841. }
  28842. this.$cellXfs.append($xf);
  28843. var count = +this.$cellXfs.children().length;
  28844. this.$cellXfs.attr('count', count);
  28845. return count - 1;
  28846. },
  28847. _addFont: function (attributes) {
  28848. if (!attributes) { return 0; }
  28849. var $font = XmlNode('font')
  28850. .append(XmlNode('sz').attr('val', attributes.sz || this.defaultStyle.font.sz))
  28851. .append(XmlNode('name').attr('val', attributes.name || this.defaultStyle.font.name))
  28852. if (attributes.bold) $font.append(XmlNode('b'));
  28853. if (attributes.underline) $font.append(XmlNode('u'));
  28854. if (attributes.italic) $font.append(XmlNode('i'));
  28855. if (attributes.strike) $font.append(XmlNode('strike'));
  28856. if (attributes.outline) $font.append(XmlNode('outline'));
  28857. if (attributes.shadow) $font.append(XmlNode('shadow'));
  28858. if (attributes.vertAlign) {
  28859. $font.append(XmlNode('vertAlign').attr('val', attributes.vertAlign))
  28860. }
  28861. if (attributes.color) {
  28862. if (attributes.color.theme) {
  28863. $font.append(XmlNode('color').attr('theme', attributes.color.theme))
  28864. if (attributes.color.tint) { //tint only if theme
  28865. $font.append(XmlNode('tint').attr('theme', attributes.color.tint))
  28866. }
  28867. } else if (attributes.color.rgb) { // not both rgb and theme
  28868. $font.append(XmlNode('color').attr('rgb', attributes.color.rgb))
  28869. }
  28870. }
  28871. this.$fonts.append($font);
  28872. var count = this.$fonts.children().length;
  28873. this.$fonts.attr('count', count);
  28874. return count - 1;
  28875. },
  28876. _addNumFmt: function (numFmt) {
  28877. if (!numFmt) { return 0; }
  28878. if (typeof numFmt == 'string') {
  28879. var numFmtIdx = fmt_table[numFmt];
  28880. if (numFmtIdx >= 0) {
  28881. return numFmtIdx; // we found a match against built in formats
  28882. }
  28883. }
  28884. if (/^[0-9]+$/.exec(numFmt)) {
  28885. return numFmt; // we're matching an integer against some known code
  28886. }
  28887. numFmt = numFmt
  28888. .replace(/&/g, '&amp;')
  28889. .replace(/</g, '&lt;')
  28890. .replace(/>/g, '&gt;')
  28891. .replace(/"/g, '&quot;')
  28892. .replace(/'/g, '&apos;');
  28893. var $numFmt = XmlNode('numFmt')
  28894. .attr('numFmtId', (++customNumFmtId))
  28895. .attr('formatCode', numFmt);
  28896. this.$numFmts.append($numFmt);
  28897. var count = this.$numFmts.children().length;
  28898. this.$numFmts.attr('count', count);
  28899. return customNumFmtId ;
  28900. },
  28901. _addFill: function (attributes) {
  28902. if (!attributes) { return 0; }
  28903. var $patternFill = XmlNode('patternFill')
  28904. .attr('patternType', attributes.patternType || 'solid');
  28905. if (attributes.fgColor) {
  28906. var $fgColor = XmlNode('fgColor');
  28907. //Excel doesn't like it when we set both rgb and theme+tint, but xlsx.parseFile() sets both
  28908. //var $fgColor = createElement('<fgColor/>', null, null, {xmlMode: true}).attr(attributes.fgColor)
  28909. if (attributes.fgColor.rgb) {
  28910. if (attributes.fgColor.rgb.length == 6) {
  28911. attributes.fgColor.rgb = "FF" + attributes.fgColor.rgb /// add alpha to an RGB as Excel expects aRGB
  28912. }
  28913. $fgColor.attr('rgb', attributes.fgColor.rgb);
  28914. $patternFill.append($fgColor);
  28915. }
  28916. else if (attributes.fgColor.theme) {
  28917. $fgColor.attr('theme', attributes.fgColor.theme);
  28918. if (attributes.fgColor.tint) {
  28919. $fgColor.attr('tint', attributes.fgColor.tint);
  28920. }
  28921. $patternFill.append($fgColor);
  28922. }
  28923. if (!attributes.bgColor) {
  28924. attributes.bgColor = { "indexed": "64"}
  28925. }
  28926. }
  28927. if (attributes.bgColor) {
  28928. var $bgColor = XmlNode('bgColor').attr(attributes.bgColor);
  28929. $patternFill.append($bgColor);
  28930. }
  28931. var $fill = XmlNode('fill')
  28932. .append($patternFill);
  28933. this.$fills.append($fill);
  28934. var count = this.$fills.children().length;
  28935. this.$fills.attr('count', count);
  28936. return count - 1;
  28937. },
  28938. _getSubBorder: function(direction, spec) {
  28939. var $direction = XmlNode(direction);
  28940. if (spec){
  28941. if (spec.style) $direction.attr('style', spec.style);
  28942. if (spec.color) {
  28943. var $color = XmlNode('color');
  28944. if (spec.color.auto) {
  28945. $color.attr('auto', spec.color.auto);
  28946. }
  28947. else if (spec.color.rgb) {
  28948. $color.attr('rgb', spec.color.rgb);
  28949. }
  28950. else if (spec.color.theme || spec.color.tint) {
  28951. $color.attr('theme', spec.color.theme || "1");
  28952. $color.attr('tint', spec.color.tint || "0");
  28953. }
  28954. $direction.append($color)
  28955. }
  28956. }
  28957. return $direction;
  28958. },
  28959. _addBorder: function (attributes) {
  28960. if (!attributes) { return 0; }
  28961. var self = this;
  28962. var $border = XmlNode('border')
  28963. .attr("diagonalUp",attributes.diagonalUp)
  28964. .attr("diagonalDown",attributes.diagonalDown);
  28965. var directions = ["left","right","top","bottom","diagonal"];
  28966. directions.forEach(function(direction) {
  28967. $border.append(self._getSubBorder(direction, attributes[direction]))
  28968. });
  28969. this.$borders.append($border);
  28970. var count = this.$borders.children().length;
  28971. this.$borders.attr('count', count);
  28972. return count -1;
  28973. },
  28974. toXml: function () {
  28975. return this.$styles.toXml();
  28976. }
  28977. }.initialize(options||{});
  28978. }
  28979. /**
  28980. * 样式魔改区 -- end
  28981. */
  28982. XLSX.parse_xlscfb = parse_xlscfb;
  28983. XLSX.parse_ods = parse_ods;
  28984. XLSX.parse_fods = parse_fods;
  28985. XLSX.write_ods = write_ods;
  28986. XLSX.parse_zip = parse_zip;
  28987. XLSX.read = readSync; //xlsread
  28988. XLSX.readFile = readFileSync; //readFile
  28989. XLSX.readFileSync = readFileSync;
  28990. XLSX.write = writeSync;
  28991. XLSX.writeFile = writeFileSync;
  28992. XLSX.writeFileSync = writeFileSync;
  28993. XLSX.writeFileAsync = writeFileAsync;
  28994. XLSX.utils = utils;
  28995. XLSX.SSF = SSF;
  28996. XLSX.CFB = CFB;
  28997. }
  28998. /*global define */
  28999. if(typeof exports !== 'undefined') make_xlsx_lib(exports);
  29000. else if(typeof module !== 'undefined' && module.exports) make_xlsx_lib(module.exports);
  29001. else if(typeof define === 'function' && define.amd) define('xlsx', function() { if(!XLSX.version) make_xlsx_lib(XLSX); return XLSX; });
  29002. else make_xlsx_lib(XLSX);
  29003. /*exported XLS, ODS */
  29004. var XLS = XLSX, ODS = XLSX;/*---------split--------*//*
  29005. Copyright (c) 2010, Linden Research, Inc.
  29006. Copyright (c) 2014, Joshua Bell
  29007. Permission is hereby granted, free of charge, to any person obtaining a copy
  29008. of this software and associated documentation files (the "Software"), to deal
  29009. in the Software without restriction, including without limitation the rights
  29010. to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  29011. copies of the Software, and to permit persons to whom the Software is
  29012. furnished to do so, subject to the following conditions:
  29013. The above copyright notice and this permission notice shall be included in
  29014. all copies or substantial portions of the Software.
  29015. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  29016. IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  29017. FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  29018. AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  29019. LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  29020. OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  29021. THE SOFTWARE.
  29022. $/LicenseInfo$
  29023. */
  29024. // Original can be found at:
  29025. // https://bitbucket.org/lindenlab/llsd
  29026. // Modifications by Joshua Bell inexorabletash@gmail.com
  29027. // https://github.com/inexorabletash/polyfill
  29028. // ES3/ES5 implementation of the Krhonos Typed Array Specification
  29029. // Ref: http://www.khronos.org/registry/typedarray/specs/latest/
  29030. // Date: 2011-02-01
  29031. //
  29032. // Variations:
  29033. // * Allows typed_array.get/set() as alias for subscripts (typed_array[])
  29034. // * Gradually migrating structure from Khronos spec to ES2015 spec
  29035. //
  29036. // Caveats:
  29037. // * Beyond 10000 or so entries, polyfilled array accessors (ta[0],
  29038. // etc) become memory-prohibitive, so array creation will fail. Set
  29039. // self.TYPED_ARRAY_POLYFILL_NO_ARRAY_ACCESSORS=true to disable
  29040. // creation of accessors. Your code will need to use the
  29041. // non-standard get()/set() instead, and will need to add those to
  29042. // native arrays for interop.
  29043. (function(global) {
  29044. 'use strict';
  29045. var undefined = (void 0); // Paranoia
  29046. // Beyond this value, index getters/setters (i.e. array[0], array[1]) are so slow to
  29047. // create, and consume so much memory, that the browser appears frozen.
  29048. var MAX_ARRAY_LENGTH = 1e5;
  29049. // Approximations of internal ECMAScript conversion functions
  29050. function Type(v) {
  29051. switch(typeof v) {
  29052. case 'undefined': return 'undefined';
  29053. case 'boolean': return 'boolean';
  29054. case 'number': return 'number';
  29055. case 'string': return 'string';
  29056. default: return v === null ? 'null' : 'object';
  29057. }
  29058. }
  29059. // Class returns internal [[Class]] property, used to avoid cross-frame instanceof issues:
  29060. function Class(v) { return Object.prototype.toString.call(v).replace(/^\[object *|\]$/g, ''); }
  29061. function IsCallable(o) { return typeof o === 'function'; }
  29062. function ToObject(v) {
  29063. if (v === null || v === undefined) throw TypeError();
  29064. return Object(v);
  29065. }
  29066. function ToInt32(v) { return v >> 0; }
  29067. function ToUint32(v) { return v >>> 0; }
  29068. // Snapshot intrinsics
  29069. var LN2 = Math.LN2,
  29070. abs = Math.abs,
  29071. floor = Math.floor,
  29072. log = Math.log,
  29073. max = Math.max,
  29074. min = Math.min,
  29075. pow = Math.pow,
  29076. round = Math.round;
  29077. // emulate ES5 getter/setter API using legacy APIs
  29078. // http://blogs.msdn.com/b/ie/archive/2010/09/07/transitioning-existing-code-to-the-es5-getter-setter-apis.aspx
  29079. // (second clause tests for Object.defineProperty() in IE<9 that only supports extending DOM prototypes, but
  29080. // note that IE<9 does not support __defineGetter__ or __defineSetter__ so it just renders the method harmless)
  29081. (function() {
  29082. var orig = Object.defineProperty;
  29083. var dom_only = !(function(){try{return Object.defineProperty({},'x',{});}catch(_){return false;}}());
  29084. if (!orig || dom_only) {
  29085. Object.defineProperty = function (o, prop, desc) {
  29086. // In IE8 try built-in implementation for defining properties on DOM prototypes.
  29087. if (orig)
  29088. try { return orig(o, prop, desc); } catch (_) {}
  29089. if (o !== Object(o))
  29090. throw TypeError('Object.defineProperty called on non-object');
  29091. if (Object.prototype.__defineGetter__ && ('get' in desc))
  29092. Object.prototype.__defineGetter__.call(o, prop, desc.get);
  29093. if (Object.prototype.__defineSetter__ && ('set' in desc))
  29094. Object.prototype.__defineSetter__.call(o, prop, desc.set);
  29095. if ('value' in desc)
  29096. o[prop] = desc.value;
  29097. return o;
  29098. };
  29099. }
  29100. }());
  29101. // ES5: Make obj[index] an alias for obj._getter(index)/obj._setter(index, value)
  29102. // for index in 0 ... obj.length
  29103. function makeArrayAccessors(obj) {
  29104. if ('TYPED_ARRAY_POLYFILL_NO_ARRAY_ACCESSORS' in global)
  29105. return;
  29106. if (obj.length > MAX_ARRAY_LENGTH) throw RangeError('Array too large for polyfill');
  29107. function makeArrayAccessor(index) {
  29108. Object.defineProperty(obj, index, {
  29109. 'get': function() { return obj._getter(index); },
  29110. 'set': function(v) { obj._setter(index, v); },
  29111. enumerable: true,
  29112. configurable: false
  29113. });
  29114. }
  29115. var i;
  29116. for (i = 0; i < obj.length; i += 1) {
  29117. makeArrayAccessor(i);
  29118. }
  29119. }
  29120. // Internal conversion functions:
  29121. // pack<Type>() - take a number (interpreted as Type), output a byte array
  29122. // unpack<Type>() - take a byte array, output a Type-like number
  29123. function as_signed(value, bits) { var s = 32 - bits; return (value << s) >> s; }
  29124. function as_unsigned(value, bits) { var s = 32 - bits; return (value << s) >>> s; }
  29125. function packI8(n) { return [n & 0xff]; }
  29126. function unpackI8(bytes) { return as_signed(bytes[0], 8); }
  29127. function packU8(n) { return [n & 0xff]; }
  29128. function unpackU8(bytes) { return as_unsigned(bytes[0], 8); }
  29129. function packU8Clamped(n) { n = round(Number(n)); return [n < 0 ? 0 : n > 0xff ? 0xff : n & 0xff]; }
  29130. function packI16(n) { return [n & 0xff, (n >> 8) & 0xff]; }
  29131. function unpackI16(bytes) { return as_signed(bytes[1] << 8 | bytes[0], 16); }
  29132. function packU16(n) { return [n & 0xff, (n >> 8) & 0xff]; }
  29133. function unpackU16(bytes) { return as_unsigned(bytes[1] << 8 | bytes[0], 16); }
  29134. function packI32(n) { return [n & 0xff, (n >> 8) & 0xff, (n >> 16) & 0xff, (n >> 24) & 0xff]; }
  29135. function unpackI32(bytes) { return as_signed(bytes[3] << 24 | bytes[2] << 16 | bytes[1] << 8 | bytes[0], 32); }
  29136. function packU32(n) { return [n & 0xff, (n >> 8) & 0xff, (n >> 16) & 0xff, (n >> 24) & 0xff]; }
  29137. function unpackU32(bytes) { return as_unsigned(bytes[3] << 24 | bytes[2] << 16 | bytes[1] << 8 | bytes[0], 32); }
  29138. function packIEEE754(v, ebits, fbits) {
  29139. var bias = (1 << (ebits - 1)) - 1;
  29140. function roundToEven(n) {
  29141. var w = floor(n), f = n - w;
  29142. if (f < 0.5)
  29143. return w;
  29144. if (f > 0.5)
  29145. return w + 1;
  29146. return w % 2 ? w + 1 : w;
  29147. }
  29148. // Compute sign, exponent, fraction
  29149. var s, e, f;
  29150. if (v !== v) {
  29151. // NaN
  29152. // http://dev.w3.org/2006/webapi/WebIDL/#es-type-mapping
  29153. e = (1 << ebits) - 1; f = pow(2, fbits - 1); s = 0;
  29154. } else if (v === Infinity || v === -Infinity) {
  29155. e = (1 << ebits) - 1; f = 0; s = (v < 0) ? 1 : 0;
  29156. } else if (v === 0) {
  29157. e = 0; f = 0; s = (1 / v === -Infinity) ? 1 : 0;
  29158. } else {
  29159. s = v < 0;
  29160. v = abs(v);
  29161. if (v >= pow(2, 1 - bias)) {
  29162. // Normalized
  29163. e = min(floor(log(v) / LN2), 1023);
  29164. var significand = v / pow(2, e);
  29165. if (significand < 1) {
  29166. e -= 1;
  29167. significand *= 2;
  29168. }
  29169. if (significand >= 2) {
  29170. e += 1;
  29171. significand /= 2;
  29172. }
  29173. var d = pow(2, fbits);
  29174. f = roundToEven(significand * d) - d;
  29175. e += bias;
  29176. if (f / d >= 1) {
  29177. e += 1;
  29178. f = 0;
  29179. }
  29180. if (e > 2 * bias) {
  29181. // Overflow
  29182. e = (1 << ebits) - 1;
  29183. f = 0;
  29184. }
  29185. } else {
  29186. // Denormalized
  29187. e = 0;
  29188. f = roundToEven(v / pow(2, 1 - bias - fbits));
  29189. }
  29190. }
  29191. // Pack sign, exponent, fraction
  29192. var bits = [], i;
  29193. for (i = fbits; i; i -= 1) { bits.push(f % 2 ? 1 : 0); f = floor(f / 2); }
  29194. for (i = ebits; i; i -= 1) { bits.push(e % 2 ? 1 : 0); e = floor(e / 2); }
  29195. bits.push(s ? 1 : 0);
  29196. bits.reverse();
  29197. var str = bits.join('');
  29198. // Bits to bytes
  29199. var bytes = [];
  29200. while (str.length) {
  29201. bytes.unshift(parseInt(str.substring(0, 8), 2));
  29202. str = str.substring(8);
  29203. }
  29204. return bytes;
  29205. }
  29206. function unpackIEEE754(bytes, ebits, fbits) {
  29207. // Bytes to bits
  29208. var bits = [], i, j, b, str,
  29209. bias, s, e, f;
  29210. for (i = 0; i < bytes.length; ++i) {
  29211. b = bytes[i];
  29212. for (j = 8; j; j -= 1) {
  29213. bits.push(b % 2 ? 1 : 0); b = b >> 1;
  29214. }
  29215. }
  29216. bits.reverse();
  29217. str = bits.join('');
  29218. // Unpack sign, exponent, fraction
  29219. bias = (1 << (ebits - 1)) - 1;
  29220. s = parseInt(str.substring(0, 1), 2) ? -1 : 1;
  29221. e = parseInt(str.substring(1, 1 + ebits), 2);
  29222. f = parseInt(str.substring(1 + ebits), 2);
  29223. // Produce number
  29224. if (e === (1 << ebits) - 1) {
  29225. return f !== 0 ? NaN : s * Infinity;
  29226. } else if (e > 0) {
  29227. // Normalized
  29228. return s * pow(2, e - bias) * (1 + f / pow(2, fbits));
  29229. } else if (f !== 0) {
  29230. // Denormalized
  29231. return s * pow(2, -(bias - 1)) * (f / pow(2, fbits));
  29232. } else {
  29233. return s < 0 ? -0 : 0;
  29234. }
  29235. }
  29236. function unpackF64(b) { return unpackIEEE754(b, 11, 52); }
  29237. function packF64(v) { return packIEEE754(v, 11, 52); }
  29238. function unpackF32(b) { return unpackIEEE754(b, 8, 23); }
  29239. function packF32(v) { return packIEEE754(v, 8, 23); }
  29240. //
  29241. // 3 The ArrayBuffer Type
  29242. //
  29243. (function() {
  29244. function ArrayBuffer(length) {
  29245. length = ToInt32(length);
  29246. if (length < 0) throw RangeError('ArrayBuffer size is not a small enough positive integer.');
  29247. Object.defineProperty(this, 'byteLength', {value: length});
  29248. Object.defineProperty(this, '_bytes', {value: Array(length)});
  29249. for (var i = 0; i < length; i += 1)
  29250. this._bytes[i] = 0;
  29251. }
  29252. global.ArrayBuffer = global.ArrayBuffer || ArrayBuffer;
  29253. //
  29254. // 5 The Typed Array View Types
  29255. //
  29256. function $TypedArray$() {
  29257. // %TypedArray% ( length )
  29258. if (!arguments.length || typeof arguments[0] !== 'object') {
  29259. return (function(length) {
  29260. length = ToInt32(length);
  29261. if (length < 0) throw RangeError('length is not a small enough positive integer.');
  29262. Object.defineProperty(this, 'length', {value: length});
  29263. Object.defineProperty(this, 'byteLength', {value: length * this.BYTES_PER_ELEMENT});
  29264. Object.defineProperty(this, 'buffer', {value: new ArrayBuffer(this.byteLength)});
  29265. Object.defineProperty(this, 'byteOffset', {value: 0});
  29266. }).apply(this, arguments);
  29267. }
  29268. // %TypedArray% ( typedArray )
  29269. if (arguments.length >= 1 &&
  29270. Type(arguments[0]) === 'object' &&
  29271. arguments[0] instanceof $TypedArray$) {
  29272. return (function(typedArray){
  29273. if (this.constructor !== typedArray.constructor) throw TypeError();
  29274. var byteLength = typedArray.length * this.BYTES_PER_ELEMENT;
  29275. Object.defineProperty(this, 'buffer', {value: new ArrayBuffer(byteLength)});
  29276. Object.defineProperty(this, 'byteLength', {value: byteLength});
  29277. Object.defineProperty(this, 'byteOffset', {value: 0});
  29278. Object.defineProperty(this, 'length', {value: typedArray.length});
  29279. for (var i = 0; i < this.length; i += 1)
  29280. this._setter(i, typedArray._getter(i));
  29281. }).apply(this, arguments);
  29282. }
  29283. // %TypedArray% ( array )
  29284. if (arguments.length >= 1 &&
  29285. Type(arguments[0]) === 'object' &&
  29286. !(arguments[0] instanceof $TypedArray$) &&
  29287. !(arguments[0] instanceof ArrayBuffer || Class(arguments[0]) === 'ArrayBuffer')) {
  29288. return (function(array) {
  29289. var byteLength = array.length * this.BYTES_PER_ELEMENT;
  29290. Object.defineProperty(this, 'buffer', {value: new ArrayBuffer(byteLength)});
  29291. Object.defineProperty(this, 'byteLength', {value: byteLength});
  29292. Object.defineProperty(this, 'byteOffset', {value: 0});
  29293. Object.defineProperty(this, 'length', {value: array.length});
  29294. for (var i = 0; i < this.length; i += 1) {
  29295. var s = array[i];
  29296. this._setter(i, Number(s));
  29297. }
  29298. }).apply(this, arguments);
  29299. }
  29300. // %TypedArray% ( buffer, byteOffset=0, length=undefined )
  29301. if (arguments.length >= 1 &&
  29302. Type(arguments[0]) === 'object' &&
  29303. (arguments[0] instanceof ArrayBuffer || Class(arguments[0]) === 'ArrayBuffer')) {
  29304. return (function(buffer, byteOffset, length) {
  29305. byteOffset = ToUint32(byteOffset);
  29306. if (byteOffset > buffer.byteLength)
  29307. throw RangeError('byteOffset out of range');
  29308. // The given byteOffset must be a multiple of the element
  29309. // size of the specific type, otherwise an exception is raised.
  29310. if (byteOffset % this.BYTES_PER_ELEMENT)
  29311. throw RangeError('buffer length minus the byteOffset is not a multiple of the element size.');
  29312. if (length === undefined) {
  29313. var byteLength = buffer.byteLength - byteOffset;
  29314. if (byteLength % this.BYTES_PER_ELEMENT)
  29315. throw RangeError('length of buffer minus byteOffset not a multiple of the element size');
  29316. length = byteLength / this.BYTES_PER_ELEMENT;
  29317. } else {
  29318. length = ToUint32(length);
  29319. byteLength = length * this.BYTES_PER_ELEMENT;
  29320. }
  29321. if ((byteOffset + byteLength) > buffer.byteLength)
  29322. throw RangeError('byteOffset and length reference an area beyond the end of the buffer');
  29323. Object.defineProperty(this, 'buffer', {value: buffer});
  29324. Object.defineProperty(this, 'byteLength', {value: byteLength});
  29325. Object.defineProperty(this, 'byteOffset', {value: byteOffset});
  29326. Object.defineProperty(this, 'length', {value: length});
  29327. }).apply(this, arguments);
  29328. }
  29329. // %TypedArray% ( all other argument combinations )
  29330. throw TypeError();
  29331. }
  29332. // Properties of the %TypedArray Instrinsic Object
  29333. // %TypedArray%.from ( source , mapfn=undefined, thisArg=undefined )
  29334. Object.defineProperty($TypedArray$, 'from', {value: function(iterable) {
  29335. return new this(iterable);
  29336. }});
  29337. // %TypedArray%.of ( ...items )
  29338. Object.defineProperty($TypedArray$, 'of', {value: function(/*...items*/) {
  29339. return new this(arguments);
  29340. }});
  29341. // %TypedArray%.prototype
  29342. var $TypedArrayPrototype$ = {};
  29343. $TypedArray$.prototype = $TypedArrayPrototype$;
  29344. // WebIDL: getter type (unsigned long index);
  29345. Object.defineProperty($TypedArray$.prototype, '_getter', {value: function(index) {
  29346. if (arguments.length < 1) throw SyntaxError('Not enough arguments');
  29347. index = ToUint32(index);
  29348. if (index >= this.length)
  29349. return undefined;
  29350. var bytes = [], i, o;
  29351. for (i = 0, o = this.byteOffset + index * this.BYTES_PER_ELEMENT;
  29352. i < this.BYTES_PER_ELEMENT;
  29353. i += 1, o += 1) {
  29354. bytes.push(this.buffer._bytes[o]);
  29355. }
  29356. return this._unpack(bytes);
  29357. }});
  29358. // NONSTANDARD: convenience alias for getter: type get(unsigned long index);
  29359. Object.defineProperty($TypedArray$.prototype, 'get', {value: $TypedArray$.prototype._getter});
  29360. // WebIDL: setter void (unsigned long index, type value);
  29361. Object.defineProperty($TypedArray$.prototype, '_setter', {value: function(index, value) {
  29362. if (arguments.length < 2) throw SyntaxError('Not enough arguments');
  29363. index = ToUint32(index);
  29364. if (index >= this.length)
  29365. return;
  29366. var bytes = this._pack(value), i, o;
  29367. for (i = 0, o = this.byteOffset + index * this.BYTES_PER_ELEMENT;
  29368. i < this.BYTES_PER_ELEMENT;
  29369. i += 1, o += 1) {
  29370. this.buffer._bytes[o] = bytes[i];
  29371. }
  29372. }});
  29373. // get %TypedArray%.prototype.buffer
  29374. // get %TypedArray%.prototype.byteLength
  29375. // get %TypedArray%.prototype.byteOffset
  29376. // -- applied directly to the object in the constructor
  29377. // %TypedArray%.prototype.constructor
  29378. Object.defineProperty($TypedArray$.prototype, 'constructor', {value: $TypedArray$});
  29379. // %TypedArray%.prototype.copyWithin (target, start, end = this.length )
  29380. Object.defineProperty($TypedArray$.prototype, 'copyWithin', {value: function(target, start) {
  29381. var end = arguments[2];
  29382. var o = ToObject(this);
  29383. var lenVal = o.length;
  29384. var len = ToUint32(lenVal);
  29385. len = max(len, 0);
  29386. var relativeTarget = ToInt32(target);
  29387. var to;
  29388. if (relativeTarget < 0)
  29389. to = max(len + relativeTarget, 0);
  29390. else
  29391. to = min(relativeTarget, len);
  29392. var relativeStart = ToInt32(start);
  29393. var from;
  29394. if (relativeStart < 0)
  29395. from = max(len + relativeStart, 0);
  29396. else
  29397. from = min(relativeStart, len);
  29398. var relativeEnd;
  29399. if (end === undefined)
  29400. relativeEnd = len;
  29401. else
  29402. relativeEnd = ToInt32(end);
  29403. var final;
  29404. if (relativeEnd < 0)
  29405. final = max(len + relativeEnd, 0);
  29406. else
  29407. final = min(relativeEnd, len);
  29408. var count = min(final - from, len - to);
  29409. var direction;
  29410. if (from < to && to < from + count) {
  29411. direction = -1;
  29412. from = from + count - 1;
  29413. to = to + count - 1;
  29414. } else {
  29415. direction = 1;
  29416. }
  29417. while (count > 0) {
  29418. o._setter(to, o._getter(from));
  29419. from = from + direction;
  29420. to = to + direction;
  29421. count = count - 1;
  29422. }
  29423. return o;
  29424. }});
  29425. // %TypedArray%.prototype.entries ( )
  29426. // -- defined in es6.js to shim browsers w/ native TypedArrays
  29427. // %TypedArray%.prototype.every ( callbackfn, thisArg = undefined )
  29428. Object.defineProperty($TypedArray$.prototype, 'every', {value: function(callbackfn) {
  29429. if (this === undefined || this === null) throw TypeError();
  29430. var t = Object(this);
  29431. var len = ToUint32(t.length);
  29432. if (!IsCallable(callbackfn)) throw TypeError();
  29433. var thisArg = arguments[1];
  29434. for (var i = 0; i < len; i++) {
  29435. if (!callbackfn.call(thisArg, t._getter(i), i, t))
  29436. return false;
  29437. }
  29438. return true;
  29439. }});
  29440. // %TypedArray%.prototype.fill (value, start = 0, end = this.length )
  29441. Object.defineProperty($TypedArray$.prototype, 'fill', {value: function(value) {
  29442. var start = arguments[1],
  29443. end = arguments[2];
  29444. var o = ToObject(this);
  29445. var lenVal = o.length;
  29446. var len = ToUint32(lenVal);
  29447. len = max(len, 0);
  29448. var relativeStart = ToInt32(start);
  29449. var k;
  29450. if (relativeStart < 0)
  29451. k = max((len + relativeStart), 0);
  29452. else
  29453. k = min(relativeStart, len);
  29454. var relativeEnd;
  29455. if (end === undefined)
  29456. relativeEnd = len;
  29457. else
  29458. relativeEnd = ToInt32(end);
  29459. var final;
  29460. if (relativeEnd < 0)
  29461. final = max((len + relativeEnd), 0);
  29462. else
  29463. final = min(relativeEnd, len);
  29464. while (k < final) {
  29465. o._setter(k, value);
  29466. k += 1;
  29467. }
  29468. return o;
  29469. }});
  29470. // %TypedArray%.prototype.filter ( callbackfn, thisArg = undefined )
  29471. Object.defineProperty($TypedArray$.prototype, 'filter', {value: function(callbackfn) {
  29472. if (this === undefined || this === null) throw TypeError();
  29473. var t = Object(this);
  29474. var len = ToUint32(t.length);
  29475. if (!IsCallable(callbackfn)) throw TypeError();
  29476. var res = [];
  29477. var thisp = arguments[1];
  29478. for (var i = 0; i < len; i++) {
  29479. var val = t._getter(i); // in case fun mutates this
  29480. if (callbackfn.call(thisp, val, i, t))
  29481. res.push(val);
  29482. }
  29483. return new this.constructor(res);
  29484. }});
  29485. // %TypedArray%.prototype.find (predicate, thisArg = undefined)
  29486. Object.defineProperty($TypedArray$.prototype, 'find', {value: function(predicate) {
  29487. var o = ToObject(this);
  29488. var lenValue = o.length;
  29489. var len = ToUint32(lenValue);
  29490. if (!IsCallable(predicate)) throw TypeError();
  29491. var t = arguments.length > 1 ? arguments[1] : undefined;
  29492. var k = 0;
  29493. while (k < len) {
  29494. var kValue = o._getter(k);
  29495. var testResult = predicate.call(t, kValue, k, o);
  29496. if (Boolean(testResult))
  29497. return kValue;
  29498. ++k;
  29499. }
  29500. return undefined;
  29501. }});
  29502. // %TypedArray%.prototype.findIndex ( predicate, thisArg = undefined )
  29503. Object.defineProperty($TypedArray$.prototype, 'findIndex', {value: function(predicate) {
  29504. var o = ToObject(this);
  29505. var lenValue = o.length;
  29506. var len = ToUint32(lenValue);
  29507. if (!IsCallable(predicate)) throw TypeError();
  29508. var t = arguments.length > 1 ? arguments[1] : undefined;
  29509. var k = 0;
  29510. while (k < len) {
  29511. var kValue = o._getter(k);
  29512. var testResult = predicate.call(t, kValue, k, o);
  29513. if (Boolean(testResult))
  29514. return k;
  29515. ++k;
  29516. }
  29517. return -1;
  29518. }});
  29519. // %TypedArray%.prototype.forEach ( callbackfn, thisArg = undefined )
  29520. Object.defineProperty($TypedArray$.prototype, 'forEach', {value: function(callbackfn) {
  29521. if (this === undefined || this === null) throw TypeError();
  29522. var t = Object(this);
  29523. var len = ToUint32(t.length);
  29524. if (!IsCallable(callbackfn)) throw TypeError();
  29525. var thisp = arguments[1];
  29526. for (var i = 0; i < len; i++)
  29527. callbackfn.call(thisp, t._getter(i), i, t);
  29528. }});
  29529. // %TypedArray%.prototype.indexOf (searchElement, fromIndex = 0 )
  29530. Object.defineProperty($TypedArray$.prototype, 'indexOf', {value: function(searchElement) {
  29531. if (this === undefined || this === null) throw TypeError();
  29532. var t = Object(this);
  29533. var len = ToUint32(t.length);
  29534. if (len === 0) return -1;
  29535. var n = 0;
  29536. if (arguments.length > 0) {
  29537. n = Number(arguments[1]);
  29538. if (n !== n) {
  29539. n = 0;
  29540. } else if (n !== 0 && n !== (1 / 0) && n !== -(1 / 0)) {
  29541. n = (n > 0 || -1) * floor(abs(n));
  29542. }
  29543. }
  29544. if (n >= len) return -1;
  29545. var k = n >= 0 ? n : max(len - abs(n), 0);
  29546. for (; k < len; k++) {
  29547. if (t._getter(k) === searchElement) {
  29548. return k;
  29549. }
  29550. }
  29551. return -1;
  29552. }});
  29553. // %TypedArray%.prototype.join ( separator )
  29554. Object.defineProperty($TypedArray$.prototype, 'join', {value: function(separator) {
  29555. if (this === undefined || this === null) throw TypeError();
  29556. var t = Object(this);
  29557. var len = ToUint32(t.length);
  29558. var tmp = Array(len);
  29559. for (var i = 0; i < len; ++i)
  29560. tmp[i] = t._getter(i);
  29561. return tmp.join(separator === undefined ? ',' : separator); // Hack for IE7
  29562. }});
  29563. // %TypedArray%.prototype.keys ( )
  29564. // -- defined in es6.js to shim browsers w/ native TypedArrays
  29565. // %TypedArray%.prototype.lastIndexOf ( searchElement, fromIndex = this.length-1 )
  29566. Object.defineProperty($TypedArray$.prototype, 'lastIndexOf', {value: function(searchElement) {
  29567. if (this === undefined || this === null) throw TypeError();
  29568. var t = Object(this);
  29569. var len = ToUint32(t.length);
  29570. if (len === 0) return -1;
  29571. var n = len;
  29572. if (arguments.length > 1) {
  29573. n = Number(arguments[1]);
  29574. if (n !== n) {
  29575. n = 0;
  29576. } else if (n !== 0 && n !== (1 / 0) && n !== -(1 / 0)) {
  29577. n = (n > 0 || -1) * floor(abs(n));
  29578. }
  29579. }
  29580. var k = n >= 0 ? min(n, len - 1) : len - abs(n);
  29581. for (; k >= 0; k--) {
  29582. if (t._getter(k) === searchElement)
  29583. return k;
  29584. }
  29585. return -1;
  29586. }});
  29587. // get %TypedArray%.prototype.length
  29588. // -- applied directly to the object in the constructor
  29589. // %TypedArray%.prototype.map ( callbackfn, thisArg = undefined )
  29590. Object.defineProperty($TypedArray$.prototype, 'map', {value: function(callbackfn) {
  29591. if (this === undefined || this === null) throw TypeError();
  29592. var t = Object(this);
  29593. var len = ToUint32(t.length);
  29594. if (!IsCallable(callbackfn)) throw TypeError();
  29595. var res = []; res.length = len;
  29596. var thisp = arguments[1];
  29597. for (var i = 0; i < len; i++)
  29598. res[i] = callbackfn.call(thisp, t._getter(i), i, t);
  29599. return new this.constructor(res);
  29600. }});
  29601. // %TypedArray%.prototype.reduce ( callbackfn [, initialValue] )
  29602. Object.defineProperty($TypedArray$.prototype, 'reduce', {value: function(callbackfn) {
  29603. if (this === undefined || this === null) throw TypeError();
  29604. var t = Object(this);
  29605. var len = ToUint32(t.length);
  29606. if (!IsCallable(callbackfn)) throw TypeError();
  29607. // no value to return if no initial value and an empty array
  29608. if (len === 0 && arguments.length === 1) throw TypeError();
  29609. var k = 0;
  29610. var accumulator;
  29611. if (arguments.length >= 2) {
  29612. accumulator = arguments[1];
  29613. } else {
  29614. accumulator = t._getter(k++);
  29615. }
  29616. while (k < len) {
  29617. accumulator = callbackfn.call(undefined, accumulator, t._getter(k), k, t);
  29618. k++;
  29619. }
  29620. return accumulator;
  29621. }});
  29622. // %TypedArray%.prototype.reduceRight ( callbackfn [, initialValue] )
  29623. Object.defineProperty($TypedArray$.prototype, 'reduceRight', {value: function(callbackfn) {
  29624. if (this === undefined || this === null) throw TypeError();
  29625. var t = Object(this);
  29626. var len = ToUint32(t.length);
  29627. if (!IsCallable(callbackfn)) throw TypeError();
  29628. // no value to return if no initial value, empty array
  29629. if (len === 0 && arguments.length === 1) throw TypeError();
  29630. var k = len - 1;
  29631. var accumulator;
  29632. if (arguments.length >= 2) {
  29633. accumulator = arguments[1];
  29634. } else {
  29635. accumulator = t._getter(k--);
  29636. }
  29637. while (k >= 0) {
  29638. accumulator = callbackfn.call(undefined, accumulator, t._getter(k), k, t);
  29639. k--;
  29640. }
  29641. return accumulator;
  29642. }});
  29643. // %TypedArray%.prototype.reverse ( )
  29644. Object.defineProperty($TypedArray$.prototype, 'reverse', {value: function() {
  29645. if (this === undefined || this === null) throw TypeError();
  29646. var t = Object(this);
  29647. var len = ToUint32(t.length);
  29648. var half = floor(len / 2);
  29649. for (var i = 0, j = len - 1; i < half; ++i, --j) {
  29650. var tmp = t._getter(i);
  29651. t._setter(i, t._getter(j));
  29652. t._setter(j, tmp);
  29653. }
  29654. return t;
  29655. }});
  29656. // %TypedArray%.prototype.set(array, offset = 0 )
  29657. // %TypedArray%.prototype.set(typedArray, offset = 0 )
  29658. // WebIDL: void set(TypedArray array, optional unsigned long offset);
  29659. // WebIDL: void set(sequence<type> array, optional unsigned long offset);
  29660. Object.defineProperty($TypedArray$.prototype, 'set', {value: function(index, value) {
  29661. if (arguments.length < 1) throw SyntaxError('Not enough arguments');
  29662. var array, sequence, offset, len,
  29663. i, s, d,
  29664. byteOffset, byteLength, tmp;
  29665. if (typeof arguments[0] === 'object' && arguments[0].constructor === this.constructor) {
  29666. // void set(TypedArray array, optional unsigned long offset);
  29667. array = arguments[0];
  29668. offset = ToUint32(arguments[1]);
  29669. if (offset + array.length > this.length) {
  29670. throw RangeError('Offset plus length of array is out of range');
  29671. }
  29672. byteOffset = this.byteOffset + offset * this.BYTES_PER_ELEMENT;
  29673. byteLength = array.length * this.BYTES_PER_ELEMENT;
  29674. if (array.buffer === this.buffer) {
  29675. tmp = [];
  29676. for (i = 0, s = array.byteOffset; i < byteLength; i += 1, s += 1) {
  29677. tmp[i] = array.buffer._bytes[s];
  29678. }
  29679. for (i = 0, d = byteOffset; i < byteLength; i += 1, d += 1) {
  29680. this.buffer._bytes[d] = tmp[i];
  29681. }
  29682. } else {
  29683. for (i = 0, s = array.byteOffset, d = byteOffset;
  29684. i < byteLength; i += 1, s += 1, d += 1) {
  29685. this.buffer._bytes[d] = array.buffer._bytes[s];
  29686. }
  29687. }
  29688. } else if (typeof arguments[0] === 'object' && typeof arguments[0].length !== 'undefined') {
  29689. // void set(sequence<type> array, optional unsigned long offset);
  29690. sequence = arguments[0];
  29691. len = ToUint32(sequence.length);
  29692. offset = ToUint32(arguments[1]);
  29693. if (offset + len > this.length) {
  29694. throw RangeError('Offset plus length of array is out of range');
  29695. }
  29696. for (i = 0; i < len; i += 1) {
  29697. s = sequence[i];
  29698. this._setter(offset + i, Number(s));
  29699. }
  29700. } else {
  29701. throw TypeError('Unexpected argument type(s)');
  29702. }
  29703. }});
  29704. // %TypedArray%.prototype.slice ( start, end )
  29705. Object.defineProperty($TypedArray$.prototype, 'slice', {value: function(start, end) {
  29706. var o = ToObject(this);
  29707. var lenVal = o.length;
  29708. var len = ToUint32(lenVal);
  29709. var relativeStart = ToInt32(start);
  29710. var k = (relativeStart < 0) ? max(len + relativeStart, 0) : min(relativeStart, len);
  29711. var relativeEnd = (end === undefined) ? len : ToInt32(end);
  29712. var final = (relativeEnd < 0) ? max(len + relativeEnd, 0) : min(relativeEnd, len);
  29713. var count = final - k;
  29714. var c = o.constructor;
  29715. var a = new c(count);
  29716. var n = 0;
  29717. while (k < final) {
  29718. var kValue = o._getter(k);
  29719. a._setter(n, kValue);
  29720. ++k;
  29721. ++n;
  29722. }
  29723. return a;
  29724. }});
  29725. // %TypedArray%.prototype.some ( callbackfn, thisArg = undefined )
  29726. Object.defineProperty($TypedArray$.prototype, 'some', {value: function(callbackfn) {
  29727. if (this === undefined || this === null) throw TypeError();
  29728. var t = Object(this);
  29729. var len = ToUint32(t.length);
  29730. if (!IsCallable(callbackfn)) throw TypeError();
  29731. var thisp = arguments[1];
  29732. for (var i = 0; i < len; i++) {
  29733. if (callbackfn.call(thisp, t._getter(i), i, t)) {
  29734. return true;
  29735. }
  29736. }
  29737. return false;
  29738. }});
  29739. // %TypedArray%.prototype.sort ( comparefn )
  29740. Object.defineProperty($TypedArray$.prototype, 'sort', {value: function(comparefn) {
  29741. if (this === undefined || this === null) throw TypeError();
  29742. var t = Object(this);
  29743. var len = ToUint32(t.length);
  29744. var tmp = Array(len);
  29745. for (var i = 0; i < len; ++i)
  29746. tmp[i] = t._getter(i);
  29747. function sortCompare(x, y) {
  29748. if (x !== x && y !== y) return +0;
  29749. if (x !== x) return 1;
  29750. if (y !== y) return -1;
  29751. if (comparefn !== undefined) {
  29752. return comparefn(x, y);
  29753. }
  29754. if (x < y) return -1;
  29755. if (x > y) return 1;
  29756. return +0;
  29757. }
  29758. tmp.sort(sortCompare);
  29759. for (i = 0; i < len; ++i)
  29760. t._setter(i, tmp[i]);
  29761. return t;
  29762. }});
  29763. // %TypedArray%.prototype.subarray(begin = 0, end = this.length )
  29764. // WebIDL: TypedArray subarray(long begin, optional long end);
  29765. Object.defineProperty($TypedArray$.prototype, 'subarray', {value: function(start, end) {
  29766. function clamp(v, min, max) { return v < min ? min : v > max ? max : v; }
  29767. start = ToInt32(start);
  29768. end = ToInt32(end);
  29769. if (arguments.length < 1) { start = 0; }
  29770. if (arguments.length < 2) { end = this.length; }
  29771. if (start < 0) { start = this.length + start; }
  29772. if (end < 0) { end = this.length + end; }
  29773. start = clamp(start, 0, this.length);
  29774. end = clamp(end, 0, this.length);
  29775. var len = end - start;
  29776. if (len < 0) {
  29777. len = 0;
  29778. }
  29779. return new this.constructor(
  29780. this.buffer, this.byteOffset + start * this.BYTES_PER_ELEMENT, len);
  29781. }});
  29782. // %TypedArray%.prototype.toLocaleString ( )
  29783. // %TypedArray%.prototype.toString ( )
  29784. // %TypedArray%.prototype.values ( )
  29785. // %TypedArray%.prototype [ @@iterator ] ( )
  29786. // get %TypedArray%.prototype [ @@toStringTag ]
  29787. // -- defined in es6.js to shim browsers w/ native TypedArrays
  29788. function makeTypedArray(elementSize, pack, unpack) {
  29789. // Each TypedArray type requires a distinct constructor instance with
  29790. // identical logic, which this produces.
  29791. var TypedArray = function() {
  29792. Object.defineProperty(this, 'constructor', {value: TypedArray});
  29793. $TypedArray$.apply(this, arguments);
  29794. makeArrayAccessors(this);
  29795. };
  29796. if ('__proto__' in TypedArray) {
  29797. TypedArray.__proto__ = $TypedArray$;
  29798. } else {
  29799. TypedArray.from = $TypedArray$.from;
  29800. TypedArray.of = $TypedArray$.of;
  29801. }
  29802. TypedArray.BYTES_PER_ELEMENT = elementSize;
  29803. var TypedArrayPrototype = function() {};
  29804. TypedArrayPrototype.prototype = $TypedArrayPrototype$;
  29805. TypedArray.prototype = new TypedArrayPrototype();
  29806. Object.defineProperty(TypedArray.prototype, 'BYTES_PER_ELEMENT', {value: elementSize});
  29807. Object.defineProperty(TypedArray.prototype, '_pack', {value: pack});
  29808. Object.defineProperty(TypedArray.prototype, '_unpack', {value: unpack});
  29809. return TypedArray;
  29810. }
  29811. var Int8Array = makeTypedArray(1, packI8, unpackI8);
  29812. var Uint8Array = makeTypedArray(1, packU8, unpackU8);
  29813. var Uint8ClampedArray = makeTypedArray(1, packU8Clamped, unpackU8);
  29814. var Int16Array = makeTypedArray(2, packI16, unpackI16);
  29815. var Uint16Array = makeTypedArray(2, packU16, unpackU16);
  29816. var Int32Array = makeTypedArray(4, packI32, unpackI32);
  29817. var Uint32Array = makeTypedArray(4, packU32, unpackU32);
  29818. var Float32Array = makeTypedArray(4, packF32, unpackF32);
  29819. var Float64Array = makeTypedArray(8, packF64, unpackF64);
  29820. global.Int8Array = global.Int8Array || Int8Array;
  29821. global.Uint8Array = global.Uint8Array || Uint8Array;
  29822. global.Uint8ClampedArray = global.Uint8ClampedArray || Uint8ClampedArray;
  29823. global.Int16Array = global.Int16Array || Int16Array;
  29824. global.Uint16Array = global.Uint16Array || Uint16Array;
  29825. global.Int32Array = global.Int32Array || Int32Array;
  29826. global.Uint32Array = global.Uint32Array || Uint32Array;
  29827. global.Float32Array = global.Float32Array || Float32Array;
  29828. global.Float64Array = global.Float64Array || Float64Array;
  29829. }());
  29830. //
  29831. // 6 The DataView View Type
  29832. //
  29833. (function() {
  29834. function r(array, index) {
  29835. return IsCallable(array.get) ? array.get(index) : array[index];
  29836. }
  29837. var IS_BIG_ENDIAN = (function() {
  29838. var u16array = new Uint16Array([0x1234]),
  29839. u8array = new Uint8Array(u16array.buffer);
  29840. return r(u8array, 0) === 0x12;
  29841. }());
  29842. // DataView(buffer, byteOffset=0, byteLength=undefined)
  29843. // WebIDL: Constructor(ArrayBuffer buffer,
  29844. // optional unsigned long byteOffset,
  29845. // optional unsigned long byteLength)
  29846. function DataView(buffer, byteOffset, byteLength) {
  29847. if (!(buffer instanceof ArrayBuffer || Class(buffer) === 'ArrayBuffer')) throw TypeError();
  29848. byteOffset = ToUint32(byteOffset);
  29849. if (byteOffset > buffer.byteLength)
  29850. throw RangeError('byteOffset out of range');
  29851. if (byteLength === undefined)
  29852. byteLength = buffer.byteLength - byteOffset;
  29853. else
  29854. byteLength = ToUint32(byteLength);
  29855. if ((byteOffset + byteLength) > buffer.byteLength)
  29856. throw RangeError('byteOffset and length reference an area beyond the end of the buffer');
  29857. Object.defineProperty(this, 'buffer', {value: buffer});
  29858. Object.defineProperty(this, 'byteLength', {value: byteLength});
  29859. Object.defineProperty(this, 'byteOffset', {value: byteOffset});
  29860. };
  29861. // get DataView.prototype.buffer
  29862. // get DataView.prototype.byteLength
  29863. // get DataView.prototype.byteOffset
  29864. // -- applied directly to instances by the constructor
  29865. function makeGetter(arrayType) {
  29866. return function GetViewValue(byteOffset, littleEndian) {
  29867. byteOffset = ToUint32(byteOffset);
  29868. if (byteOffset + arrayType.BYTES_PER_ELEMENT > this.byteLength)
  29869. throw RangeError('Array index out of range');
  29870. byteOffset += this.byteOffset;
  29871. var uint8Array = new Uint8Array(this.buffer, byteOffset, arrayType.BYTES_PER_ELEMENT),
  29872. bytes = [];
  29873. for (var i = 0; i < arrayType.BYTES_PER_ELEMENT; i += 1)
  29874. bytes.push(r(uint8Array, i));
  29875. if (Boolean(littleEndian) === Boolean(IS_BIG_ENDIAN))
  29876. bytes.reverse();
  29877. return r(new arrayType(new Uint8Array(bytes).buffer), 0);
  29878. };
  29879. }
  29880. Object.defineProperty(DataView.prototype, 'getUint8', {value: makeGetter(Uint8Array)});
  29881. Object.defineProperty(DataView.prototype, 'getInt8', {value: makeGetter(Int8Array)});
  29882. Object.defineProperty(DataView.prototype, 'getUint16', {value: makeGetter(Uint16Array)});
  29883. Object.defineProperty(DataView.prototype, 'getInt16', {value: makeGetter(Int16Array)});
  29884. Object.defineProperty(DataView.prototype, 'getUint32', {value: makeGetter(Uint32Array)});
  29885. Object.defineProperty(DataView.prototype, 'getInt32', {value: makeGetter(Int32Array)});
  29886. Object.defineProperty(DataView.prototype, 'getFloat32', {value: makeGetter(Float32Array)});
  29887. Object.defineProperty(DataView.prototype, 'getFloat64', {value: makeGetter(Float64Array)});
  29888. function makeSetter(arrayType) {
  29889. return function SetViewValue(byteOffset, value, littleEndian) {
  29890. byteOffset = ToUint32(byteOffset);
  29891. if (byteOffset + arrayType.BYTES_PER_ELEMENT > this.byteLength)
  29892. throw RangeError('Array index out of range');
  29893. // Get bytes
  29894. var typeArray = new arrayType([value]),
  29895. byteArray = new Uint8Array(typeArray.buffer),
  29896. bytes = [], i, byteView;
  29897. for (i = 0; i < arrayType.BYTES_PER_ELEMENT; i += 1)
  29898. bytes.push(r(byteArray, i));
  29899. // Flip if necessary
  29900. if (Boolean(littleEndian) === Boolean(IS_BIG_ENDIAN))
  29901. bytes.reverse();
  29902. // Write them
  29903. byteView = new Uint8Array(this.buffer, byteOffset, arrayType.BYTES_PER_ELEMENT);
  29904. byteView.set(bytes);
  29905. };
  29906. }
  29907. Object.defineProperty(DataView.prototype, 'setUint8', {value: makeSetter(Uint8Array)});
  29908. Object.defineProperty(DataView.prototype, 'setInt8', {value: makeSetter(Int8Array)});
  29909. Object.defineProperty(DataView.prototype, 'setUint16', {value: makeSetter(Uint16Array)});
  29910. Object.defineProperty(DataView.prototype, 'setInt16', {value: makeSetter(Int16Array)});
  29911. Object.defineProperty(DataView.prototype, 'setUint32', {value: makeSetter(Uint32Array)});
  29912. Object.defineProperty(DataView.prototype, 'setInt32', {value: makeSetter(Int32Array)});
  29913. Object.defineProperty(DataView.prototype, 'setFloat32', {value: makeSetter(Float32Array)});
  29914. Object.defineProperty(DataView.prototype, 'setFloat64', {value: makeSetter(Float64Array)});
  29915. global.DataView = global.DataView || DataView;
  29916. }());
  29917. }(self));