When the data contains control characters, such as "U+0002" (Start of Text) or other control characters, the document would fail to generate.
Docxtemplater now has an option called stripInvalidXMLChars
which will strip the bad control characters automatically.
Use it like this :
const doc = new Docxtemplater(zip, {
stripInvalidXMLChars: true,
paragraphLoop: true,
linebreaks: true,
});
Fix small issues with expressions.js
-
Add
compiled
property to be able to access the internal expression :If the tag is simply :
{myVal}
In your code, you can use :
const expressionParser = require("docxtemplater/expressions.js"); const doc = new Docxtemplater(zip, { parser: (tag) => { const result = expressionParser(tag); const firstExpression = result.compiled.ast.body[0].expression; expect(firstExpression.type).to.equal("Identifier"); expect(firstExpression.name).to.equal("myVal"); expect(firstExpression.constant).to.equal(false); return result; }, });
-
Use better error message if using non string value for expressions.js
-
Do not fail if trying to access property of null inside a loop with expressions.js
Update to fix inspectModule.getAllTags() for all less used modules (qrcode, expand-loop for xlsx).
Use hasOwnProperty.call() for better compatibility.
Bugfix to fix inspectModule.getAllTags() when using image module ({%image} tag inside alt text) or table merge cell module.
In case when you were using the features of the module, the getAllTags would fail with a stacktrace.
When using a loop inside a table, if there is a bookmark right after the table, a new empty paragraph would be added.
Now an empty paragraph will not be added in this specific case (an empty paragraph is added between two tables for example, this is mandatory to generate a valid docx document).
Angular-expressions had a vulnerability (CVE-2024-54152), and was patched at the same time of the release of docxtemplater@3.55.5.
The new version is angular-expressions@1.4.3.
All users of docxtemplater that use angular-expressions
should upgrade to angular-expressions@1.4.3
as soon as possible..
With version angular-expressions@1.4.3 and version<=3.55.4 of docxtemplater, the
following template using this
with a computation would not return the correct value
// If the scope is `2`
{this + this} // ✕ Will return undefined with angular-expressions@1.4.3 and docxtemplater@3.55.4
{this + this} // ✓ Will return 4 with angular-expressions@1.4.3 and docxtemplater@3.55.5
// Behavior with older angular-expressions
{this + this} // ✓ Will return 4 with angular-expressions@1.4.2 and docxtemplater@3.55.5
{this + this} // ✓ Will return 4 with angular-expressions@1.4.2 and docxtemplater@3.55.4
Similarly, the following templates would also produce a non correct value :
{this|square}
{. + .}
Most expressions (Probably 99.5%) are not affected and continued to work without an upgrade in docxtemplater, such as {users | filter}, {age > 18}, {company.address}, ...
With version 3.55.5, the correct behavior is restored for the 0.5% cases using this with computations.
Improve handling of XLSX files for the inspectModule.getTags() function.
It now works correctly with innerloops, like this :
{#l1} | {#l2}
{name} | {otherName}
| {/}{/}
Previously, the "otherName" key would appear as a direct child of l1 (instead of inside l2).
Hotfix, 3.55.2 was published and did not contain the full fix, 3.55.3 now really fixes the issue with the XlsxModule and odt files.
Bugfix issue introduced in 3.55.1 when using the XlsxModule and trying to load an odt file, an unexpected error was thrown instead of an error explaining that odt files are not supported.
Add internal xmlContentTypes API for setting xmlFileNames easily from modules.
-
Performance improvements : docxtemplater now uses for of loops instead of forEach to avoid creating many anonymous functions.
-
Options immutability : in previous versions, when changing the delimiters.start from the optionsTransformer for one instance, it would change the options for all future doc instances.
If for some reason, you were changing the DocUtils.defaults, like this, this is no longer possible : (This was not documented anywhere so it is very unlikely that you're doing this).
require("docxtemplater").DocUtils.defaults.paragraphLoop = true;
This code will not have any effect starting from 3.55.0, you have to pass the options in the constructor.
-
Refactor InspectModule.getAllTags() to work correctly on xlsx files to properly show nesting of data.
Bugfix so that "errorChecker" is called when resolveTags fails (and not later in render).
This fixes a bug when using the xlsx + error location module in async mode, so that errors are correctly shown.
Add console.warn messages when using deprecated methods :
compile, attachModule, setData, resolveData, loadZip
Read more about the migration here
Add support for replacing placeholders such as {name}
in Smart art shapes.
Add syntax.allowUnclosedTag
option.
This allows to write : Hello {user
and not have an error in your template.
Improve typescript typings :
- Add position and tag to
DXT.Part
- Add void return value to
getFileType?(opts: any): string | void;
- Add
targets
to Docxtemplater instance
Update to be able to write {#loop}{. | filter}{/}
so that the variable passed
to filter is not of type Proxy.
Previously, the variable passed to the filter would be of type Proxy.
This requires angular-expressions@1.4.0
Add support for module.preZip
function, which is useful for the subtemplate and the meta module.
After upgrading to 3.51.0, if you use any of the paid modules, please also run the upgrade for all your modules with this command :
npm install docxtemplater && npx -y update-docxtemplater && npm install
Update moduleApiVersion to 3.41.0.
In the continuity of the "evaluateIdentifier" feature added in 3.49.0, we added the setIdentifier
option for the expressions.js file :
This is useful if you want to do assignments in your template, like this :
{$$globalVar = 3}
You can then write :
const expressionParser = require("docxtemplater/expressions.js");
const globalData = {};
const doc = new Docxtemplater(zip, {
parser: expressionParser.configure({
setIdentifier(tag, value) {
const matchGlobal = /^\$\$/g;
if (matchGlobal.test(tag)) {
globalData[tag] = value;
return true;
}
},
evaluateIdentifier(tag) {
const matchGlobal = /^\$\$/g;
if (matchGlobal.test(tag)) {
return globalData[tag];
}
},
}),
});
doc.render(/* data */);
In this case, all of your assignments to variable that start with "$$" will be assigned to the "globalData" object.
Also tags that contain one assignment and then a statement will now return the statement.
So for example, you can write :
Hello { $$admin=user; $$admin }
In this case, it will render "Hello John" (if the data is {user: "John"}
)
Bugfix corruption that could appear when using the vertical loop module.
Previously, the vertical loop module could sometimes produce empty tables that would not be cleaned.
For example, with following template :
--------------
| {:vt#loop} |
--------------
| XXX |
| {:vt/} |
--------------
If the loop was an empty array, the output would produce a corrupt document.
The table is now correctly removed in this case.
The table will
Add doc.keepStyles
and doc.includeSections
to Typescript definition.
Add possibility, when using the angular parser, to use "magic" keys to return some specific values. (This feature cannot be implemented if you use the "docxtemplater/expressions-ie11.js"
package).
In your template, if you write :
{#loop}
{__val}
{/}
This will retrieve the "val" value from the scope that is above the current scope (it retrieves the value of "val" in the scope outside of the loop).
const expressionParser = require("docxtemplater/expressions.js");
const doc = new Docxtemplater(zip, {
parser: expressionParser.configure({
evaluateIdentifier(tag, scope, scopeList, context) {
const matchesParent = /^(_{2,})(.*)/g;
if (matchesParent.test(tag)) {
const parentCount = tag.replace(matchesParent, "$1").length - 1;
tag = tag.replace(matchesParent, "$2");
if (parentCount >= 1) {
for (let i = scopeList.length - 1 - parentCount; i >= 0; i--) {
const s = scopeList[i];
if (s[tag] != null) {
const property = s[tag];
return typeof property === "function"
? property.bind(s)
: property;
}
}
}
}
},
}),
});
doc.render({
loop: [
{
val: "This value",
},
],
val: "Other value", // <= This value will be retrieved
});
Allow to configure the behavior of the "change delimiter syntax".
As documented here :
https://docxtemplater.com/docs/tag-types/#set-delimiter
You can for example use :
{=[[ ]]=}
[[name]]
It is possible to change the special behavior that will catch tags that start with a "=".
It is either possible to set the syntax.changeDelimiterPrefix
to null so that it won't be possible to change the delimiters inside the template, or you can change the char that is used.
For example :
const doc = new Docxtemplater(zip, {
syntax: {
changeDelimiterPrefix: null,
},
});
or
const doc = new Docxtemplater(zip, {
syntax: {
changeDelimiterPrefix: "$",
},
});
Add correct typescript typings for isIdentifierStart
and isIdentifierContinue
.
Improve getStructuredTags and getTags of the inspectModule to allow to get tags present in image attributes.
(This is to work together with the image-module 3.28.0)
Bugfix internal api mechanism :
It internally allows to have multiple traits.expandToOne().
Fixes bugs with the subtemplate and subsection module.
Update moduleApiVersion to 3.40.0.
If zip file is not a docx file, show the following error message now :
The filetype for this file could not be identified, is this file corrupted ? Zip file contains : world.txt,xxx.log
In previous versions, the following message was shown :
The filetype for this file could not be identified, is this file corrupted ?
Make it possible to dynamically allow to use a given tag for a module.
For example, you can write :
const doc = new Docxtemplater(zip, {
modules: [
{
optionsTransformer(options, doc) {
for (module of doc.modules) {
if (module.name === "RawXmlModule") {
module.prefix = function (placeholderContent) {
if (placeholderContent === "raw") {
return "raw";
}
if (placeholderContent[0] === "@") {
return placeholderContent.substr(1);
}
};
}
}
return options;
},
},
],
});
This code means that if you write : {raw} in your document (without the "@" prefix), that tag will be used as a rawxml tag.
Add "synced-zip" event that is run right after the zip is prepared.
Update moduleApiVersion to version 3.39.0, which is used by the latest subtemplate module.
Fix typescript issue with TxtTemplater
When using a loop inside a powerpoint table, if the result is an empty table, correctly drop the table from the presentation.
Add getObjectIdentifiers to expressionParser, which can be used like this :
const expressionParser = require("docxtemplater/expressions.js");
expressionParser("a.b.c").getObjectIdentifiers();
// returns { a: { b: { c: {} } } }
Bugfix for proofstate module : Following error was thrown when using this module :
Unnamed module
Now the module should work correctly
Make it possible to configure the angular parser for just one docxtemplater instance.
(This needs angular-expressions version 1.2.0)
Use following code :
const expressionParser = require("docxtemplater/expressions.js");
new Docxtemplater(zip, {
parser: expressionParser.configure({
csp: true, // this disables the use of "new Function", useful for Vercel, Deno, ...
filters: {
uppercase: (input) => {
if (typeof input === "string") {
return input.toUpperCase();
}
return input;
},
},
}),
});
Improve Typescript support to use the NodeNext moduleResolution setting.
See the explanation here.
Fixed in this pull request thanks to @benasher44.
Add getResolvedId calculation in docxtemplater to all template parts so that
all modules can store a value for each templated part. This value is guaranteed
to be the same for a given {placeholder} and data attribute between resolve
and render
.
Throw specific error if two modules with the same name are attached.
Bugfix issue on TxtTemplater when using {paragraphLoop: true}
.
Bugfix of internal API change, which was published in v3.42.5
Bugfix for TxtTemplating :
const TxtTemplater = require("docxtemplater/text.js");
The following template :
<p>Foobar</p>
Would be rendered as :
<p>Foobar</p>
Also, errors such as unclosed loops, like in :
{#users}Foo
would produce an internal stacktrace.
Now, a MultiError is thrown which contains the list of all errors inside error.properties.errors
Avoid issue Cannot read properties of undefined (reading 'length')
when using renderAsync
.
Now, the correct error message should be shown.
Bugfix to avoid following error when runnig iModule.getStructuredTags()
:
TypeError: Cannot read properties of undefined (reading 'replace')
Now, the tags are correctly returned.
Bugfix to add clone method to the assertion module and to the inspect module
Bugfix for inspect module when used together with qrcode/xlsx or table module, in some specific cases, the getTags function would return values correctly, but also return a key named "undefined", like this :
const tags = iModule.getAllTags();
console.log(tags); // would return : { name: {}, undefined: {}}
In order to apply the fix, you have to update the following modules (if you use them) :
- qrcode module to 3.4.7
- table module to 3.19.9
- xlsx module to 3.14.2
[Internal] Add filePath to each "inspect" call, which fixes a bug with the chart module when used together with the "getTags" feature of the inspect module.
If you update to this version, it is important that you also upgrade following modules if you use them :
- slides module to version 3.5.3
- pptx-sub module to version 3.1.4
Correctly show error in a multi error if the scope parser execution fails inside the render function
Previously, following error was thrown :
Error: Scope parser execution failed
at new XTScopeParserError (....)
with following template :
{#users | sortBy:'foo'}
Foo
{/}
expressionParser.filters.sortBy = function (input, ...fields) {
if (!input) return input;
return sortBy(input, fields);
};
(when sortBy is not imported correctly).
Now, the error will show a multierror with the list of errors that are happening + the tags that are causing the error.
Add support for angularExpressions.compile(), angularExpressions.Parser, and expressionParser.Lexer
Fix issue when having {tag} inside title in pptx (or docx) and using the linebreak option.
Bugfix to not add "w:sdt" inside "w:sdtContent".
Fixes a corruption on a particular type of document.
Bugfix when using docxtemplater asynchronously, and having some module inside a loop.
The "contentType" and some other properties were not transfered correctly to the elements inside the loop.
This specifically caused an issue in the HTML module to return the correct pageHeight inside the getSize
and getImage
function.
This could also lead to some other bugs that were happening only when having some specific tag present in the loop.
-
In previous versions the following code will throw an error:
new Docxtemplater(zip, { paragraphLoop: true, linebreaks: true, delimiters: { start: "$(", end: ")", }, });
$(last_name) $(first_name) Some text (Some text) Some text $(last_name) $(first_name)
MultiError { name: "TemplateError", id: "unopened_tag", explanation: "The tag beginning with \") Some text\" is unopened" }
The syntax can now be made more lenient to permit closing tags even when there are no corresponding opening tags. In your code, write :
new Docxtemplater(zip, { paragraphLoop: true, linebreaks: true, syntax: { allowUnopenedTag: true, }, });
For now, the only available property for
syntax
object isallowUnopenedTag
(it makes it possible to use the end delimiter tag as a text and not to parse it as a closing tag and cause syntax error). Fixes #726.The default behavior for the parser without setting the syntax option is the same as in 3.39.2, meaning without the
syntax.allowUnopenedTag: true
option, placeholders that are closed but not opened will throw an error. -
Internal: Refactor
getDelimiterErrors
function to be cleaner and more performant -
Internal: Add tests for new functionality
- Internal: Remove mergeObjects from doc-utils.js
- Internal: Small refactoring in regex
- Internal: Avoid calling dropUnsupportedFileTypesModules too many times
Always add {tag} in second argument to parser, like this :
parser(tag, options) {
console.log(options.tag.module); // for image tag, it will log "open-xml-templating/docxtemplater-image-module"
}
Add preResolve
API for modules that allows to run some code before the calls to resolve.
Update moduleApiVersion to version 3.37.0
Add support for templating content which is in comments.
Bugfix following error, when calling setOptions
and then getFullText
:
Cannot read properties of null (reading 'tagsXmlTextArray')
at XmlTemplater.getFullText (es6/xml-templater.js:53:56)
at Docxtemplater.getFullText (es6/docxtemplater.js:484:5)
In powerpoint, the inspect module will now return correctly for the getAllTags
and getStructuredTags
methods :
For this to work, you need to install version 3.4.10 of slides module or above.
For following template with the slides module attached :
{:loop}
{$index}{name}
The output of inspectModule.getAllTags()
will now be :
{
loop: {
$index: {},
name: {},
},
}
For the following template
Hi {#products}{# . }-{ . }-{/}{/}
This did not work correctly with following data with expressions parser :
doc.render({
products: [
[1, 2, 3, 4],
[4, 5, 6, 7],
],
});
It rendered :
Hi -1,2,3,4--4,5,6,7-
(which is incorrect)
This is because the docxtemplater/expressions.js
parser was returning an object instead of the array in this case.
Update handling of "." in angular parser.
Docxtemplater now supports the following expression :
{ . | myFilter }
This is the same as :
{ this | myFilter }
Also, the following will also work now to access the "user-name" property of the root object (synonym of this["user-name"]
:
{.["user-name"]}
Important bugfix for modules, if you are still using the legacy constructor (eg if you still have attachModule
somewhere in your code.
For example, one issue that could happen is with the HTML module, you could have following stacktrace :
Cannot read property 'getElementsByTagName' of undefined` error
This would happen when adding lists, and only if you're using one of following methods : attachModule
, setOptions
, or loadZip
.
Update to render parts of the documents in the most natural order :
First all header parts, than the main document body, than the footer part.
When using the following code :
const expressionParser = require("docxtemplater/expressions.js");
const doc = new Docxtemplater(zip, { parser: expressionParser });
doc.render();
This would always fail (when the scope was not set), but this should be allowed.
This will now work correctly.
Correctly calculate the endLindex for loop module.
This fixes a bug of the Segmentmodule (part of the subtemplate module) where the segment would show wrongly an error of "Unclosed loop" when using the segment module with following template :
{:segment s}
{#loop}{#loop}{value} {/}{/}
{:segment/}
{#loop}
{:includesegment s}
{/}
This fix also needs the latest subtemplate module : version 3.12.3
Template docProps/app.xml before word/document.xml.
This way, users can write assignments in the word/settings.xml, and use the exposed variables in the word/document.xml
With 3.37.5, require("docxtemplater/text") would throw the following error :
Cannot find module './lexer.js' from 'node_modules/docxtemplater/text.js'
This is now fixed in version 3.37.6
The TxtTemplater feature is now available using require("docxtemplater/text")
const TxtTemplater = require("docxtemplater/text.js");
const doc = new TxtTemplater("Hello {user}, how are you ?");
const result = doc.render({ user: "John" });
console.log(result); // Shows : "Hello John, how are you ?"
Previously this was only available at "docxtemplater/js/text.js", but now both are supported.
Add typings definitions for docxtemplater/js/text.js
(Fixes #715)
Add better typings to expressions.js (including typings for filters).
Add support to get identifiers when using the docxtemplater/expressions.js
package :
const expressionParser = require("docxtemplater/expressions.js");
const identifiers = expressionParser("x+0+users").getIdentifiers();
// identifiers will be : ["x", "users"]
Add typescript typings to expressions.js
Improve the way {$index} is handled with the expressions parser.
Previously, if you wrote the following :
{#todos}
{#important}{$index}.! {text}{/}
{^important}{$index}. ({text}){/}
{/}
The $index
value would always be equal to 0
.
This was because the $index
would use the closest condition or loop.
Now, the library will look whether the {#important}
is using an array or a boolean. Only for arrays will it calculate the {$index}
, it will ignore any section that is a condition.
This means that the output of the following will correctly be :
0.! Do the dishes
1.! Invite XYZ 2. (Other thing)
Previously, the same template would show just 0 for the index.
Bugfix when using following in the template :
{this["first name"]}
With following file :
doc.render({
"first name": "John",
});
This was incorrectly rendering undefined.
The bug was present since version 3.32.0
Version 3.31.6 and before were not affected by this bug.
Now, the code will correctly render : "John" in this case
Bugfix issue #707 : Correctly handle usage of {#.}{.}{/} with angular parser
Update moduleApiVersion to version 3.36.0
-
Now the modules that define their
supportedFileTypes
will correctly be removed if the filetype does'nt match even when using theattachModule
API. -
Bugfix in FixDocPRCorruptionModule : when using the following code :
const fixDocPrCorruption = require("docxtemplater/js/modules/fix-doc-pr-corruption.js"); const doc = new Docxtemplater(zip, { modules: [fixDocPrCorruption] });
The issue was that if you attached the same module to multiple docxtemplater instances in parallel, because of badly handled state, the state for the fixDocPrCorruption was overwritten
const doc1 = new Docxtemplater(zip, { modules: [fixDocPrCorruption] }); const doc2 = new Docxtemplater(zip, { modules: [fixDocPrCorruption] }); doc1.render(); // In this situation, the fixDocPrCorruption would use data from the doc2, which is incorrect, and could result in a corrupt document
Now, the fixDocPrCorruption can be used on multiple docxtemplater instances without causing any issue.
Fix typescript definition for constructor / zip instance.
Allows to correctly autocomplete after doing doc.getZip().generate()
Thanks to @oleksandr-danylchenko #704
Fix typescript definition for getAllStructuredTags
: remove file argument.
Fixed in #702 thanks to @oleksandr-danylchenko
Bugfix in require("docxtemplater/expressions.js")
:
Avoid error const expressionParser = require("docxtemplater/expressions.js");
TypeError: 'set' on proxy: trap returned falsish for property 'x'
This will no more happen now, in the case where for example you wanted to set a property to any falsy value, like this :
{x=0}
Add support to reorder modules automatically using module.priority.
Fetch data from _rels/.rels
and pass it to each module function using the relType
attribute.
Fixes issue in HTML module for some particular input that contains some tp/document-orig.xml
file.
Add support for templating text files (or simple strings).
Usage is like this :
const TxtTemplater = require("docxtemplater/js/text.js");
const doc = new TxtTemplater("Hello {user}, how are you ?");
const result = doc.render({ user: "John" });
console.log(result); // Shows : "Hello John, how are you ?"
This also works with loops and options can be set (parser for angular expressions)
Automatically template footnotes
When having a comment inside a placeholder, the document could get corrupt.
This is now fixed.
Update resolveOffset algorithm to improve slides module compatibility. Internal update of moduleApiVersion to 3.34.0
Bugfix of version 3.32.2, 3.32.1, 3.32.0
When using this document :
Hello {name}!
if the data is { name: "" }
with the "docxtemplater/expresssions.js" option, the tag will render the following :
Hello !
In version 3.32.2, this would have rendered Hello undefined!
which is incorrect.
Bugfix of version 3.32.1 and 3.32.0.
Correctly expose require("docxtemplater/expressions.js")
.
Add support to accented characters in tags when using the "docxtemplater/expressions.js" parser.
Tags such as {être} will not throw an error.
Expose "docxtemplater/expressions.js" to simplify the parser option for angular parser.
You now can replace your expressionParser code with the following :
const expressionParser = require("docxtemplater/expressions.js");
new Docxtemplater(zip, { parser: expressionParser });
For IE11 or other runtimes that do not support "Proxy", you can use instead :
const expressionParser = require("docxtemplater/expressions.js");
new Docxtemplater(zip, { parser: expressionParser });
Both examples require the angular-expressions
package which is an external dependency :
npm install --save angular-expressions
Add replaceFirstSection
and replaceLastSection
booleans types for typescript and the subsection module.
Bugfix for pptx files not keeping correct font properties (font-size) when using {linebreak: true}
option.
Bugfix to correctly handle empty loops.
Fixes #680
Previously the following stack trace would be shown :
TypeError: Cannot read properties of undefined (reading 'lIndex')
at .../docxtemplater/js/modules/loop.js:331:42
at Array.some (<anonymous>)
at LoopModule.postparse (.../docxtemplater/js/modules/loop.js:322:15)
at .../docxtemplater/js/parser.js:226:24
at Array.reduce (<anonymous>)
at _postparse (.../docxtemplater/js/parser.js:225:22)
at postparse (.../docxtemplater/js/parser.js:228:20)
at .../docxtemplater/js/modules/expand-pair-trait.js:268:30
at Array.reduce (<anonymous>)
at Object.postparse (.../docxtemplater/js/modules/expand-pair-trait.js:248:32)
Now the template is rendered correctly.
Bugfix for table module : merge-cells-col did not work correctly when placed inside a loop.
Fixes : #671
Bugfix to avoid throwing following error :
New Delimiters cannot be parsed
When the template contains an equal sign right after a closing tag.
For example, the following template would throw that error in previous versions :
Hello {name}===
Now, no error is thrown.
Correctly handle case when having a manual section break of type "nextPage", that is within a loop.
Add "cp:contentStatus" to templated tags
When having a table, that after the generation, has no table rows (<w:tr> elements), the whole <w:tbl>
element is dropped.
This automatically fixes a corruption that would happen when using a loop within table rows.
If you are using the subtemplate module, you have to update to subtemplate module 3.11.3
Throw an error when calling doc.render()
twice on the same instance.
Bugfix in fix-doc-pr-corruption module to work with xmlDocuments too.
Bugfix to correctly add xml:space="preserve"
for each type of placeholder, not just for the loop module.
This fixes spacing issues that happened rarely in many modules that have an "inline mode" (word-run, image, html, styling, paragraph-placeholder).
Make it possible to have a tag that contains multiple lines, like this :
Hello {
name = "John";
name;
}, how are you ?
And remove the paragraphs correctly.
Internal update of moduleApiVersion to 3.33.0
After upgrading this, you will need to upgrade :
- image-module to 3.14.2 or higher
- xlsx-module to 3.10.2 or higher
- styling-module to 3.6.14 or higher
Bugfix to make loop module work well even on dotx files.
Internal update of moduleApiVersion to 3.32.0
Improve corruption handling of adding <w:p/>
after some tables, even when the table is added inside a loop.
Avoid corruption when having a table without a <w:p/>
. This corruption only happens on very rare cases, for example when having a table containing a table that has no paragraph after it.
With the paragraphLoop option turned on, when using a loop that was containing a selfclosing paragraph : <w:p/>
, the generated output could become corrupt.
Now the output is valid.
Handle following input when using loops with array :
{#users}
Hello {name}
{/users}
doc.renderAsync({
users: [
new Promise((resolve, reject) => {
resolve({ name: "John" });
}),
new Promise((resolve, reject) => {
resolve({ name: "Mary" });
}),
],
});
Log errors on multiple lines instead of on one line.
Previously, error messages were shown in one line only, making the output hard to read.
Now, error messages are shown on multiple lines, using normal JSON indentation.
You can use the previous behavior (one big JSON line) by writing the following :
var doc = new Docxtemplater(zip, {
paragraphLoop: true,
linebreaks: true,
errorLogging: "jsonl", // JSONL stands for JSON Line, one big JSON entry on a single line
// Other options for errorLogging are false which means do not log anything, or "json" (which is now the default)
});
Template document properties that are set inside <vt:lpwstr>
in docProps/custom.xml
Add slideLayout files to list of templated files
Internal update of moduleApiVersion to 3.31.0
Bugfix issue introduced in 3.28.3 when using expandOne trait
Update expandOne trait to work even when having nested <w:p>
and <w:txbxContent>
tags
Fixes issue in the styling module in particular.
This release might slow down documents having many rawxml tags
Throw specific error when using rawxml tag and the type of the output is not a string.
For example, if you do the following :
{@raw}
doc.render({
raw: 42,
});
This will throw the following error :
Non string values are not allowed for rawXML tags
Typing files : make them Typescript 3.x compatible (a change in 3.28.0 of docxtemplater made the typings only work with typescript 4+)
Add type for "errorLogging" parameter
Internal update of moduleApiVersion to 3.30.0
Add matchers
API for modules, to replace the internal parse
function.
This internal change fixes bugs that can be triggered between for example the SlidesModule and the TableGridPptxModule.
When two modules have two prefixes that contain each other, for example the SlidesModule has a prefix of :
as in {:users}
, and the TableGridPptxModule is :#
as in {:#1}
.
In versions before this version, the tag {:#1}
would be Interpreted as a SlidesModule tag depending on the order of the modules.
Since this version, the matchers
API makes it possible for docxtemplater to intelligently decide that the tag belongs to the TableGridPptxModule.
The algorithm used is to use a "priority" integer if present, or to use the tag that has the longest prefix.
This change requires updates in following modules :
- chart 3.4.0 or higher
- footnotes 3.3.0 or higher
- html-pptx 3.3.0 or higher
- html 3.29.0 or higher
- image 3.12.0 or higher
- paragraph-placeholder 3.3.0 or higher
- pptx-sub 3.1.0 or higher
- slides 3.4.0 or higher
- styling 3.5.0 or higher
- subsection 3.5.0 or higher
- subtemplate 3.9.0 or higher
- table 3.15.0 or higher
- word-run 3.2.0 or higher
- xlsx 3.8.0 or higher
Internal bugfix that would show a stacktrace instead of the real underlying RenderingError.
The stacktrace was "Cannot read property indexOf of undefined" in the isStarting
function
Tag names containing "non-breaking-spaces" (Ascii code 160) will be converted to normal spaces.
Bugfix issue introduced in 3.27.0
When using the option {linebreaks: true}
, documents could be made corrupt on version 3.27.0
This version fixes the corruption
Internal update of moduleApiVersion to 3.29.0
Add support to output docx files that are bigger than 500MB.
Please make sure to update the following modules if you use them :
- chart 3.3.0 or higher
- error-location 3.4.0 or higher
- html 3.28.0 or higher
- image 3.11.0 or higher
- subsection 3.4.0 or higher
- subtemplate 3.8.0 or higher
- table 3.14.0 or higher
- styling 3.4.0 or higher
Previously, after a certain limit (usually about 500MB), the error "Invalid String Length" would be thrown by Node, because that is the max string length allowed.
Fixes issue reported here : https://stackoverflow.com/questions/68578216/docxtemplater-rangeerror-invalid-string-length
A test has been created, which you can run with npm run memorytest
if you clone this repository. It will create a file of about 550MB. This test need more memory than the default tests, and takes about 75 seconds on my computer.
Bugfix to template header and footers created by Office365.
Previously, only files matching header\d.xml would be templated. Now, also header.xml (without any digit) will be templated.
Bugfix issue when having tab character in the document, that would, after rendering, appear as " " in the document.
When updating to this version, you also need to update
- the xlsx module to 3.7.2
- the error-location module to 3.3.1
Internal update of moduleApiVersion to 3.28.0
Bugfix issue "Cannot read property 'tag' of undefined" when having an empty condition, like :
Hello {#a}{/a}
Altough there is not really a good reason to create such an empty condition, it is better to not fail with an obscure error message.
Add code for fix-doc-pr-corruption accessible by doing :
const fixDocPrCorruption = require("docxtemplater/js/modules/fix-doc-pr-corruption.js");
const doc = new Docxtemplater(zip, { modules: [fixDocPrCorruption] });
Add automatic error logging using console.log to make code samples easier.
You can replace the following code :
// The error object contains additional information when logged
// with JSON.stringify (it contains a properties object containing all suberrors).
function replaceErrors(key, value) {
if (value instanceof Error) {
return Object.getOwnPropertyNames(value).reduce(function (error, key) {
error[key] = value[key];
return error;
}, {});
}
return value;
}
function errorHandler(error) {
console.log(JSON.stringify({ error: error }, replaceErrors));
if (error.properties && error.properties.errors instanceof Array) {
const errorMessages = error.properties.errors
.map(function (error) {
return error.properties.explanation;
})
.join("\n");
console.log("errorMessages", errorMessages);
// errorMessages is a humanly readable message looking like this:
// 'The tag beginning with "foobar" is unopened'
}
throw error;
}
var zip = new PizZip(content);
var doc;
try {
doc = new Docxtemplater(zip, {
paragraphLoop: true,
linebreaks: true,
});
} catch (error) {
// Catch compilation errors
// (errors caused by the compilation of the template: misplaced tags)
errorHandler(error);
}
try {
// render the document
// (replace all occurences of {first_name} by John, {last_name} by Doe, ...)
doc.render({
first_name: "John",
last_name: "Doe",
phone: "0652455478",
description: "New Website",
});
} catch (error) {
// Catch rendering errors
// (errors relating to the rendering of the template:
// for example when the expressionParser throws an error)
errorHandler(error);
}
to this :
var zip = new PizZip(content);
var doc = new Docxtemplater(zip, {
paragraphLoop: true,
linebreaks: true,
});
// render the document
// (replace all occurences of {first_name} by John, {last_name} by Doe, ...)
doc.render({
first_name: "John",
last_name: "Doe",
phone: "0652455478",
description: "New Website",
});
To disable this automatic errorLogging, use :
var doc = new Docxtemplater(zip, {
paragraphLoop: true,
linebreaks: true,
errorLogging: false,
});
Add specific error message when using a module without instantiating it.
When doing :
const HtmlModule = require("docxtemplater-html-module");
const doc = new Docxtemplater(zip, { modules: [HtmlModule] });
The error message shown will now be :
Cannot attach a class/function as a module. Most probably you forgot to call new on the module.
If you get this error, you should simply write :
const HtmlModule = require("docxtemplater-html-module");
const doc = new Docxtemplater(zip, { modules: [new HtmlModule()] });
Bugfix when having loop containing hebrew, the text would be escaped once for each iteration
The regression was introduced in version 3.25.2 and is now fixed
Fix issue in rendering of tables generated with loop module for Powerpoint documents by deduplicating a16:rowId tags.
Previously, when having a loop, the following would be generated :
<a:tr>
<a:t>Content</a:t>
<a:extLst>
<a:ext>
<a16:rowId val="1379104516"/>
</a:ext>
</a:extLst>
</a:tr>
<a:tr>
<a:t>Content</a:t>
<a:extLst>
<a:ext>
<a16:rowId val="1379104516"/>
</a:ext>
</a:extLst>
</a:tr>
The duplicate val
attribute for the a16:rowId caused rendering issues on office live.
Now, the values are incremented after each loop, like this :
<a:ext>
<a16:rowId val="1379104516"/>
</a:ext>
<a:ext>
<a16:rowId val="1379104517"/>
</a:ext>
When having a loop inside a pptx table, the height of the frame was not updated if a tag in the form of : <a:ext uri="{11111111-1111-1111-1111-111111111111}">
was present in the document.
Now, those tags are ignored and the height of the table should be updated appropriately.
When having a loop inside a pptx table, the height of the frame will automatically be updated if some rows are added or removed.
In previous versions, the table would keep the previous height, meaning the added rows would not be shown, except after forcing a rerendering of the table by changing the fontsize.
Add support for "lambdas", eg if a value in the data is a function, that function will be called with the scope
and the scopeManager
.
You now can write :
const doc = new Docxtemplater(zip);
doc.render({
userGreeting: (scope) => {
return "How is it going, " + scope.user + " ? ";
},
users: [
{
name: "John",
},
{
name: "Mary",
},
],
});
With the following template :
{#users}
{userGreeting}
{/}
Add support to remove the call to setData
or resolveData
.
(The setData function and resolveData will still work, but will be dropped in Docxtemplater v4)
You can now do :
const doc = new Docxtemplater(zip, { linebreaks: true });
doc.render({
user: "John",
});
or, in the async version :
const doc = new Docxtemplater(zip, { linebreaks: true });
doc
.renderAsync({
user: new Promise(function (resolve, reject) {
resolve("John");
}),
})
.then(function () {
const zip = doc.getZip().generate({
type: "nodebuffer",
compression: "DEFLATE",
});
});
Bugfix error : Cannot read property 'type' of undefined
when having a loop inside a loop that has an empty content, such as :
{#a}
{#b}{/}{/}
Add typescript typings to InspectModule
Internal cleanup :
-
Remove internal unused function
getNearestLeftIndex
,getNearestRightIndex
,getNearestLeftIndexWithCache
,getNearestRightIndexWithCache
from doc-utils. -
Add explanation to all errors that have id
Use @xmldom/xmldom instead of xmldom
See : xmldom/xmldom#271
Bugfix to avoid loosing an image that is contained inside a paragraph loop.
The image would disappear only when the image was followed directly by the end of the loop, like this :
{#loop}
[IMAGE]
{/loop}
Bugfix to avoid error message Cannot read property 'offset' of undefined
on some rare templates.
The internal mergesort
function had a bug.
Bugfix to avoid dropping section when having a loop right before a section break
When having a loop containing a page section break, the generated document had continuous section breaks in the output.
With this version, the page section break should be rendered as page section breaks, and create as many pages as there are items in the loop.
Update InspectModule to make getTags function work well with XLSXModule
Disallow calling loadZip if using v4 constructor
Correctly handle <<tag>>
delimiters when saved in word as <<tag>>
Fixes #606
- Add resolveOffset internal property on scope managers to be able to handle the slides loop case correctly.
- Update moduleApiVersion to 3.26.0.
Internal change to allow to match tags with non-breaking space.
When entering Ctrl+Shift+Space
, a "non-breaking" space is created.
When using a non-breaking space in the tag {:subtemplate doc}
, or `{:table data}, the tag would not be replaced correctly. You need to upgrade the PRO modules as well to their latest version in order to fix the bug.
Bugfix issue with spaces that would disappear after rendering :
Correctly add xml:space="preserve" to all relevant w:t tags, ie all <w:t>
tags that contain a placeholder.
Update algorithm for loops to have more chance to show the underlying error.
For example, with the following template :
{#a}
{#b}
{#c}
{/d}
{/d}
{/c}
{/b}
{/a}
The algorithm will now show that the {/d} tags are involved in the issue.
Automatically add empty paragraph after table if the table is the last element in a document.
Change how the resolve algorithm works internally.
This fixes a bug in the slides module and image module, when used together, in async mode, where all image tags coming after a condition or loop would not be shown.
Because this induces an internal incompatible change, if you use the footnotes module, you have to upgrade the footnotes module to the version 3.2.0
Update moduleApiVersion to 3.25.0.
When having a document that has breaks to change the amount of columns, and loops surrounding the breaks, some pagebreaks could appear in the document.
Now, the section breaks should no more be transformed into page breaks.
When having a document containing a table like this :
================================
| {#users} abc |
--------------------------------
| {/users}{#cond} def {/cond} |
================================
Docxtemplater will now throw an error : id: "unbalanced_loop_tags"
,
and explanation: Unbalanced loop tags {#users}{/users}{#cond}{/cond}
,
In previous versions, this would most likely lead to a corrupt document.
The reason is that when a loop tag is spanning across multiple cells of a table, it will automatically expand to "row-mode", eg it will create 2 rows for each user in the users iterable.
However, the {#cond} loop tag is in one single cell, hence it will not expand to "row-mode".
Instead, the template should be written like this if the cond loop should be part of the users loop :
================================
| {#users} abc |
--------------------------------
| {#cond} def {/cond}{/users} |
================================
or like this if the loops are to be kept separate
===========================
| {#users} abc {/users} |
---------------------------
| {#cond} def {/cond} |
===========================
Don't override global configuration for docx/pptx when using a module, instead, a new fresh configuration is passed to a docxtemplater instance.
Fail with a clear error message when instantiating Docxtemplater with a null value, like this :
new Docxtemplater(null);
This will now throw :
The first argument of docxtemplater's constructor must be a valid zip file (jszip v2 or pizzip v3)
Fix speed regression when having big document with many rawxml tag.
This was caused by the fact that expandToOne was a recursive function.
Now that the function uses simple for loop, the rendering is way faster.
If you use the PRO word-run module, you should update the word-run-module to version 3.1.1
Bugfix when data contains double array, the scope parser would not do the right thing.
With the following template + data :
{#a}{.}{/a}
and the following data :
{ a: [[ "b", "c"]] }
This would render just the "c", but it should render the array [ "b", "c" ]
which will render as b,c
Avoid corruption when having self closing <w:sdtContent/>
in the document.
Throw an error when calling render on an invalid template.
Before this version, it was possible to do the following on an invalid template :
try {
doc.compile();
}
catch (e) {
doc.render();
}
And, this would produce, most of the times, a generated document that is corrupted.
Now, doc.render()
will also throw in this case the error "You should not call .render on a document that had compilation errors"
Update typescript bindings to be able to write doc.resolveData()
[Internal Only, for tests] : Rewrite xml-prettify to handle canonicalization of <w:t>Hello></w:t>
When there are errors both in the header and the footer, all errors are shown up instead of seeing only the errors of the first parsed file. This helps to find errors more quickly.
Huge performance improvements when using resolveData with loops with many iterations or with huge content inside the loop. On a document with loops of 8000 iterations, a total of 130000 Promises were created, now only about 8000 are created, (This is a 16x improvement in this case). This will produce noticeable improvements in particular in browser environments, where creating many promises costs more than in Node JS.
Some other improvements to call the parser()
function only when needed, and using some internal caching.
If you're using the PRO xlsx module, you should upgrade the module to version 3.3.4 or higher, or you might see some rendered values duplicated at multiple points in the generated spreadsheet.
- Throw error if calling
.resolveData(data)
before.compile()
- Add TypeScript typings
Bugfix corruption when having loops containing one section.
In that case, the generated file would be marked as corrupt by Word.
Do not mutate options when calling setOptions.
This means that if you do :
const options = { paragraphLoop: true };
doc.setOptions(options);
The options object will not be mutated at all anymore. Before this release, this could lead to fatal errors if using the options object across multiple Docxtemplater instances.
When using docxtemplater in async mode, inside loops, rejections would be ignored.
With this version, if one or more tags turn into a rejected Promise,
doc.resolveData(data)
will also reject (with a multi error containing all
suberrors)
Add support for dotx and dotm file formats
- Make expandToOne recursive, to allow to have multiple tags expand to the same level. (for example, for multiple "paragraph-placeholder" tags in the same paragraph).
- Update moduleApiVersion to 3.24.0.
- Make sure all postparse (also recursive ones) get the options such as filePath and contentType
Make docxtemplater fail with following error messages when using the v4 constructor with either setOptions or attachModule :
setOptions() should not be called manually when using the v4 constructor
attachModule() should not be called manually when using the v4 constructor
You should not write :
const doc = new Docxtemplater(zip);
doc.setOptions({
delimiters: {
start: "<",
end: ">",
},
});
doc.attachModule(new ImageModule())
You should instead write :
const doc = new Docxtemplater(zip, {
modules: [new ImageModule()],
delimiters: {
start: "<",
end: ">",
},
});
- Update moduleApiVersion to 3.23.0.
- Add
contentType
property in options passed to parse, postparse, render and postrender. - Bugfix in apiversion check, the patch version was not taken into account at all previously
Add proofstate module to allow to remove the <w:proofState w:grammar="clean" w:spelling="clean"/>
tag during .render()
- Add support for automatically detaching modules that do not support the current filetype when using constructor v4. In previous versions, you would do the following:
let doc = new Docxtemplater();
const zip = new PizZip(buffer)
doc.loadZip(zip);
if (doc.fileType === "pptx") {
doc.attachModule(new SlidesModule());
}
Now it is possible to write the following, without needing the condition on filetype:
const zip = new PizZip(buffer)
const options = {
modules: [new SlidesModule()],
}
try {
const doc = new Docxtemplater(zip, options);
}
catch (e) {
console.log(e);
// error handler
}
- Update moduleApiVersion to 3.22.0.
- Add a constructor method that accepts zip and optionally modules and other options. This constructor will be the official constructor in docxtemplater v4 and the methods:
loadZip
,attachModule
,setOptions
andcompile
will no more be available.
You can migrate the following code:
const doc = new Docxtemplater();
doc.loadZip(zip)
doc.setOptions({
delimiters: {
start: "<",
end: ">",
},
});
doc.attachModule(new ImageModule())
doc.attachModule(new Htmlmodule())
doc.attachModule(new Pptxmodule())
try {
doc.compile();
}
catch (e) {
// error handler
}
to
const options = {
modules: [new ImageModule(), new Htmlmodule(), new Pptxmodule()],
delimiters: {
start: "<",
end: ">",
},
}
try {
const doc = new Docxtemplater(zip, options);
}
catch (e) {
// error handler
}
- This change is backward compatible, meaning that you can continue to use the constructor with no arguments for the time being.
- Add specific error (
Duplicate open tag
andDuplicate close tag
) when using{{foobar}}
in a template when the delimiters are just one single{
and}
- Avoid error
TypeError: Cannot set property 'file' of undefined
that hides the real error
- Properly decode
&gt;
into>
in templates, in previous versions, the value was decoded to>
- Pass in
{match, getValue, getValues}
functions to second argument of module.parse - Allow to have a promise at the root level in resolveData
- Update moduleApiVersion to 3.21.0
(Internal, for modules) Pass whole part instead of just part.value to explanation function
Bugfix with paragraphLoops containing pagebreaks + tables, the pagebreak got reordered
Bugfix corrupt document when using dashloop : {-a:p loop}
inside table cell which renders false
Other bugfixes related to pagebreaks and loops
Bugfix issue with paragraphLoop and pagebreaks.
In some situations, the pagebreak got removed if the start of the loop is at the top of the page (right after the w:br
element).
- Add options.lIndex and other information about the current token in the
module.parse
method. - Update moduleApiVersion to 3.20.0
Since this version, modules using the expandToOne trait will be able to throw a specific error instead of the one related to the raw-xml module, ("Raw tag not in paragraph")
Bugfix for loop module contained inside grid slides module : Do not fail with error "Unopened loop"
Bugfix for loop module and raw module : when having an "execution" error in the parser (eg on render, after the compilation of the template, for example with the angular parser when a filter is called with bad arguments), if one {#looptag}
or {@rawtag}
had an execution error, the code would immediately stop collecting other errors. Now docxtemplater will collect all errors, for simple tags, for loop tags, and for rawxml tags.
This means that if your template was like this :
{#foobar|badfilter1}test{/}
{#foobar|badfilter2}test{/}
{@rawvalue|badfilter3}
Before version 3.16.0, calling render
on this template would throw an error with just the first tag erroring.
Since version 3.16.0, a multi error is thrown with all three tags in error being in .properties.errors
.
This is fixed in sync mode (when calling doc.render
directly) and also in async mode (when calling doc.resolveData
and then doc.render
)
- Add scopePathLength to scopemanager to be able to write "$isLast".
- Update moduleApiVersion to 3.19.0
Add offset property to following errors :
"No w:p was found at the left"
"No w:p was found at the right"
Bugfix when having dashloop inside normal loop, like this :
{#foo}test {-w:p loop}loop{/}{/}
we now throw a multi error instead of just the first error encountered.
- Update moduleApiVersion to 3.18.0
- (internal) Use array of string for shouldContain to fix corruptions
- Do not add
<w:p>
in<w:sdtContent>
if it contains a<w:r>
- If you use the xlsx module, please update the xlsx module to 3.1.1 or you will get a stacktrace
Cannot set property 'parse' of undefined
- Update moduleApiVersion to 3.17.0
- Add
options.parse
function inparse(placeholder, options)
to allow "recursive" parsing - Fill empty
<w:sdtContent>
with<w:p>
to avoid corruption
- Update moduleApiVersion to 3.16.0
- Disallow to call attachModule twice on a given module
- Add options to
parse
andpostparse
such as filePath
Use {}
(empty object) instead of "false" for the fs
shim in the browser.
- Bugfix for joinUncorrupt method, which produced a corrupt document when having a paragraphLoop containing a table containing a
{-w:tr loop}
loop.
-
Update moduleApiVersion to 3.15.0
-
Add
joinUncorrupt
tooptions
inmodule.render
to be able to fix corruptions -
Fixes corruption when using
{#loops}
that contain tables with{@rawXML}
evaluating to empty.
- Add p:txBody to pptx-lexed to handle corruption on pptx
-
Update moduleApiVersion to 3.14.0
-
Add
getNearestLeftIndex
,getNearestRightIndex
,getNearestLeftIndexWithCache
,getNearestRightIndexWithCache
to doc-utils. -
Add
preparse
method to modules to be able to retrieve info about document before the parsing of the whole document by other modules. -
Add
tagShouldContain
to the filetype API to be able to make the documents valid, which would make it possible to remove thepart.emptyValue
trick.
Bugfix when using paragraphLoop section with a pagebreak : at the end of the paragraphLoop, the pagebreak was removed by mistake. This is no longer the case since this version
Bugfix speed issue when having many loop tags in the same paragraph (over 50 loop tags in one paragraph).
(internal) Update verify moduleAPI to 3.13.0 (addition of getNearestLeftWithCache
, getNearestRightWithCache
, buildNearestCache
)
Throw Error with offsets when having loop start tag inside table and loop end tag outside table
Update xmlprettify and test it
Update dependencies
Add support for .docm
format
Document the usage of PizZip instead of JSZip
Improved fix for 3.13.5, to still create a scope but that is the same as the parent scope (without deep copy)
Bugfix condition when value is truthy, the subscope was incorrect
(This is a fix on top of 3.13.3, which fixed a bug for the value true
but not for other truthy values such as integers, strings).
Fix support for Node 6 (also for Nashorn): Rebuild js after upgrade to latest babel version
Bugfix condition when value is === true, the subscope was incorrect
Auto verify moduleAPI version on attachModule
Fix undefined is not a function evaluating 'object.getOwnPropertyDescriptors', by downgrading to babel 7.4
-
Throw multi Error with offsets when having rendering error (makes it possible to use the error location module on rendering errors).
-
Avoid false positive error "The filetype for this file could not be identified" when the document uses
<Default>
in the[Content_Types].xml
file -
Add getFileType module API to add potential support for other filetypes
Add support for setting the same delimiter on the start and beginning, for example :
%user%
, by doing doc.setOptions({delimiters: {start: "%", end: "%"}});
Remove (internal) scopeManager.clone, which is no more used
Add offset property to unopened/unclosed loop error
Parse selfclosing tag correctly, such as in <w:rPr><w:noProof/></w:rPr>
- Bugfix issue with conditions in async mode conflicting sometimes
For example with the template :
{#loop}
{#cond2}
{label}
{/cond2}
{#cond3}
{label}
{/cond3}
{/loop}
and the data :
{
label: "outer",
loop: [
{
cond2: true,
label: "inner",
},
],
}
The label would render as outer
even though it should render as inner
.
- Bugfix speed issue introduced in 3.10.0 for rawXmlModule
-
Bugfix corruption when using paragraphLoop with a template containing
<w:p>
as direct childs of other<w:p>
. -
Update getLeft, getRight, getNearestLeft, getNearestRight, getLeftOrNull, getRightOrNull to take into account the nesting of tags.
-
Ensure that the
f
functor is always called when using chunkBy (like the native.map
function) -
In expandOne trait, call
getLeft
beforegetRight
- Improve detection of filetype to read information directly from
[Content_Types].xml
instead of from zip files.
- WIPED - Published by mistake.
-
Add support for multiple elements in getRightOrNull and getLeftOrNull (to be able to specify array of elements in traits.expandToOne
-
Add postparse to expandToOne
-
Add
index
andprefix
property torender
method in the second argument (options.prefix
andoptions.render
) -
Make sure that postrender returns an array of the same length as it gets
-
Update moduleApiVersion to 3.9.0
- Update moduleApiVersion to 3.8.0
When using a paragraphLoop inside a table like this :
{#loop} Content {/loop}
When loop was falsey, the table cell caused a corruption in the resulting document.
This now fixes the issue, an empty "<w:p>/w:p" is insereted.
Performance fix in inspectModule. This performance fix concerns you only if you do doc.attachModule(inspectModule)
.
On some templates, and when reusing the same inspectModule instance, the rendering could be very slow (20 minutes in some cases). On a simple test case, the rendering time has decreased from 6 seconds to 90ms.
Render all spaces when using linebreak option
Remove .babelrc from published module
Expose Lexer
to modules (API Version 3-7-0)
Fix issue with nested loops with paragraphLoop
:
{#user}{#pets}
{name}
{/}{/}
would produce No tag "w:p" was found at the left
It now renders the same way as without paragraphLoop
-
Add possibility to change the prefix of the rawxmlmodule and of the loopmodule
-
In parsed output, also add raw containing the full tag, for example {#loop} will be parsed to : {value: "loop", raw: "#loop"}
Fix bug with {linebreaks: true} throwing a.split is not a function
Add templating of more meta data of the document, including :
Author, Title, Topic, marks, Categories, Comments, Company
Fix bug with resolveData algorithm, which raised the error : "Cannot read property value of undefined" in the render call.
Fix bug wrong styling when using linebreaks.
- Add linebreaks option for pptx and docx
Revert : Add back lIndex to parsed in addition to endLindex : Fixes issue with async rendering of multiple tags (images, qrcode, ...)
-
Add nullGetter module API
-
Update inspectModule to have :
- Unused variables (nullValues)
- filetype
- data from setData()
- templatedFiles
- list of tags
In firefox, fix : Argument 1 of DOMParser.constructor does not implement interface Principal
Fix issue of remaining xmlns=""
in some docx files
Fix issue with loopmodule eating up whitespace characters
Allow to set delimiters to {start: '<', end: '>'}
#403
Add meta context argument to custom parser with information about the tag for each types of tags
Fix infinite loop when XML is invalid (throw an explicit error instead)
Fixes #398
Fix #397 : Add more information to context
in the parser to be able to write {#users}{$index} - {name}
.
See https://docxtemplater.com/docs/configuration#custom-parser for full doc.
Add scopeManager
argument to nullGetter to know what variable is undefined.
-
Move cli out of main repository : https://github.com/open-xml-templating/docxtemplater-cli
-
Get meta attributes with raw-xml tag.
- Fix #226 when using {@rawXml} tag inside table column, the document is no longer corrupted
- Use JSZipUtils in tests
-
Add resolveData function for async data resolving
-
Fix bugs with spacing : Now space="preserve" is applied in the first and last
<w:t>
of each placeholder
- Update getAllTags to work with multiple loops by merging fields.
-
Add getAllTags to inspectModule
-
Add base-modules after
loadZip
instead of oncompile
Breaking change : The syntax of change delimiter has been changed from :
{=<% %>}
to:
{=<% %>=}
To change to <%
and %>
delimiters.
The reason is that this allows to parse the delimiters without any assumption.
For example {={{ }}}
was not possible to parse at version 3.4.0, but by adding the ending equal sign : {={{ }}=}
, the ambiguity is removed.
Add change delimiter syntax from inside template :
For example :
* {default_tags}
{=<% %>}
* <% erb_style_tags %>
<%={ }%>
* { default_tags_again }
- Add
getTags
toInspectModule
.
-
Automatically strip empty namespaces in xml files
-
Do not throw postparsed errors for tags that are unclosed/unopened
Throw error if the output contains invalid XML characters
Take into account paragraphLoop for PPTX documents
Correctly replace titles of PPTX documents
Add support for Office365 generated documents (with word/document2.xml
file)
- Fix rendering issues with
paragraphLoop
When setting paragraphLoop
, the intention is to have a special case for loops when you write :
{#users}
{name}
{/users}
Eg : both the start of the loop and the end of the loop are in a paragraph, surrounded by no other content. In that particular case, we render the content of the loop ({name}
) in this use case, in a new paragraph each time, so that there would be no additional whitespace added to the loop.
On version 3.2.1, the paragraphLoop would change the rendering for most of the loops, for example, if you wrote :
My paragraph {#users}
{name}
{/users}
the paragraphLoop code was triggered, and if users was [], even the text "My paragraph" would be removed, which was a bug.
This release fixes that bug.
- Fix bug with tr loop inside
paragraphLoop
If doing
{#par}
======================
| {#row} | {/row} |
======================
{/par}
An unexpected error 'No "w:tr" found at the left' would be raised.
Add paragraphLoop
option, that permits to have better rendering for spaces (Fixes #272)
It is recommended to turn that option on, because the templates are more readable. However since it breaks backwards-compatibility, it is turned off by default.
Inspect postparsed in compile() method instead of render()
- Add support for self-closing tag in xmlMatcher -
- Add tag information in parsed/postparsed
- Review testing code to always use "real" docx
- Add support for setting the prefix when attaching a module.
- Bugfix when looping over selfclosing tag
Add error loop_position_invalid
when the position of the loop would produce invalid xml (for example, when putting the start of a loop in a table and the end outside the loop
Use createFolders JSZip option to avoid docx corruption
Show clear error if file is ODT, escape " as "
Template values in docProps/core.xml and docProps/app.xml
Add possibility to have RenderingErrors (if data is incorrect)
Fix RangeError: Maximum call stack size exceeded
with very big document
Handle unclosed tag <a:t /> and <w:t />
Bugfix loop over string value (fixes https://github.com/open-xml-templating/docxtemplater/issues/309\)
Add support for multi errors :
docxtemplater doesn't fail on the first error, but instead, will throw multiple errors at the time. See https://docxtemplater.com/docs/errors for a detailled explanation.
Add js/tests to npm package (for modules)
Reduce size of the package
Reduce size of the package
Reduce size of the package (718.9 MB to 1.0 MB), and add prepublish script to ensure the size will never exceed 1.5MB
Add expanded property when using expandTo
Bugfix : Do not decode utf8 in xmlDocuments
- When using getRenderedMap in the modules, we now pass the filepath of the file that will be generated instead of the path of the template.
- Remove cycle between traits and docutils
- Make sure fileTypeConfig is uptodate
-
Autodetection of filetype : if you use pptx, you don't have to write doc.setOptions({fileType: "pptx"}) anymore.
-
Intelligent tagging for pptx (Fixes issue #284)
- Update documentation
- Completely remove intelligentTagging (which is on for everyone)
- Performance improvements for arrays: prefer push over concat
- Add automatic module wrapping
- Add mecanism to change which files to template with modules
- The modules are now ordered in the following order : baseModules, and then attachedModules. Before version 3.0.2, the attachedModules came before baseModules. This fixes : https://github.com/open-xml-templating/docxtemplater-image-module/issues/76
This release was published by error, and should not be used at all.
-
The rendering of the templates has now multiple steps : Lexing, Parsing, and Rendering. This makes the code much more robust, they might be bugs at the beginning, but in the long run, the code is much easier to understand/debug/change.
-
The module system was completely changed, no compatibility is kept between version 2 and version 3, please make sure to update your modules if needed.
-
You can't have the same startTag and endTag (for example
$foo$
), because this would make the code more complex and the errorHandling quite impossible. -
All extended features (loop, rawxml, invertedloops), are written as v3 modules. We dogfood our own module system, and will hopefully improve it that way.
-
The constructor arguments have been removed, and you are responsible to load the JSZip.
Instead of :
var doc = new Docxtemplater(content);
You now should do :
var zip = new JSZip(content); var doc=new Docxtemplater().loadZip(zip)
-
getTags() has been removed. It is now not easily possible to get the tags. See #258 for a alternate solution
The JSZip version that you use should be 2.x, the 3.x is now exclusively async, which Docxtemplater doesn't handle yet.
- Fix stacktrace not showing up (#245)
- Fixed a memory leak when using large document (10Mb document.xml) (#237)
- Add fileTypeConfig options in setOptions to define your own fileType config (#235)
- {@rawXml} has been fixed for pptx
- Add possibility to close loopTag with {/} #192
- Bug fix : Some times, docxtemplater could eat lots of memory due to the new "compilation" feature that was only experimental. As this feature is not yet used, it was completely removed from the codebase.
- Performance : The code has been made a little bit faster.
- Speed Improvements : docxtemplater had a regression causing it to be slow for loops. The regression would cause the code to run in O(n²) instead of O(n) where n is the length of the loops (with {#users}{name}{/users}. The bug is now fixed, and docxtemplater gained a lot of speed for users of lengthy loops.
- Breaking : To choose between docx or pptx, you now have to pass doc.setOptions({fileType:'docx'}) where the fileTypes are one of 'pptx', 'docx' (default is 'docx')
- Using es6 instead of coffeescript to code (code is still compiled to es5, to be usable with node v0.{10,12} or in the browser)
- Add finalize step in render function to throw an error if a tag is unclosed
- Module API has been updated, notably the tagXml property doesn't exist anymore, you should use the properties in
fileTypeConfig
- You should check if your modules have updated for the new module API, for instance, you should use version 1.0 of the docxtemplater image module
For example :
var xmlTemplater = this.manager.getInstance("xmlTemplater");
var tagXml = xmlTemplater.fileTypeConfig.tagsXmlArray[0];
- It is now possible to use tags inside of docx equations
- This release adds error management, all errors that are thrown by docxtemplater are now typed and contain additional properties.
- fix : When using loops with images inside, the modulemanager was not updated and would continue to return 'image' for the type of the tag even if the type changed
- feature : it is now possible to set the nullgetter for simple tags and for raw xml tags.
- bugfix for the modulemanager : it was not in sync in some cases
- They now is a default value for rawtags {@rawXml}, which is '' (this will delete the paragraph)
- bugfix (There was still '{' and '}' hardcoded)
- It is now possible to output the delimiters in the output (for example output "Mark } Z" with the template {name}
- scopeManager now doesn't return the string 'undefined' when the parser returns null. That behaviour is moved to the xmlTemplater class
- docxtemplater is now much faster to process loops #131
- docutils should not be used any more except for
DocUtils.defaults
(eg docutils is deprecated) - Module maintainers: please don't rely on docUtils anymore, except for docUtils.defaults
- Some templates would output corrupt templates, this should not happen anymore (if it still does, please open an issue)
Upgrade guide :
- If you use modules (image-module, chart-module, or others) its best to update those because they shouldn't use DocUtils anymore
- Add ScopeManager.loopOverValue
- {@rawXml} works in pptx
- Created new docxtemplater.com website
- Using mocha instead of jasmine-node
- New FAQ section in docs
- Corrupt documents when
<w:t>}{/body}</w:t>
- New events for moduleManager :
xmlRendering
andxmlRendered
- pptx generation was repaired 2be10b69d47e8c4ba0f541e4d201b29ef6281505
- header generation works with any amount of headers/footers
- slide generation works with any amount of slides
- doc.setOptions({delimiters:{start:”[“,end:”]”}}) now works e2d06dedd88860d2dac3d598b590bf81e2d113a6
- allowing same tag as start and end delimiter (eg @user@ instead of {user}) 32b9a7645f659ae835fd695e4d8ea99cc6bbec94
- made it possible to use objects to loop over (eg {#user} {name} {/user}) 97411cb3537be08f48ff707ac34d6aac8b008c50
- docxtemplater doesn’t depend on fs anymore, for transparency and security reasons.
loadFromFile
has been removed. You now just have to pass the content to Docxtemplater’s constructor.setTags
has been renamed tosetData
applyTags
has been renamed torender
- the constructor has changed: eg
new Docxtemplater(content,options)
will now callJSzip.load(content,options)
. - To pass options (such as the parser), you will have to call
setOptions
- The
output
function has been removed. You should now callgetZip().generate(options)
where the options are documented here: https://stuk.github.io/jszip/documentation/api_jszip/generate.html - the qrcode module has been removed, and will be migrated in an other package that will be attached to docxtemplater
The QrCode Option had a security issue. If you don’t upgrade according to this, the functionality should continue to work but the leak will still be there. If you are running in the browser, the vulnerability will not affect you (no access to the filesystem). If the users can’t change the qrCodes or their value, you’re safe too.
If you set qrCode:true, you are affected. The Command Line is not affected as of v0.7.0 (but was on 0.6.3 and less). However the command line as of v0.7.0 is not more able to load images over the filesystem.
You should set qrCode to a function now, according to https://docxtemplater.com/docs/configuration#image-replacing.