Merge pull request #1552 from dolanmiu/feat/word-break-asian
#1529 Add word break feature
This commit is contained in:
37
demo/72-word-wrap.ts
Normal file
37
demo/72-word-wrap.ts
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
// Example on how to preserve word wrap text. Works with all languages.
|
||||||
|
// Import from 'docx' rather than '../build' if you install from npm
|
||||||
|
import * as fs from "fs";
|
||||||
|
import { Document, Packer, Paragraph, TextRun, SpaceType } from "../build";
|
||||||
|
|
||||||
|
const doc = new Document({
|
||||||
|
sections: [
|
||||||
|
{
|
||||||
|
children: [
|
||||||
|
new Paragraph({
|
||||||
|
children: [
|
||||||
|
new TextRun("我今天遛狗去公园"),
|
||||||
|
new TextRun({
|
||||||
|
text: "456435234523456435564745673456345456435234523456435564745673456345456435234523456435564745673456345456435234523456435564745673456345456435234523456435564745673456345",
|
||||||
|
space: SpaceType.PRESERVE,
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
new Paragraph({
|
||||||
|
children: [
|
||||||
|
new TextRun(
|
||||||
|
"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua",
|
||||||
|
),
|
||||||
|
new TextRun({
|
||||||
|
text: "456435234523456435564745673456345456435234523456435564745673456345456435234523456435564745673456345456435234523456435564745673456345456435234523456435564745673456345",
|
||||||
|
space: SpaceType.PRESERVE,
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
Packer.toBuffer(doc).then((buffer) => {
|
||||||
|
fs.writeFileSync("My Document.docx", buffer);
|
||||||
|
});
|
@ -43,19 +43,27 @@ describe("Compiler", () => {
|
|||||||
sections: [
|
sections: [
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
default: new Header(),
|
default: new Header({
|
||||||
|
children: [new Paragraph("test")],
|
||||||
|
}),
|
||||||
},
|
},
|
||||||
footers: {
|
footers: {
|
||||||
default: new Footer(),
|
default: new Footer({
|
||||||
|
children: [new Paragraph("test")],
|
||||||
|
}),
|
||||||
},
|
},
|
||||||
children: [],
|
children: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
default: new Header(),
|
default: new Header({
|
||||||
|
children: [new Paragraph("test")],
|
||||||
|
}),
|
||||||
},
|
},
|
||||||
footers: {
|
footers: {
|
||||||
default: new Footer(),
|
default: new Footer({
|
||||||
|
children: [new Paragraph("test")],
|
||||||
|
}),
|
||||||
},
|
},
|
||||||
children: [],
|
children: [],
|
||||||
},
|
},
|
||||||
@ -99,5 +107,39 @@ describe("Compiler", () => {
|
|||||||
compiler.compile(file);
|
compiler.compile(file);
|
||||||
expect(spy.callCount).to.equal(12);
|
expect(spy.callCount).to.equal(12);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should work with media datas", () => {
|
||||||
|
// This test is required because before, there was a case where Document was formatted twice, which was inefficient
|
||||||
|
// This also caused issues such as running prepForXml multiple times as format() was ran multiple times.
|
||||||
|
const paragraph = new Paragraph("");
|
||||||
|
const file = new File({
|
||||||
|
sections: [
|
||||||
|
{
|
||||||
|
properties: {},
|
||||||
|
children: [paragraph],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
// tslint:disable-next-line: no-string-literal
|
||||||
|
sinon.stub(compiler["imageReplacer"], "getMediaData").returns([
|
||||||
|
{
|
||||||
|
stream: Buffer.from(""),
|
||||||
|
fileName: "test",
|
||||||
|
transformation: {
|
||||||
|
pixels: {
|
||||||
|
x: 100,
|
||||||
|
y: 100,
|
||||||
|
},
|
||||||
|
emus: {
|
||||||
|
x: 100,
|
||||||
|
y: 100,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
compiler.compile(file);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -19,3 +19,4 @@ export * from "./shared";
|
|||||||
export * from "./border";
|
export * from "./border";
|
||||||
export * from "./values";
|
export * from "./values";
|
||||||
export * from "./vertical-align";
|
export * from "./vertical-align";
|
||||||
|
export * from "./space-type";
|
||||||
|
@ -1,10 +1,8 @@
|
|||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
import { IPageReferenceOptions } from "./pageref-properties";
|
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
import { TextAttributes } from "../run/text-attributes";
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
import { IPageReferenceOptions } from "./pageref-properties";
|
||||||
}
|
|
||||||
|
|
||||||
export class PageReferenceFieldInstruction extends XmlComponent {
|
export class PageReferenceFieldInstruction extends XmlComponent {
|
||||||
constructor(bookmarkId: string, options: IPageReferenceOptions = {}) {
|
constructor(bookmarkId: string, options: IPageReferenceOptions = {}) {
|
||||||
|
@ -1,9 +1,7 @@
|
|||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
import { TextAttributes } from "./text-attributes";
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
|
||||||
}
|
|
||||||
|
|
||||||
export class Page extends XmlComponent {
|
export class Page extends XmlComponent {
|
||||||
constructor() {
|
constructor() {
|
||||||
|
@ -1,8 +1,9 @@
|
|||||||
import { ChangeAttributes, IChangedAttributesProperties } from "../../track-revision/track-revision";
|
|
||||||
|
|
||||||
import { BorderElement, IBorderOptions } from "file/border";
|
import { BorderElement, IBorderOptions } from "file/border";
|
||||||
import { IShadingAttributesProperties, Shading } from "file/shading";
|
import { IShadingAttributesProperties, Shading } from "file/shading";
|
||||||
|
import { SpaceType } from "file/space-type";
|
||||||
|
import { ChangeAttributes, IChangedAttributesProperties } from "file/track-revision/track-revision";
|
||||||
import { HpsMeasureElement, IgnoreIfEmptyXmlComponent, OnOffElement, StringValueElement, XmlComponent } from "file/xml-components";
|
import { HpsMeasureElement, IgnoreIfEmptyXmlComponent, OnOffElement, StringValueElement, XmlComponent } from "file/xml-components";
|
||||||
|
|
||||||
import { EmphasisMark, EmphasisMarkType } from "./emphasis-mark";
|
import { EmphasisMark, EmphasisMarkType } from "./emphasis-mark";
|
||||||
import { CharacterSpacing, Color, Highlight, HighlightComplexScript } from "./formatting";
|
import { CharacterSpacing, Color, Highlight, HighlightComplexScript } from "./formatting";
|
||||||
import { IFontAttributesProperties, RunFonts } from "./run-fonts";
|
import { IFontAttributesProperties, RunFonts } from "./run-fonts";
|
||||||
@ -45,6 +46,7 @@ export interface IRunStylePropertiesOptions {
|
|||||||
readonly imprint?: boolean;
|
readonly imprint?: boolean;
|
||||||
readonly revision?: IRunPropertiesChangeOptions;
|
readonly revision?: IRunPropertiesChangeOptions;
|
||||||
readonly border?: IBorderOptions;
|
readonly border?: IBorderOptions;
|
||||||
|
readonly space?: SpaceType;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IRunPropertiesOptions extends IRunStylePropertiesOptions {
|
export interface IRunPropertiesOptions extends IRunStylePropertiesOptions {
|
||||||
|
@ -1,9 +1,7 @@
|
|||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
import { TextAttributes } from "../text-attributes";
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
|
||||||
}
|
|
||||||
|
|
||||||
export class Text extends XmlComponent {
|
export class Text extends XmlComponent {
|
||||||
constructor(text: string) {
|
constructor(text: string) {
|
||||||
|
@ -4,6 +4,7 @@ import { Formatter } from "export/formatter";
|
|||||||
import { BorderStyle } from "file/border";
|
import { BorderStyle } from "file/border";
|
||||||
// import { FootnoteReferenceRun } from "file/footnotes/footnote/run/reference-run";
|
// import { FootnoteReferenceRun } from "file/footnotes/footnote/run/reference-run";
|
||||||
import { ShadingType } from "file/shading";
|
import { ShadingType } from "file/shading";
|
||||||
|
import { SpaceType } from "file/space-type";
|
||||||
|
|
||||||
import { Run } from "./";
|
import { Run } from "./";
|
||||||
import { EmphasisMarkType } from "./emphasis-mark";
|
import { EmphasisMarkType } from "./emphasis-mark";
|
||||||
@ -521,4 +522,20 @@ describe("Run", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("#space", () => {
|
||||||
|
it("should correctly set the border", () => {
|
||||||
|
const run = new Run({
|
||||||
|
space: SpaceType.PRESERVE,
|
||||||
|
});
|
||||||
|
const tree = new Formatter().format(run);
|
||||||
|
expect(tree).to.deep.equal({
|
||||||
|
"w:r": {
|
||||||
|
_attr: {
|
||||||
|
"xml:space": "preserve",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -3,11 +3,13 @@ import { XmlComponent } from "file/xml-components";
|
|||||||
|
|
||||||
import { FootnoteReferenceRun } from "file/footnotes/footnote/run/reference-run";
|
import { FootnoteReferenceRun } from "file/footnotes/footnote/run/reference-run";
|
||||||
import { FieldInstruction } from "file/table-of-contents/field-instruction";
|
import { FieldInstruction } from "file/table-of-contents/field-instruction";
|
||||||
|
|
||||||
import { Break } from "./break";
|
import { Break } from "./break";
|
||||||
import { Begin, End, Separate } from "./field";
|
import { Begin, End, Separate } from "./field";
|
||||||
import { NumberOfPages, NumberOfPagesSection, Page } from "./page-number";
|
import { NumberOfPages, NumberOfPagesSection, Page } from "./page-number";
|
||||||
import { IRunPropertiesOptions, RunProperties } from "./properties";
|
import { IRunPropertiesOptions, RunProperties } from "./properties";
|
||||||
import { Text } from "./run-components/text";
|
import { Text } from "./run-components/text";
|
||||||
|
import { TextAttributes } from "./text-attributes";
|
||||||
|
|
||||||
export interface IRunOptions extends IRunPropertiesOptions {
|
export interface IRunOptions extends IRunPropertiesOptions {
|
||||||
readonly children?: (Begin | FieldInstruction | Separate | End | PageNumber | FootnoteReferenceRun | string)[];
|
readonly children?: (Begin | FieldInstruction | Separate | End | PageNumber | FootnoteReferenceRun | string)[];
|
||||||
@ -35,6 +37,10 @@ export class Run extends XmlComponent {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (options.space) {
|
||||||
|
this.root.push(new TextAttributes({ space: options.space }));
|
||||||
|
}
|
||||||
|
|
||||||
if (options.children) {
|
if (options.children) {
|
||||||
for (const child of options.children) {
|
for (const child of options.children) {
|
||||||
if (typeof child === "string") {
|
if (typeof child === "string") {
|
||||||
|
@ -1,10 +1,8 @@
|
|||||||
// http://officeopenxml.com/WPfieldInstructions.php
|
// http://officeopenxml.com/WPfieldInstructions.php
|
||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
import { TextAttributes } from "./text-attributes";
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
|
||||||
}
|
|
||||||
|
|
||||||
export class SequentialIdentifierInstruction extends XmlComponent {
|
export class SequentialIdentifierInstruction extends XmlComponent {
|
||||||
constructor(identifier: string) {
|
constructor(identifier: string) {
|
||||||
|
6
src/file/paragraph/run/text-attributes.ts
Normal file
6
src/file/paragraph/run/text-attributes.ts
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
import { SpaceType } from "file/space-type";
|
||||||
|
import { XmlAttributeComponent } from "file/xml-components";
|
||||||
|
|
||||||
|
export class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
||||||
|
protected readonly xmlKeys = { space: "xml:space" };
|
||||||
|
}
|
@ -1,11 +1,9 @@
|
|||||||
// http://officeopenxml.com/WPfieldInstructions.php
|
// http://officeopenxml.com/WPfieldInstructions.php
|
||||||
|
import { TextAttributes } from "file/paragraph/run/text-attributes";
|
||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
import { ITableOfContentsOptions } from "./table-of-contents-properties";
|
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
import { ITableOfContentsOptions } from "./table-of-contents-properties";
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
|
||||||
}
|
|
||||||
|
|
||||||
export class FieldInstruction extends XmlComponent {
|
export class FieldInstruction extends XmlComponent {
|
||||||
private readonly properties: ITableOfContentsOptions;
|
private readonly properties: ITableOfContentsOptions;
|
||||||
|
@ -1,9 +1,6 @@
|
|||||||
|
import { TextAttributes } from "file/paragraph/run/text-attributes";
|
||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
|
||||||
}
|
|
||||||
|
|
||||||
export class DeletedPage extends XmlComponent {
|
export class DeletedPage extends XmlComponent {
|
||||||
constructor() {
|
constructor() {
|
||||||
|
@ -1,9 +1,6 @@
|
|||||||
|
import { TextAttributes } from "file/paragraph/run/text-attributes";
|
||||||
import { SpaceType } from "file/space-type";
|
import { SpaceType } from "file/space-type";
|
||||||
import { XmlAttributeComponent, XmlComponent } from "file/xml-components";
|
import { XmlComponent } from "file/xml-components";
|
||||||
|
|
||||||
class TextAttributes extends XmlAttributeComponent<{ readonly space: SpaceType }> {
|
|
||||||
protected readonly xmlKeys = { space: "xml:space" };
|
|
||||||
}
|
|
||||||
|
|
||||||
export class DeletedText extends XmlComponent {
|
export class DeletedText extends XmlComponent {
|
||||||
constructor(text: string) {
|
constructor(text: string) {
|
||||||
|
Reference in New Issue
Block a user