jspdf-md-renderer 1.6.2 → 1.6.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,81 @@
1
+ import { default as default_2 } from 'jspdf';
2
+ import { jsPDFOptions } from 'jspdf';
3
+
4
+ declare type FontItem = {
5
+ name: string;
6
+ style: string;
7
+ };
8
+
9
+ /**
10
+ * Parses markdown into tokens and converts to a custom parsed structure.
11
+ *
12
+ * @param text - The markdown content to parse.
13
+ * @returns Parsed markdown elements.
14
+ */
15
+ export declare const MdTextParser: (text: string) => Promise<ParsedElement[]>;
16
+
17
+ /**
18
+ * Renders parsed markdown text into jsPDF document.
19
+ *
20
+ * @param doc - The jsPDF document.
21
+ * @param text - The markdown content to render.
22
+ * @param options - The render options (fonts, page margins, etc.).
23
+ */
24
+ export declare const MdTextRender: (doc: default_2, text: string, options: RenderOption) => Promise<void>;
25
+
26
+ declare type ParsedElement = {
27
+ type: string;
28
+ content?: string;
29
+ depth?: number;
30
+ items?: ParsedElement[];
31
+ ordered?: boolean;
32
+ start?: number;
33
+ lang?: string;
34
+ code?: string;
35
+ src?: string;
36
+ alt?: string;
37
+ href?: string;
38
+ text?: string;
39
+ header?: {
40
+ type?: string;
41
+ content?: any;
42
+ };
43
+ rows?: {
44
+ type?: string;
45
+ content?: any;
46
+ };
47
+ };
48
+
49
+ export declare type RenderOption = {
50
+ cursor: {
51
+ x: number;
52
+ y: number;
53
+ };
54
+ page: {
55
+ format?: string | number[];
56
+ unit?: jsPDFOptions['unit'];
57
+ orientation?: jsPDFOptions['orientation'];
58
+ maxContentWidth: number;
59
+ maxContentHeight: number;
60
+ lineSpace: number;
61
+ defaultLineHeightFactor: number;
62
+ defaultFontSize: number;
63
+ defaultTitleFontSize: number;
64
+ topmargin: number;
65
+ xpading: number;
66
+ xmargin: number;
67
+ indent: number;
68
+ };
69
+ font: {
70
+ bold: FontItem;
71
+ regular: FontItem;
72
+ light: FontItem;
73
+ };
74
+ content?: {
75
+ textAlignment: 'left' | 'right' | 'center' | 'justify';
76
+ };
77
+ pageBreakHandler?: () => void;
78
+ endCursorYHandler: (y: number) => void;
79
+ };
80
+
81
+ export { }
package/dist/index.js ADDED
@@ -0,0 +1,4 @@
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const c=require("marked");var h=(t=>(t.Heading="heading",t.Paragraph="paragraph",t.List="list",t.ListItem="list_item",t.Blockquote="blockquote",t.Code="code",t.CodeSpan="codespan",t.Table="table",t.Html="html",t.Hr="hr",t.Image="image",t.Link="link",t.Strong="strong",t.Em="em",t.TableHeader="table_header",t.TableCell="table_cell",t.Raw="raw",t.Text="text",t))(h||{});const z=async t=>{const a=await c.marked.lexer(t,{async:!0});return C(a)},C=t=>{const a=[];return t.forEach(e=>{try{const g=I[e.type];g?a.push(g(e)):a.push({type:h.Raw,content:e.raw})}catch(g){console.error("Failed to handle token ==>",e,g)}}),a},I={[h.Heading]:t=>({type:h.Heading,depth:t.depth,content:t.text,items:t.tokens?C(t.tokens):[]}),[h.Paragraph]:t=>({type:h.Paragraph,content:t.text,items:t.tokens?C(t.tokens):[]}),[h.List]:t=>({type:h.List,ordered:t.ordered,start:t.start,items:t.items?C(t.items):[]}),[h.ListItem]:t=>({type:h.ListItem,content:t.text,items:t.tokens?C(t.tokens):[]}),[h.Code]:t=>({type:h.Code,lang:t.lang,code:t.text}),[h.Table]:t=>({type:h.Table,header:t.header.map(a=>({type:h.TableHeader,content:a})),rows:t.rows.map(a=>a.map(e=>({type:h.TableCell,content:e})))}),[h.Image]:t=>({type:h.Image,src:t.href,alt:t.text}),[h.Link]:t=>({type:h.Link,href:t.href,text:t.text,items:t.tokens?C(t.tokens):[]}),[h.Strong]:t=>({type:h.Strong,content:t.text,items:t.tokens?C(t.tokens):[]}),[h.Em]:t=>({type:h.Em,content:t.text,items:t.tokens?C(t.tokens):[]}),[h.Text]:t=>({type:h.Text,content:t.text,items:t.tokens?C(t.tokens):[]}),[h.Hr]:t=>({type:h.Hr,content:t.raw,items:t.tokens?C(t.tokens):[]}),[h.CodeSpan]:t=>({type:h.CodeSpan,content:t.text,items:t.tokens?C(t.tokens):[]})},S=(t,a)=>{var e,g;typeof a.pageBreakHandler=="function"?a.pageBreakHandler():t.addPage((e=a.page)==null?void 0:e.format,(g=a.page)==null?void 0:g.orientation)},p=(t,a)=>t.getTextDimensions("H").h*a.page.defaultLineHeightFactor,E=(t,a,e,g,i,n)=>{const s=6-((a==null?void 0:a.depth)??0)>0?6-((a==null?void 0:a.depth)??0):0;if(t.setFontSize(i.page.defaultFontSize+s),a!=null&&a.items&&(a==null?void 0:a.items.length)>0)for(const f of(a==null?void 0:a.items)??[])e=n(f,g,!1);else t.text((a==null?void 0:a.content)??"",e.x+g,e.y,{align:"left",maxWidth:i.page.maxContentWidth-g}),e.y+=1.5*p(t,i);return t.setFontSize(i.page.defaultFontSize),e},P=(t,a,e,g,i,n,s,f,r)=>{const y=(f-e)/(a.length-1);let x=i;const l=n+g*s;for(const b of a)t.text(b.text,x,l,{align:"justify",lineHeightFactor:r,maxWidth:f}),x+=b.wordLength+y},j=(t,a,e,g,i,n,s,f)=>{const r=t.map(y=>y.text).join(" ");a.text(r,e,g+i*n,{align:"justify",lineHeightFactor:f,maxWidth:s})},F=(t,a,e,g,i,n)=>{const s={x:e,y:g},f=t.getTextDimensions("A").h*n,r=a.split(" ");let y=0,x=[],l=0;for(const b of r){const W=t.getTextWidth(b+"a");W+l>=i&&(P(t,x,l,y++,e,g,f,i,n),x=[],l=0),x.push({text:b,wordLength:W}),l+=W}return x.length>0&&j(x,t,e,g,y,f,i,n),s.y=g+y*f,s.x=t.getTextWidth(a)+e,s},D=(t,a,e,g,i,n)=>{t.setFontSize(i.page.defaultFontSize);let s=a.content;const f=t.getTextDimensions("A").h*i.page.defaultLineHeightFactor;if(a!=null&&a.items&&(a==null?void 0:a.items.length)>0)for(const r of(a==null?void 0:a.items)??[])e=n(r,g,!1);else{if(e.y+t.splitTextToSize(s??"",i.page.maxContentWidth-g).length*f-3*f>=i.page.maxContentHeight){const r=t.splitTextToSize(s??"",i.page.maxContentWidth-g),y=[],x=e.y;for(let l=0;l<r.length;l++)if(e.y-2*f<i.page.maxContentHeight)y.push(r[l]),e.y+=i.page.lineSpace;else{l<=r.length-1&&(s=r.slice(l).join(""));break}y.length>0&&(e=F(t,y.join(" "),e.x+g,x,i.page.maxContentWidth-g,i.page.defaultLineHeightFactor)),S(t,i),e.y=i.page.topmargin}e.y=F(t,s??"",e.x+g,e.y,i.page.maxContentWidth-g,i.page.defaultLineHeightFactor).y+p(t,i),e.x=i.page.xpading}return e},R=(t,a,e,g,i,n)=>{var s;t.setFontSize(i.page.defaultFontSize);for(const[f,r]of((s=a==null?void 0:a.items)==null?void 0:s.entries())??[]){const y=a.ordered?(a.start??0)+f:a.start;e=n(r,g+1,!0,y,a.ordered),e.y+=p(t,i)*.2}return e},w=(t,a,e,g,i)=>{const n=t.getFont().fontName,s=t.getFont().fontStyle,f=t.getFontSize(),r=x=>{switch(x){case"normal":return 0;case"bold":return 1;case"italic":return 1.5;case"bolditalic":return 1.5;default:return 0}},y=(x,l)=>{var d;l==="bold"?t.setFont(i.font.bold.name&&i.font.bold.name!==""?i.font.bold.name:n,i.font.bold.style||"bold"):l==="italic"?t.setFont(i.font.regular.name,"italic"):l==="bolditalic"?t.setFont(i.font.bold.name&&i.font.bold.name!==""?i.font.bold.name:n,"bolditalic"):t.setFont(i.font.regular.name,s);const b=i.page.maxContentWidth-g-e.x,W=t.splitTextToSize(x,b);if(W.length>1){const H=W[0],m=(d=W==null?void 0:W.slice(1))==null?void 0:d.join(" ");t.text(H,e.x+(g>=2?g+2*r(l):0),e.y,{baseline:"top",maxWidth:b}),e.x=i.page.xpading+g,e.y+=p(t,i);const L=i.page.maxContentWidth-g-i.page.xpading-i.page.xmargin;t.splitTextToSize(m,L).forEach(T=>{t.text(T,e.x+g,e.y,{baseline:"top",maxWidth:L}),e.x=g,e.y+=p(t,i)})}else t.text(x,e.x+g,e.y,{baseline:"top",maxWidth:b}),e.x+=t.getTextDimensions(x).w+(g>=2?x.split(" ").length+2:2)*r(l)};if(a.type==="text"&&a.items&&a.items.length>0)for(const x of a.items)if(x.type==="em"||x.type==="strong"){const l=x.type==="em"?"italic":"bold";if(x.items&&x.items.length>0)for(const b of x.items)b.type==="strong"&&l==="italic"||b.type==="em"&&l==="bold"?y(b.content||"","bolditalic"):y(b.content||"",l);else y(x.content||"",l)}else y(x.content||"","normal");else a.type==="em"?y(a.content||"","italic"):a.type==="strong"?y(a.content||"","bold"):y(a.content||"","normal");return t.setFont(n,s),t.setFontSize(f),e},_=(t,a,e,g,i,n,s,f)=>{const r=g*i.page.indent,y=f?`${s}. `:"• ";e.y+p(t,i)>=i.page.maxContentHeight&&(S(t,i),e.y=i.page.topmargin),t.setFont(i.font.regular.name,i.font.regular.style),t.text(y,e.x+r,e.y,{baseline:"top"});const x=t.getTextWidth(y);if(e.x+=x,a.items&&a.items.length>0)for(const l of a.items){if(e.y+p(t,i)>=i.page.maxContentHeight&&(S(t,i),e.y=i.page.topmargin),l.type===h.List)n(l,g+1,!0,s,l.ordered??!1);else if(l.type===h.ListItem){const b=a.type===h.List?g:g+1;n(l,b,!0,s,f)}else e=w(t,l,e,r,i);e.x=i.page.xpading,e.y+=p(t,i)}else if(a.content){const l=e.x+r,b=t.getTextWidth(y),W=i.page.maxContentWidth-r-b,d=t.splitTextToSize(a.content,W);if(d.length>0){t.text(d[0],l+b,e.y,{baseline:"top",maxWidth:W});for(let m=1;m<d.length;m++)e.y+=p(t,i),t.text(d[m],l+b,e.y,{baseline:"top",maxWidth:W});e.y+=p(t,i),e.x=i.page.xmargin+r;const H=t.getTextWidth(a.content);e.x+=H}}return e},q=(t,a,e,g,i,n,s,f,r,y=!0)=>{if(a!=null&&a.items&&(a==null?void 0:a.items.length)>0)for(const x of(a==null?void 0:a.items)??[])e=s(x,g,i,f,r,y);else{const x=g*n.page.indent,l=i?r?`${f}. `:"• ":"";if(i&&l){const b=t.getTextWidth(l),W=n.page.maxContentWidth-x-b,d=t.splitTextToSize(a.content||"",W);if(d.length>0){t.text(l,e.x+x,e.y,{baseline:"top"}),t.text(d[0],e.x+x+b,e.y,{baseline:"top",maxWidth:W});for(let m=1;m<d.length;m++)e.x=n.page.xpading,e.y+=p(t,n),t.text(d[m],e.x+x+b,e.y,{baseline:"top",maxWidth:W});e.y+=p(t,n),e.x=n.page.xpading+x;const H=t.getTextWidth(a.content||"");e.x+=H}}else{const b=t.splitTextToSize(a.content||"",n.page.maxContentWidth-x);e.y+b.length*p(t,n)>=n.page.maxContentHeight&&(S(t,n),e.y=n.page.topmargin),y?(e.y=F(t,a.content||"",e.x+x,e.y,n.page.maxContentWidth-x,n.page.defaultLineHeightFactor).y+p(t,n),e.x=n.page.xpading):(t.text(a.content||"",e.x+x,e.y,{baseline:"top"}),e.x+=t.getTextWidth(a.content||""),e.x>=n.page.xpading+n.page.maxContentWidth&&(S(t,n),e.x=n.page.xpading,e.y=n.page.topmargin))}}return e},B=(t,a,e)=>{const g=t.internal.pageSize.getWidth();return t.setLineDashPattern([1,1],0),t.setLineWidth(.1),t.line(e.page.xpading,a.y,g-e.page.xpading,a.y),t.setLineWidth(.1),t.setLineDashPattern([],0),a.y+=p(t,e),a},J=(t,a,e,g,i,n)=>{const s=g*n.page.indent;e.y+t.splitTextToSize(a.code??"",n.page.maxContentWidth-s).length*p(t,n)-2*p(t,n)>=n.page.maxContentHeight&&(S(t,n),e.y=n.page.topmargin);const f=t.splitTextToSize(a.code??"",n.page.maxContentWidth-s).length*p(t,n);return e.y+=n.page.lineSpace,t.setFillColor("#EEEEEE"),t.setDrawColor("#eee"),t.roundedRect(e.x,e.y-n.page.lineSpace,n.page.maxContentWidth,f,2,2,"FD"),t.setFontSize(10),t.text(a.lang??"",e.x+n.page.maxContentWidth-t.getTextWidth(a.lang??"")-n.page.lineSpace/2,e.y),t.setFontSize(n.page.defaultFontSize),t.text(a.code??"",e.x+4,e.y),e.y+=f,e},$=async(t,a,e)=>{const g=await z(a);let i={x:e.cursor.x,y:e.cursor.y};const n=(s,f=0,r=!1,y=0,x=!1,l=!0)=>{const b=f*e.page.indent;switch(i.y+t.splitTextToSize(s.content??"",e.page.maxContentWidth-b).length*p(t,e)>=e.page.maxContentHeight&&(S(t,e),i.y=e.page.topmargin),s.type){case h.Heading:i=E(t,s,i,b,e,n);break;case h.Paragraph:i=D(t,s,i,b,e,n);break;case h.List:i=R(t,s,i,f,e,n);break;case h.ListItem:i=_(t,s,i,f,e,n,y,x);break;case h.Hr:i=B(t,i,e);break;case h.Code:i=J(t,s,i,f,r,e);break;case h.Strong:case h.Em:i=w(t,s,i,b,e);break;case h.Raw:case h.Text:i=q(t,s,i,f,r,e,n,y,x,l);break;default:console.warn(`Warning: Unsupported element type encountered: ${s.type}.
2
+ If you believe this element type should be supported, please create an issue at:
3
+ https://github.com/JeelGajera/jspdf-md-renderer/issues
4
+ with details of the element and expected behavior. Thank you for helping improve this library!`);break}return i};for(const s of g)n(s);e.endCursorYHandler(i.y)};exports.MdTextParser=z;exports.MdTextRender=$;
package/dist/index.mjs ADDED
@@ -0,0 +1,534 @@
1
+ import { marked as I } from "marked";
2
+ var h = /* @__PURE__ */ ((t) => (t.Heading = "heading", t.Paragraph = "paragraph", t.List = "list", t.ListItem = "list_item", t.Blockquote = "blockquote", t.Code = "code", t.CodeSpan = "codespan", t.Table = "table", t.Html = "html", t.Hr = "hr", t.Image = "image", t.Link = "link", t.Strong = "strong", t.Em = "em", t.TableHeader = "table_header", t.TableCell = "table_cell", t.Raw = "raw", t.Text = "text", t))(h || {});
3
+ const T = async (t) => {
4
+ const a = await I.lexer(t, { async: !0 });
5
+ return d(a);
6
+ }, d = (t) => {
7
+ const a = [];
8
+ return t.forEach((e) => {
9
+ try {
10
+ const g = c[e.type];
11
+ g ? a.push(g(e)) : a.push({
12
+ type: h.Raw,
13
+ content: e.raw
14
+ });
15
+ } catch (g) {
16
+ console.error("Failed to handle token ==>", e, g);
17
+ }
18
+ }), a;
19
+ }, c = {
20
+ [h.Heading]: (t) => ({
21
+ type: h.Heading,
22
+ depth: t.depth,
23
+ content: t.text,
24
+ items: t.tokens ? d(t.tokens) : []
25
+ }),
26
+ [h.Paragraph]: (t) => ({
27
+ type: h.Paragraph,
28
+ content: t.text,
29
+ items: t.tokens ? d(t.tokens) : []
30
+ }),
31
+ [h.List]: (t) => ({
32
+ type: h.List,
33
+ ordered: t.ordered,
34
+ start: t.start,
35
+ items: t.items ? d(t.items) : []
36
+ }),
37
+ [h.ListItem]: (t) => ({
38
+ type: h.ListItem,
39
+ content: t.text,
40
+ items: t.tokens ? d(t.tokens) : []
41
+ }),
42
+ [h.Code]: (t) => ({
43
+ type: h.Code,
44
+ lang: t.lang,
45
+ code: t.text
46
+ }),
47
+ [h.Table]: (t) => ({
48
+ type: h.Table,
49
+ header: t.header.map((a) => ({
50
+ type: h.TableHeader,
51
+ content: a
52
+ })),
53
+ rows: t.rows.map(
54
+ (a) => a.map((e) => ({
55
+ type: h.TableCell,
56
+ content: e
57
+ }))
58
+ )
59
+ }),
60
+ [h.Image]: (t) => ({
61
+ type: h.Image,
62
+ src: t.href,
63
+ alt: t.text
64
+ }),
65
+ [h.Link]: (t) => ({
66
+ type: h.Link,
67
+ href: t.href,
68
+ text: t.text,
69
+ items: t.tokens ? d(t.tokens) : []
70
+ }),
71
+ [h.Strong]: (t) => ({
72
+ type: h.Strong,
73
+ content: t.text,
74
+ items: t.tokens ? d(t.tokens) : []
75
+ }),
76
+ [h.Em]: (t) => ({
77
+ type: h.Em,
78
+ content: t.text,
79
+ items: t.tokens ? d(t.tokens) : []
80
+ }),
81
+ [h.Text]: (t) => ({
82
+ type: h.Text,
83
+ content: t.text,
84
+ items: t.tokens ? d(t.tokens) : []
85
+ }),
86
+ [h.Hr]: (t) => ({
87
+ type: h.Hr,
88
+ content: t.raw,
89
+ items: t.tokens ? d(t.tokens) : []
90
+ }),
91
+ [h.CodeSpan]: (t) => ({
92
+ type: h.CodeSpan,
93
+ content: t.text,
94
+ items: t.tokens ? d(t.tokens) : []
95
+ })
96
+ }, H = (t, a) => {
97
+ var e, g;
98
+ typeof a.pageBreakHandler == "function" ? a.pageBreakHandler() : t.addPage((e = a.page) == null ? void 0 : e.format, (g = a.page) == null ? void 0 : g.orientation);
99
+ }, p = (t, a) => t.getTextDimensions("H").h * a.page.defaultLineHeightFactor, E = (t, a, e, g, i, n) => {
100
+ const s = 6 - ((a == null ? void 0 : a.depth) ?? 0) > 0 ? 6 - ((a == null ? void 0 : a.depth) ?? 0) : 0;
101
+ if (t.setFontSize(i.page.defaultFontSize + s), a != null && a.items && (a == null ? void 0 : a.items.length) > 0)
102
+ for (const f of (a == null ? void 0 : a.items) ?? [])
103
+ e = n(f, g, !1);
104
+ else
105
+ t.text((a == null ? void 0 : a.content) ?? "", e.x + g, e.y, {
106
+ align: "left",
107
+ maxWidth: i.page.maxContentWidth - g
108
+ }), e.y += 1.5 * p(t, i);
109
+ return t.setFontSize(i.page.defaultFontSize), e;
110
+ }, P = (t, a, e, g, i, n, s, f, r) => {
111
+ const y = (f - e) / (a.length - 1);
112
+ let x = i;
113
+ const l = n + g * s;
114
+ for (const b of a)
115
+ t.text(b.text, x, l, {
116
+ align: "justify",
117
+ lineHeightFactor: r,
118
+ maxWidth: f
119
+ }), x += b.wordLength + y;
120
+ }, j = (t, a, e, g, i, n, s, f) => {
121
+ const r = t.map((y) => y.text).join(" ");
122
+ a.text(r, e, g + i * n, {
123
+ align: "justify",
124
+ lineHeightFactor: f,
125
+ maxWidth: s
126
+ });
127
+ }, F = (t, a, e, g, i, n) => {
128
+ const s = {
129
+ x: e,
130
+ y: g
131
+ }, f = t.getTextDimensions("A").h * n, r = a.split(" ");
132
+ let y = 0, x = [], l = 0;
133
+ for (const b of r) {
134
+ const W = t.getTextWidth(b + "a");
135
+ W + l >= i && (P(
136
+ t,
137
+ x,
138
+ l,
139
+ y++,
140
+ e,
141
+ g,
142
+ f,
143
+ i,
144
+ n
145
+ ), x = [], l = 0), x.push({ text: b, wordLength: W }), l += W;
146
+ }
147
+ return x.length > 0 && j(
148
+ x,
149
+ t,
150
+ e,
151
+ g,
152
+ y,
153
+ f,
154
+ i,
155
+ n
156
+ ), s.y = g + y * f, s.x = t.getTextWidth(a) + e, s;
157
+ }, D = (t, a, e, g, i, n) => {
158
+ t.setFontSize(i.page.defaultFontSize);
159
+ let s = a.content;
160
+ const f = t.getTextDimensions("A").h * i.page.defaultLineHeightFactor;
161
+ if (a != null && a.items && (a == null ? void 0 : a.items.length) > 0)
162
+ for (const r of (a == null ? void 0 : a.items) ?? [])
163
+ e = n(r, g, !1);
164
+ else {
165
+ if (e.y + t.splitTextToSize(
166
+ s ?? "",
167
+ i.page.maxContentWidth - g
168
+ ).length * f - 3 * f >= i.page.maxContentHeight) {
169
+ const r = t.splitTextToSize(
170
+ s ?? "",
171
+ i.page.maxContentWidth - g
172
+ ), y = [], x = e.y;
173
+ for (let l = 0; l < r.length; l++)
174
+ if (e.y - 2 * f < i.page.maxContentHeight)
175
+ y.push(r[l]), e.y += i.page.lineSpace;
176
+ else {
177
+ l <= r.length - 1 && (s = r.slice(l).join(""));
178
+ break;
179
+ }
180
+ y.length > 0 && (e = F(
181
+ t,
182
+ y.join(" "),
183
+ e.x + g,
184
+ x,
185
+ i.page.maxContentWidth - g,
186
+ i.page.defaultLineHeightFactor
187
+ )), H(t, i), e.y = i.page.topmargin;
188
+ }
189
+ e.y = F(
190
+ t,
191
+ s ?? "",
192
+ e.x + g,
193
+ e.y,
194
+ i.page.maxContentWidth - g,
195
+ i.page.defaultLineHeightFactor
196
+ ).y + p(t, i), e.x = i.page.xpading;
197
+ }
198
+ return e;
199
+ }, R = (t, a, e, g, i, n) => {
200
+ var s;
201
+ t.setFontSize(i.page.defaultFontSize);
202
+ for (const [f, r] of ((s = a == null ? void 0 : a.items) == null ? void 0 : s.entries()) ?? []) {
203
+ const y = a.ordered ? (a.start ?? 0) + f : a.start;
204
+ e = n(
205
+ r,
206
+ g + 1,
207
+ !0,
208
+ y,
209
+ a.ordered
210
+ ), e.y += p(t, i) * 0.2;
211
+ }
212
+ return e;
213
+ }, z = (t, a, e, g, i) => {
214
+ const n = t.getFont().fontName, s = t.getFont().fontStyle, f = t.getFontSize(), r = (x) => {
215
+ switch (x) {
216
+ case "normal":
217
+ return 0;
218
+ case "bold":
219
+ return 1;
220
+ case "italic":
221
+ return 1.5;
222
+ case "bolditalic":
223
+ return 1.5;
224
+ default:
225
+ return 0;
226
+ }
227
+ }, y = (x, l) => {
228
+ var C;
229
+ l === "bold" ? t.setFont(
230
+ i.font.bold.name && i.font.bold.name !== "" ? i.font.bold.name : n,
231
+ i.font.bold.style || "bold"
232
+ ) : l === "italic" ? t.setFont(i.font.regular.name, "italic") : l === "bolditalic" ? t.setFont(
233
+ i.font.bold.name && i.font.bold.name !== "" ? i.font.bold.name : n,
234
+ "bolditalic"
235
+ ) : t.setFont(i.font.regular.name, s);
236
+ const b = i.page.maxContentWidth - g - e.x, W = t.splitTextToSize(x, b);
237
+ if (W.length > 1) {
238
+ const S = W[0], m = (C = W == null ? void 0 : W.slice(1)) == null ? void 0 : C.join(" ");
239
+ t.text(
240
+ S,
241
+ e.x + (g >= 2 ? g + 2 * r(l) : 0),
242
+ e.y,
243
+ {
244
+ baseline: "top",
245
+ maxWidth: b
246
+ }
247
+ ), e.x = i.page.xpading + g, e.y += p(t, i);
248
+ const L = i.page.maxContentWidth - g - i.page.xpading - i.page.xmargin;
249
+ t.splitTextToSize(m, L).forEach((w) => {
250
+ t.text(w, e.x + g, e.y, {
251
+ baseline: "top",
252
+ maxWidth: L
253
+ }), e.x = g, e.y += p(t, i);
254
+ });
255
+ } else
256
+ t.text(x, e.x + g, e.y, {
257
+ baseline: "top",
258
+ maxWidth: b
259
+ }), e.x += t.getTextDimensions(x).w + (g >= 2 ? x.split(" ").length + 2 : 2) * r(l);
260
+ };
261
+ if (a.type === "text" && a.items && a.items.length > 0)
262
+ for (const x of a.items)
263
+ if (x.type === "em" || x.type === "strong") {
264
+ const l = x.type === "em" ? "italic" : "bold";
265
+ if (x.items && x.items.length > 0)
266
+ for (const b of x.items)
267
+ b.type === "strong" && l === "italic" || b.type === "em" && l === "bold" ? y(
268
+ b.content || "",
269
+ "bolditalic"
270
+ ) : y(
271
+ b.content || "",
272
+ l
273
+ );
274
+ else
275
+ y(x.content || "", l);
276
+ } else
277
+ y(x.content || "", "normal");
278
+ else a.type === "em" ? y(a.content || "", "italic") : a.type === "strong" ? y(a.content || "", "bold") : y(a.content || "", "normal");
279
+ return t.setFont(n, s), t.setFontSize(f), e;
280
+ }, _ = (t, a, e, g, i, n, s, f) => {
281
+ const r = g * i.page.indent, y = f ? `${s}. ` : "• ";
282
+ e.y + p(t, i) >= i.page.maxContentHeight && (H(t, i), e.y = i.page.topmargin), t.setFont(i.font.regular.name, i.font.regular.style), t.text(y, e.x + r, e.y, { baseline: "top" });
283
+ const x = t.getTextWidth(y);
284
+ if (e.x += x, a.items && a.items.length > 0)
285
+ for (const l of a.items) {
286
+ if (e.y + p(t, i) >= i.page.maxContentHeight && (H(t, i), e.y = i.page.topmargin), l.type === h.List)
287
+ n(
288
+ l,
289
+ g + 1,
290
+ !0,
291
+ s,
292
+ l.ordered ?? !1
293
+ );
294
+ else if (l.type === h.ListItem) {
295
+ const b = a.type === h.List ? g : g + 1;
296
+ n(
297
+ l,
298
+ b,
299
+ !0,
300
+ s,
301
+ f
302
+ );
303
+ } else
304
+ e = z(
305
+ t,
306
+ l,
307
+ e,
308
+ r,
309
+ i
310
+ );
311
+ e.x = i.page.xpading, e.y += p(t, i);
312
+ }
313
+ else if (a.content) {
314
+ const l = e.x + r, b = t.getTextWidth(y), W = i.page.maxContentWidth - r - b, C = t.splitTextToSize(a.content, W);
315
+ if (C.length > 0) {
316
+ t.text(C[0], l + b, e.y, {
317
+ baseline: "top",
318
+ maxWidth: W
319
+ });
320
+ for (let m = 1; m < C.length; m++)
321
+ e.y += p(t, i), t.text(C[m], l + b, e.y, {
322
+ baseline: "top",
323
+ maxWidth: W
324
+ });
325
+ e.y += p(t, i), e.x = i.page.xmargin + r;
326
+ const S = t.getTextWidth(a.content);
327
+ e.x += S;
328
+ }
329
+ }
330
+ return e;
331
+ }, B = (t, a, e, g, i, n, s, f, r, y = !0) => {
332
+ if (a != null && a.items && (a == null ? void 0 : a.items.length) > 0)
333
+ for (const x of (a == null ? void 0 : a.items) ?? [])
334
+ e = s(
335
+ x,
336
+ g,
337
+ i,
338
+ f,
339
+ r,
340
+ y
341
+ );
342
+ else {
343
+ const x = g * n.page.indent, l = i ? r ? `${f}. ` : "• " : "";
344
+ if (i && l) {
345
+ const b = t.getTextWidth(l), W = n.page.maxContentWidth - x - b, C = t.splitTextToSize(
346
+ a.content || "",
347
+ W
348
+ );
349
+ if (C.length > 0) {
350
+ t.text(l, e.x + x, e.y, {
351
+ baseline: "top"
352
+ }), t.text(
353
+ C[0],
354
+ e.x + x + b,
355
+ e.y,
356
+ {
357
+ baseline: "top",
358
+ maxWidth: W
359
+ }
360
+ );
361
+ for (let m = 1; m < C.length; m++)
362
+ e.x = n.page.xpading, e.y += p(t, n), t.text(
363
+ C[m],
364
+ e.x + x + b,
365
+ e.y,
366
+ {
367
+ baseline: "top",
368
+ maxWidth: W
369
+ }
370
+ );
371
+ e.y += p(t, n), e.x = n.page.xpading + x;
372
+ const S = t.getTextWidth(a.content || "");
373
+ e.x += S;
374
+ }
375
+ } else {
376
+ const b = t.splitTextToSize(
377
+ a.content || "",
378
+ n.page.maxContentWidth - x
379
+ );
380
+ e.y + b.length * p(t, n) >= n.page.maxContentHeight && (H(t, n), e.y = n.page.topmargin), y ? (e.y = F(
381
+ t,
382
+ a.content || "",
383
+ e.x + x,
384
+ e.y,
385
+ n.page.maxContentWidth - x,
386
+ n.page.defaultLineHeightFactor
387
+ ).y + p(t, n), e.x = n.page.xpading) : (t.text(a.content || "", e.x + x, e.y, {
388
+ baseline: "top"
389
+ }), e.x += t.getTextWidth(a.content || ""), e.x >= n.page.xpading + n.page.maxContentWidth && (H(t, n), e.x = n.page.xpading, e.y = n.page.topmargin));
390
+ }
391
+ }
392
+ return e;
393
+ }, J = (t, a, e) => {
394
+ const g = t.internal.pageSize.getWidth();
395
+ return t.setLineDashPattern([1, 1], 0), t.setLineWidth(0.1), t.line(
396
+ e.page.xpading,
397
+ a.y,
398
+ g - e.page.xpading,
399
+ a.y
400
+ ), t.setLineWidth(0.1), t.setLineDashPattern([], 0), a.y += p(t, e), a;
401
+ }, $ = (t, a, e, g, i, n) => {
402
+ const s = g * n.page.indent;
403
+ e.y + t.splitTextToSize(
404
+ a.code ?? "",
405
+ n.page.maxContentWidth - s
406
+ ).length * p(t, n) - 2 * p(t, n) >= n.page.maxContentHeight && (H(t, n), e.y = n.page.topmargin);
407
+ const f = t.splitTextToSize(
408
+ a.code ?? "",
409
+ n.page.maxContentWidth - s
410
+ ).length * p(t, n);
411
+ return e.y += n.page.lineSpace, t.setFillColor("#EEEEEE"), t.setDrawColor("#eee"), t.roundedRect(
412
+ e.x,
413
+ e.y - n.page.lineSpace,
414
+ n.page.maxContentWidth,
415
+ f,
416
+ 2,
417
+ 2,
418
+ "FD"
419
+ ), t.setFontSize(10), t.text(
420
+ a.lang ?? "",
421
+ e.x + n.page.maxContentWidth - t.getTextWidth(a.lang ?? "") - n.page.lineSpace / 2,
422
+ e.y
423
+ ), t.setFontSize(n.page.defaultFontSize), t.text(a.code ?? "", e.x + 4, e.y), e.y += f, e;
424
+ }, N = async (t, a, e) => {
425
+ const g = await T(a);
426
+ let i = {
427
+ x: e.cursor.x,
428
+ y: e.cursor.y
429
+ };
430
+ const n = (s, f = 0, r = !1, y = 0, x = !1, l = !0) => {
431
+ const b = f * e.page.indent;
432
+ switch (i.y + t.splitTextToSize(
433
+ s.content ?? "",
434
+ e.page.maxContentWidth - b
435
+ ).length * p(t, e) >= e.page.maxContentHeight && (H(t, e), i.y = e.page.topmargin), s.type) {
436
+ case h.Heading:
437
+ i = E(
438
+ t,
439
+ s,
440
+ i,
441
+ b,
442
+ e,
443
+ n
444
+ );
445
+ break;
446
+ case h.Paragraph:
447
+ i = D(
448
+ t,
449
+ s,
450
+ i,
451
+ b,
452
+ e,
453
+ n
454
+ );
455
+ break;
456
+ case h.List:
457
+ i = R(
458
+ t,
459
+ s,
460
+ i,
461
+ f,
462
+ e,
463
+ n
464
+ );
465
+ break;
466
+ case h.ListItem:
467
+ i = _(
468
+ t,
469
+ s,
470
+ i,
471
+ f,
472
+ e,
473
+ n,
474
+ y,
475
+ x
476
+ );
477
+ break;
478
+ case h.Hr:
479
+ i = J(t, i, e);
480
+ break;
481
+ case h.Code:
482
+ i = $(
483
+ t,
484
+ s,
485
+ i,
486
+ f,
487
+ r,
488
+ e
489
+ );
490
+ break;
491
+ case h.Strong:
492
+ case h.Em:
493
+ i = z(
494
+ t,
495
+ s,
496
+ i,
497
+ b,
498
+ e
499
+ );
500
+ break;
501
+ case h.Raw:
502
+ case h.Text:
503
+ i = B(
504
+ t,
505
+ s,
506
+ i,
507
+ f,
508
+ r,
509
+ e,
510
+ n,
511
+ y,
512
+ x,
513
+ l
514
+ );
515
+ break;
516
+ default:
517
+ console.warn(
518
+ `Warning: Unsupported element type encountered: ${s.type}.
519
+ If you believe this element type should be supported, please create an issue at:
520
+ https://github.com/JeelGajera/jspdf-md-renderer/issues
521
+ with details of the element and expected behavior. Thank you for helping improve this library!`
522
+ );
523
+ break;
524
+ }
525
+ return i;
526
+ };
527
+ for (const s of g)
528
+ n(s);
529
+ e.endCursorYHandler(i.y);
530
+ };
531
+ export {
532
+ T as MdTextParser,
533
+ N as MdTextRender
534
+ };
package/package.json CHANGED
@@ -1,68 +1,68 @@
1
1
  {
2
- "name": "jspdf-md-renderer",
3
- "version": "1.6.2",
4
- "description": "A jsPDF utility to render Markdown directly into formatted PDFs with custom designs.",
5
- "main": "dist/index.js",
6
- "module": "dist/index.mjs",
7
- "types": "dist/index.d.ts",
8
- "exports": {
9
- ".": {
10
- "import": "./dist/index.mjs",
11
- "require": "./dist/index.js",
12
- "types": "./dist/index.d.ts"
13
- },
14
- "./types": "./dist/types/index.d.ts"
2
+ "name": "jspdf-md-renderer",
3
+ "version": "1.6.4",
4
+ "description": "A jsPDF utility to render Markdown directly into formatted PDFs with custom designs.",
5
+ "main": "dist/index.js",
6
+ "module": "dist/index.mjs",
7
+ "types": "dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "import": "./dist/index.mjs",
11
+ "require": "./dist/index.js",
12
+ "types": "./dist/index.d.ts"
15
13
  },
16
- "files": [
17
- "dist",
18
- "types",
19
- "LICENSE",
20
- "README.md"
21
- ],
22
- "scripts": {
23
- "dev": "npm run format && npm run lint:fix && npm run build",
24
- "build": "rimraf dist && vite build",
25
- "lint": "eslint src/**",
26
- "lint:fix": "eslint src/** --fix",
27
- "format": "prettier --write .",
28
- "test": "echo \"Error: no test specified\" && exit 1",
29
- "watch": "vite build --watch"
30
- },
31
- "repository": {
32
- "type": "git",
33
- "url": "git+https://github.com/JeelGajera/jspdf-md-renderer.git"
34
- },
35
- "keywords": [
36
- "jspdf",
37
- "markdown",
38
- "pdf",
39
- "renderer"
40
- ],
41
- "author": "Jeel Gajera <jeelgajera200@gmail.com>",
42
- "license": "MIT",
43
- "bugs": {
44
- "url": "https://github.com/JeelGajera/jspdf-md-renderer/issues"
45
- },
46
- "homepage": "https://github.com/JeelGajera/jspdf-md-renderer#readme",
47
- "dependencies": {
48
- "jspdf": "^3.0.1",
49
- "jspdf-md-renderer": "file:",
50
- "marked": "^15.0.3"
51
- },
52
- "devDependencies": {
53
- "@eslint/js": "^9.16.0",
54
- "@types/node": "^22.10.2",
55
- "@typescript-eslint/eslint-plugin": "^8.18.0",
56
- "@typescript-eslint/parser": "^8.18.0",
57
- "eslint": "^9.16.0",
58
- "eslint-config-prettier": "^10.1.2",
59
- "eslint-plugin-prettier": "^5.2.1",
60
- "globals": "^16.0.0",
61
- "prettier": "^3.4.2",
62
- "rimraf": "^6.0.1",
63
- "typescript": "^5.7.2",
64
- "typescript-eslint": "^8.18.0",
65
- "vite": "^6.2.4",
66
- "vite-plugin-dts": "^4.5.3"
67
- }
14
+ "./types": "./dist/types/index.d.ts"
15
+ },
16
+ "files": [
17
+ "dist",
18
+ "types",
19
+ "LICENSE",
20
+ "README.md"
21
+ ],
22
+ "scripts": {
23
+ "dev": "npm run format && npm run lint:fix && npm run build",
24
+ "build": "rimraf dist && vite build",
25
+ "lint": "eslint src/**",
26
+ "lint:fix": "eslint src/** --fix",
27
+ "format": "prettier --write .",
28
+ "test": "echo \"Error: no test specified\" && exit 1",
29
+ "watch": "vite build --watch"
30
+ },
31
+ "repository": {
32
+ "type": "git",
33
+ "url": "git+https://github.com/JeelGajera/jspdf-md-renderer.git"
34
+ },
35
+ "keywords": [
36
+ "jspdf",
37
+ "markdown",
38
+ "pdf",
39
+ "renderer"
40
+ ],
41
+ "author": "Jeel Gajera <jeelgajera200@gmail.com>",
42
+ "license": "MIT",
43
+ "bugs": {
44
+ "url": "https://github.com/JeelGajera/jspdf-md-renderer/issues"
45
+ },
46
+ "homepage": "https://github.com/JeelGajera/jspdf-md-renderer#readme",
47
+ "dependencies": {
48
+ "jspdf": "^3.0.1",
49
+ "jspdf-md-renderer": "file:",
50
+ "marked": "^15.0.3"
51
+ },
52
+ "devDependencies": {
53
+ "@eslint/js": "^9.16.0",
54
+ "@types/node": "^22.10.2",
55
+ "@typescript-eslint/eslint-plugin": "^8.18.0",
56
+ "@typescript-eslint/parser": "^8.18.0",
57
+ "eslint": "^9.16.0",
58
+ "eslint-config-prettier": "^10.1.2",
59
+ "eslint-plugin-prettier": "^5.2.1",
60
+ "globals": "^16.0.0",
61
+ "prettier": "^3.4.2",
62
+ "rimraf": "^6.0.1",
63
+ "typescript": "^5.7.2",
64
+ "typescript-eslint": "^8.18.0",
65
+ "vite": "^6.2.4",
66
+ "vite-plugin-dts": "^4.5.3"
67
+ }
68
68
  }