html-to-pdfmake
Advanced tools
Comparing version 1.0.6 to 1.0.7
@@ -48,2 +48,4 @@ (function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.htmlToPdfmake = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){ | ||
var inlineTags = [ 'p', 'li', 'span', 'strong', 'em', 'b', 'i', 'u' ]; | ||
/** | ||
@@ -59,4 +61,2 @@ * Takes an HTML string, converts to HTML using a DOM parser and recursivly parses | ||
// Cleanup of dirty html would happen here | ||
// Create a HTML DOM tree out of html string | ||
@@ -101,7 +101,4 @@ var parser = new wndw.DOMParser(); | ||
if (parentNodeName) { | ||
// do we have a default style to apply? | ||
// for 'p' we want to apply it from the parent | ||
if (parentNodeName !== 'p') { | ||
applyDefaultStyle(ret, parentNodeName); | ||
} | ||
// check if we have inherent styles to apply when a text is inside several <tag> | ||
applyParentsStyle(ret, element); | ||
@@ -113,2 +110,8 @@ // for links | ||
// for 'td' and 'th' we check if we have "rowspan" or "colspan" | ||
if (parentNodeName === "td" || parentNodeName === "th") { | ||
if (parentNode.getAttribute("rowspan")) ret.rowSpan = parentNode.getAttribute("rowspan")*1; | ||
if (parentNode.getAttribute("colspan")) ret.colSpan = parentNode.getAttribute("colspan")*1; | ||
} | ||
// is there any class to this element? | ||
@@ -137,4 +140,2 @@ cssClass = parentNode.getAttribute("class"); | ||
[].forEach.call(element.childNodes, function(child) { | ||
// for THEAD and TBODY we go straight to the TR | ||
//if (child.nodeName === "THEAD" || child.nodeName === "TBODY" || child.nodeName === "TFOOTER") continue; | ||
child = parseElement(child, element); | ||
@@ -176,5 +177,35 @@ if (child) { | ||
if (re.stack) { | ||
var td = [] | ||
re.stack.forEach(function(r) { | ||
var td = [], | ||
rowspan = {}; | ||
re.stack.forEach(function(r, indexRow) { | ||
if (r.stack) { | ||
// do we have a rowspan to apply from previous rows? | ||
if (rowspan[indexRow]) { | ||
// insert empty cell due to rowspan | ||
rowspan[indexRow].forEach(function(cell) { | ||
r.stack.splice(cell.index, 0, {text:'', style: ['html-td', 'html-tr'], colSpan:cell.colspan}); | ||
}); | ||
} | ||
// insert empty cells due to colspan | ||
r.stack.forEach(function(cell, index) { | ||
if (cell.colSpan > 1) { | ||
for (var i=0; i<cell.colSpan-1; i++) { | ||
r.stack.splice(index+1, 0, {text:'', style: ['html-td', 'html-tr']}) | ||
} | ||
} | ||
}) | ||
// check rowspan for the current row in order to then apply it to the next ones | ||
var indexCell = 0; | ||
r.stack.forEach(function(cell) { | ||
if (cell.rowSpan) { | ||
for (var i=0; i<cell.rowSpan; i++) { | ||
if (!rowspan[indexRow+i]) rowspan[indexRow+i] = []; | ||
// we also remember the colSpan for cells with both rowspan and colspan | ||
rowspan[indexRow+i].push({index:indexCell, colspan:cell.colSpan||1}); | ||
} | ||
} | ||
indexCell += cell.colSpan || 1; | ||
}); | ||
ret.table.body.push(r.stack) | ||
@@ -219,5 +250,9 @@ } else { | ||
// add a custom class to let the user customize the element | ||
// "tr" elements should always contain an array | ||
// "tr" elements should always contain an array | ||
if (ret.length === 1 && nodeName !== "tr") { | ||
ret=ret[0]; | ||
if (ret.text) { | ||
applyDefaultStyle(ret, nodeName); | ||
setComputedStyle(ret, element.getAttribute("style")); | ||
} | ||
ret.style = (ret.style||[]).concat(['html-'+nodeName]); | ||
@@ -227,7 +262,12 @@ // for TD and TH we want to include the style from TR | ||
} else { | ||
ret = (nodeName==='p' ? {text:ret} : {stack:ret}); | ||
// we apply the default style if it's a "p" | ||
if (nodeName === 'p') { | ||
applyDefaultStyle(ret, 'p'); | ||
var isInlineTag = (inlineTags.indexOf(nodeName) > -1); | ||
// if we have an inline tag, then we check if we have a non-inline tag in its section | ||
ret = (!isInlineTag || /{"(stack|table|ol|ul|image)"/.test(JSON.stringify(ret)) ? {stack:ret} : {text:ret}); | ||
// we apply the default style for the inline tags | ||
if (isInlineTag) { | ||
applyDefaultStyle(ret, nodeName); | ||
} | ||
ret.style = ['html-'+nodeName]; | ||
@@ -245,2 +285,13 @@ } | ||
} | ||
// retrieve the class from the parent | ||
cssClass = element.getAttribute("class"); | ||
if (cssClass && typeof ret === 'object') { | ||
// apply all the classes not there yet | ||
ret.style = (ret.style || []) | ||
.concat(cssClass.split(' ')) | ||
.filter(function (value, index, self) { | ||
return self.indexOf(value) === index; | ||
}); | ||
} | ||
} | ||
@@ -264,2 +315,26 @@ | ||
var applyParentsStyle = function(ret, node) { | ||
// while the parents are an inline tag, we want to apply the default style and the class to the children too | ||
var classes = [], defaultStyles = [], cssClass; | ||
var inlineParentNode=node.parentNode; | ||
while (inlineParentNode) { | ||
var defaultStyle = {}; | ||
var inlineParentNodeName=inlineParentNode.nodeName.toLowerCase(); | ||
if (inlineTags.indexOf(inlineParentNodeName) > -1) { | ||
cssClass = inlineParentNode.getAttribute("class"); | ||
classes = classes.concat(['html-'+inlineParentNodeName], cssClass||[]); | ||
applyDefaultStyle(defaultStyle, inlineParentNodeName); | ||
defaultStyles.push(defaultStyle); | ||
inlineParentNode=inlineParentNode.parentNode; | ||
} else break; | ||
} | ||
ret.style = (ret.style||[]).concat(classes); | ||
defaultStyles.forEach(function(defaultStyle) { | ||
for (var key in defaultStyle) { | ||
if (key.indexOf("margin") === -1 && ret[key] === undefined) ret[key] = defaultStyle[key]; | ||
} | ||
}) | ||
} | ||
/** | ||
@@ -266,0 +341,0 @@ * Transform a CSS expression (e.g. 'margin:10px') in the PDFMake version |
@@ -32,4 +32,4 @@ var pdfMake = require("pdfmake/build/pdfmake"); | ||
<ul> | ||
<li>Sub First item</li> | ||
<li>Sub Second item</li> | ||
<li>Sub First <b>bolded</b> item</li> | ||
<li>Sub Second <u>underlined</u> item</li> | ||
<li>With a sub sub unordered list: | ||
@@ -110,2 +110,39 @@ <ul style="background-color:yellow"> | ||
<table> | ||
<tr> | ||
<th>Col A</th> | ||
<th>Col B</th> | ||
<th>Col C</th> | ||
<th>Col D</th> | ||
</tr> | ||
<tr> | ||
<td>Cell A1</td> | ||
<td rowspan="2"> | ||
Cell B1 & B2 | ||
</td> | ||
<td>Cell C1</td> | ||
<td rowspan="2"> | ||
Cell D1 & D2 | ||
</td> | ||
</tr> | ||
<tr> | ||
<td>Cell A2</td> | ||
<td>Cell C2</td> | ||
</tr> | ||
<tr> | ||
<td>Cell A3</td> | ||
<td colspan="2">Cell B3 & C3</td> | ||
<td>Cell D3</td> | ||
</tr> | ||
<tr> | ||
<td rowspan="2" colspan="3"> | ||
Cell A4 & A5 & B4 & B5 & C4 & C5 | ||
</td> | ||
<td>Cell D4</td> | ||
</tr> | ||
<tr> | ||
<td>Cell D5</td> | ||
</tr> | ||
</table> | ||
<div style="margin-top:20px"> | ||
@@ -116,2 +153,4 @@ An image: <img width="54" style="height:70px" src=""> | ||
<p style="text-align: center;"> <span style="font-size: 14px;"><em><strong>Bold italic centered text</strong></em></span> </p> | ||
<span class="a">text "bold" <span class="b">text "bold & italic" <span class="c">text "bold & italic & red"</span> text "bold & italic"</span> text "bold"</span> | ||
`, window); | ||
@@ -132,2 +171,15 @@ | ||
color:'purple' | ||
}, | ||
'html-strong':{ | ||
color:'purple' | ||
}, | ||
'a':{ | ||
bold:true | ||
}, | ||
'b':{ | ||
italics: true | ||
}, | ||
'c':{ | ||
color:'red', | ||
italics: false | ||
} | ||
@@ -134,0 +186,0 @@ } |
107
index.js
@@ -47,2 +47,4 @@ // source: https://github.com/OpenSlides/OpenSlides/blob/f4f8b8422f9b3fbab58e35ac3f8f870d35813b7d/client/src/app/core/ui-services/html-to-pdf.service.ts | ||
var inlineTags = [ 'p', 'li', 'span', 'strong', 'em', 'b', 'i', 'u' ]; | ||
/** | ||
@@ -58,4 +60,2 @@ * Takes an HTML string, converts to HTML using a DOM parser and recursivly parses | ||
// Cleanup of dirty html would happen here | ||
// Create a HTML DOM tree out of html string | ||
@@ -96,11 +96,8 @@ var parser = new wndw.DOMParser(); | ||
if (element.textContent) { | ||
text = element.textContent.replace(/\n(\s+)?/g, "").trim(); | ||
text = element.textContent.replace(/\n(\s+)?/g, ""); | ||
if (text) { | ||
ret = {'text': text}; | ||
if (parentNodeName) { | ||
// do we have a default style to apply? | ||
// for 'p' we want to apply it from the parent | ||
if (parentNodeName !== 'p') { | ||
applyDefaultStyle(ret, parentNodeName); | ||
} | ||
// check if we have inherent styles to apply when a text is inside several <tag> | ||
applyParentsStyle(ret, element); | ||
@@ -112,2 +109,8 @@ // for links | ||
// for 'td' and 'th' we check if we have "rowspan" or "colspan" | ||
if (parentNodeName === "td" || parentNodeName === "th") { | ||
if (parentNode.getAttribute("rowspan")) ret.rowSpan = parentNode.getAttribute("rowspan")*1; | ||
if (parentNode.getAttribute("colspan")) ret.colSpan = parentNode.getAttribute("colspan")*1; | ||
} | ||
// is there any class to this element? | ||
@@ -136,4 +139,2 @@ cssClass = parentNode.getAttribute("class"); | ||
[].forEach.call(element.childNodes, function(child) { | ||
// for THEAD and TBODY we go straight to the TR | ||
//if (child.nodeName === "THEAD" || child.nodeName === "TBODY" || child.nodeName === "TFOOTER") continue; | ||
child = parseElement(child, element); | ||
@@ -175,5 +176,35 @@ if (child) { | ||
if (re.stack) { | ||
var td = [] | ||
re.stack.forEach(function(r) { | ||
var td = [], | ||
rowspan = {}; | ||
re.stack.forEach(function(r, indexRow) { | ||
if (r.stack) { | ||
// do we have a rowspan to apply from previous rows? | ||
if (rowspan[indexRow]) { | ||
// insert empty cell due to rowspan | ||
rowspan[indexRow].forEach(function(cell) { | ||
r.stack.splice(cell.index, 0, {text:'', style: ['html-td', 'html-tr'], colSpan:cell.colspan}); | ||
}); | ||
} | ||
// insert empty cells due to colspan | ||
r.stack.forEach(function(cell, index) { | ||
if (cell.colSpan > 1) { | ||
for (var i=0; i<cell.colSpan-1; i++) { | ||
r.stack.splice(index+1, 0, {text:'', style: ['html-td', 'html-tr']}) | ||
} | ||
} | ||
}) | ||
// check rowspan for the current row in order to then apply it to the next ones | ||
var indexCell = 0; | ||
r.stack.forEach(function(cell) { | ||
if (cell.rowSpan) { | ||
for (var i=0; i<cell.rowSpan; i++) { | ||
if (!rowspan[indexRow+i]) rowspan[indexRow+i] = []; | ||
// we also remember the colSpan for cells with both rowspan and colspan | ||
rowspan[indexRow+i].push({index:indexCell, colspan:cell.colSpan||1}); | ||
} | ||
} | ||
indexCell += cell.colSpan || 1; | ||
}); | ||
ret.table.body.push(r.stack) | ||
@@ -218,7 +249,5 @@ } else { | ||
// add a custom class to let the user customize the element | ||
// "tr" elements should always contain an array | ||
// "tr" elements should always contain an array | ||
if (ret.length === 1 && nodeName !== "tr") { | ||
ret=ret[0]; | ||
// check if we have a default css style to apply when a text is inside several <tag> | ||
// e.g. <strong><em>text</em></strong> | ||
if (ret.text) { | ||
@@ -232,7 +261,12 @@ applyDefaultStyle(ret, nodeName); | ||
} else { | ||
ret = (nodeName==='p' ? {text:ret} : {stack:ret}); | ||
// we apply the default style if it's a "p" | ||
if (nodeName === 'p') { | ||
applyDefaultStyle(ret, 'p'); | ||
var isInlineTag = (inlineTags.indexOf(nodeName) > -1); | ||
// if we have an inline tag, then we check if we have a non-inline tag in its section | ||
ret = (!isInlineTag || /{"(stack|table|ol|ul|image)"/.test(JSON.stringify(ret)) ? {stack:ret} : {text:ret}); | ||
// we apply the default style for the inline tags | ||
if (isInlineTag) { | ||
applyDefaultStyle(ret, nodeName); | ||
} | ||
ret.style = ['html-'+nodeName]; | ||
@@ -250,2 +284,13 @@ } | ||
} | ||
// retrieve the class from the parent | ||
cssClass = element.getAttribute("class"); | ||
if (cssClass && typeof ret === 'object') { | ||
// apply all the classes not there yet | ||
ret.style = (ret.style || []) | ||
.concat(cssClass.split(' ')) | ||
.filter(function (value, index, self) { | ||
return self.indexOf(value) === index; | ||
}); | ||
} | ||
} | ||
@@ -269,2 +314,26 @@ | ||
var applyParentsStyle = function(ret, node) { | ||
// while the parents are an inline tag, we want to apply the default style and the class to the children too | ||
var classes = [], defaultStyles = [], cssClass; | ||
var inlineParentNode=node.parentNode; | ||
while (inlineParentNode) { | ||
var defaultStyle = {}; | ||
var inlineParentNodeName=inlineParentNode.nodeName.toLowerCase(); | ||
if (inlineTags.indexOf(inlineParentNodeName) > -1) { | ||
cssClass = inlineParentNode.getAttribute("class"); | ||
classes = classes.concat(['html-'+inlineParentNodeName], cssClass||[]); | ||
applyDefaultStyle(defaultStyle, inlineParentNodeName); | ||
defaultStyles.push(defaultStyle); | ||
inlineParentNode=inlineParentNode.parentNode; | ||
} else break; | ||
} | ||
ret.style = (ret.style||[]).concat(classes); | ||
defaultStyles.forEach(function(defaultStyle) { | ||
for (var key in defaultStyle) { | ||
if (key.indexOf("margin") === -1 && ret[key] === undefined) ret[key] = defaultStyle[key]; | ||
} | ||
}) | ||
} | ||
/** | ||
@@ -271,0 +340,0 @@ * Transform a CSS expression (e.g. 'margin:10px') in the PDFMake version |
{ | ||
"name": "html-to-pdfmake", | ||
"version": "1.0.6", | ||
"version": "1.0.7", | ||
"description": "Convert HTML code to PDFMake", | ||
@@ -9,3 +9,4 @@ "main": "index.js", | ||
"test": "node test/unit.js", | ||
"browser": "browserify index.js --standalone htmlToPdfmake > docs/browser.js" | ||
"browser": "browserify index.js --standalone htmlToPdfmake > docs/browser.js", | ||
"publish": "npm run browser && npm publish" | ||
}, | ||
@@ -27,3 +28,4 @@ "repository": { | ||
"simple-test-framework": "^0.1.7" | ||
} | ||
}, | ||
"dependencies": {} | ||
} |
@@ -327,2 +327,76 @@ var htmlToPdfMake = require('../index.js'); | ||
test("table (rowspan/colspan)", function(t) { | ||
var html = `<table> | ||
<tr> | ||
<th>Col A</th> | ||
<th>Col B</th> | ||
<th>Col C</th> | ||
<th>Col D</th> | ||
</tr> | ||
<tr> | ||
<td>Cell A1</td> | ||
<td rowspan="2"> | ||
Cell B1 & B2 | ||
</td> | ||
<td>Cell C1</td> | ||
<td rowspan="2"> | ||
Cell D1 & D2 | ||
</td> | ||
</tr> | ||
<tr> | ||
<td>Cell A2</td> | ||
<td>Cell C2</td> | ||
</tr> | ||
<tr> | ||
<td>Cell A3</td> | ||
<td colspan="2">Cell B3 & C3</td> | ||
<td>Cell D3</td> | ||
</tr> | ||
<tr> | ||
<td rowspan="2" colspan="3"> | ||
Cell A4 & A5 & B4 & B5 & C4 & C5 | ||
</td> | ||
<td>Cell D4</td> | ||
</tr> | ||
<tr> | ||
<td>Cell D5</td> | ||
</tr> | ||
</table>`; | ||
var ret = htmlToPdfMake(html, window); | ||
t.check(Array.isArray(ret) && ret.length===1, "return is OK"); | ||
ret = ret[0]; | ||
t.check( | ||
ret.table && | ||
Array.isArray(ret.table.body) && | ||
ret.table.body.length === 6 && | ||
ret.table.body[1][0].text === "Cell A1" && | ||
ret.table.body[1][0].style[0] === 'html-td' && | ||
ret.table.body[1][0].style[1] === 'html-tr' && | ||
ret.table.body[1][1].text === "Cell B1 & B2" && | ||
ret.table.body[1][2].text === "Cell C1" && | ||
ret.table.body[1][3].text === "Cell D1 & D2" && | ||
ret.table.body[2][0].text === "Cell A2" && | ||
ret.table.body[2][1].text === "" && | ||
ret.table.body[2][2].text === "Cell C2" && | ||
ret.table.body[2][3].text === "" && | ||
ret.table.body[3][0].text === "Cell A3" && | ||
ret.table.body[3][1].text === "Cell B3 & C3" && | ||
ret.table.body[3][2].text === "" && | ||
ret.table.body[3][3].text === "Cell D3" && | ||
ret.table.body[4][0].text === "Cell A4 & A5 & B4 & B5 & C4 & C5" && | ||
ret.table.body[4][1].text === "" && | ||
ret.table.body[4][2].text === "" && | ||
ret.table.body[4][3].text === "Cell D4" && | ||
ret.table.body[5][0].text === "" && | ||
ret.table.body[5][1].text === "" && | ||
ret.table.body[5][2].text === "" && | ||
ret.table.body[5][3].text === "Cell D5" && | ||
Array.isArray(ret.style) && | ||
ret.style[0] === 'html-table', | ||
"table (rowspan/colspan)"); | ||
t.finish(); | ||
}) | ||
test("img",function(t) { | ||
@@ -343,3 +417,3 @@ var ret = htmlToPdfMake('<img width="10" style="height:10px" src="data:image/jpeg;base64,...encodedContent...">', window); | ||
test("cascade_tags", function(t) { | ||
var ret = htmlToPdfMake('<p style="text-align: center;"> <span style="font-size: 14px;"><em><strong>test</strong></em></span> </p>', window); | ||
var ret = htmlToPdfMake('<p style="text-align: center;"><span style="font-size: 14px;"><em><strong>test</strong></em></span></p>', window); | ||
t.check(Array.isArray(ret) && ret.length===1, "return is OK"); | ||
@@ -346,0 +420,0 @@ ret = ret[0]; |
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
104983
12
1448