jspdf-md-renderer 1.4.1 → 1.5.1

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.
Files changed (41) hide show
  1. package/README.md +14 -4
  2. package/dist/index.d.ts +81 -3
  3. package/dist/index.js +4 -3
  4. package/dist/index.mjs +445 -0
  5. package/package.json +13 -8
  6. package/dist/enums/mdTokenType.d.ts +0 -20
  7. package/dist/enums/mdTokenType.js +0 -21
  8. package/dist/parser/MdTextParser.d.ts +0 -8
  9. package/dist/parser/MdTextParser.js +0 -119
  10. package/dist/renderer/MdTextRender.d.ts +0 -10
  11. package/dist/renderer/MdTextRender.js +0 -63
  12. package/dist/renderer/components/code.d.ts +0 -5
  13. package/dist/renderer/components/code.js +0 -28
  14. package/dist/renderer/components/heading.d.ts +0 -8
  15. package/dist/renderer/components/heading.js +0 -25
  16. package/dist/renderer/components/hr.d.ts +0 -4
  17. package/dist/renderer/components/hr.js +0 -11
  18. package/dist/renderer/components/index.d.ts +0 -7
  19. package/dist/renderer/components/index.js +0 -7
  20. package/dist/renderer/components/list.d.ts +0 -5
  21. package/dist/renderer/components/list.js +0 -13
  22. package/dist/renderer/components/listItem.d.ts +0 -5
  23. package/dist/renderer/components/listItem.js +0 -33
  24. package/dist/renderer/components/paragraph.d.ts +0 -8
  25. package/dist/renderer/components/paragraph.js +0 -51
  26. package/dist/renderer/components/rawItem.d.ts +0 -5
  27. package/dist/renderer/components/rawItem.js +0 -26
  28. package/dist/types/index.d.ts +0 -2
  29. package/dist/types/index.js +0 -2
  30. package/dist/types/parsedElement.d.ts +0 -22
  31. package/dist/types/parsedElement.js +0 -1
  32. package/dist/types/renderOption.d.ts +0 -37
  33. package/dist/types/renderOption.js +0 -1
  34. package/dist/types/wordInfo.d.ts +0 -4
  35. package/dist/types/wordInfo.js +0 -1
  36. package/dist/utils/doc-helpers.d.ts +0 -3
  37. package/dist/utils/doc-helpers.js +0 -3
  38. package/dist/utils/handlePageBreak.d.ts +0 -6
  39. package/dist/utils/handlePageBreak.js +0 -11
  40. package/dist/utils/justifyText.d.ts +0 -15
  41. package/dist/utils/justifyText.js +0 -53
package/README.md CHANGED
@@ -69,6 +69,14 @@ This section showcases how to create simple and nested lists.
69
69
  1. Nested Subtopic 2.2.1
70
70
  2. Nested Subtopic 2.2.2
71
71
 
72
+ ### Emphasis and Strong Emphasis
73
+ - *Italic* text using asterisks.
74
+ - _Italic_ text using underscores.
75
+ - **Bold** text using double asterisks.
76
+ - __Bold__ text using double underscores.
77
+ - ***Bold and Italic*** text using triple asterisks.
78
+ - ___Bold and Italic___ text using triple underscores.
79
+
72
80
  `;
73
81
 
74
82
  const generatePDF = async () => {
@@ -146,17 +154,19 @@ The following Markdown elements are currently supported by `jspdf-md-renderer`:
146
154
  - **Lists**:
147
155
  - Unordered lists: `-`, `*`, `+`
148
156
  - Ordered lists: `1.`, `2.`, `3.`, etc.
157
+ - **Horizontal Rules**: `---`, `***`, `___`
158
+ - **Text Styles**:
159
+ - Bold: `**bold**` or `__bold__`
160
+ - Italic: `*italic*` or `_italic_`
161
+ - Bold Italic: `***bold italic***` or `___bold italic___`
149
162
  - **Code Blocks** (fenced and indented):
150
163
  ````markdown
151
164
  ```js
