1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895 |
- (function (global, factory) {
- typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('jszip')) :
- typeof define === 'function' && define.amd ? define(['exports', 'jszip'], factory) :
- (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.docx = {}, global.JSZip));
- })(this, (function (exports, JSZip) {
- 'use strict';
- var RelationshipTypes;
- (function (RelationshipTypes) {
- RelationshipTypes["OfficeDocument"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/officeDocument";
- RelationshipTypes["FontTable"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/fontTable";
- RelationshipTypes["Image"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/image";
- RelationshipTypes["Numbering"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/numbering";
- RelationshipTypes["Styles"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles";
- RelationshipTypes["StylesWithEffects"] = "http://schemas.microsoft.com/office/2007/relationships/stylesWithEffects";
- RelationshipTypes["Theme"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/theme";
- RelationshipTypes["Settings"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/settings";
- RelationshipTypes["WebSettings"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/webSettings";
- RelationshipTypes["Hyperlink"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/hyperlink";
- RelationshipTypes["Footnotes"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/footnotes";
- RelationshipTypes["Endnotes"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/endnotes";
- RelationshipTypes["Footer"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/footer";
- RelationshipTypes["Header"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/header";
- RelationshipTypes["ExtendedProperties"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/extended-properties";
- RelationshipTypes["CoreProperties"] = "http://schemas.openxmlformats.org/package/2006/relationships/metadata/core-properties";
- RelationshipTypes["CustomProperties"] = "http://schemas.openxmlformats.org/package/2006/relationships/metadata/custom-properties";
- RelationshipTypes["Comments"] = "http://schemas.openxmlformats.org/officeDocument/2006/relationships/comments";
- RelationshipTypes["CommentsExtended"] = "http://schemas.microsoft.com/office/2011/relationships/commentsExtended";
- })(RelationshipTypes || (RelationshipTypes = {}));
- function parseRelationships (root, xml) {
- return xml.elements(root).map(e => ({
- id: xml.attr(e, "Id"),
- type: xml.attr(e, "Type"),
- target: xml.attr(e, "Target"),
- targetMode: xml.attr(e, "TargetMode")
- }));
- }
- const ns$1 = {
- wordml: "http://schemas.openxmlformats.org/wordprocessingml/2006/main",
- drawingml: "http://schemas.openxmlformats.org/drawingml/2006/main",
- picture: "http://schemas.openxmlformats.org/drawingml/2006/picture",
- compatibility: "http://schemas.openxmlformats.org/markup-compatibility/2006",
- math: "http://schemas.openxmlformats.org/officeDocument/2006/math"
- };
- const LengthUsage = {
- Dxa: { mul: 0.05, unit: "pt" },
- Emu: { mul: 1 / 12700, unit: "pt" },
- FontSize: { mul: 0.5, unit: "pt" },
- Border: { mul: 0.125, unit: "pt" },
- Point: { mul: 1, unit: "pt" },
- Percent: { mul: 0.02, unit: "%" },
- LineHeight: { mul: 1 / 240, unit: "" },
- VmlEmu: { mul: 1 / 12700, unit: "" },
- };
- function convertLength (val, usage = LengthUsage.Dxa) {
- if (val == null || /.+(p[xt]|[%])$/.test(val)) {
- return val;
- }
- return `${(parseInt(val) * usage.mul).toFixed(2)}${usage.unit}`;
- }
- function convertBoolean (v, defaultValue = false) {
- switch (v) {
- case "1": return true;
- case "0": return false;
- case "on": return true;
- case "off": return false;
- case "true": return true;
- case "false": return false;
- default: return defaultValue;
- }
- }
- function parseCommonProperty (elem, props, xml) {
- if (elem.namespaceURI != ns$1.wordml)
- return false;
- switch (elem.localName) {
- case "color":
- props.color = xml.attr(elem, "val");
- break;
- case "sz":
- props.fontSize = xml.lengthAttr(elem, "val", LengthUsage.FontSize);
- break;
- default:
- return false;
- }
- return true;
- }
- function parseXmlString (xmlString, trimXmlDeclaration = false) {
- if (trimXmlDeclaration)
- xmlString = xmlString.replace(/<[?].*[?]>/, "");
- xmlString = removeUTF8BOM(xmlString);
- const result = new DOMParser().parseFromString(xmlString, "application/xml");
- const errorText = hasXmlParserError(result);
- if (errorText)
- throw new Error(errorText);
- return result;
- }
- function hasXmlParserError (doc) {
- return doc.getElementsByTagName("parsererror")[0]?.textContent;
- }
- function removeUTF8BOM (data) {
- return data.charCodeAt(0) === 0xFEFF ? data.substring(1) : data;
- }
- function serializeXmlString (elem) {
- return new XMLSerializer().serializeToString(elem);
- }
- class XmlParser {
- elements (elem, localName = null) {
- const result = [];
- for (let i = 0, l = elem.childNodes.length; i < l; i++) {
- let c = elem.childNodes.item(i);
- if (c.nodeType == 1 && (localName == null || c.localName == localName))
- result.push(c);
- }
- return result;
- }
- element (elem, localName) {
- for (let i = 0, l = elem.childNodes.length; i < l; i++) {
- let c = elem.childNodes.item(i);
- if (c.nodeType == 1 && c.localName == localName)
- return c;
- }
- return null;
- }
- elementAttr (elem, localName, attrLocalName) {
- var el = this.element(elem, localName);
- return el ? this.attr(el, attrLocalName) : undefined;
- }
- attrs (elem) {
- return Array.from(elem.attributes);
- }
- attr (elem, localName) {
- for (let i = 0, l = elem.attributes.length; i < l; i++) {
- let a = elem.attributes.item(i);
- if (a.localName == localName)
- return a.value;
- }
- return null;
- }
- intAttr (node, attrName, defaultValue = null) {
- var val = this.attr(node, attrName);
- return val ? parseInt(val) : defaultValue;
- }
- hexAttr (node, attrName, defaultValue = null) {
- var val = this.attr(node, attrName);
- return val ? parseInt(val, 16) : defaultValue;
- }
- floatAttr (node, attrName, defaultValue = null) {
- var val = this.attr(node, attrName);
- return val ? parseFloat(val) : defaultValue;
- }
- boolAttr (node, attrName, defaultValue = null) {
- return convertBoolean(this.attr(node, attrName), defaultValue);
- }
- lengthAttr (node, attrName, usage = LengthUsage.Dxa) {
- return convertLength(this.attr(node, attrName), usage);
- }
- }
- const globalXmlParser = new XmlParser();
- class Part {
- constructor(_package, path) {
- this._package = _package;
- this.path = path;
- }
- async load () {
- this.rels = await this._package.loadRelationships(this.path);
- const xmlText = await this._package.load(this.path);
- const xmlDoc = this._package.parseXmlDocument(xmlText);
- if (this._package.options.keepOrigin) {
- this._xmlDocument = xmlDoc;
- }
- this.parseXml(xmlDoc.firstElementChild);
- }
- save () {
- this._package.update(this.path, serializeXmlString(this._xmlDocument));
- }
- parseXml (root) {
- }
- }
- const embedFontTypeMap = {
- embedRegular: 'regular',
- embedBold: 'bold',
- embedItalic: 'italic',
- embedBoldItalic: 'boldItalic',
- };
- function parseFonts (root, xml) {
- return xml.elements(root).map(el => parseFont(el, xml));
- }
- function parseFont (elem, xml) {
- let result = {
- name: xml.attr(elem, "name"),
- embedFontRefs: []
- };
- for (let el of xml.elements(elem)) {
- switch (el.localName) {
- case "family":
- result.family = xml.attr(el, "val");
- break;
- case "altName":
- result.altName = xml.attr(el, "val");
- break;
- case "embedRegular":
- case "embedBold":
- case "embedItalic":
- case "embedBoldItalic":
- result.embedFontRefs.push(parseEmbedFontRef(el, xml));
- break;
- }
- }
- return result;
- }
- function parseEmbedFontRef (elem, xml) {
- return {
- id: xml.attr(elem, "id"),
- key: xml.attr(elem, "fontKey"),
- type: embedFontTypeMap[elem.localName]
- };
- }
- class FontTablePart extends Part {
- parseXml (root) {
- this.fonts = parseFonts(root, this._package.xmlParser);
- }
- }
- function escapeClassName (className) {
- return className?.replace(/[ .]+/g, '-').replace(/[&]+/g, 'and').toLowerCase();
- }
- function splitPath (path) {
- let si = path.lastIndexOf('/') + 1;
- let folder = si == 0 ? "" : path.substring(0, si);
- let fileName = si == 0 ? path : path.substring(si);
- return [folder, fileName];
- }
- function resolvePath (path, base) {
- try {
- const prefix = "http://docx/";
- const url = new URL(path, prefix + base).toString();
- return url.substring(prefix.length);
- }
- catch {
- return `${base}${path}`;
- }
- }
- function keyBy (array, by) {
- return array.reduce((a, x) => {
- a[by(x)] = x;
- return a;
- }, {});
- }
- function blobToBase64 (blob) {
- return new Promise((resolve, reject) => {
- const reader = new FileReader();
- reader.onloadend = () => resolve(reader.result);
- reader.onerror = () => reject();
- reader.readAsDataURL(blob);
- });
- }
- function isObject (item) {
- return item && typeof item === 'object' && !Array.isArray(item);
- }
- function isString (item) {
- return typeof item === 'string' || item instanceof String;
- }
- function mergeDeep (target, ...sources) {
- if (!sources.length)
- return target;
- const source = sources.shift();
- if (isObject(target) && isObject(source)) {
- for (const key in source) {
- if (isObject(source[key])) {
- const val = target[key] ?? (target[key] = {});
- mergeDeep(val, source[key]);
- }
- else {
- target[key] = source[key];
- }
- }
- }
- return mergeDeep(target, ...sources);
- }
- function asArray (val) {
- return Array.isArray(val) ? val : [val];
- }
- class OpenXmlPackage {
- constructor(_zip, options) {
- this._zip = _zip;
- this.options = options;
- this.xmlParser = new XmlParser();
- }
- get (path) {
- const p = normalizePath(path);
- return this._zip.files[p] ?? this._zip.files[p.replace(/\//g, '\\')];
- }
- update (path, content) {
- this._zip.file(path, content);
- }
- static async load (input, options) {
- const zip = await JSZip.loadAsync(input);
- return new OpenXmlPackage(zip, options);
- }
- save (type = "blob") {
- return this._zip.generateAsync({ type });
- }
- load (path, type = "string") {
- return this.get(path)?.async(type) ?? Promise.resolve(null);
- }
- async loadRelationships (path = null) {
- let relsPath = `_rels/.rels`;
- if (path != null) {
- const [f, fn] = splitPath(path);
- relsPath = `${f}_rels/${fn}.rels`;
- }
- const txt = await this.load(relsPath);
- return txt ? parseRelationships(this.parseXmlDocument(txt).firstElementChild, this.xmlParser) : null;
- }
- parseXmlDocument (txt) {
- return parseXmlString(txt, this.options.trimXmlDeclaration);
- }
- }
- function normalizePath (path) {
- return path.startsWith('/') ? path.substr(1) : path;
- }
- class DocumentPart extends Part {
- constructor(pkg, path, parser) {
- super(pkg, path);
- this._documentParser = parser;
- }
- parseXml (root) {
- this.body = this._documentParser.parseDocumentFile(root);
- }
- }
- function parseBorder (elem, xml) {
- return {
- type: xml.attr(elem, "val"),
- color: xml.attr(elem, "color"),
- size: xml.lengthAttr(elem, "sz", LengthUsage.Border),
- offset: xml.lengthAttr(elem, "space", LengthUsage.Point),
- frame: xml.boolAttr(elem, 'frame'),
- shadow: xml.boolAttr(elem, 'shadow')
- };
- }
- function parseBorders (elem, xml) {
- var result = {};
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "left":
- result.left = parseBorder(e, xml);
- break;
- case "top":
- result.top = parseBorder(e, xml);
- break;
- case "right":
- result.right = parseBorder(e, xml);
- break;
- case "bottom":
- result.bottom = parseBorder(e, xml);
- break;
- }
- }
- return result;
- }
- var SectionType;
- (function (SectionType) {
- SectionType["Continuous"] = "continuous";
- SectionType["NextPage"] = "nextPage";
- SectionType["NextColumn"] = "nextColumn";
- SectionType["EvenPage"] = "evenPage";
- SectionType["OddPage"] = "oddPage";
- })(SectionType || (SectionType = {}));
- function parseSectionProperties (elem, xml = globalXmlParser) {
- var section = {};
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "pgSz":
- section.pageSize = {
- width: xml.lengthAttr(e, "w"),
- height: xml.lengthAttr(e, "h"),
- orientation: xml.attr(e, "orient")
- };
- break;
- case "type":
- section.type = xml.attr(e, "val");
- break;
- case "pgMar":
- section.pageMargins = {
- left: xml.lengthAttr(e, "left"),
- right: xml.lengthAttr(e, "right"),
- top: xml.lengthAttr(e, "top"),
- bottom: xml.lengthAttr(e, "bottom"),
- header: xml.lengthAttr(e, "header"),
- footer: xml.lengthAttr(e, "footer"),
- gutter: xml.lengthAttr(e, "gutter"),
- };
- break;
- case "cols":
- section.columns = parseColumns(e, xml);
- break;
- case "headerReference":
- (section.headerRefs ?? (section.headerRefs = [])).push(parseFooterHeaderReference(e, xml));
- break;
- case "footerReference":
- (section.footerRefs ?? (section.footerRefs = [])).push(parseFooterHeaderReference(e, xml));
- break;
- case "titlePg":
- section.titlePage = xml.boolAttr(e, "val", true);
- break;
- case "pgBorders":
- section.pageBorders = parseBorders(e, xml);
- break;
- case "pgNumType":
- section.pageNumber = parsePageNumber(e, xml);
- break;
- }
- }
- return section;
- }
- function parseColumns (elem, xml) {
- return {
- numberOfColumns: xml.intAttr(elem, "num"),
- space: xml.lengthAttr(elem, "space"),
- separator: xml.boolAttr(elem, "sep"),
- equalWidth: xml.boolAttr(elem, "equalWidth", true),
- columns: xml.elements(elem, "col")
- .map(e => ({
- width: xml.lengthAttr(e, "w"),
- space: xml.lengthAttr(e, "space")
- }))
- };
- }
- function parsePageNumber (elem, xml) {
- return {
- chapSep: xml.attr(elem, "chapSep"),
- chapStyle: xml.attr(elem, "chapStyle"),
- format: xml.attr(elem, "fmt"),
- start: xml.intAttr(elem, "start")
- };
- }
- function parseFooterHeaderReference (elem, xml) {
- return {
- id: xml.attr(elem, "id"),
- type: xml.attr(elem, "type"),
- };
- }
- function parseLineSpacing (elem, xml) {
- return {
- before: xml.lengthAttr(elem, "before"),
- after: xml.lengthAttr(elem, "after"),
- line: xml.intAttr(elem, "line"),
- lineRule: xml.attr(elem, "lineRule")
- };
- }
- function parseRunProperties (elem, xml) {
- let result = {};
- for (let el of xml.elements(elem)) {
- parseRunProperty(el, result, xml);
- }
- return result;
- }
- function parseRunProperty (elem, props, xml) {
- if (parseCommonProperty(elem, props, xml))
- return true;
- return false;
- }
- function parseParagraphProperties (elem, xml) {
- let result = {};
- for (let el of xml.elements(elem)) {
- parseParagraphProperty(el, result, xml);
- }
- return result;
- }
- function parseParagraphProperty (elem, props, xml) {
- if (elem.namespaceURI != ns$1.wordml)
- return false;
- if (parseCommonProperty(elem, props, xml))
- return true;
- switch (elem.localName) {
- case "tabs":
- props.tabs = parseTabs(elem, xml);
- break;
- case "sectPr":
- props.sectionProps = parseSectionProperties(elem, xml);
- break;
- case "numPr":
- props.numbering = parseNumbering$1(elem, xml);
- break;
- case "spacing":
- props.lineSpacing = parseLineSpacing(elem, xml);
- return false;
- case "textAlignment":
- props.textAlignment = xml.attr(elem, "val");
- return false;
- case "keepLines":
- props.keepLines = xml.boolAttr(elem, "val", true);
- break;
- case "keepNext":
- props.keepNext = xml.boolAttr(elem, "val", true);
- break;
- case "pageBreakBefore":
- props.pageBreakBefore = xml.boolAttr(elem, "val", true);
- break;
- case "outlineLvl":
- props.outlineLevel = xml.intAttr(elem, "val");
- break;
- case "pStyle":
- props.styleName = xml.attr(elem, "val");
- break;
- case "rPr":
- props.runProps = parseRunProperties(elem, xml);
- break;
- default:
- return false;
- }
- return true;
- }
- function parseTabs (elem, xml) {
- return xml.elements(elem, "tab")
- .map(e => ({
- position: xml.lengthAttr(e, "pos"),
- leader: xml.attr(e, "leader"),
- style: xml.attr(e, "val")
- }));
- }
- function parseNumbering$1 (elem, xml) {
- var result = {};
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "numId":
- result.id = xml.attr(e, "val");
- break;
- case "ilvl":
- result.level = xml.intAttr(e, "val");
- break;
- }
- }
- return result;
- }
- function parseNumberingPart (elem, xml) {
- let result = {
- numberings: [],
- abstractNumberings: [],
- bulletPictures: []
- };
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "num":
- result.numberings.push(parseNumbering(e, xml));
- break;
- case "abstractNum":
- result.abstractNumberings.push(parseAbstractNumbering(e, xml));
- break;
- case "numPicBullet":
- result.bulletPictures.push(parseNumberingBulletPicture(e, xml));
- break;
- }
- }
- return result;
- }
- function parseNumbering (elem, xml) {
- let result = {
- id: xml.attr(elem, 'numId'),
- overrides: []
- };
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "abstractNumId":
- result.abstractId = xml.attr(e, "val");
- break;
- case "lvlOverride":
- result.overrides.push(parseNumberingLevelOverrride(e, xml));
- break;
- }
- }
- return result;
- }
- function parseAbstractNumbering (elem, xml) {
- let result = {
- id: xml.attr(elem, 'abstractNumId'),
- levels: []
- };
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "name":
- result.name = xml.attr(e, "val");
- break;
- case "multiLevelType":
- result.multiLevelType = xml.attr(e, "val");
- break;
- case "numStyleLink":
- result.numberingStyleLink = xml.attr(e, "val");
- break;
- case "styleLink":
- result.styleLink = xml.attr(e, "val");
- break;
- case "lvl":
- result.levels.push(parseNumberingLevel(e, xml));
- break;
- }
- }
- return result;
- }
- function parseNumberingLevel (elem, xml) {
- let result = {
- level: xml.intAttr(elem, 'ilvl')
- };
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "start":
- result.start = xml.attr(e, "val");
- break;
- case "lvlRestart":
- result.restart = xml.intAttr(e, "val");
- break;
- case "numFmt":
- result.format = xml.attr(e, "val");
- break;
- case "lvlText":
- result.text = xml.attr(e, "val");
- break;
- case "lvlJc":
- result.justification = xml.attr(e, "val");
- break;
- case "lvlPicBulletId":
- result.bulletPictureId = xml.attr(e, "val");
- break;
- case "pStyle":
- result.paragraphStyle = xml.attr(e, "val");
- break;
- case "pPr":
- result.paragraphProps = parseParagraphProperties(e, xml);
- break;
- case "rPr":
- result.runProps = parseRunProperties(e, xml);
- break;
- }
- }
- return result;
- }
- function parseNumberingLevelOverrride (elem, xml) {
- let result = {
- level: xml.intAttr(elem, 'ilvl')
- };
- for (let e of xml.elements(elem)) {
- switch (e.localName) {
- case "startOverride":
- result.start = xml.intAttr(e, "val");
- break;
- case "lvl":
- result.numberingLevel = parseNumberingLevel(e, xml);
- break;
- }
- }
- return result;
- }
- function parseNumberingBulletPicture (elem, xml) {
- var pict = xml.element(elem, "pict");
- var shape = pict && xml.element(pict, "shape");
- var imagedata = shape && xml.element(shape, "imagedata");
- return imagedata ? {
- id: xml.attr(elem, "numPicBulletId"),
- referenceId: xml.attr(imagedata, "id"),
- style: xml.attr(shape, "style")
- } : null;
- }
- class NumberingPart extends Part {
- constructor(pkg, path, parser) {
- super(pkg, path);
- this._documentParser = parser;
- }
- parseXml (root) {
- Object.assign(this, parseNumberingPart(root, this._package.xmlParser));
- this.domNumberings = this._documentParser.parseNumberingFile(root);
- }
- }
- class StylesPart extends Part {
- constructor(pkg, path, parser) {
- super(pkg, path);
- this._documentParser = parser;
- }
- parseXml (root) {
- this.styles = this._documentParser.parseStylesFile(root);
- }
- }
- var DomType;
- (function (DomType) {
- DomType["Document"] = "document";
- DomType["Paragraph"] = "paragraph";
- DomType["Run"] = "run";
- DomType["Break"] = "break";
- DomType["NoBreakHyphen"] = "noBreakHyphen";
- DomType["Table"] = "table";
- DomType["Row"] = "row";
- DomType["Cell"] = "cell";
- DomType["Hyperlink"] = "hyperlink";
- DomType["SmartTag"] = "smartTag";
- DomType["Drawing"] = "drawing";
- DomType["Image"] = "image";
- DomType["Text"] = "text";
- DomType["Tab"] = "tab";
- DomType["Symbol"] = "symbol";
- DomType["BookmarkStart"] = "bookmarkStart";
- DomType["BookmarkEnd"] = "bookmarkEnd";
- DomType["Footer"] = "footer";
- DomType["Header"] = "header";
- DomType["FootnoteReference"] = "footnoteReference";
- DomType["EndnoteReference"] = "endnoteReference";
- DomType["Footnote"] = "footnote";
- DomType["Endnote"] = "endnote";
- DomType["SimpleField"] = "simpleField";
- DomType["ComplexField"] = "complexField";
- DomType["Instruction"] = "instruction";
- DomType["VmlPicture"] = "vmlPicture";
- DomType["MmlMath"] = "mmlMath";
- DomType["MmlMathParagraph"] = "mmlMathParagraph";
- DomType["MmlFraction"] = "mmlFraction";
- DomType["MmlFunction"] = "mmlFunction";
- DomType["MmlFunctionName"] = "mmlFunctionName";
- DomType["MmlNumerator"] = "mmlNumerator";
- DomType["MmlDenominator"] = "mmlDenominator";
- DomType["MmlRadical"] = "mmlRadical";
- DomType["MmlBase"] = "mmlBase";
- DomType["MmlDegree"] = "mmlDegree";
- DomType["MmlSuperscript"] = "mmlSuperscript";
- DomType["MmlSubscript"] = "mmlSubscript";
- DomType["MmlPreSubSuper"] = "mmlPreSubSuper";
- DomType["MmlSubArgument"] = "mmlSubArgument";
- DomType["MmlSuperArgument"] = "mmlSuperArgument";
- DomType["MmlNary"] = "mmlNary";
- DomType["MmlDelimiter"] = "mmlDelimiter";
- DomType["MmlRun"] = "mmlRun";
- DomType["MmlEquationArray"] = "mmlEquationArray";
- DomType["MmlLimit"] = "mmlLimit";
- DomType["MmlLimitLower"] = "mmlLimitLower";
- DomType["MmlMatrix"] = "mmlMatrix";
- DomType["MmlMatrixRow"] = "mmlMatrixRow";
- DomType["MmlBox"] = "mmlBox";
- DomType["MmlBar"] = "mmlBar";
- DomType["MmlGroupChar"] = "mmlGroupChar";
- DomType["VmlElement"] = "vmlElement";
- DomType["Inserted"] = "inserted";
- DomType["Deleted"] = "deleted";
- DomType["DeletedText"] = "deletedText";
- DomType["Comment"] = "comment";
- DomType["CommentReference"] = "commentReference";
- DomType["CommentRangeStart"] = "commentRangeStart";
- DomType["CommentRangeEnd"] = "commentRangeEnd";
- })(DomType || (DomType = {}));
- class OpenXmlElementBase {
- constructor() {
- this.children = [];
- this.cssStyle = {};
- }
- }
- class WmlHeader extends OpenXmlElementBase {
- constructor() {
- super(...arguments);
- this.type = DomType.Header;
- }
- }
- class WmlFooter extends OpenXmlElementBase {
- constructor() {
- super(...arguments);
- this.type = DomType.Footer;
- }
- }
- class BaseHeaderFooterPart extends Part {
- constructor(pkg, path, parser) {
- super(pkg, path);
- this._documentParser = parser;
- }
- parseXml (root) {
- this.rootElement = this.createRootElement();
- this.rootElement.children = this._documentParser.parseBodyElements(root);
- }
- }
- class HeaderPart extends BaseHeaderFooterPart {
- createRootElement () {
- return new WmlHeader();
- }
- }
- class FooterPart extends BaseHeaderFooterPart {
- createRootElement () {
- return new WmlFooter();
- }
- }
- function parseExtendedProps (root, xmlParser) {
- const result = {};
- for (let el of xmlParser.elements(root)) {
- switch (el.localName) {
- case "Template":
- result.template = el.textContent;
- break;
- case "Pages":
- result.pages = safeParseToInt(el.textContent);
- break;
- case "Words":
- result.words = safeParseToInt(el.textContent);
- break;
- case "Characters":
- result.characters = safeParseToInt(el.textContent);
- break;
- case "Application":
- result.application = el.textContent;
- break;
- case "Lines":
- result.lines = safeParseToInt(el.textContent);
- break;
- case "Paragraphs":
- result.paragraphs = safeParseToInt(el.textContent);
- break;
- case "Company":
- result.company = el.textContent;
- break;
- case "AppVersion":
- result.appVersion = el.textContent;
- break;
- }
- }
- return result;
- }
- function safeParseToInt (value) {
- if (typeof value === 'undefined')
- return;
- return parseInt(value);
- }
- class ExtendedPropsPart extends Part {
- parseXml (root) {
- this.props = parseExtendedProps(root, this._package.xmlParser);
- }
- }
- function parseCoreProps (root, xmlParser) {
- const result = {};
- for (let el of xmlParser.elements(root)) {
- switch (el.localName) {
- case "title":
- result.title = el.textContent;
- break;
- case "description":
- result.description = el.textContent;
- break;
- case "subject":
- result.subject = el.textContent;
- break;
- case "creator":
- result.creator = el.textContent;
- break;
- case "keywords":
- result.keywords = el.textContent;
- break;
- case "language":
- result.language = el.textContent;
- break;
- case "lastModifiedBy":
- result.lastModifiedBy = el.textContent;
- break;
- case "revision":
- el.textContent && (result.revision = parseInt(el.textContent));
- break;
- }
- }
- return result;
- }
- class CorePropsPart extends Part {
- parseXml (root) {
- this.props = parseCoreProps(root, this._package.xmlParser);
- }
- }
- class DmlTheme {
- }
- function parseTheme (elem, xml) {
- var result = new DmlTheme();
- var themeElements = xml.element(elem, "themeElements");
- for (let el of xml.elements(themeElements)) {
- switch (el.localName) {
- case "clrScheme":
- result.colorScheme = parseColorScheme(el, xml);
- break;
- case "fontScheme":
- result.fontScheme = parseFontScheme(el, xml);
- break;
- }
- }
- return result;
- }
- function parseColorScheme (elem, xml) {
- var result = {
- name: xml.attr(elem, "name"),
- colors: {}
- };
- for (let el of xml.elements(elem)) {
- var srgbClr = xml.element(el, "srgbClr");
- var sysClr = xml.element(el, "sysClr");
- if (srgbClr) {
- result.colors[el.localName] = xml.attr(srgbClr, "val");
- }
- else if (sysClr) {
- result.colors[el.localName] = xml.attr(sysClr, "lastClr");
- }
- }
- return result;
- }
- function parseFontScheme (elem, xml) {
- var result = {
- name: xml.attr(elem, "name"),
- };
- for (let el of xml.elements(elem)) {
- switch (el.localName) {
- case "majorFont":
- result.majorFont = parseFontInfo(el, xml);
- break;
- case "minorFont":
- result.minorFont = parseFontInfo(el, xml);
- break;
- }
- }
- return result;
- }
- function parseFontInfo (elem, xml) {
- return {
- latinTypeface: xml.elementAttr(elem, "latin", "typeface"),
- eaTypeface: xml.elementAttr(elem, "ea", "typeface"),
- csTypeface: xml.elementAttr(elem, "cs", "typeface"),
- };
- }
- class ThemePart extends Part {
- constructor(pkg, path) {
- super(pkg, path);
- }
- parseXml (root) {
- this.theme = parseTheme(root, this._package.xmlParser);
- }
- }
- class WmlBaseNote {
- }
- class WmlFootnote extends WmlBaseNote {
- constructor() {
- super(...arguments);
- this.type = DomType.Footnote;
- }
- }
- class WmlEndnote extends WmlBaseNote {
- constructor() {
- super(...arguments);
- this.type = DomType.Endnote;
- }
- }
- class BaseNotePart extends Part {
- constructor(pkg, path, parser) {
- super(pkg, path);
- this._documentParser = parser;
- }
- }
- class FootnotesPart extends BaseNotePart {
- constructor(pkg, path, parser) {
- super(pkg, path, parser);
- }
- parseXml (root) {
- this.notes = this._documentParser.parseNotes(root, "footnote", WmlFootnote);
- }
- }
- class EndnotesPart extends BaseNotePart {
- constructor(pkg, path, parser) {
- super(pkg, path, parser);
- }
- parseXml (root) {
- this.notes = this._documentParser.parseNotes(root, "endnote", WmlEndnote);
- }
- }
- function parseSettings (elem, xml) {
- var result = {};
- for (let el of xml.elements(elem)) {
- switch (el.localName) {
- case "defaultTabStop":
- result.defaultTabStop = xml.lengthAttr(el, "val");
- break;
- case "footnotePr":
- result.footnoteProps = parseNoteProperties(el, xml);
- break;
- case "endnotePr":
- result.endnoteProps = parseNoteProperties(el, xml);
- break;
- case "autoHyphenation":
- result.autoHyphenation = xml.boolAttr(el, "val");
- break;
- }
- }
- return result;
- }
- function parseNoteProperties (elem, xml) {
- var result = {
- defaultNoteIds: []
- };
- for (let el of xml.elements(elem)) {
- switch (el.localName) {
- case "numFmt":
- result.nummeringFormat = xml.attr(el, "val");
- break;
- case "footnote":
- case "endnote":
- result.defaultNoteIds.push(xml.attr(el, "id"));
- break;
- }
- }
- return result;
- }
- class SettingsPart extends Part {
- constructor(pkg, path) {
- super(pkg, path);
- }
- parseXml (root) {
- this.settings = parseSettings(root, this._package.xmlParser);
- }
- }
- function parseCustomProps (root, xml) {
- return xml.elements(root, "property").map(e => {
- const firstChild = e.firstChild;
- return {
- formatId: xml.attr(e, "fmtid"),
- name: xml.attr(e, "name"),
- type: firstChild.nodeName,
- value: firstChild.textContent
- };
- });
- }
- class CustomPropsPart extends Part {
- parseXml (root) {
- this.props = parseCustomProps(root, this._package.xmlParser);
- }
- }
- class CommentsPart extends Part {
- constructor(pkg, path, parser) {
- super(pkg, path);
- this._documentParser = parser;
- }
- parseXml (root) {
- this.comments = this._documentParser.parseComments(root);
- this.commentMap = keyBy(this.comments, x => x.id);
- }
- }
- class CommentsExtendedPart extends Part {
- constructor(pkg, path) {
- super(pkg, path);
- this.comments = [];
- }
- parseXml (root) {
- const xml = this._package.xmlParser;
- for (let el of xml.elements(root, "commentEx")) {
- this.comments.push({
- paraId: xml.attr(el, 'paraId'),
- paraIdParent: xml.attr(el, 'paraIdParent'),
- done: xml.boolAttr(el, 'done')
- });
- }
- this.commentMap = keyBy(this.comments, x => x.paraId);
- }
- }
- const topLevelRels = [
- { type: RelationshipTypes.OfficeDocument, target: "word/document.xml" },
- { type: RelationshipTypes.ExtendedProperties, target: "docProps/app.xml" },
- { type: RelationshipTypes.CoreProperties, target: "docProps/core.xml" },
- { type: RelationshipTypes.CustomProperties, target: "docProps/custom.xml" },
- ];
- class WordDocument {
- constructor() {
- this.parts = [];
- this.partsMap = {};
- }
- static async load (blob, parser, options) {
- var d = new WordDocument();
- d._options = options;
- d._parser = parser;
- d._package = await OpenXmlPackage.load(blob, options);
- d.rels = await d._package.loadRelationships();
- await Promise.all(topLevelRels.map(rel => {
- const r = d.rels.find(x => x.type === rel.type) ?? rel;
- return d.loadRelationshipPart(r.target, r.type);
- }));
- return d;
- }
- save (type = "blob") {
- return this._package.save(type);
- }
- async loadRelationshipPart (path, type) {
- if (this.partsMap[path])
- return this.partsMap[path];
- if (!this._package.get(path))
- return null;
- let part = null;
- switch (type) {
- case RelationshipTypes.OfficeDocument:
- this.documentPart = part = new DocumentPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.FontTable:
- this.fontTablePart = part = new FontTablePart(this._package, path);
- break;
- case RelationshipTypes.Numbering:
- this.numberingPart = part = new NumberingPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.Styles:
- this.stylesPart = part = new StylesPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.Theme:
- this.themePart = part = new ThemePart(this._package, path);
- break;
- case RelationshipTypes.Footnotes:
- this.footnotesPart = part = new FootnotesPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.Endnotes:
- this.endnotesPart = part = new EndnotesPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.Footer:
- part = new FooterPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.Header:
- part = new HeaderPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.CoreProperties:
- this.corePropsPart = part = new CorePropsPart(this._package, path);
- break;
- case RelationshipTypes.ExtendedProperties:
- this.extendedPropsPart = part = new ExtendedPropsPart(this._package, path);
- break;
- case RelationshipTypes.CustomProperties:
- part = new CustomPropsPart(this._package, path);
- break;
- case RelationshipTypes.Settings:
- this.settingsPart = part = new SettingsPart(this._package, path);
- break;
- case RelationshipTypes.Comments:
- this.commentsPart = part = new CommentsPart(this._package, path, this._parser);
- break;
- case RelationshipTypes.CommentsExtended:
- this.commentsExtendedPart = part = new CommentsExtendedPart(this._package, path);
- break;
- }
- if (part == null)
- return Promise.resolve(null);
- this.partsMap[path] = part;
- this.parts.push(part);
- await part.load();
- if (part.rels?.length > 0) {
- const [folder] = splitPath(part.path);
- await Promise.all(part.rels.map(rel => this.loadRelationshipPart(resolvePath(rel.target, folder), rel.type)));
- }
- return part;
- }
- async loadDocumentImage (id, part) {
- const x = await this.loadResource(part ?? this.documentPart, id, "blob");
- return this.blobToURL(x);
- }
- async loadNumberingImage (id) {
- const x = await this.loadResource(this.numberingPart, id, "blob");
- return this.blobToURL(x);
- }
- async loadFont (id, key) {
- const x = await this.loadResource(this.fontTablePart, id, "uint8array");
- return x ? this.blobToURL(new Blob([deobfuscate(x, key)])) : x;
- }
- blobToURL (blob) {
- if (!blob)
- return null;
- if (this._options.useBase64URL) {
- return blobToBase64(blob);
- }
- return URL.createObjectURL(blob);
- }
- findPartByRelId (id, basePart = null) {
- var rel = (basePart.rels ?? this.rels).find(r => r.id == id);
- const folder = basePart ? splitPath(basePart.path)[0] : '';
- return rel ? this.partsMap[resolvePath(rel.target, folder)] : null;
- }
- getPathById (part, id) {
- const rel = part.rels.find(x => x.id == id);
- const [folder] = splitPath(part.path);
- return rel ? resolvePath(rel.target, folder) : null;
- }
- loadResource (part, id, outputType) {
- const path = this.getPathById(part, id);
- return path ? this._package.load(path, outputType) : Promise.resolve(null);
- }
- }
- function deobfuscate (data, guidKey) {
- const len = 16;
- const trimmed = guidKey.replace(/{|}|-/g, "");
- const numbers = new Array(len);
- for (let i = 0; i < len; i++)
- numbers[len - i - 1] = parseInt(trimmed.substr(i * 2, 2), 16);
- for (let i = 0; i < 32; i++)
- data[i] = data[i] ^ numbers[i % len];
- return data;
- }
- function parseBookmarkStart (elem, xml) {
- return {
- type: DomType.BookmarkStart,
- id: xml.attr(elem, "id"),
- name: xml.attr(elem, "name"),
- colFirst: xml.intAttr(elem, "colFirst"),
- colLast: xml.intAttr(elem, "colLast")
- };
- }
- function parseBookmarkEnd (elem, xml) {
- return {
- type: DomType.BookmarkEnd,
- id: xml.attr(elem, "id")
- };
- }
- class VmlElement extends OpenXmlElementBase {
- constructor() {
- super(...arguments);
- this.type = DomType.VmlElement;
- this.attrs = {};
- }
- }
- function parseVmlElement (elem, parser) {
- var result = new VmlElement();
- switch (elem.localName) {
- case "rect":
- result.tagName = "rect";
- Object.assign(result.attrs, { width: '100%', height: '100%' });
- break;
- case "oval":
- result.tagName = "ellipse";
- Object.assign(result.attrs, { cx: "50%", cy: "50%", rx: "50%", ry: "50%" });
- break;
- case "line":
- result.tagName = "line";
- break;
- case "shape":
- result.tagName = "g";
- break;
- case "textbox":
- result.tagName = "foreignObject";
- Object.assign(result.attrs, { width: '100%', height: '100%' });
- break;
- default:
- return null;
- }
- for (const at of globalXmlParser.attrs(elem)) {
- switch (at.localName) {
- case "style":
- result.cssStyleText = at.value;
- break;
- case "fillcolor":
- result.attrs.fill = at.value;
- break;
- case "from":
- const [x1, y1] = parsePoint(at.value);
- Object.assign(result.attrs, { x1, y1 });
- break;
- case "to":
- const [x2, y2] = parsePoint(at.value);
- Object.assign(result.attrs, { x2, y2 });
- break;
- }
- }
- for (const el of globalXmlParser.elements(elem)) {
- switch (el.localName) {
- case "stroke":
- Object.assign(result.attrs, parseStroke(el));
- break;
- case "fill":
- Object.assign(result.attrs, parseFill());
- break;
- case "imagedata":
- result.tagName = "image";
- Object.assign(result.attrs, { width: '100%', height: '100%' });
- result.imageHref = {
- id: globalXmlParser.attr(el, "id"),
- title: globalXmlParser.attr(el, "title"),
- };
- break;
- case "txbxContent":
- result.children.push(...parser.parseBodyElements(el));
- break;
- default:
- const child = parseVmlElement(el, parser);
- child && result.children.push(child);
- break;
- }
- }
- return result;
- }
- function parseStroke (el) {
- return {
- 'stroke': globalXmlParser.attr(el, "color"),
- 'stroke-width': globalXmlParser.lengthAttr(el, "weight", LengthUsage.Emu) ?? '1px'
- };
- }
- function parseFill (el) {
- return {};
- }
- function parsePoint (val) {
- return val.split(",");
- }
- class WmlComment extends OpenXmlElementBase {
- constructor() {
- super(...arguments);
- this.type = DomType.Comment;
- }
- }
- class WmlCommentReference extends OpenXmlElementBase {
- constructor(id) {
- super();
- this.id = id;
- this.type = DomType.CommentReference;
- }
- }
- class WmlCommentRangeStart extends OpenXmlElementBase {
- constructor(id) {
- super();
- this.id = id;
- this.type = DomType.CommentRangeStart;
- }
- }
- class WmlCommentRangeEnd extends OpenXmlElementBase {
- constructor(id) {
- super();
- this.id = id;
- this.type = DomType.CommentRangeEnd;
- }
- }
- var autos = {
- shd: "inherit",
- color: "black",
- borderColor: "black",
- highlight: "transparent"
- };
- const supportedNamespaceURIs = [];
- const mmlTagMap = {
- "oMath": DomType.MmlMath,
- "oMathPara": DomType.MmlMathParagraph,
- "f": DomType.MmlFraction,
- "func": DomType.MmlFunction,
- "fName": DomType.MmlFunctionName,
- "num": DomType.MmlNumerator,
- "den": DomType.MmlDenominator,
- "rad": DomType.MmlRadical,
- "deg": DomType.MmlDegree,
- "e": DomType.MmlBase,
- "sSup": DomType.MmlSuperscript,
- "sSub": DomType.MmlSubscript,
- "sPre": DomType.MmlPreSubSuper,
- "sup": DomType.MmlSuperArgument,
- "sub": DomType.MmlSubArgument,
- "d": DomType.MmlDelimiter,
- "nary": DomType.MmlNary,
- "eqArr": DomType.MmlEquationArray,
- "lim": DomType.MmlLimit,
- "limLow": DomType.MmlLimitLower,
- "m": DomType.MmlMatrix,
- "mr": DomType.MmlMatrixRow,
- "box": DomType.MmlBox,
- "bar": DomType.MmlBar,
- "groupChr": DomType.MmlGroupChar
- };
- class DocumentParser {
- constructor(options) {
- this.options = {
- ignoreWidth: false,
- debug: false,
- ...options
- };
- }
- parseNotes (xmlDoc, elemName, elemClass) {
- var result = [];
- for (let el of globalXmlParser.elements(xmlDoc, elemName)) {
- const node = new elemClass();
- node.id = globalXmlParser.attr(el, "id");
- node.noteType = globalXmlParser.attr(el, "type");
- node.children = this.parseBodyElements(el);
- result.push(node);
- }
- return result;
- }
- parseComments (xmlDoc) {
- var result = [];
- for (let el of globalXmlParser.elements(xmlDoc, "comment")) {
- const item = new WmlComment();
- item.id = globalXmlParser.attr(el, "id");
- item.author = globalXmlParser.attr(el, "author");
- item.initials = globalXmlParser.attr(el, "initials");
- item.date = globalXmlParser.attr(el, "date");
- item.children = this.parseBodyElements(el);
- result.push(item);
- }
- return result;
- }
- parseDocumentFile (xmlDoc) {
- var xbody = globalXmlParser.element(xmlDoc, "body");
- var background = globalXmlParser.element(xmlDoc, "background");
- var sectPr = globalXmlParser.element(xbody, "sectPr");
- return {
- type: DomType.Document,
- children: this.parseBodyElements(xbody),
- props: sectPr ? parseSectionProperties(sectPr, globalXmlParser) : {},
- cssStyle: background ? this.parseBackground(background) : {},
- };
- }
- parseBackground (elem) {
- var result = {};
- var color = xmlUtil.colorAttr(elem, "color");
- if (color) {
- result["background-color"] = color;
- }
- return result;
- }
- parseBodyElements (element) {
- var children = [];
- for (let elem of globalXmlParser.elements(element)) {
- switch (elem.localName) {
- case "p":
- children.push(this.parseParagraph(elem));
- break;
- case "tbl":
- children.push(this.parseTable(elem));
- break;
- case "sdt":
- children.push(...this.parseSdt(elem, e => this.parseBodyElements(e)));
- break;
- }
- }
- return children;
- }
- parseStylesFile (xstyles) {
- var result = [];
- xmlUtil.foreach(xstyles, n => {
- switch (n.localName) {
- case "style":
- result.push(this.parseStyle(n));
- break;
- case "docDefaults":
- result.push(this.parseDefaultStyles(n));
- break;
- }
- });
- return result;
- }
- parseDefaultStyles (node) {
- var result = {
- id: null,
- name: null,
- target: null,
- basedOn: null,
- styles: []
- };
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "rPrDefault":
- var rPr = globalXmlParser.element(c, "rPr");
- if (rPr)
- result.styles.push({
- target: "span",
- values: this.parseDefaultProperties(rPr, {})
- });
- break;
- case "pPrDefault":
- var pPr = globalXmlParser.element(c, "pPr");
- if (pPr)
- result.styles.push({
- target: "p",
- values: this.parseDefaultProperties(pPr, {})
- });
- break;
- }
- });
- return result;
- }
- parseStyle (node) {
- var result = {
- id: globalXmlParser.attr(node, "styleId"),
- isDefault: globalXmlParser.boolAttr(node, "default"),
- name: null,
- target: null,
- basedOn: null,
- styles: [],
- linked: null
- };
- switch (globalXmlParser.attr(node, "type")) {
- case "paragraph":
- result.target = "p";
- break;
- case "table":
- result.target = "table";
- break;
- case "character":
- result.target = "span";
- break;
- }
- xmlUtil.foreach(node, n => {
- switch (n.localName) {
- case "basedOn":
- result.basedOn = globalXmlParser.attr(n, "val");
- break;
- case "name":
- result.name = globalXmlParser.attr(n, "val");
- break;
- case "link":
- result.linked = globalXmlParser.attr(n, "val");
- break;
- case "next":
- result.next = globalXmlParser.attr(n, "val");
- break;
- case "aliases":
- result.aliases = globalXmlParser.attr(n, "val").split(",");
- break;
- case "pPr":
- result.styles.push({
- target: "p",
- values: this.parseDefaultProperties(n, {})
- });
- result.paragraphProps = parseParagraphProperties(n, globalXmlParser);
- break;
- case "rPr":
- result.styles.push({
- target: "span",
- values: this.parseDefaultProperties(n, {})
- });
- result.runProps = parseRunProperties(n, globalXmlParser);
- break;
- case "tblPr":
- case "tcPr":
- result.styles.push({
- target: "td",
- values: this.parseDefaultProperties(n, {})
- });
- break;
- case "tblStylePr":
- for (let s of this.parseTableStyle(n))
- result.styles.push(s);
- break;
- case "rsid":
- case "qFormat":
- case "hidden":
- case "semiHidden":
- case "unhideWhenUsed":
- case "autoRedefine":
- case "uiPriority":
- break;
- default:
- this.options.debug && console.warn(`DOCX: Unknown style element: ${n.localName}`);
- }
- });
- return result;
- }
- parseTableStyle (node) {
- var result = [];
- var type = globalXmlParser.attr(node, "type");
- var selector = "";
- var modificator = "";
- switch (type) {
- case "firstRow":
- modificator = ".first-row";
- selector = "tr.first-row td";
- break;
- case "lastRow":
- modificator = ".last-row";
- selector = "tr.last-row td";
- break;
- case "firstCol":
- modificator = ".first-col";
- selector = "td.first-col";
- break;
- case "lastCol":
- modificator = ".last-col";
- selector = "td.last-col";
- break;
- case "band1Vert":
- modificator = ":not(.no-vband)";
- selector = "td.odd-col";
- break;
- case "band2Vert":
- modificator = ":not(.no-vband)";
- selector = "td.even-col";
- break;
- case "band1Horz":
- modificator = ":not(.no-hband)";
- selector = "tr.odd-row";
- break;
- case "band2Horz":
- modificator = ":not(.no-hband)";
- selector = "tr.even-row";
- break;
- default: return [];
- }
- xmlUtil.foreach(node, n => {
- switch (n.localName) {
- case "pPr":
- result.push({
- target: `${selector} p`,
- mod: modificator,
- values: this.parseDefaultProperties(n, {})
- });
- break;
- case "rPr":
- result.push({
- target: `${selector} span`,
- mod: modificator,
- values: this.parseDefaultProperties(n, {})
- });
- break;
- case "tblPr":
- case "tcPr":
- result.push({
- target: selector,
- mod: modificator,
- values: this.parseDefaultProperties(n, {})
- });
- break;
- }
- });
- return result;
- }
- parseNumberingFile (xnums) {
- var result = [];
- var mapping = {};
- var bullets = [];
- xmlUtil.foreach(xnums, n => {
- switch (n.localName) {
- case "abstractNum":
- this.parseAbstractNumbering(n, bullets)
- .forEach(x => result.push(x));
- break;
- case "numPicBullet":
- bullets.push(this.parseNumberingPicBullet(n));
- break;
- case "num":
- var numId = globalXmlParser.attr(n, "numId");
- var abstractNumId = globalXmlParser.elementAttr(n, "abstractNumId", "val");
- mapping[abstractNumId] = numId;
- break;
- }
- });
- result.forEach(x => x.id = mapping[x.id]);
- return result;
- }
- parseNumberingPicBullet (elem) {
- var pict = globalXmlParser.element(elem, "pict");
- var shape = pict && globalXmlParser.element(pict, "shape");
- var imagedata = shape && globalXmlParser.element(shape, "imagedata");
- return imagedata ? {
- id: globalXmlParser.intAttr(elem, "numPicBulletId"),
- src: globalXmlParser.attr(imagedata, "id"),
- style: globalXmlParser.attr(shape, "style")
- } : null;
- }
- parseAbstractNumbering (node, bullets) {
- var result = [];
- var id = globalXmlParser.attr(node, "abstractNumId");
- xmlUtil.foreach(node, n => {
- switch (n.localName) {
- case "lvl":
- result.push(this.parseNumberingLevel(id, n, bullets));
- break;
- }
- });
- return result;
- }
- parseNumberingLevel (id, node, bullets) {
- var result = {
- id: id,
- level: globalXmlParser.intAttr(node, "ilvl"),
- start: 1,
- pStyleName: undefined,
- pStyle: {},
- rStyle: {},
- suff: "tab"
- };
- xmlUtil.foreach(node, n => {
- switch (n.localName) {
- case "start":
- result.start = globalXmlParser.intAttr(n, "val");
- break;
- case "pPr":
- this.parseDefaultProperties(n, result.pStyle);
- break;
- case "rPr":
- this.parseDefaultProperties(n, result.rStyle);
- break;
- case "lvlPicBulletId":
- var id = globalXmlParser.intAttr(n, "val");
- result.bullet = bullets.find(x => x?.id == id);
- break;
- case "lvlText":
- result.levelText = globalXmlParser.attr(n, "val");
- break;
- case "pStyle":
- result.pStyleName = globalXmlParser.attr(n, "val");
- break;
- case "numFmt":
- result.format = globalXmlParser.attr(n, "val");
- break;
- case "suff":
- result.suff = globalXmlParser.attr(n, "val");
- break;
- }
- });
- return result;
- }
- parseSdt (node, parser) {
- const sdtContent = globalXmlParser.element(node, "sdtContent");
- return sdtContent ? parser(sdtContent) : [];
- }
- parseInserted (node, parentParser) {
- return {
- type: DomType.Inserted,
- children: parentParser(node)?.children ?? []
- };
- }
- parseDeleted (node, parentParser) {
- return {
- type: DomType.Deleted,
- children: parentParser(node)?.children ?? []
- };
- }
- parseParagraph (node) {
- var result = { type: DomType.Paragraph, children: [] };
- for (let el of globalXmlParser.elements(node)) {
- switch (el.localName) {
- case "pPr":
- this.parseParagraphProperties(el, result);
- break;
- case "r":
- result.children.push(this.parseRun(el, result));
- break;
- case "hyperlink":
- result.children.push(this.parseHyperlink(el, result));
- break;
- case "smartTag":
- result.children.push(this.parseSmartTag(el, result));
- break;
- case "bookmarkStart":
- result.children.push(parseBookmarkStart(el, globalXmlParser));
- break;
- case "bookmarkEnd":
- result.children.push(parseBookmarkEnd(el, globalXmlParser));
- break;
- case "commentRangeStart":
- result.children.push(new WmlCommentRangeStart(globalXmlParser.attr(el, "id")));
- break;
- case "commentRangeEnd":
- result.children.push(new WmlCommentRangeEnd(globalXmlParser.attr(el, "id")));
- break;
- case "oMath":
- case "oMathPara":
- result.children.push(this.parseMathElement(el));
- break;
- case "sdt":
- result.children.push(...this.parseSdt(el, e => this.parseParagraph(e).children));
- break;
- case "ins":
- result.children.push(this.parseInserted(el, e => this.parseParagraph(e)));
- break;
- case "del":
- result.children.push(this.parseDeleted(el, e => this.parseParagraph(e)));
- break;
- }
- }
- return result;
- }
- parseParagraphProperties (elem, paragraph) {
- this.parseDefaultProperties(elem, paragraph.cssStyle = {}, null, c => {
- if (parseParagraphProperty(c, paragraph, globalXmlParser))
- return true;
- switch (c.localName) {
- case "pStyle":
- paragraph.styleName = globalXmlParser.attr(c, "val");
- break;
- case "cnfStyle":
- paragraph.className = values.classNameOfCnfStyle(c);
- break;
- case "framePr":
- this.parseFrame(c, paragraph);
- break;
- case "rPr":
- break;
- default:
- return false;
- }
- return true;
- });
- }
- parseFrame (node, paragraph) {
- var dropCap = globalXmlParser.attr(node, "dropCap");
- if (dropCap == "drop")
- paragraph.cssStyle["float"] = "left";
- }
- parseHyperlink (node, parent) {
- var result = { type: DomType.Hyperlink, parent: parent, children: [] };
- var anchor = globalXmlParser.attr(node, "anchor");
- var relId = globalXmlParser.attr(node, "id");
- if (anchor)
- result.href = "#" + anchor;
- if (relId)
- result.id = relId;
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "r":
- result.children.push(this.parseRun(c, result));
- break;
- }
- });
- return result;
- }
- parseSmartTag (node, parent) {
- var result = { type: DomType.SmartTag, parent, children: [] };
- var uri = globalXmlParser.attr(node, "uri");
- var element = globalXmlParser.attr(node, "element");
- if (uri)
- result.uri = uri;
- if (element)
- result.element = element;
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "r":
- result.children.push(this.parseRun(c, result));
- break;
- }
- });
- return result;
- }
- parseRun (node, parent) {
- var result = { type: DomType.Run, parent: parent, children: [] };
- xmlUtil.foreach(node, c => {
- c = this.checkAlternateContent(c);
- switch (c.localName) {
- case "t":
- result.children.push({
- type: DomType.Text,
- text: c.textContent
- });
- break;
- case "delText":
- result.children.push({
- type: DomType.DeletedText,
- text: c.textContent
- });
- break;
- case "commentReference":
- result.children.push(new WmlCommentReference(globalXmlParser.attr(c, "id")));
- break;
- case "fldSimple":
- result.children.push({
- type: DomType.SimpleField,
- instruction: globalXmlParser.attr(c, "instr"),
- lock: globalXmlParser.boolAttr(c, "lock", false),
- dirty: globalXmlParser.boolAttr(c, "dirty", false)
- });
- break;
- case "instrText":
- result.fieldRun = true;
- result.children.push({
- type: DomType.Instruction,
- text: c.textContent
- });
- break;
- case "fldChar":
- result.fieldRun = true;
- result.children.push({
- type: DomType.ComplexField,
- charType: globalXmlParser.attr(c, "fldCharType"),
- lock: globalXmlParser.boolAttr(c, "lock", false),
- dirty: globalXmlParser.boolAttr(c, "dirty", false)
- });
- break;
- case "noBreakHyphen":
- result.children.push({ type: DomType.NoBreakHyphen });
- break;
- case "br":
- result.children.push({
- type: DomType.Break,
- break: globalXmlParser.attr(c, "type") || "textWrapping"
- });
- break;
- case "lastRenderedPageBreak":
- result.children.push({
- type: DomType.Break,
- break: "lastRenderedPageBreak"
- });
- break;
- case "sym":
- result.children.push({
- type: DomType.Symbol,
- font: globalXmlParser.attr(c, "font"),
- char: globalXmlParser.attr(c, "char")
- });
- break;
- case "tab":
- result.children.push({ type: DomType.Tab });
- break;
- case "footnoteReference":
- result.children.push({
- type: DomType.FootnoteReference,
- id: globalXmlParser.attr(c, "id")
- });
- break;
- case "endnoteReference":
- result.children.push({
- type: DomType.EndnoteReference,
- id: globalXmlParser.attr(c, "id")
- });
- break;
- case "drawing":
- let d = this.parseDrawing(c);
- if (d)
- result.children = [d];
- break;
- case "pict":
- result.children.push(this.parseVmlPicture(c));
- break;
- case "rPr":
- this.parseRunProperties(c, result);
- break;
- }
- });
- return result;
- }
- parseMathElement (elem) {
- const propsTag = `${elem.localName}Pr`;
- const result = { type: mmlTagMap[elem.localName], children: [] };
- for (const el of globalXmlParser.elements(elem)) {
- const childType = mmlTagMap[el.localName];
- if (childType) {
- result.children.push(this.parseMathElement(el));
- }
- else if (el.localName == "r") {
- var run = this.parseRun(el);
- run.type = DomType.MmlRun;
- result.children.push(run);
- }
- else if (el.localName == propsTag) {
- result.props = this.parseMathProperies(el);
- }
- }
- return result;
- }
- parseMathProperies (elem) {
- const result = {};
- for (const el of globalXmlParser.elements(elem)) {
- switch (el.localName) {
- case "chr":
- result.char = globalXmlParser.attr(el, "val");
- break;
- case "vertJc":
- result.verticalJustification = globalXmlParser.attr(el, "val");
- break;
- case "pos":
- result.position = globalXmlParser.attr(el, "val");
- break;
- case "degHide":
- result.hideDegree = globalXmlParser.boolAttr(el, "val");
- break;
- case "begChr":
- result.beginChar = globalXmlParser.attr(el, "val");
- break;
- case "endChr":
- result.endChar = globalXmlParser.attr(el, "val");
- break;
- }
- }
- return result;
- }
- parseRunProperties (elem, run) {
- this.parseDefaultProperties(elem, run.cssStyle = {}, null, c => {
- switch (c.localName) {
- case "rStyle":
- run.styleName = globalXmlParser.attr(c, "val");
- break;
- case "vertAlign":
- run.verticalAlign = values.valueOfVertAlign(c, true);
- break;
- default:
- return false;
- }
- return true;
- });
- }
- parseVmlPicture (elem) {
- const result = { type: DomType.VmlPicture, children: [] };
- for (const el of globalXmlParser.elements(elem)) {
- const child = parseVmlElement(el, this);
- child && result.children.push(child);
- }
- return result;
- }
- checkAlternateContent (elem) {
- if (elem.localName != 'AlternateContent')
- return elem;
- var choice = globalXmlParser.element(elem, "Choice");
- if (choice) {
- var requires = globalXmlParser.attr(choice, "Requires");
- var namespaceURI = elem.lookupNamespaceURI(requires);
- if (supportedNamespaceURIs.includes(namespaceURI))
- return choice.firstElementChild;
- }
- return globalXmlParser.element(elem, "Fallback")?.firstElementChild;
- }
- parseDrawing (node) {
- for (var n of globalXmlParser.elements(node)) {
- switch (n.localName) {
- case "inline":
- case "anchor":
- return this.parseDrawingWrapper(n);
- }
- }
- }
- parseDrawingWrapper (node) {
- var result = { type: DomType.Drawing, children: [], cssStyle: {} };
- var isAnchor = node.localName == "anchor";
- let wrapType = null;
- let simplePos = globalXmlParser.boolAttr(node, "simplePos");
- globalXmlParser.boolAttr(node, "behindDoc");
- let posX = { relative: "page", align: "left", offset: "0" };
- let posY = { relative: "page", align: "top", offset: "0" };
- for (var n of globalXmlParser.elements(node)) {
- switch (n.localName) {
- case "simplePos":
- if (simplePos) {
- posX.offset = globalXmlParser.lengthAttr(n, "x", LengthUsage.Emu);
- posY.offset = globalXmlParser.lengthAttr(n, "y", LengthUsage.Emu);
- }
- break;
- case "extent":
- result.cssStyle["width"] = globalXmlParser.lengthAttr(n, "cx", LengthUsage.Emu);
- result.cssStyle["height"] = globalXmlParser.lengthAttr(n, "cy", LengthUsage.Emu);
- break;
- case "positionH":
- case "positionV":
- if (!simplePos) {
- let pos = n.localName == "positionH" ? posX : posY;
- var alignNode = globalXmlParser.element(n, "align");
- var offsetNode = globalXmlParser.element(n, "posOffset");
- pos.relative = globalXmlParser.attr(n, "relativeFrom") ?? pos.relative;
- if (alignNode)
- pos.align = alignNode.textContent;
- if (offsetNode)
- pos.offset = xmlUtil.sizeValue(offsetNode, LengthUsage.Emu);
- }
- break;
- case "wrapTopAndBottom":
- wrapType = "wrapTopAndBottom";
- break;
- case "wrapNone":
- wrapType = "wrapNone";
- break;
- case "graphic":
- var g = this.parseGraphic(n);
- if (g)
- result.children.push(g);
- break;
- }
- }
- if (wrapType == "wrapTopAndBottom") {
- result.cssStyle['display'] = 'block';
- if (posX.align) {
- result.cssStyle['text-align'] = posX.align;
- result.cssStyle['width'] = "100%";
- }
- }
- else if (wrapType == "wrapNone") {
- result.cssStyle['display'] = 'block';
- result.cssStyle['position'] = 'relative';
- result.cssStyle["width"] = "0px";
- result.cssStyle["height"] = "0px";
- if (posX.offset)
- result.cssStyle["left"] = posX.offset;
- if (posY.offset)
- result.cssStyle["top"] = posY.offset;
- }
- else if (isAnchor && (posX.align == 'left' || posX.align == 'right')) {
- result.cssStyle["float"] = posX.align;
- }
- return result;
- }
- parseGraphic (elem) {
- var graphicData = globalXmlParser.element(elem, "graphicData");
- for (let n of globalXmlParser.elements(graphicData)) {
- switch (n.localName) {
- case "pic":
- return this.parsePicture(n);
- }
- }
- return null;
- }
- parsePicture (elem) {
- var result = { type: DomType.Image, src: "", cssStyle: {} };
- var blipFill = globalXmlParser.element(elem, "blipFill");
- var blip = globalXmlParser.element(blipFill, "blip");
- result.src = globalXmlParser.attr(blip, "embed");
- var spPr = globalXmlParser.element(elem, "spPr");
- var xfrm = globalXmlParser.element(spPr, "xfrm");
- result.cssStyle["position"] = "relative";
- for (var n of globalXmlParser.elements(xfrm)) {
- switch (n.localName) {
- case "ext":
- result.cssStyle["width"] = globalXmlParser.lengthAttr(n, "cx", LengthUsage.Emu);
- result.cssStyle["height"] = globalXmlParser.lengthAttr(n, "cy", LengthUsage.Emu);
- break;
- case "off":
- result.cssStyle["left"] = globalXmlParser.lengthAttr(n, "x", LengthUsage.Emu);
- result.cssStyle["top"] = globalXmlParser.lengthAttr(n, "y", LengthUsage.Emu);
- break;
- }
- }
- return result;
- }
- parseTable (node) {
- var result = { type: DomType.Table, children: [] };
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "tr":
- result.children.push(this.parseTableRow(c));
- break;
- case "tblGrid":
- result.columns = this.parseTableColumns(c);
- break;
- case "tblPr":
- this.parseTableProperties(c, result);
- break;
- }
- });
- return result;
- }
- parseTableColumns (node) {
- var result = [];
- xmlUtil.foreach(node, n => {
- switch (n.localName) {
- case "gridCol":
- result.push({ width: globalXmlParser.lengthAttr(n, "w") });
- break;
- }
- });
- return result;
- }
- parseTableProperties (elem, table) {
- table.cssStyle = {};
- table.cellStyle = {};
- this.parseDefaultProperties(elem, table.cssStyle, table.cellStyle, c => {
- switch (c.localName) {
- case "tblStyle":
- table.styleName = globalXmlParser.attr(c, "val");
- break;
- case "tblLook":
- table.className = values.classNameOftblLook(c);
- break;
- case "tblpPr":
- this.parseTablePosition(c, table);
- break;
- case "tblStyleColBandSize":
- table.colBandSize = globalXmlParser.intAttr(c, "val");
- break;
- case "tblStyleRowBandSize":
- table.rowBandSize = globalXmlParser.intAttr(c, "val");
- break;
- default:
- return false;
- }
- return true;
- });
- switch (table.cssStyle["text-align"]) {
- case "center":
- delete table.cssStyle["text-align"];
- table.cssStyle["margin-left"] = "auto";
- table.cssStyle["margin-right"] = "auto";
- break;
- case "right":
- delete table.cssStyle["text-align"];
- table.cssStyle["margin-left"] = "auto";
- break;
- }
- }
- parseTablePosition (node, table) {
- var topFromText = globalXmlParser.lengthAttr(node, "topFromText");
- var bottomFromText = globalXmlParser.lengthAttr(node, "bottomFromText");
- var rightFromText = globalXmlParser.lengthAttr(node, "rightFromText");
- var leftFromText = globalXmlParser.lengthAttr(node, "leftFromText");
- table.cssStyle["float"] = 'left';
- table.cssStyle["margin-bottom"] = values.addSize(table.cssStyle["margin-bottom"], bottomFromText);
- table.cssStyle["margin-left"] = values.addSize(table.cssStyle["margin-left"], leftFromText);
- table.cssStyle["margin-right"] = values.addSize(table.cssStyle["margin-right"], rightFromText);
- table.cssStyle["margin-top"] = values.addSize(table.cssStyle["margin-top"], topFromText);
- }
- parseTableRow (node) {
- var result = { type: DomType.Row, children: [] };
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "tc":
- result.children.push(this.parseTableCell(c));
- break;
- case "trPr":
- this.parseTableRowProperties(c, result);
- break;
- }
- });
- return result;
- }
- parseTableRowProperties (elem, row) {
- row.cssStyle = this.parseDefaultProperties(elem, {}, null, c => {
- switch (c.localName) {
- case "cnfStyle":
- row.className = values.classNameOfCnfStyle(c);
- break;
- case "tblHeader":
- row.isHeader = globalXmlParser.boolAttr(c, "val");
- break;
- default:
- return false;
- }
- return true;
- });
- }
- parseTableCell (node) {
- var result = { type: DomType.Cell, children: [] };
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "tbl":
- result.children.push(this.parseTable(c));
- break;
- case "p":
- result.children.push(this.parseParagraph(c));
- break;
- case "tcPr":
- this.parseTableCellProperties(c, result);
- break;
- }
- });
- return result;
- }
- parseTableCellProperties (elem, cell) {
- cell.cssStyle = this.parseDefaultProperties(elem, {}, null, c => {
- switch (c.localName) {
- case "gridSpan":
- cell.span = globalXmlParser.intAttr(c, "val", null);
- break;
- case "vMerge":
- cell.verticalMerge = globalXmlParser.attr(c, "val") ?? "continue";
- break;
- case "cnfStyle":
- cell.className = values.classNameOfCnfStyle(c);
- break;
- default:
- return false;
- }
- return true;
- });
- }
- parseDefaultProperties (elem, style = null, childStyle = null, handler = null) {
- style = style || {};
- xmlUtil.foreach(elem, c => {
- if (handler?.(c))
- return;
- switch (c.localName) {
- case "jc":
- style["text-align"] = values.valueOfJc(c);
- break;
- case "textAlignment":
- style["vertical-align"] = values.valueOfTextAlignment(c);
- break;
- case "color":
- style["color"] = xmlUtil.colorAttr(c, "val", null, autos.color);
- break;
- case "sz":
- style["font-size"] = style["min-height"] = globalXmlParser.lengthAttr(c, "val", LengthUsage.FontSize);
- break;
- case "shd":
- style["background-color"] = xmlUtil.colorAttr(c, "fill", null, autos.shd);
- break;
- case "highlight":
- style["background-color"] = xmlUtil.colorAttr(c, "val", null, autos.highlight);
- break;
- case "vertAlign":
- break;
- case "position":
- style.verticalAlign = globalXmlParser.lengthAttr(c, "val", LengthUsage.FontSize);
- break;
- case "tcW":
- if (this.options.ignoreWidth)
- break;
- case "tblW":
- style["width"] = values.valueOfSize(c, "w");
- break;
- case "trHeight":
- this.parseTrHeight(c, style);
- break;
- case "strike":
- style["text-decoration"] = globalXmlParser.boolAttr(c, "val", true) ? "line-through" : "none";
- break;
- case "b":
- style["font-weight"] = globalXmlParser.boolAttr(c, "val", true) ? "bold" : "normal";
- break;
- case "i":
- style["font-style"] = globalXmlParser.boolAttr(c, "val", true) ? "italic" : "normal";
- break;
- case "caps":
- style["text-transform"] = globalXmlParser.boolAttr(c, "val", true) ? "uppercase" : "none";
- break;
- case "smallCaps":
- style["font-variant"] = globalXmlParser.boolAttr(c, "val", true) ? "small-caps" : "none";
- break;
- case "u":
- this.parseUnderline(c, style);
- break;
- case "ind":
- case "tblInd":
- this.parseIndentation(c, style);
- break;
- case "rFonts":
- this.parseFont(c, style);
- break;
- case "tblBorders":
- this.parseBorderProperties(c, childStyle || style);
- break;
- case "tblCellSpacing":
- style["border-spacing"] = values.valueOfMargin(c);
- style["border-collapse"] = "separate";
- break;
- case "pBdr":
- this.parseBorderProperties(c, style);
- break;
- case "bdr":
- style["border"] = values.valueOfBorder(c);
- break;
- case "tcBorders":
- this.parseBorderProperties(c, style);
- break;
- case "vanish":
- if (globalXmlParser.boolAttr(c, "val", true))
- style["display"] = "none";
- break;
- case "kern":
- break;
- case "noWrap":
- break;
- case "tblCellMar":
- case "tcMar":
- this.parseMarginProperties(c, childStyle || style);
- break;
- case "tblLayout":
- style["table-layout"] = values.valueOfTblLayout(c);
- break;
- case "vAlign":
- style["vertical-align"] = values.valueOfTextAlignment(c);
- break;
- case "spacing":
- if (elem.localName == "pPr")
- this.parseSpacing(c, style);
- break;
- case "wordWrap":
- if (globalXmlParser.boolAttr(c, "val"))
- style["overflow-wrap"] = "break-word";
- break;
- case "suppressAutoHyphens":
- style["hyphens"] = globalXmlParser.boolAttr(c, "val", true) ? "none" : "auto";
- break;
- case "lang":
- style["$lang"] = globalXmlParser.attr(c, "val");
- break;
- case "bCs":
- case "iCs":
- case "szCs":
- case "tabs":
- case "outlineLvl":
- case "contextualSpacing":
- case "tblStyleColBandSize":
- case "tblStyleRowBandSize":
- case "webHidden":
- case "pageBreakBefore":
- case "suppressLineNumbers":
- case "keepLines":
- case "keepNext":
- case "widowControl":
- case "bidi":
- case "rtl":
- case "noProof":
- break;
- default:
- if (this.options.debug)
- console.warn(`DOCX: Unknown document element: ${elem.localName}.${c.localName}`);
- break;
- }
- });
- return style;
- }
- parseUnderline (node, style) {
- var val = globalXmlParser.attr(node, "val");
- if (val == null)
- return;
- switch (val) {
- case "dash":
- case "dashDotDotHeavy":
- case "dashDotHeavy":
- case "dashedHeavy":
- case "dashLong":
- case "dashLongHeavy":
- case "dotDash":
- case "dotDotDash":
- style["text-decoration"] = "underline dashed";
- break;
- case "dotted":
- case "dottedHeavy":
- style["text-decoration"] = "underline dotted";
- break;
- case "double":
- style["text-decoration"] = "underline double";
- break;
- case "single":
- case "thick":
- style["text-decoration"] = "underline";
- break;
- case "wave":
- case "wavyDouble":
- case "wavyHeavy":
- style["text-decoration"] = "underline wavy";
- break;
- case "words":
- style["text-decoration"] = "underline";
- break;
- case "none":
- style["text-decoration"] = "none";
- break;
- }
- var col = xmlUtil.colorAttr(node, "color");
- if (col)
- style["text-decoration-color"] = col;
- }
- parseFont (node, style) {
- var ascii = globalXmlParser.attr(node, "ascii");
- var asciiTheme = values.themeValue(node, "asciiTheme");
- var fonts = [ascii, asciiTheme].filter(x => x).join(', ');
- if (fonts.length > 0)
- style["font-family"] = fonts;
- }
- parseIndentation (node, style) {
- var firstLine = globalXmlParser.lengthAttr(node, "firstLine");
- var hanging = globalXmlParser.lengthAttr(node, "hanging");
- var left = globalXmlParser.lengthAttr(node, "left");
- var start = globalXmlParser.lengthAttr(node, "start");
- var right = globalXmlParser.lengthAttr(node, "right");
- var end = globalXmlParser.lengthAttr(node, "end");
- if (firstLine)
- style["text-indent"] = firstLine;
- if (hanging)
- style["text-indent"] = `-${hanging}`;
- if (left || start)
- style["margin-left"] = left || start;
- if (right || end)
- style["margin-right"] = right || end;
- }
- parseSpacing (node, style) {
- var before = globalXmlParser.lengthAttr(node, "before");
- var after = globalXmlParser.lengthAttr(node, "after");
- var line = globalXmlParser.intAttr(node, "line", null);
- var lineRule = globalXmlParser.attr(node, "lineRule");
- if (before)
- style["margin-top"] = before;
- if (after)
- style["margin-bottom"] = after;
- if (line !== null) {
- switch (lineRule) {
- case "auto":
- style["line-height"] = `${(line / 240).toFixed(2)}`;
- break;
- case "atLeast":
- style["line-height"] = `calc(100% + ${line / 20}pt)`;
- break;
- default:
- style["line-height"] = style["min-height"] = `${line / 20}pt`;
- break;
- }
- }
- }
- parseMarginProperties (node, output) {
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "left":
- output["padding-left"] = values.valueOfMargin(c);
- break;
- case "right":
- output["padding-right"] = values.valueOfMargin(c);
- break;
- case "top":
- output["padding-top"] = values.valueOfMargin(c);
- break;
- case "bottom":
- output["padding-bottom"] = values.valueOfMargin(c);
- break;
- }
- });
- }
- parseTrHeight (node, output) {
- switch (globalXmlParser.attr(node, "hRule")) {
- case "exact":
- output["height"] = globalXmlParser.lengthAttr(node, "val");
- break;
- case "atLeast":
- default:
- output["height"] = globalXmlParser.lengthAttr(node, "val");
- break;
- }
- }
- parseBorderProperties (node, output) {
- xmlUtil.foreach(node, c => {
- switch (c.localName) {
- case "start":
- case "left":
- output["border-left"] = values.valueOfBorder(c);
- break;
- case "end":
- case "right":
- output["border-right"] = values.valueOfBorder(c);
- break;
- case "top":
- output["border-top"] = values.valueOfBorder(c);
- break;
- case "bottom":
- output["border-bottom"] = values.valueOfBorder(c);
- break;
- }
- });
- }
- }
- const knownColors = ['black', 'blue', 'cyan', 'darkBlue', 'darkCyan', 'darkGray', 'darkGreen', 'darkMagenta', 'darkRed', 'darkYellow', 'green', 'lightGray', 'magenta', 'none', 'red', 'white', 'yellow'];
- class xmlUtil {
- static foreach (node, cb) {
- for (var i = 0; i < node.childNodes.length; i++) {
- let n = node.childNodes[i];
- if (n.nodeType == Node.ELEMENT_NODE)
- cb(n);
- }
- }
- static colorAttr (node, attrName, defValue = null, autoColor = 'black') {
- var v = globalXmlParser.attr(node, attrName);
- if (v) {
- if (v == "auto") {
- return autoColor;
- }
- else if (knownColors.includes(v)) {
- return v;
- }
- return `#${v}`;
- }
- var themeColor = globalXmlParser.attr(node, "themeColor");
- return themeColor ? `var(--docx-${themeColor}-color)` : defValue;
- }
- static sizeValue (node, type = LengthUsage.Dxa) {
- return convertLength(node.textContent, type);
- }
- }
- class values {
- static themeValue (c, attr) {
- var val = globalXmlParser.attr(c, attr);
- return val ? `var(--docx-${val}-font)` : null;
- }
- static valueOfSize (c, attr) {
- var type = LengthUsage.Dxa;
- switch (globalXmlParser.attr(c, "type")) {
- case "dxa": break;
- case "pct":
- type = LengthUsage.Percent;
- break;
- case "auto": return "auto";
- }
- return globalXmlParser.lengthAttr(c, attr, type);
- }
- static valueOfMargin (c) {
- return globalXmlParser.lengthAttr(c, "w");
- }
- static valueOfBorder (c) {
- var type = globalXmlParser.attr(c, "val");
- if (type == "nil")
- return "none";
- var color = xmlUtil.colorAttr(c, "color");
- var size = globalXmlParser.lengthAttr(c, "sz", LengthUsage.Border);
- return `${size} solid ${color == "auto" ? autos.borderColor : color}`;
- }
- static valueOfTblLayout (c) {
- var type = globalXmlParser.attr(c, "val");
- return type == "fixed" ? "fixed" : "auto";
- }
- static classNameOfCnfStyle (c) {
- const val = globalXmlParser.attr(c, "val");
- const classes = [
- 'first-row', 'last-row', 'first-col', 'last-col',
- 'odd-col', 'even-col', 'odd-row', 'even-row',
- 'ne-cell', 'nw-cell', 'se-cell', 'sw-cell'
- ];
- return classes.filter((_, i) => val[i] == '1').join(' ');
- }
- static valueOfJc (c) {
- var type = globalXmlParser.attr(c, "val");
- switch (type) {
- case "start":
- case "left": return "left";
- case "center": return "center";
- case "end":
- case "right": return "right";
- case "both": return "justify";
- }
- return type;
- }
- static valueOfVertAlign (c, asTagName = false) {
- var type = globalXmlParser.attr(c, "val");
- switch (type) {
- case "subscript": return "sub";
- case "superscript": return asTagName ? "sup" : "super";
- }
- return asTagName ? null : type;
- }
- static valueOfTextAlignment (c) {
- var type = globalXmlParser.attr(c, "val");
- switch (type) {
- case "auto":
- case "baseline": return "baseline";
- case "top": return "top";
- case "center": return "middle";
- case "bottom": return "bottom";
- }
- return type;
- }
- static addSize (a, b) {
- if (a == null)
- return b;
- if (b == null)
- return a;
- return `calc(${a} + ${b})`;
- }
- static classNameOftblLook (c) {
- const val = globalXmlParser.hexAttr(c, "val", 0);
- let className = "";
- if (globalXmlParser.boolAttr(c, "firstRow") || (val & 0x0020))
- className += " first-row";
- if (globalXmlParser.boolAttr(c, "lastRow") || (val & 0x0040))
- className += " last-row";
- if (globalXmlParser.boolAttr(c, "firstColumn") || (val & 0x0080))
- className += " first-col";
- if (globalXmlParser.boolAttr(c, "lastColumn") || (val & 0x0100))
- className += " last-col";
- if (globalXmlParser.boolAttr(c, "noHBand") || (val & 0x0200))
- className += " no-hband";
- if (globalXmlParser.boolAttr(c, "noVBand") || (val & 0x0400))
- className += " no-vband";
- return className.trim();
- }
- }
- const defaultTab = { pos: 0, leader: "none", style: "left" };
- const maxTabs = 50;
- function computePixelToPoint (container = document.body) {
- const temp = document.createElement("div");
- temp.style.width = '100pt';
- container.appendChild(temp);
- const result = 100 / temp.offsetWidth;
- container.removeChild(temp);
- return result;
- }
- function updateTabStop (elem, tabs, defaultTabSize, pixelToPoint = 72 / 96) {
- const p = elem.closest("p");
- const ebb = elem.getBoundingClientRect();
- const pbb = p.getBoundingClientRect();
- const pcs = getComputedStyle(p);
- const tabStops = tabs?.length > 0 ? tabs.map(t => ({
- pos: lengthToPoint(t.position),
- leader: t.leader,
- style: t.style
- })).sort((a, b) => a.pos - b.pos) : [defaultTab];
- const lastTab = tabStops[tabStops.length - 1];
- const pWidthPt = pbb.width * pixelToPoint;
- const size = lengthToPoint(defaultTabSize);
- let pos = lastTab.pos + size;
- if (pos < pWidthPt) {
- for (; pos < pWidthPt && tabStops.length < maxTabs; pos += size) {
- tabStops.push({ ...defaultTab, pos: pos });
- }
- }
- const marginLeft = parseFloat(pcs.marginLeft);
- const pOffset = pbb.left + marginLeft;
- const left = (ebb.left - pOffset) * pixelToPoint;
- const tab = tabStops.find(t => t.style != "clear" && t.pos > left);
- if (tab == null)
- return;
- let width = 1;
- if (tab.style == "right" || tab.style == "center") {
- const tabStops = Array.from(p.querySelectorAll(`.${elem.className}`));
- const nextIdx = tabStops.indexOf(elem) + 1;
- const range = document.createRange();
- range.setStart(elem, 1);
- if (nextIdx < tabStops.length) {
- range.setEndBefore(tabStops[nextIdx]);
- }
- else {
- range.setEndAfter(p);
- }
- const mul = tab.style == "center" ? 0.5 : 1;
- const nextBB = range.getBoundingClientRect();
- const offset = nextBB.left + mul * nextBB.width - (pbb.left - marginLeft);
- width = tab.pos - offset * pixelToPoint;
- }
- else {
- width = tab.pos - left;
- }
- elem.innerHTML = " ";
- elem.style.textDecoration = "inherit";
- elem.style.wordSpacing = `${width.toFixed(0)}pt`;
- switch (tab.leader) {
- case "dot":
- case "middleDot":
- elem.style.textDecoration = "underline";
- elem.style.textDecorationStyle = "dotted";
- break;
- case "hyphen":
- case "heavy":
- case "underscore":
- elem.style.textDecoration = "underline";
- break;
- }
- }
- function lengthToPoint (length) {
- return parseFloat(length);
- }
- const ns = {
- svg: "http://www.w3.org/2000/svg",
- mathML: "http://www.w3.org/1998/Math/MathML"
- };
- class HtmlRenderer {
- constructor(htmlDocument) {
- this.htmlDocument = htmlDocument;
- this.className = "docx";
- this.styleMap = {};
- this.currentPart = null;
- this.tableVerticalMerges = [];
- this.currentVerticalMerge = null;
- this.tableCellPositions = [];
- this.currentCellPosition = null;
- this.footnoteMap = {};
- this.endnoteMap = {};
- this.currentEndnoteIds = [];
- this.usedHederFooterParts = [];
- this.currentTabs = [];
- this.tabsTimeout = 0;
- this.commentMap = {};
- this.tasks = [];
- this.postRenderTasks = [];
- this.createElement = createElement;
- }
- render (document, bodyContainer, styleContainer = null, options) {
- this.document = document;
- this.options = options;
- this.className = options.className;
- this.rootSelector = options.inWrapper ? `.${this.className}-wrapper` : ':root';
- this.styleMap = null;
- this.tasks = [];
- if (this.options.renderComments && globalThis.Highlight) {
- this.commentHighlight = new Highlight();
- }
- styleContainer = styleContainer || bodyContainer;
- removeAllElements(styleContainer);
- removeAllElements(bodyContainer);
- appendComment(styleContainer, "docxjs library predefined styles");
- styleContainer.appendChild(this.renderDefaultStyle());
- if (document.themePart) {
- appendComment(styleContainer, "docxjs document theme values");
- this.renderTheme(document.themePart, styleContainer);
- }
- if (document.stylesPart != null) {
- this.styleMap = this.processStyles(document.stylesPart.styles);
- appendComment(styleContainer, "docxjs document styles");
- styleContainer.appendChild(this.renderStyles(document.stylesPart.styles));
- }
- if (document.numberingPart) {
- this.prodessNumberings(document.numberingPart.domNumberings);
- appendComment(styleContainer, "docxjs document numbering styles");
- styleContainer.appendChild(this.renderNumbering(document.numberingPart.domNumberings, styleContainer));
- }
- if (document.footnotesPart) {
- this.footnoteMap = keyBy(document.footnotesPart.notes, x => x.id);
- }
- if (document.endnotesPart) {
- this.endnoteMap = keyBy(document.endnotesPart.notes, x => x.id);
- }
- if (document.settingsPart) {
- this.defaultTabSize = document.settingsPart.settings?.defaultTabStop;
- }
- if (!options.ignoreFonts && document.fontTablePart)
- this.renderFontTable(document.fontTablePart, styleContainer);
- var sectionElements = this.renderSections(document.documentPart.body);
- if (this.options.inWrapper) {
- bodyContainer.appendChild(this.renderWrapper(sectionElements));
- }
- else {
- appendChildren(bodyContainer, sectionElements);
- }
- if (this.commentHighlight && options.renderComments) {
- CSS.highlights.set(`${this.className}-comments`, this.commentHighlight);
- }
- this.refreshTabStops();
- this.postRenderTasks.forEach(t => t());
- }
- renderTheme (themePart, styleContainer) {
- const variables = {};
- const fontScheme = themePart.theme?.fontScheme;
- if (fontScheme) {
- if (fontScheme.majorFont) {
- variables['--docx-majorHAnsi-font'] = fontScheme.majorFont.latinTypeface;
- }
- if (fontScheme.minorFont) {
- variables['--docx-minorHAnsi-font'] = fontScheme.minorFont.latinTypeface;
- }
- }
- const colorScheme = themePart.theme?.colorScheme;
- if (colorScheme) {
- for (let [k, v] of Object.entries(colorScheme.colors)) {
- variables[`--docx-${k}-color`] = `#${v}`;
- }
- }
- const cssText = this.styleToString(`.${this.className}`, variables);
- styleContainer.appendChild(createStyleElement(cssText));
- }
- renderFontTable (fontsPart, styleContainer) {
- for (let f of fontsPart.fonts) {
- for (let ref of f.embedFontRefs) {
- this.tasks.push(this.document.loadFont(ref.id, ref.key).then(fontData => {
- const cssValues = {
- 'font-family': f.name,
- 'src': `url(${fontData})`
- };
- if (ref.type == "bold" || ref.type == "boldItalic") {
- cssValues['font-weight'] = 'bold';
- }
- if (ref.type == "italic" || ref.type == "boldItalic") {
- cssValues['font-style'] = 'italic';
- }
- appendComment(styleContainer, `docxjs ${f.name} font`);
- const cssText = this.styleToString("@font-face", cssValues);
- styleContainer.appendChild(createStyleElement(cssText));
- this.refreshTabStops();
- }));
- }
- }
- }
- processStyleName (className) {
- return className ? `${this.className}_${escapeClassName(className)}` : this.className;
- }
- processStyles (styles) {
- const stylesMap = keyBy(styles.filter(x => x.id != null), x => x.id);
- for (const style of styles.filter(x => x.basedOn)) {
- var baseStyle = stylesMap[style.basedOn];
- if (baseStyle) {
- style.paragraphProps = mergeDeep(style.paragraphProps, baseStyle.paragraphProps);
- style.runProps = mergeDeep(style.runProps, baseStyle.runProps);
- for (const baseValues of baseStyle.styles) {
- const styleValues = style.styles.find(x => x.target == baseValues.target);
- if (styleValues) {
- this.copyStyleProperties(baseValues.values, styleValues.values);
- }
- else {
- style.styles.push({ ...baseValues, values: { ...baseValues.values } });
- }
- }
- }
- else if (this.options.debug)
- console.warn(`Can't find base style ${style.basedOn}`);
- }
- for (let style of styles) {
- style.cssName = this.processStyleName(style.id);
- }
- return stylesMap;
- }
- prodessNumberings (numberings) {
- for (let num of numberings.filter(n => n.pStyleName)) {
- const style = this.findStyle(num.pStyleName);
- if (style?.paragraphProps?.numbering) {
- style.paragraphProps.numbering.level = num.level;
- }
- }
- }
- processElement (element) {
- if (element.children) {
- for (var e of element.children) {
- e.parent = element;
- if (e.type == DomType.Table) {
- this.processTable(e);
- }
- else {
- this.processElement(e);
- }
- }
- }
- }
- processTable (table) {
- for (var r of table.children) {
- for (var c of r.children) {
- c.cssStyle = this.copyStyleProperties(table.cellStyle, c.cssStyle, [
- "border-left", "border-right", "border-top", "border-bottom",
- "padding-left", "padding-right", "padding-top", "padding-bottom"
- ]);
- this.processElement(c);
- }
- }
- }
- copyStyleProperties (input, output, attrs = null) {
- if (!input)
- return output;
- if (output == null)
- output = {};
- if (attrs == null)
- attrs = Object.getOwnPropertyNames(input);
- for (var key of attrs) {
- if (input.hasOwnProperty(key) && !output.hasOwnProperty(key))
- output[key] = input[key];
- }
- return output;
- }
- createPageElement (className, props) {
- var elem = this.createElement("section", { className });
- if (props) {
- if (props.pageMargins) {
- elem.style.paddingLeft = props.pageMargins.left;
- elem.style.paddingRight = props.pageMargins.right;
- elem.style.paddingTop = props.pageMargins.top;
- elem.style.paddingBottom = props.pageMargins.bottom;
- }
- if (props.pageSize) {
- if (!this.options.ignoreWidth)
- elem.style.width = props.pageSize.width;
- if (!this.options.ignoreHeight)
- elem.style.minHeight = props.pageSize.height;
- }
- }
- return elem;
- }
- createSectionContent (props) {
- var elem = this.createElement("article");
- if (props.columns && props.columns.numberOfColumns) {
- elem.style.columnCount = `${props.columns.numberOfColumns}`;
- elem.style.columnGap = props.columns.space;
- if (props.columns.separator) {
- elem.style.columnRule = "1px solid black";
- }
- }
- return elem;
- }
- renderSections (document) {
- const result = [];
- this.processElement(document);
- const sections = this.splitBySection(document.children, document.props);
- const pages = this.groupByPageBreaks(sections);
- let prevProps = null;
- for (let i = 0, l = pages.length; i < l; i++) {
- this.currentFootnoteIds = [];
- const section = pages[i][0];
- let props = section.sectProps;
- const pageElement = this.createPageElement(this.className, props);
- this.renderStyleValues(document.cssStyle, pageElement);
- this.options.renderHeaders && this.renderHeaderFooter(props.headerRefs, props, result.length, prevProps != props, pageElement);
- for (const sect of pages[i]) {
- var contentElement = this.createSectionContent(sect.sectProps);
- this.renderElements(sect.elements, contentElement);
- pageElement.appendChild(contentElement);
- props = sect.sectProps;
- }
- if (this.options.renderFootnotes) {
- this.renderNotes(this.currentFootnoteIds, this.footnoteMap, pageElement);
- }
- if (this.options.renderEndnotes && i == l - 1) {
- this.renderNotes(this.currentEndnoteIds, this.endnoteMap, pageElement);
- }
- this.options.renderFooters && this.renderHeaderFooter(props.footerRefs, props, result.length, prevProps != props, pageElement);
- result.push(pageElement);
- prevProps = props;
- }
- return result;
- }
- renderHeaderFooter (refs, props, page, firstOfSection, into) {
- if (!refs)
- return;
- var ref = (props.titlePage && firstOfSection ? refs.find(x => x.type == "first") : null)
- ?? (page % 2 == 1 ? refs.find(x => x.type == "even") : null)
- ?? refs.find(x => x.type == "default");
- var part = ref && this.document.findPartByRelId(ref.id, this.document.documentPart);
- if (part) {
- this.currentPart = part;
- if (!this.usedHederFooterParts.includes(part.path)) {
- this.processElement(part.rootElement);
- this.usedHederFooterParts.push(part.path);
- }
- const [el] = this.renderElements([part.rootElement], into);
- if (props?.pageMargins) {
- if (part.rootElement.type === DomType.Header) {
- el.style.marginTop = `calc(${props.pageMargins.header} - ${props.pageMargins.top})`;
- el.style.minHeight = `calc(${props.pageMargins.top} - ${props.pageMargins.header})`;
- }
- else if (part.rootElement.type === DomType.Footer) {
- el.style.marginBottom = `calc(${props.pageMargins.footer} - ${props.pageMargins.bottom})`;
- el.style.minHeight = `calc(${props.pageMargins.bottom} - ${props.pageMargins.footer})`;
- }
- }
- this.currentPart = null;
- }
- }
- isPageBreakElement (elem) {
- if (elem.type != DomType.Break)
- return false;
- if (elem.break == "lastRenderedPageBreak")
- return !this.options.ignoreLastRenderedPageBreak;
- return elem.break == "page";
- }
- isPageBreakSection (prev, next) {
- if (!prev)
- return false;
- if (!next)
- return false;
- return prev.pageSize?.orientation != next.pageSize?.orientation
- || prev.pageSize?.width != next.pageSize?.width
- || prev.pageSize?.height != next.pageSize?.height;
- }
- splitBySection (elements, defaultProps) {
- var current = { sectProps: null, elements: [], pageBreak: false };
- var result = [current];
- for (let elem of elements) {
- if (elem.type == DomType.Paragraph) {
- const s = this.findStyle(elem.styleName);
- if (s?.paragraphProps?.pageBreakBefore) {
- current.sectProps = sectProps;
- current.pageBreak = true;
- current = { sectProps: null, elements: [], pageBreak: false };
- result.push(current);
- }
- }
- current.elements.push(elem);
- if (elem.type == DomType.Paragraph) {
- const p = elem;
- var sectProps = p.sectionProps;
- var pBreakIndex = -1;
- var rBreakIndex = -1;
- if (this.options.breakPages && p.children) {
- pBreakIndex = p.children.findIndex(r => {
- rBreakIndex = r.children?.findIndex(this.isPageBreakElement.bind(this)) ?? -1;
- return rBreakIndex != -1;
- });
- }
- if (sectProps || pBreakIndex != -1) {
- current.sectProps = sectProps;
- current.pageBreak = pBreakIndex != -1;
- current = { sectProps: null, elements: [], pageBreak: false };
- result.push(current);
- }
- if (pBreakIndex != -1) {
- let breakRun = p.children[pBreakIndex];
- let splitRun = rBreakIndex < breakRun.children.length - 1;
- if (pBreakIndex < p.children.length - 1 || splitRun) {
- var children = elem.children;
- var newParagraph = { ...elem, children: children.slice(pBreakIndex) };
- elem.children = children.slice(0, pBreakIndex);
- current.elements.push(newParagraph);
- if (splitRun) {
- let runChildren = breakRun.children;
- let newRun = { ...breakRun, children: runChildren.slice(0, rBreakIndex) };
- elem.children.push(newRun);
- breakRun.children = runChildren.slice(rBreakIndex);
- }
- }
- }
- }
- }
- let currentSectProps = null;
- for (let i = result.length - 1; i >= 0; i--) {
- if (result[i].sectProps == null) {
- result[i].sectProps = currentSectProps ?? defaultProps;
- }
- else {
- currentSectProps = result[i].sectProps;
- }
- }
- return result;
- }
- groupByPageBreaks (sections) {
- let current = [];
- let prev;
- const result = [current];
- for (let s of sections) {
- current.push(s);
- if (this.options.ignoreLastRenderedPageBreak || s.pageBreak || this.isPageBreakSection(prev, s.sectProps))
- result.push(current = []);
- prev = s.sectProps;
- }
- return result.filter(x => x.length > 0);
- }
- renderWrapper (children) {
- return this.createElement("div", { className: `${this.className}-wrapper` }, children);
- }
- renderDefaultStyle () {
- var c = this.className;
- var styleText = `
- .${c}-wrapper { background: gray; padding: 30px; padding-bottom: 0px; display: flex; flex-flow: column; align-items: center; }
- .${c}-wrapper>section.${c} { background: white; box-shadow: 0 0 10px rgba(0, 0, 0, 0.5); margin-bottom: 30px; }
- .${c} { color: black; hyphens: auto; text-underline-position: from-font; }
- section.${c} { box-sizing: border-box; display: flex; flex-flow: column nowrap; position: relative; overflow: hidden; }
- section.${c}>article { margin-bottom: auto; z-index: 1; }
- section.${c}>footer { z-index: 1; }
- .${c} table { border-collapse: collapse; }
- .${c} table td, .${c} table th { vertical-align: top; }
- .${c} p { margin: 0pt; min-height: 1em; }
- .${c} span { white-space: pre-wrap; overflow-wrap: break-word; }
- .${c} a { color: inherit; text-decoration: inherit; }
- .${c} svg { fill: transparent; }
- `;
- if (this.options.renderComments) {
- styleText += `
- .${c}-comment-ref { cursor: default; }
- .${c}-comment-popover { display: none; z-index: 1000; padding: 0.5rem; background: white; position: absolute; box-shadow: 0 0 0.25rem rgba(0, 0, 0, 0.25); width: 30ch; }
- .${c}-comment-ref:hover~.${c}-comment-popover { display: block; }
- .${c}-comment-author,.${c}-comment-date { font-size: 0.875rem; color: #888; }
- `;
- }
- return createStyleElement(styleText);
- }
- renderNumbering (numberings, styleContainer) {
- var styleText = "";
- var resetCounters = [];
- for (var num of numberings) {
- var selector = `p.${this.numberingClass(num.id, num.level)}`;
- var listStyleType = "none";
- if (num.bullet) {
- let valiable = `--${this.className}-${num.bullet.src}`.toLowerCase();
- styleText += this.styleToString(`${selector}:before`, {
- "content": "' '",
- "display": "inline-block",
- "background": `var(${valiable})`
- }, num.bullet.style);
- this.tasks.push(this.document.loadNumberingImage(num.bullet.src).then(data => {
- var text = `${this.rootSelector} { ${valiable}: url(${data}) }`;
- styleContainer.appendChild(createStyleElement(text));
- }));
- }
- else if (num.levelText) {
- let counter = this.numberingCounter(num.id, num.level);
- const counterReset = counter + " " + (num.start - 1);
- if (num.level > 0) {
- styleText += this.styleToString(`p.${this.numberingClass(num.id, num.level - 1)}`, {
- "counter-reset": counterReset
- });
- }
- resetCounters.push(counterReset);
- styleText += this.styleToString(`${selector}:before`, {
- "content": this.levelTextToContent(num.levelText, num.suff, num.id, this.numFormatToCssValue(num.format)),
- "counter-increment": counter,
- ...num.rStyle,
- });
- }
- else {
- listStyleType = this.numFormatToCssValue(num.format);
- }
- styleText += this.styleToString(selector, {
- "display": "list-item",
- "list-style-position": "inside",
- "list-style-type": listStyleType,
- ...num.pStyle
- });
- }
- if (resetCounters.length > 0) {
- styleText += this.styleToString(this.rootSelector, {
- "counter-reset": resetCounters.join(" ")
- });
- }
- return createStyleElement(styleText);
- }
- renderStyles (styles) {
- var styleText = "";
- const stylesMap = this.styleMap;
- const defautStyles = keyBy(styles.filter(s => s.isDefault), s => s.target);
- for (const style of styles) {
- var subStyles = style.styles;
- if (style.linked) {
- var linkedStyle = style.linked && stylesMap[style.linked];
- if (linkedStyle)
- subStyles = subStyles.concat(linkedStyle.styles);
- else if (this.options.debug)
- console.warn(`Can't find linked style ${style.linked}`);
- }
- for (const subStyle of subStyles) {
- var selector = `${style.target ?? ''}.${style.cssName}`;
- if (style.target != subStyle.target)
- selector += ` ${subStyle.target}`;
- if (defautStyles[style.target] == style)
- selector = `.${this.className} ${style.target}, ` + selector;
- styleText += this.styleToString(selector, subStyle.values);
- }
- }
- return createStyleElement(styleText);
- }
- renderNotes (noteIds, notesMap, into) {
- var notes = noteIds.map(id => notesMap[id]).filter(x => x);
- if (notes.length > 0) {
- var result = this.createElement("ol", null, this.renderElements(notes));
- into.appendChild(result);
- }
- }
- renderElement (elem) {
- switch (elem.type) {
- case DomType.Paragraph:
- return this.renderParagraph(elem);
- case DomType.BookmarkStart:
- return this.renderBookmarkStart(elem);
- case DomType.BookmarkEnd:
- return null;
- case DomType.Run:
- return this.renderRun(elem);
- case DomType.Table:
- return this.renderTable(elem);
- case DomType.Row:
- return this.renderTableRow(elem);
- case DomType.Cell:
- return this.renderTableCell(elem);
- case DomType.Hyperlink:
- return this.renderHyperlink(elem);
- case DomType.SmartTag:
- return this.renderSmartTag(elem);
- case DomType.Drawing:
- return this.renderDrawing(elem);
- case DomType.Image:
- return this.renderImage(elem);
- case DomType.Text:
- return this.renderText(elem);
- case DomType.Text:
- return this.renderText(elem);
- case DomType.DeletedText:
- return this.renderDeletedText(elem);
- case DomType.Tab:
- return this.renderTab(elem);
- case DomType.Symbol:
- return this.renderSymbol(elem);
- case DomType.Break:
- return this.renderBreak(elem);
- case DomType.Footer:
- return this.renderContainer(elem, "footer");
- case DomType.Header:
- return this.renderContainer(elem, "header");
- case DomType.Footnote:
- case DomType.Endnote:
- return this.renderContainer(elem, "li");
- case DomType.FootnoteReference:
- return this.renderFootnoteReference(elem);
- case DomType.EndnoteReference:
- return this.renderEndnoteReference(elem);
- case DomType.NoBreakHyphen:
- return this.createElement("wbr");
- case DomType.VmlPicture:
- return this.renderVmlPicture(elem);
- case DomType.VmlElement:
- return this.renderVmlElement(elem);
- case DomType.MmlMath:
- return this.renderContainerNS(elem, ns.mathML, "math", { xmlns: ns.mathML });
- case DomType.MmlMathParagraph:
- return this.renderContainer(elem, "span");
- case DomType.MmlFraction:
- return this.renderContainerNS(elem, ns.mathML, "mfrac");
- case DomType.MmlBase:
- return this.renderContainerNS(elem, ns.mathML, elem.parent.type == DomType.MmlMatrixRow ? "mtd" : "mrow");
- case DomType.MmlNumerator:
- case DomType.MmlDenominator:
- case DomType.MmlFunction:
- case DomType.MmlLimit:
- case DomType.MmlBox:
- return this.renderContainerNS(elem, ns.mathML, "mrow");
- case DomType.MmlGroupChar:
- return this.renderMmlGroupChar(elem);
- case DomType.MmlLimitLower:
- return this.renderContainerNS(elem, ns.mathML, "munder");
- case DomType.MmlMatrix:
- return this.renderContainerNS(elem, ns.mathML, "mtable");
- case DomType.MmlMatrixRow:
- return this.renderContainerNS(elem, ns.mathML, "mtr");
- case DomType.MmlRadical:
- return this.renderMmlRadical(elem);
- case DomType.MmlSuperscript:
- return this.renderContainerNS(elem, ns.mathML, "msup");
- case DomType.MmlSubscript:
- return this.renderContainerNS(elem, ns.mathML, "msub");
- case DomType.MmlDegree:
- case DomType.MmlSuperArgument:
- case DomType.MmlSubArgument:
- return this.renderContainerNS(elem, ns.mathML, "mn");
- case DomType.MmlFunctionName:
- return this.renderContainerNS(elem, ns.mathML, "ms");
- case DomType.MmlDelimiter:
- return this.renderMmlDelimiter(elem);
- case DomType.MmlRun:
- return this.renderMmlRun(elem);
- case DomType.MmlNary:
- return this.renderMmlNary(elem);
- case DomType.MmlPreSubSuper:
- return this.renderMmlPreSubSuper(elem);
- case DomType.MmlBar:
- return this.renderMmlBar(elem);
- case DomType.MmlEquationArray:
- return this.renderMllList(elem);
- case DomType.Inserted:
- return this.renderInserted(elem);
- case DomType.Deleted:
- return this.renderDeleted(elem);
- case DomType.CommentRangeStart:
- return this.renderCommentRangeStart(elem);
- case DomType.CommentRangeEnd:
- return this.renderCommentRangeEnd(elem);
- case DomType.CommentReference:
- return this.renderCommentReference(elem);
- }
- return null;
- }
- renderChildren (elem, into) {
- return this.renderElements(elem.children, into);
- }
- renderElements (elems, into) {
- if (elems == null)
- return null;
- var result = elems.flatMap(e => this.renderElement(e)).filter(e => e != null);
- if (into)
- appendChildren(into, result);
- return result;
- }
- renderContainer (elem, tagName, props) {
- return this.createElement(tagName, props, this.renderChildren(elem));
- }
- renderContainerNS (elem, ns, tagName, props) {
- return createElementNS(ns, tagName, props, this.renderChildren(elem));
- }
- renderParagraph (elem) {
- var result = this.createElement("p");
- const style = this.findStyle(elem.styleName);
- elem.tabs ?? (elem.tabs = style?.paragraphProps?.tabs);
- this.renderClass(elem, result);
- this.renderChildren(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- this.renderCommonProperties(result.style, elem);
- const numbering = elem.numbering ?? style?.paragraphProps?.numbering;
- if (numbering) {
- result.classList.add(this.numberingClass(numbering.id, numbering.level));
- }
- return result;
- }
- renderRunProperties (style, props) {
- this.renderCommonProperties(style, props);
- }
- renderCommonProperties (style, props) {
- if (props == null)
- return;
- if (props.color) {
- style["color"] = props.color;
- }
- if (props.fontSize) {
- style["font-size"] = props.fontSize;
- }
- }
- renderHyperlink (elem) {
- var result = this.createElement("a");
- this.renderChildren(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- if (elem.href) {
- result.href = elem.href;
- }
- else if (elem.id) {
- const rel = this.document.documentPart.rels
- .find(it => it.id == elem.id && it.targetMode === "External");
- result.href = rel?.target;
- }
- return result;
- }
- renderSmartTag (elem) {
- var result = this.createElement("span");
- this.renderChildren(elem, result);
- return result;
- }
- renderCommentRangeStart (commentStart) {
- if (!this.options.renderComments)
- return null;
- const rng = new Range();
- this.commentHighlight?.add(rng);
- const result = this.htmlDocument.createComment(`start of comment #${commentStart.id}`);
- this.later(() => rng.setStart(result, 0));
- this.commentMap[commentStart.id] = rng;
- return result;
- }
- renderCommentRangeEnd (commentEnd) {
- if (!this.options.renderComments)
- return null;
- const rng = this.commentMap[commentEnd.id];
- const result = this.htmlDocument.createComment(`end of comment #${commentEnd.id}`);
- this.later(() => rng?.setEnd(result, 0));
- return result;
- }
- renderCommentReference (commentRef) {
- if (!this.options.renderComments)
- return null;
- var comment = this.document.commentsPart?.commentMap[commentRef.id];
- if (!comment)
- return null;
- const frg = new DocumentFragment();
- const commentRefEl = createElement("span", { className: `${this.className}-comment-ref` }, ['💬']);
- const commentsContainerEl = createElement("div", { className: `${this.className}-comment-popover` });
- this.renderCommentContent(comment, commentsContainerEl);
- frg.appendChild(this.htmlDocument.createComment(`comment #${comment.id} by ${comment.author} on ${comment.date}`));
- frg.appendChild(commentRefEl);
- frg.appendChild(commentsContainerEl);
- return frg;
- }
- renderCommentContent (comment, container) {
- container.appendChild(createElement('div', { className: `${this.className}-comment-author` }, [comment.author]));
- container.appendChild(createElement('div', { className: `${this.className}-comment-date` }, [new Date(comment.date).toLocaleString()]));
- this.renderChildren(comment, container);
- }
- renderDrawing (elem) {
- var result = this.createElement("div");
- result.style.display = "inline-block";
- result.style.position = "relative";
- result.style.textIndent = "0px";
- this.renderChildren(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- return result;
- }
- renderImage (elem) {
- let result = this.createElement("img");
- this.renderStyleValues(elem.cssStyle, result);
- if (this.document) {
- this.tasks.push(this.document.loadDocumentImage(elem.src, this.currentPart).then(x => {
- result.src = x;
- }));
- }
- return result;
- }
- renderText (elem) {
- return this.htmlDocument.createTextNode(elem.text);
- }
- renderDeletedText (elem) {
- return this.options.renderEndnotes ? this.htmlDocument.createTextNode(elem.text) : null;
- }
- renderBreak (elem) {
- if (elem.break == "textWrapping") {
- return this.createElement("br");
- }
- return null;
- }
- renderInserted (elem) {
- if (this.options.renderChanges)
- return this.renderContainer(elem, "ins");
- return this.renderChildren(elem);
- }
- renderDeleted (elem) {
- if (this.options.renderChanges)
- return this.renderContainer(elem, "del");
- return null;
- }
- renderSymbol (elem) {
- var span = this.createElement("span");
- span.style.fontFamily = elem.font;
- span.innerHTML = `&#x${elem.char};`;
- return span;
- }
- renderFootnoteReference (elem) {
- var result = this.createElement("sup");
- this.currentFootnoteIds.push(elem.id);
- result.textContent = `${this.currentFootnoteIds.length}`;
- return result;
- }
- renderEndnoteReference (elem) {
- var result = this.createElement("sup");
- this.currentEndnoteIds.push(elem.id);
- result.textContent = `${this.currentEndnoteIds.length}`;
- return result;
- }
- renderTab (elem) {
- var tabSpan = this.createElement("span");
- tabSpan.innerHTML = " ";
- if (this.options.experimental) {
- tabSpan.className = this.tabStopClass();
- var stops = findParent(elem, DomType.Paragraph)?.tabs;
- this.currentTabs.push({ stops, span: tabSpan });
- }
- return tabSpan;
- }
- renderBookmarkStart (elem) {
- var result = this.createElement("span");
- result.id = elem.name;
- return result;
- }
- renderRun (elem) {
- if (elem.fieldRun)
- return null;
- const result = this.createElement("span");
- if (elem.id)
- result.id = elem.id;
- this.renderClass(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- if (elem.verticalAlign) {
- const wrapper = this.createElement(elem.verticalAlign);
- this.renderChildren(elem, wrapper);
- result.appendChild(wrapper);
- }
- else {
- this.renderChildren(elem, result);
- }
- return result;
- }
- renderTable (elem) {
- let result = this.createElement("table");
- this.tableCellPositions.push(this.currentCellPosition);
- this.tableVerticalMerges.push(this.currentVerticalMerge);
- this.currentVerticalMerge = {};
- this.currentCellPosition = { col: 0, row: 0 };
- if (elem.columns)
- result.appendChild(this.renderTableColumns(elem.columns));
- this.renderClass(elem, result);
- this.renderChildren(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- this.currentVerticalMerge = this.tableVerticalMerges.pop();
- this.currentCellPosition = this.tableCellPositions.pop();
- return result;
- }
- renderTableColumns (columns) {
- let result = this.createElement("colgroup");
- for (let col of columns) {
- let colElem = this.createElement("col");
- if (col.width)
- colElem.style.width = col.width;
- result.appendChild(colElem);
- }
- return result;
- }
- renderTableRow (elem) {
- let result = this.createElement("tr");
- this.currentCellPosition.col = 0;
- this.renderClass(elem, result);
- this.renderChildren(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- this.currentCellPosition.row++;
- return result;
- }
- renderTableCell (elem) {
- let result = this.createElement("td");
- const key = this.currentCellPosition.col;
- if (elem.verticalMerge) {
- if (elem.verticalMerge == "restart") {
- this.currentVerticalMerge[key] = result;
- result.rowSpan = 1;
- }
- else if (this.currentVerticalMerge[key]) {
- this.currentVerticalMerge[key].rowSpan += 1;
- result.style.display = "none";
- }
- }
- else {
- this.currentVerticalMerge[key] = null;
- }
- this.renderClass(elem, result);
- this.renderChildren(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- if (elem.span)
- result.colSpan = elem.span;
- this.currentCellPosition.col += result.colSpan;
- return result;
- }
- renderVmlPicture (elem) {
- var result = createElement("div");
- this.renderChildren(elem, result);
- return result;
- }
- renderVmlElement (elem) {
- var container = createSvgElement("svg");
- container.setAttribute("style", elem.cssStyleText);
- const result = this.renderVmlChildElement(elem);
- if (elem.imageHref?.id) {
- this.tasks.push(this.document?.loadDocumentImage(elem.imageHref.id, this.currentPart)
- .then(x => result.setAttribute("href", x)));
- }
- container.appendChild(result);
- requestAnimationFrame(() => {
- const bb = container.firstElementChild.getBBox();
- container.setAttribute("width", `${Math.ceil(bb.x + bb.width)}`);
- container.setAttribute("height", `${Math.ceil(bb.y + bb.height)}`);
- });
- return container;
- }
- renderVmlChildElement (elem) {
- const result = createSvgElement(elem.tagName);
- Object.entries(elem.attrs).forEach(([k, v]) => result.setAttribute(k, v));
- for (let child of elem.children) {
- if (child.type == DomType.VmlElement) {
- result.appendChild(this.renderVmlChildElement(child));
- }
- else {
- result.appendChild(...asArray(this.renderElement(child)));
- }
- }
- return result;
- }
- renderMmlRadical (elem) {
- const base = elem.children.find(el => el.type == DomType.MmlBase);
- if (elem.props?.hideDegree) {
- return createElementNS(ns.mathML, "msqrt", null, this.renderElements([base]));
- }
- const degree = elem.children.find(el => el.type == DomType.MmlDegree);
- return createElementNS(ns.mathML, "mroot", null, this.renderElements([base, degree]));
- }
- renderMmlDelimiter (elem) {
- const children = [];
- children.push(createElementNS(ns.mathML, "mo", null, [elem.props.beginChar ?? '(']));
- children.push(...this.renderElements(elem.children));
- children.push(createElementNS(ns.mathML, "mo", null, [elem.props.endChar ?? ')']));
- return createElementNS(ns.mathML, "mrow", null, children);
- }
- renderMmlNary (elem) {
- const children = [];
- const grouped = keyBy(elem.children, x => x.type);
- const sup = grouped[DomType.MmlSuperArgument];
- const sub = grouped[DomType.MmlSubArgument];
- const supElem = sup ? createElementNS(ns.mathML, "mo", null, asArray(this.renderElement(sup))) : null;
- const subElem = sub ? createElementNS(ns.mathML, "mo", null, asArray(this.renderElement(sub))) : null;
- const charElem = createElementNS(ns.mathML, "mo", null, [elem.props?.char ?? '\u222B']);
- if (supElem || subElem) {
- children.push(createElementNS(ns.mathML, "munderover", null, [charElem, subElem, supElem]));
- }
- else if (supElem) {
- children.push(createElementNS(ns.mathML, "mover", null, [charElem, supElem]));
- }
- else if (subElem) {
- children.push(createElementNS(ns.mathML, "munder", null, [charElem, subElem]));
- }
- else {
- children.push(charElem);
- }
- children.push(...this.renderElements(grouped[DomType.MmlBase].children));
- return createElementNS(ns.mathML, "mrow", null, children);
- }
- renderMmlPreSubSuper (elem) {
- const children = [];
- const grouped = keyBy(elem.children, x => x.type);
- const sup = grouped[DomType.MmlSuperArgument];
- const sub = grouped[DomType.MmlSubArgument];
- const supElem = sup ? createElementNS(ns.mathML, "mo", null, asArray(this.renderElement(sup))) : null;
- const subElem = sub ? createElementNS(ns.mathML, "mo", null, asArray(this.renderElement(sub))) : null;
- const stubElem = createElementNS(ns.mathML, "mo", null);
- children.push(createElementNS(ns.mathML, "msubsup", null, [stubElem, subElem, supElem]));
- children.push(...this.renderElements(grouped[DomType.MmlBase].children));
- return createElementNS(ns.mathML, "mrow", null, children);
- }
- renderMmlGroupChar (elem) {
- const tagName = elem.props.verticalJustification === "bot" ? "mover" : "munder";
- const result = this.renderContainerNS(elem, ns.mathML, tagName);
- if (elem.props.char) {
- result.appendChild(createElementNS(ns.mathML, "mo", null, [elem.props.char]));
- }
- return result;
- }
- renderMmlBar (elem) {
- const result = this.renderContainerNS(elem, ns.mathML, "mrow");
- switch (elem.props.position) {
- case "top":
- result.style.textDecoration = "overline";
- break;
- case "bottom":
- result.style.textDecoration = "underline";
- break;
- }
- return result;
- }
- renderMmlRun (elem) {
- const result = createElementNS(ns.mathML, "ms");
- this.renderClass(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- this.renderChildren(elem, result);
- return result;
- }
- renderMllList (elem) {
- const result = createElementNS(ns.mathML, "mtable");
- this.renderClass(elem, result);
- this.renderStyleValues(elem.cssStyle, result);
- this.renderChildren(elem);
- for (let child of this.renderChildren(elem)) {
- result.appendChild(createElementNS(ns.mathML, "mtr", null, [
- createElementNS(ns.mathML, "mtd", null, [child])
- ]));
- }
- return result;
- }
- renderStyleValues (style, ouput) {
- for (let k in style) {
- if (k.startsWith("$")) {
- ouput.setAttribute(k.slice(1), style[k]);
- }
- else {
- ouput.style[k] = style[k];
- }
- }
- }
- renderClass (input, ouput) {
- if (input.className)
- ouput.className = input.className;
- if (input.styleName)
- ouput.classList.add(this.processStyleName(input.styleName));
- }
- findStyle (styleName) {
- return styleName && this.styleMap?.[styleName];
- }
- numberingClass (id, lvl) {
- return `${this.className}-num-${id}-${lvl}`;
- }
- tabStopClass () {
- return `${this.className}-tab-stop`;
- }
- styleToString (selectors, values, cssText = null) {
- let result = `${selectors} {\r\n`;
- for (const key in values) {
- if (key.startsWith('$'))
- continue;
- result += ` ${key}: ${values[key]};\r\n`;
- }
- if (cssText)
- result += cssText;
- return result + "}\r\n";
- }
- numberingCounter (id, lvl) {
- return `${this.className}-num-${id}-${lvl}`;
- }
- levelTextToContent (text, suff, id, numformat) {
- const suffMap = {
- "tab": "\\9",
- "space": "\\a0",
- };
- var result = text.replace(/%\d*/g, s => {
- let lvl = parseInt(s.substring(1), 10) - 1;
- return `"counter(${this.numberingCounter(id, lvl)}, ${numformat})"`;
- });
- return `"${result}${suffMap[suff] ?? ""}"`;
- }
- numFormatToCssValue (format) {
- var mapping = {
- none: "none",
- bullet: "disc",
- decimal: "decimal",
- lowerLetter: "lower-alpha",
- upperLetter: "upper-alpha",
- lowerRoman: "lower-roman",
- upperRoman: "upper-roman",
- decimalZero: "decimal-leading-zero",
- aiueo: "katakana",
- aiueoFullWidth: "katakana",
- chineseCounting: "simp-chinese-informal",
- chineseCountingThousand: "simp-chinese-informal",
- chineseLegalSimplified: "simp-chinese-formal",
- chosung: "hangul-consonant",
- ideographDigital: "cjk-ideographic",
- ideographTraditional: "cjk-heavenly-stem",
- ideographLegalTraditional: "trad-chinese-formal",
- ideographZodiac: "cjk-earthly-branch",
- iroha: "katakana-iroha",
- irohaFullWidth: "katakana-iroha",
- japaneseCounting: "japanese-informal",
- japaneseDigitalTenThousand: "cjk-decimal",
- japaneseLegal: "japanese-formal",
- thaiNumbers: "thai",
- koreanCounting: "korean-hangul-formal",
- koreanDigital: "korean-hangul-formal",
- koreanDigital2: "korean-hanja-informal",
- hebrew1: "hebrew",
- hebrew2: "hebrew",
- hindiNumbers: "devanagari",
- ganada: "hangul",
- taiwaneseCounting: "cjk-ideographic",
- taiwaneseCountingThousand: "cjk-ideographic",
- taiwaneseDigital: "cjk-decimal",
- };
- return mapping[format] ?? format;
- }
- refreshTabStops () {
- if (!this.options.experimental)
- return;
- clearTimeout(this.tabsTimeout);
- this.tabsTimeout = setTimeout(() => {
- const pixelToPoint = computePixelToPoint();
- for (let tab of this.currentTabs) {
- updateTabStop(tab.span, tab.stops, this.defaultTabSize, pixelToPoint);
- }
- }, 500);
- }
- later (func) {
- this.postRenderTasks.push(func);
- }
- }
- function createElement (tagName, props, children) {
- return createElementNS(undefined, tagName, props, children);
- }
- function createSvgElement (tagName, props, children) {
- return createElementNS(ns.svg, tagName, props, children);
- }
- function createElementNS (ns, tagName, props, children) {
- var result = ns ? document.createElementNS(ns, tagName) : document.createElement(tagName);
- Object.assign(result, props);
- children && appendChildren(result, children);
- return result;
- }
- function removeAllElements (elem) {
- elem.innerHTML = '';
- }
- function appendChildren (elem, children) {
- children.forEach(c => elem.appendChild(isString(c) ? document.createTextNode(c) : c));
- }
- function createStyleElement (cssText) {
- return createElement("style", { innerHTML: cssText });
- }
- function appendComment (elem, comment) {
- elem.appendChild(document.createComment(comment));
- }
- function findParent (elem, type) {
- var parent = elem.parent;
- while (parent != null && parent.type != type)
- parent = parent.parent;
- return parent;
- }
- const defaultOptions = {
- ignoreHeight: false,
- ignoreWidth: false,
- ignoreFonts: false,
- breakPages: true,
- debug: false,
- experimental: false,
- className: "docx",
- inWrapper: true,
- trimXmlDeclaration: true,
- ignoreLastRenderedPageBreak: true,
- renderHeaders: true,
- renderFooters: true,
- renderFootnotes: true,
- renderEndnotes: true,
- useBase64URL: false,
- renderChanges: false,
- renderComments: false
- };
- function parseAsync (data, userOptions) {
- const ops = { ...defaultOptions, ...userOptions };
- return WordDocument.load(data, new DocumentParser(ops), ops);
- }
- async function renderDocument (document, bodyContainer, styleContainer, userOptions) {
- const ops = { ...defaultOptions, ...userOptions };
- const renderer = new HtmlRenderer(window.document);
- renderer.render(document, bodyContainer, styleContainer, ops);
- return Promise.allSettled(renderer.tasks);
- }
- async function renderAsync (data, bodyContainer, styleContainer, userOptions) {
- const doc = await parseAsync(data, userOptions);
- await renderDocument(doc, bodyContainer, styleContainer, userOptions);
- return doc;
- }
- exports.defaultOptions = defaultOptions;
- exports.parseAsync = parseAsync;
- exports.renderAsync = renderAsync;
- exports.renderDocument = renderDocument;
- }));
- //# sourceMappingURL=docx-preview.js.map
|