152
165
  console.log('Hello, world!');
153
166
  ```
154
167
  ````
168
+
155
169
  ### Proposed for Future Implementation:
156
- - **Emphasis**:
157
- - **Bold**: `**bold text**` or `__bold text__`
158
- - _Italic_: `*italic text*` or `_italic text_`
159
- - **_Bold Italic_**: `***bold italic***`
160
170
  - **Blockquotes**:
161
171
  ```markdown
162
172
  > This is a blockquote.
package/dist/index.d.ts CHANGED
@@ -1,3 +1,81 @@
1
- import { MdTextRender } from './renderer/MdTextRender';
2
- import { MdTextParser } from './parser/MdTextParser';
3
- export { MdTextRender, MdTextParser };
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 CHANGED
@@ -1,3 +1,4 @@
1
- import { MdTextRender } from './renderer/MdTextRender';
2
- import { MdTextParser } from './parser/MdTextParser';
3
- export { MdTextRender, MdTextParser };
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const W=require("marked");var r=(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))(r||{});const S=async t=>{const a=await W.marked.lexer(t,{async:!0});return p(a)},p=t=>{const a=[];return t.forEach(e=>{try{const g=m[e.type];g?a.push(g(e)):a.push({type:r.Raw,content:e.raw})}catch(g){console.error("Failed to handle token ==>",e,g)}}),a},m={[r.Heading]:t=>({type:r.Heading,depth:t.depth,content:t.text,items:t.tokens?p(t.tokens):[]}),[r.Paragraph]:t=>({type:r.Paragraph,content:t.text,items:t.tokens?p(t.tokens):[]}),[r.List]:t=>({type:r.List,ordered:t.ordered,start:t.start,items:t.items?p(t.items):[]}),[r.ListItem]:t=>({type:r.ListItem,content:t.text,items:t.tokens?p(t.tokens):[]}),[r.Code]:t=>({type:r.Code,lang:t.lang,code:t.text}),[r.Table]:t=>({type:r.Table,header:t.header.map(a=>({type:r.TableHeader,content:a})),rows:t.rows.map(a=>a.map(e=>({type:r.TableCell,content:e})))}),[r.Image]:t=>({type:r.Image,src:t.href,alt:t.text}),[r.Link]:t=>({type:r.Link,href:t.href,text:t.text,items:t.tokens?p(t.tokens):[]}),[r.Strong]:t=>({type:r.Strong,content:t.text,items:t.tokens?p(t.tokens):[]}),[r.Em]:t=>({type:r.Em,content:t.text,items:t.tokens?p(t.tokens):[]}),[r.Text]:t=>({type:r.Text,content:t.text,items:t.tokens?p(t.tokens):[]}),[r.Hr]:t=>({type:r.Hr,content:t.raw,items:t.tokens?p(t.tokens):[]}),[r.CodeSpan]:t=>({type:r.CodeSpan,content:t.text,items:t.tokens?p(t.tokens):[]})},c=(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)},d=(t,a)=>t.getTextDimensions("H").h*a.page.defaultLineHeightFactor,w=(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 x of(a==null?void 0:a.items)??[])e=n(x,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*d(t,i);return t.setFontSize(i.page.defaultFontSize),e},L=(t,a,e,g,i,n,s,x,y)=>{const l=(x-e)/(a.length-1);let h=i;const f=n+g*s;for(const b of a)t.text(b.text,h,f,{align:"justify",lineHeightFactor:y,maxWidth:x}),h+=b.wordLength+l},z=(t,a,e,g,i,n,s,x)=>{const y=t.map(l=>l.text).join(" ");a.text(y,e,g+i*n,{align:"justify",lineHeightFactor:x,maxWidth:s})},H=(t,a,e,g,i,n)=>{const s={x:e,y:g},x=t.getTextDimensions("A").h*n,y=a.split(" ");let l=0,h=[],f=0;for(const b of y){const C=t.getTextWidth(b+"a");C+f>=i&&(L(t,h,f,l++,e,g,x,i,n),h=[],f=0),h.push({text:b,wordLength:C}),f+=C}return h.length>0&&z(h,t,e,g,l,x,i,n),s.y=g+l*x,s.x=t.getTextWidth(a)+e,s},E=(t,a,e,g,i,n)=>{t.setFontSize(i.page.defaultFontSize);let s=a.content;const x=t.getTextDimensions("A").h*i.page.defaultLineHeightFactor;if(a!=null&&a.items&&(a==null?void 0:a.items.length)>0)for(const y of(a==null?void 0:a.items)??[])e=n(y,g,!1);else{if(e.y+t.splitTextToSize(s??"",i.page.maxContentWidth-g).length*x-3*x>=i.page.maxContentHeight){const y=t.splitTextToSize(s??"",i.page.maxContentWidth-g),l=[],h=e.y;for(let f=0;f<y.length;f++)if(e.y-2*x<i.page.maxContentHeight)l.push(y[f]),e.y+=i.page.lineSpace;else{f<=y.length-1&&(s=y.slice(f).join(""));break}l.length>0&&(e=H(t,l.join(" "),e.x+g,h,i.page.maxContentWidth-g,i.page.defaultLineHeightFactor)),c(t,i),e.y=i.page.topmargin}e.y=H(t,s??"",e.x+g,e.y,i.page.maxContentWidth-g,i.page.defaultLineHeightFactor).y+d(t,i),e.x=i.page.xpading}return e},I=(t,a,e,g,i,n)=>{var s;t.setFontSize(i.page.defaultFontSize);for(const[x,y]of((s=a==null?void 0:a.items)==null?void 0:s.entries())??[]){const l=a.ordered?(a.start??0)+x:a.start;e=n(y,g+1,!0,l,a.ordered),e.y+=d(t,i)*.2}return e},F=(t,a,e,g,i)=>{const n=t.getFont().fontName,s=t.getFont().fontStyle,x=t.getFontSize(),y=h=>{switch(h){case"normal":return 0;case"bold":return 1;case"italic":return 1.5;case"bolditalic":return 2.5;default:return 0}},l=(h,f)=>{f==="bold"?t.setFont(i.font.bold.name&&i.font.bold.name!==""?i.font.bold.name:n,i.font.bold.style||"bold"):f==="italic"?t.setFont(i.font.regular.name,"italic"):f==="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=t.getTextWidth(h);e.x+b>i.page.xpading+i.page.maxContentWidth&&(e.x=i.page.xpading,e.y+=d(t,i));const C=t.getTextWidth(" ");t.text(h,e.x+g,e.y,{baseline:"top"}),e.x+=b+C*y(f)+(f==="normal"?g*.7:0)};if(a.type==="text"&&a.items&&a.items.length>0)for(const h of a.items)if(h.type==="em"||h.type==="strong"){const f=h.type==="em"?"italic":"bold";if(h.items&&h.items.length>0)for(const b of h.items)b.type==="strong"&&f==="italic"||b.type==="em"&&f==="bold"?l(b.content||"","bolditalic"):l(b.content||"",f);else l(h.content||"",f)}else l(h.content||"","normal");else a.type==="em"?l(a.content||"","italic"):a.type==="strong"?l(a.content||"","bold"):l(a.content||"","normal");return t.setFont(n,s),t.setFontSize(x),e},u=(t,a,e,g,i,n,s,x)=>{const y=g*i.page.indent,l=x?`${s}. `:"• ";if(e.y+d(t,i)>=i.page.maxContentHeight&&(c(t,i),e.y=i.page.topmargin),t.setFont(i.font.regular.name,i.font.regular.style),t.text(l,e.x+y,e.y,{baseline:"top"}),e.x+=t.getTextWidth(l),a.items&&a.items.length>0)for(const h of a.items)e=F(t,h,e,y,i);else a.content&&(t.text(a.content,e.x,e.y,{baseline:"top"}),e.x+=t.getTextWidth(a.content));return e.y+=d(t,i),e.x=i.page.xpading,e},P=(t,a,e,g,i,n,s,x,y,l=!0)=>{if(a!=null&&a.items&&(a==null?void 0:a.items.length)>0)for(const h of(a==null?void 0:a.items)??[])e=s(h,g,i,x,y,l);else{const h=g*n.page.indent,f=i?y?`${x}. `:"• ":"",b=t.splitTextToSize(f+a.content,n.page.maxContentWidth-h);e.y+b.length*d(t,n)>=n.page.maxContentHeight&&(c(t,n),e.y=n.page.topmargin),l?(e.y=H(t,f+a.content,e.x+h,e.y,n.page.maxContentWidth-h,n.page.defaultLineHeightFactor).y+d(t,n),e.x=n.page.xpading):(t.text(f+a.content,e.x+h,e.y,{baseline:"top"}),e.x+=t.getTextWidth(f+a.content),e.x>=n.page.xpading+n.page.maxContentWidth&&(c(t,n),e.x=n.page.xpading,e.y+=d(t,n)))}return e},T=(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+=d(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*d(t,n)-2*d(t,n)>=n.page.maxContentHeight&&(c(t,n),e.y=n.page.topmargin);const x=t.splitTextToSize(a.code??"",n.page.maxContentWidth-s).length*d(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,x,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+=x,e},R=async(t,a,e)=>{const g=await S(a);console.log(g);let i={x:e.cursor.x,y:e.cursor.y};const n=(s,x=0,y=!1,l=0,h=!1,f=!0)=>{const b=x*e.page.indent;switch(i.y+t.splitTextToSize(s.content??"",e.page.maxContentWidth-b).length*d(t,e)>=e.page.maxContentHeight&&(c(t,e),i.y=e.page.topmargin),s.type){case r.Heading:i=w(t,s,i,b,e,n);break;case r.Paragraph:i=E(t,s,i,b,e,n);break;case r.List:i=I(t,s,i,x,e,n);break;case r.ListItem:i=u(t,s,i,x,e,n,l,h);break;case r.Hr:i=T(t,i,e);break;case r.Code:i=j(t,s,i,x,y,e);break;case r.Strong:case r.Em:i=F(t,s,i,b,e);break;case r.Raw:case r.Text:i=P(t,s,i,x,y,e,n,l,h,f);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=S;exports.MdTextRender=R;
package/dist/index.mjs ADDED
@@ -0,0 +1,445 @@
1
+ import { marked as S } 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 W = async (t) => {
4
+ const a = await S.lexer(t, { async: !0 });
5
+ return d(a);
6
+ }, d = (t) => {
7
+ const a = [];
8
+ return t.forEach((e) => {
9
+ try {
10
+ const g = m[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
+ }, m = {
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
+ }, c = (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, w = (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 r of (a == null ? void 0 : a.items) ?? [])
103
+ e = n(r, 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
+ }, L = (t, a, e, g, i, n, s, r, y) => {
111
+ const f = (r - 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: y,
118
+ maxWidth: r
119
+ }), x += b.wordLength + f;
120
+ }, z = (t, a, e, g, i, n, s, r) => {
121
+ const y = t.map((f) => f.text).join(" ");
122
+ a.text(y, e, g + i * n, {
123
+ align: "justify",
124
+ lineHeightFactor: r,
125
+ maxWidth: s
126
+ });
127
+ }, H = (t, a, e, g, i, n) => {
128
+ const s = {
129
+ x: e,
130
+ y: g
131
+ }, r = t.getTextDimensions("A").h * n, y = a.split(" ");
132
+ let f = 0, x = [], l = 0;
133
+ for (const b of y) {
134
+ const C = t.getTextWidth(b + "a");
135
+ C + l >= i && (L(
136
+ t,
137
+ x,
138
+ l,
139
+ f++,
140
+ e,
141
+ g,
142
+ r,
143
+ i,
144
+ n
145
+ ), x = [], l = 0), x.push({ text: b, wordLength: C }), l += C;
146
+ }
147
+ return x.length > 0 && z(
148
+ x,
149
+ t,
150
+ e,
151
+ g,
152
+ f,
153
+ r,
154
+ i,
155
+ n
156
+ ), s.y = g + f * r, s.x = t.getTextWidth(a) + e, s;
157
+ }, E = (t, a, e, g, i, n) => {
158
+ t.setFontSize(i.page.defaultFontSize);
159
+ let s = a.content;
160
+ const r = t.getTextDimensions("A").h * i.page.defaultLineHeightFactor;
161
+ if (a != null && a.items && (a == null ? void 0 : a.items.length) > 0)
162
+ for (const y of (a == null ? void 0 : a.items) ?? [])
163
+ e = n(y, g, !1);
164
+ else {
165
+ if (e.y + t.splitTextToSize(
166
+ s ?? "",
167
+ i.page.maxContentWidth - g
168
+ ).length * r - 3 * r >= i.page.maxContentHeight) {
169
+ const y = t.splitTextToSize(
170
+ s ?? "",
171
+ i.page.maxContentWidth - g
172
+ ), f = [], x = e.y;
173
+ for (let l = 0; l < y.length; l++)
174
+ if (e.y - 2 * r < i.page.maxContentHeight)
175
+ f.push(y[l]), e.y += i.page.lineSpace;
176
+ else {
177
+ l <= y.length - 1 && (s = y.slice(l).join(""));
178
+ break;
179
+ }
180
+ f.length > 0 && (e = H(
181
+ t,
182
+ f.join(" "),
183
+ e.x + g,
184
+ x,
185
+ i.page.maxContentWidth - g,
186
+ i.page.defaultLineHeightFactor
187
+ )), c(t, i), e.y = i.page.topmargin;
188
+ }
189
+ e.y = H(
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
+ }, I = (t, a, e, g, i, n) => {
200
+ var s;
201
+ t.setFontSize(i.page.defaultFontSize);
202
+ for (const [r, y] of ((s = a == null ? void 0 : a.items) == null ? void 0 : s.entries()) ?? []) {
203
+ const f = a.ordered ? (a.start ?? 0) + r : a.start;
204
+ e = n(
205
+ y,
206
+ g + 1,
207
+ !0,
208
+ f,
209
+ a.ordered
210
+ ), e.y += p(t, i) * 0.2;
211
+ }
212
+ return e;
213
+ }, F = (t, a, e, g, i) => {
214
+ const n = t.getFont().fontName, s = t.getFont().fontStyle, r = t.getFontSize(), y = (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 2.5;
224
+ default:
225
+ return 0;
226
+ }
227
+ }, f = (x, l) => {
228
+ l === "bold" ? t.setFont(
229
+ i.font.bold.name && i.font.bold.name !== "" ? i.font.bold.name : n,
230
+ i.font.bold.style || "bold"
231
+ ) : l === "italic" ? t.setFont(i.font.regular.name, "italic") : l === "bolditalic" ? t.setFont(
232
+ i.font.bold.name && i.font.bold.name !== "" ? i.font.bold.name : n,
233
+ "bolditalic"
234
+ ) : t.setFont(i.font.regular.name, s);
235
+ const b = t.getTextWidth(x);
236
+ e.x + b > i.page.xpading + i.page.maxContentWidth && (e.x = i.page.xpading, e.y += p(t, i));
237
+ const C = t.getTextWidth(" ");
238
+ t.text(x, e.x + g, e.y, { baseline: "top" }), e.x += b + C * y(l) + (l === "normal" ? g * 0.7 : 0);
239
+ };
240
+ if (a.type === "text" && a.items && a.items.length > 0)
241
+ for (const x of a.items)
242
+ if (x.type === "em" || x.type === "strong") {
243
+ const l = x.type === "em" ? "italic" : "bold";
244
+ if (x.items && x.items.length > 0)
245
+ for (const b of x.items)
246
+ b.type === "strong" && l === "italic" || b.type === "em" && l === "bold" ? f(
247
+ b.content || "",
248
+ "bolditalic"
249
+ ) : f(
250
+ b.content || "",
251
+ l
252
+ );
253
+ else
254
+ f(x.content || "", l);
255
+ } else
256
+ f(x.content || "", "normal");
257
+ else a.type === "em" ? f(a.content || "", "italic") : a.type === "strong" ? f(a.content || "", "bold") : f(a.content || "", "normal");
258
+ return t.setFont(n, s), t.setFontSize(r), e;
259
+ }, P = (t, a, e, g, i, n, s, r) => {
260
+ const y = g * i.page.indent, f = r ? `${s}. ` : "• ";
261
+ if (e.y + p(t, i) >= i.page.maxContentHeight && (c(t, i), e.y = i.page.topmargin), t.setFont(i.font.regular.name, i.font.regular.style), t.text(f, e.x + y, e.y, { baseline: "top" }), e.x += t.getTextWidth(f), a.items && a.items.length > 0)
262
+ for (const x of a.items)
263
+ e = F(
264
+ t,
265
+ x,
266
+ e,
267
+ y,
268
+ i
269
+ );
270
+ else a.content && (t.text(a.content, e.x, e.y, { baseline: "top" }), e.x += t.getTextWidth(a.content));
271
+ return e.y += p(t, i), e.x = i.page.xpading, e;
272
+ }, u = (t, a, e, g, i, n, s, r, y, f = !0) => {
273
+ if (a != null && a.items && (a == null ? void 0 : a.items.length) > 0)
274
+ for (const x of (a == null ? void 0 : a.items) ?? [])
275
+ e = s(
276
+ x,
277
+ g,
278
+ i,
279
+ r,
280
+ y,
281
+ f
282
+ );
283
+ else {
284
+ const x = g * n.page.indent, l = i ? y ? `${r}. ` : "• " : "", b = t.splitTextToSize(
285
+ l + a.content,
286
+ n.page.maxContentWidth - x
287
+ );
288
+ e.y + b.length * p(t, n) >= n.page.maxContentHeight && (c(t, n), e.y = n.page.topmargin), f ? (e.y = H(
289
+ t,
290
+ l + a.content,
291
+ e.x + x,
292
+ e.y,
293
+ n.page.maxContentWidth - x,
294
+ n.page.defaultLineHeightFactor
295
+ ).y + p(t, n), e.x = n.page.xpading) : (t.text(
296
+ l + a.content,
297
+ e.x + x,
298
+ e.y,
299
+ { baseline: "top" }
300
+ ), e.x += t.getTextWidth(l + a.content), e.x >= n.page.xpading + n.page.maxContentWidth && (c(t, n), e.x = n.page.xpading, e.y += p(t, n)));
301
+ }
302
+ return e;
303
+ }, j = (t, a, e) => {
304
+ const g = t.internal.pageSize.getWidth();
305
+ return t.setLineDashPattern([1, 1], 0), t.setLineWidth(0.1), t.line(
306
+ e.page.xpading,
307
+ a.y,
308
+ g - e.page.xpading,
309
+ a.y
310
+ ), t.setLineWidth(0.1), t.setLineDashPattern([], 0), a.y += p(t, e), a;
311
+ }, D = (t, a, e, g, i, n) => {
312
+ const s = g * n.page.indent;
313
+ e.y + t.splitTextToSize(
314
+ a.code ?? "",
315
+ n.page.maxContentWidth - s
316
+ ).length * p(t, n) - 2 * p(t, n) >= n.page.maxContentHeight && (c(t, n), e.y = n.page.topmargin);
317
+ const r = t.splitTextToSize(
318
+ a.code ?? "",
319
+ n.page.maxContentWidth - s
320
+ ).length * p(t, n);
321
+ return e.y += n.page.lineSpace, t.setFillColor("#EEEEEE"), t.setDrawColor("#eee"), t.roundedRect(
322
+ e.x,
323
+ e.y - n.page.lineSpace,
324
+ n.page.maxContentWidth,
325
+ r,
326
+ 2,
327
+ 2,
328
+ "FD"
329
+ ), t.setFontSize(10), t.text(
330
+ a.lang ?? "",
331
+ e.x + n.page.maxContentWidth - t.getTextWidth(a.lang ?? "") - n.page.lineSpace / 2,
332
+ e.y
333
+ ), t.setFontSize(n.page.defaultFontSize), t.text(a.code ?? "", e.x + 4, e.y), e.y += r, e;
334
+ }, T = async (t, a, e) => {
335
+ const g = await W(a);
336
+ console.log(g);
337
+ let i = {
338
+ x: e.cursor.x,
339
+ y: e.cursor.y
340
+ };
341
+ const n = (s, r = 0, y = !1, f = 0, x = !1, l = !0) => {
342
+ const b = r * e.page.indent;
343
+ switch (i.y + t.splitTextToSize(
344
+ s.content ?? "",
345
+ e.page.maxContentWidth - b
346
+ ).length * p(t, e) >= e.page.maxContentHeight && (c(t, e), i.y = e.page.topmargin), s.type) {
347
+ case h.Heading:
348
+ i = w(
349
+ t,
350
+ s,
351
+ i,
352
+ b,
353
+ e,
354
+ n
355
+ );
356
+ break;
357
+ case h.Paragraph:
358
+ i = E(
359
+ t,
360
+ s,
361
+ i,
362
+ b,
363
+ e,
364
+ n
365
+ );
366
+ break;
367
+ case h.List:
368
+ i = I(
369
+ t,
370
+ s,
371
+ i,
372
+ r,
373
+ e,
374
+ n
375
+ );
376
+ break;
377
+ case h.ListItem:
378
+ i = P(
379
+ t,
380
+ s,
381
+ i,
382
+ r,
383
+ e,
384
+ n,
385
+ f,
386
+ x
387
+ );
388
+ break;
389
+ case h.Hr:
390
+ i = j(t, i, e);
391
+ break;
392
+ case h.Code:
393
+ i = D(
394
+ t,
395
+ s,
396
+ i,
397
+ r,
398
+ y,
399
+ e
400
+ );
401
+ break;
402
+ case h.Strong:
403
+ case h.Em:
404
+ i = F(
405
+ t,
406
+ s,
407
+ i,
408
+ b,
409
+ e
410
+ );
411
+ break;
412
+ case h.Raw:
413
+ case h.Text:
414
+ i = u(
415
+ t,
416
+ s,
417
+ i,
418
+ r,
419
+ y,
420
+ e,
421
+ n,
422
+ f,
423
+ x,
424
+ l
425
+ );
426
+ break;
427
+ default:
428
+ console.warn(
429
+ `Warning: Unsupported element type encountered: ${s.type}.
430
+ If you believe this element type should be supported, please create an issue at:
431
+ https://github.com/JeelGajera/jspdf-md-renderer/issues
432
+ with details of the element and expected behavior. Thank you for helping improve this library!`
433
+ );
434
+ break;
435
+ }
436
+ return i;
437
+ };
438
+ for (const s of g)
439
+ n(s);
440
+ e.endCursorYHandler(i.y);
441
+ };
442
+ export {
443
+ W as MdTextParser,
444
+ T as MdTextRender
445
+ };
package/package.json CHANGED
@@ -1,12 +1,13 @@
1
1
  {
2
2
  "name": "jspdf-md-renderer",
3
- "version": "1.4.1",
3
+ "version": "1.5.1",
4
4
  "description": "A jsPDF utility to render Markdown directly into formatted PDFs with custom designs.",
5
5
  "main": "dist/index.js",
6
+ "module": "dist/index.mjs",
6
7
  "types": "dist/index.d.ts",
7
8
  "exports": {
8
9
  ".": {
9
- "import": "./dist/index.js",
10
+ "import": "./dist/index.mjs",
10
11
  "require": "./dist/index.js",
11
12
  "types": "./dist/index.d.ts"
12
13
  },
@@ -19,13 +20,14 @@
19
20
  "README.md"
20
21
  ],
21
22
  "scripts": {
22
- "build": "rimraf dist && tsc",
23
+ "dev": "npm run format && npm run lint:fix && npm run build",
24
+ "build": "rimraf dist && vite build",
23
25
  "lint": "eslint src/**",
24
26
  "lint:fix": "eslint src/** --fix",
25
27
  "format": "prettier --write .",
26
28
  "test": "echo \"Error: no test specified\" && exit 1",
27
29
  "prepare": "npm run build",
28
- "watch": "tsc --watch"
30
+ "watch": "vite build --watch"
29
31
  },
30
32
  "repository": {
31
33
  "type": "git",
@@ -45,6 +47,7 @@
45
47
  "homepage": "https://github.com/JeelGajera/jspdf-md-renderer#readme",
46
48
  "dependencies": {
47
49
  "jspdf": "^3.0.1",
50
+ "jspdf-md-renderer": "file:",
48
51
  "marked": "^15.0.3"
49
52
  },
50
53
  "devDependencies": {
@@ -53,12 +56,14 @@
53
56
  "@typescript-eslint/eslint-plugin": "^8.18.0",
54
57
  "@typescript-eslint/parser": "^8.18.0",
55
58
  "eslint": "^9.16.0",
56
- "eslint-config-prettier": "^10.1.1",
59
+ "eslint-config-prettier": "^9.1.0",
57
60
  "eslint-plugin-prettier": "^5.2.1",
58
- "globals": "^16.0.0",
61
+ "globals": "^15.13.0",
59
62
  "prettier": "^3.4.2",
60
63
  "rimraf": "^6.0.1",
61
64
  "typescript": "^5.7.2",
62
- "typescript-eslint": "^8.18.0"
65
+ "typescript-eslint": "^8.18.0",
66
+ "vite": "^6.2.4",
67
+ "vite-plugin-dts": "^4.5.3"
63
68
  }
64
- }
69
+ }
@@ -1,20 +0,0 @@
1
- export declare enum MdTokenType {
2
- Heading = "heading",
3
- Paragraph = "paragraph",
4
- List = "list",
5
- ListItem = "list_item",
6
- Blockquote = "blockquote",
7
- Code = "code",
8
- CodeSpan = "codespan",
9
- Table = "table",
10
- Html = "html",
11
- Hr = "hr",
12
- Image = "image",
13
- Link = "link",
14
- Strong = "strong",
15
- Em = "em",
16
- TableHeader = "table_header",
17
- TableCell = "table_cell",
18
- Raw = "raw",
19
- Text = "text"
20
- }
@@ -1,21 +0,0 @@
1
- export var MdTokenType;
2
- (function (MdTokenType) {
3
- MdTokenType["Heading"] = "heading";
4
- MdTokenType["Paragraph"] = "paragraph";
5
- MdTokenType["List"] = "list";
6
- MdTokenType["ListItem"] = "list_item";
7
- MdTokenType["Blockquote"] = "blockquote";
8
- MdTokenType["Code"] = "code";
9
- MdTokenType["CodeSpan"] = "codespan";
10
- MdTokenType["Table"] = "table";
11
- MdTokenType["Html"] = "html";
12
- MdTokenType["Hr"] = "hr";
13
- MdTokenType["Image"] = "image";
14
- MdTokenType["Link"] = "link";
15
- MdTokenType["Strong"] = "strong";
16
- MdTokenType["Em"] = "em";
17
- MdTokenType["TableHeader"] = "table_header";
18
- MdTokenType["TableCell"] = "table_cell";
19
- MdTokenType["Raw"] = "raw";
20
- MdTokenType["Text"] = "text";
21
- })(MdTokenType || (MdTokenType = {}));