/**
* Created by Tony on 2017/4/1.
*/
let JV = require('../rpt_component/jpc_value_define');
let fs = require('fs');
let JSZip = require("jszip");
let strUtil = require('../../../public/stringUtil');
let jpcCmnHelper = require('../rpt_component/helper/jpc_helper_common');
let DPI = jpcCmnHelper.getScreenDPI()[0];
let fsUtil = require('../../../public/fsUtil');
const dftHeadXml = '';
const uuidV1 = require('uuid/v1');
function writeContentTypes(sheets, isSinglePage) {
let rst = [];
rst.push(dftHeadXml + '\r\n');
rst.push('');
//...
rst.push('');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
if (isSinglePage) {
rst.push('');
} else {
for (let i = 0; i < sheets.length; i++) {
rst.push('');
}
}
rst.push('');
rst.push('');
return rst;
}
function writeRootRels(){
let rst = [];
rst.push(dftHeadXml + '\r\n');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
return rst;
}
function writeApp(sheets, isSinglePage) {
let rst = [];
rst.push(dftHeadXml + '\r\n');
rst.push('');
rst.push('Microsoft Excel');
rst.push('0');
rst.push('false');
rst.push('');
rst.push('');
rst.push('工作表');
if (isSinglePage) rst.push('1')
else rst.push('' + sheets.length + '');
rst.push('');
rst.push('');
rst.push('');
if (isSinglePage) {
rst.push('');
rst.push('' + sheets[0].sheetName + '');
} else {
rst.push('');
for (let i = 0; i < sheets.length; i++) {
rst.push('' + sheets[i].sheetName + '');
}
}
rst.push('');
rst.push('');
rst.push('SmartCost');
rst.push('false');
rst.push('false');
rst.push('false');
rst.push('12.0000');
rst.push('');
return rst;
}
function writeCore() {
let rst = [];
let p_fillZero = function(val){
let rst = val;
if (val < 10) {
rst = '0' + val;
}
return rst;
};
rst.push(dftHeadXml + '\r\n');
rst.push('');
rst.push('SmartCost');
rst.push('SmartCost');
let dt = new Date();
dt.setDate(dt.getDate() - 8/24); //it's GMT time, so please add the server offset time ( -8 hours )
let dtStr = dt.getFullYear() + '-' + p_fillZero(dt.getMonth()+1) + '-' + p_fillZero(dt.getDate()) + 'T' +
p_fillZero(dt.getHours()) + ':' + p_fillZero(dt.getMinutes()) + ':' + p_fillZero(dt.getSeconds()) + 'Z';
rst.push('' + dtStr + '');
rst.push('' + dtStr + '');
//rst.push('');
rst.push('');
return rst;
}
function writeXlWorkBook(sheets, isSinglePage){
let rst = [];
rst.push(dftHeadXml + '\r\n');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
if (isSinglePage) {
rst.push('');
} else {
for (let i = 0; i < sheets.length; i++) {
rst.push('');
}
}
rst.push('');
rst.push('');
rst.push('');
return rst;
}
function writeXlRels(sheets, isSinglePage){
let rst = [], idx = 1;
rst.push(dftHeadXml + '\r\n');
rst.push('');
if (isSinglePage) {
rst.push('');
idx++;
} else {
for (let i = 0; i < sheets.length; i++) {
rst.push('');
idx++;
}
}
rst.push('');
idx++;
rst.push('');
idx++;
rst.push('');
rst.push('');
return rst;
}
function writeTheme(){
let rst = fs.readFileSync(__dirname + '/excel_base_files/theme1.xml', 'utf8', 'r');
return rst;
}
function writeStyles(stylesObj){
let rst = [];
rst.push(dftHeadXml + '\r\n');
rst.push('');
//1. push fonts
rst.push('');
for (let i = 0; i < stylesObj.fonts.length; i++) {
let font = stylesObj.fonts[i];
rst.push('');
if (strUtil.convertStrToBoolean(font[JV.FONT_PROPS[3]])) {
rst.push('');
}
if (strUtil.convertStrToBoolean(font[JV.FONT_PROPS[4]])) {
rst.push('');
}
if (strUtil.convertStrToBoolean(font[JV.FONT_PROPS[5]])) {
rst.push('');
}
rst.push('');
rst.push('');
rst.push('');
rst.push('');
rst.push('');
}
rst.push('');
//2. push default fills
rst.push('');
//3. push borders
rst.push('');
let private_setBorder = function(border, borderDirection) {
if (parseInt(border[borderDirection][JV.PROP_LINE_WEIGHT]) === 0) {
rst.push('<' + borderDirection.toLowerCase() + '/>');
} else {
let bW = 'thin';
if (parseInt(border[borderDirection][JV.PROP_LINE_WEIGHT]) === 2) bW = 'medium';
if (parseInt(border[borderDirection][JV.PROP_LINE_WEIGHT]) > 2) bW = 'thick';
rst.push('<' + borderDirection.toLowerCase() + ' style="' + bW + '">' + '' + '' + borderDirection.toLowerCase() + '>');
}
};
for (let i = 0; i < stylesObj.borders.length; i++) {
let border = stylesObj.borders[i];
rst.push('');
private_setBorder(border, JV.PROP_LEFT);
private_setBorder(border, JV.PROP_RIGHT);
private_setBorder(border, JV.PROP_TOP);
private_setBorder(border, JV.PROP_BOTTOM);
rst.push('');
rst.push('');
}
rst.push('');
//4. push cellStyleXfs
rst.push('');
//5. push cellXfs
rst.push('');
for (let i = 0; i < stylesObj.cellXfs.length; i++) {
let excelStyle = stylesObj.cellXfs[i];
rst.push('');
//pageData[JV.NODE_FONT_COLLECTION] excelStyle.fontId
let alignStr = " 0) {
textRotation = 180;
if (newHorizontal === "left") {
tmpV = 'top';
} else if (newHorizontal === "right") {
tmpV = 'bottom';
} else {
tmpV = 'center';
}
if (newVertical === "top") {
tmpH = 'right';
} else if (newVertical === "bottom") {
tmpH = 'left';
} else if (newVertical === "justify") {
tmpH = 'justify';
} else {
tmpH = 'center';
}
} else {
textRotation = 90;
if (newHorizontal === "left") {
tmpV = 'bottom';
} else if (newHorizontal === "right") {
tmpV = 'top';
} else {
tmpV = 'center';
}
if (newVertical === "top") {
tmpH = 'left';
} else if (newVertical === "bottom") {
tmpH = 'right';
} else {
tmpH = 'center';
}
}
newHorizontal = tmpH;
newVertical = tmpV;
}
alignStr += ' horizontal="' + newHorizontal + '" vertical="' + newVertical + '"';
if (strUtil.convertStrToBoolean(excelStyle[JV.CONTROL_PROPS[0]])) {
alignStr += ' shrinkToFit="1"';
}
if (strUtil.convertStrToBoolean(excelStyle[JV.CONTROL_PROPS[4]]) && !strUtil.convertStrToBoolean(excelStyle[JV.CONTROL_PROPS[6]])) {
alignStr += ' wrapText="1"';
}
if (textRotation !== 0) {
alignStr += ' textRotation="' + textRotation + '"';
}
alignStr += '/>';
rst.push(alignStr);
rst.push('');
}
rst.push('');
//6. others (xfl style / dxfs / tableStyles)
rst.push('');
rst.push('');
rst.push('');
rst.push('');
return rst;
}
function writeSharedString(sharedStrList){
let rst = [];
let pri_func_write = function(cellVal) {
if (cellVal !== null) {
if ((typeof cellVal === 'string') && cellVal.indexOf(' ') === 0) {
rst.push('' + cellVal + '');
} else {
rst.push('' + cellVal + '');
}
}
};
if (sharedStrList && sharedStrList.length > 0) {
rst.push(dftHeadXml + '\r\n');
rst.push('');
let regExp = new RegExp("<", "gm");
for (let i = 0; i < sharedStrList.length; i++) {
if (typeof sharedStrList[i] === 'string') {
//转换特殊字符,如 < , 则需要转义一下
sharedStrList[i] = sharedStrList[i].replace(regExp, "<");
if (sharedStrList[i].indexOf('|') >= 0) {
//rst.push('' + sharedStrList[i].split('|').join('\r\n') + '');
// rst.push('' + sharedStrList[i].split('|').join('\n') + '');
pri_func_write(sharedStrList[i].split('|').join('\n'));
} else {
// rst.push('' + sharedStrList[i] + '');
pri_func_write(sharedStrList[i]);
}
// rst.push('' + sharedStrList[i].replace('|','\r\n') + '');
} else {
// rst.push('' + sharedStrList[i] + '');
pri_func_write(sharedStrList[i]);
}
}
rst.push('');
}
return rst;
}
function writeSheets(pageData, paperSize, sharedStrList, stylesObj, isSinglePage, custSheetMergeBands){
let rst = [];
let private_pushDftFont = function(){
let font = {};
if (!(stylesObj.fonts)) {
stylesObj.fonts = [];
}
font[JV.FONT_PROPS[0]] = "宋体"; //font name
font.size = 12;
font.charset = 134;
font.colorIdx = "8";
stylesObj.fonts.push(font);
};
let private_buildFirstDftStyle = function () {
stylesObj.cellXfs = [];
stylesObj.borders = [];
let fontId = 0;
let borderId = 0;
let border = {};
border[JV.PROP_LEFT] = {};
border[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT] = 0;
border[JV.PROP_RIGHT] = {};
border[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT] = 0;
border[JV.PROP_TOP] = {};
border[JV.PROP_TOP][JV.PROP_LINE_WEIGHT] = 0;
border[JV.PROP_BOTTOM] = {};
border[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT] = 0;
stylesObj.borders.push(border);
let cellControl = pageData[JV.NODE_CONTROL_COLLECTION].Default;
let sheetControl = {};
sheetControl.fontId = fontId;
sheetControl.borderId = borderId;
sheetControl.fontAngle = 0;
for (let i = 0; i < JV.CONTROL_PROPS.length; i++) {
sheetControl[JV.CONTROL_PROPS[i]] = cellControl[JV.CONTROL_PROPS[i]];
}
stylesObj.cellXfs.push(sheetControl);
}
private_pushDftFont();
private_buildFirstDftStyle();
if (isSinglePage) {
rst.push(writeSheet(pageData, null, paperSize, sharedStrList, stylesObj, null, true));
} else {
for (let i = 0; i < pageData.items.length; i++) {
let appointedMergeBand = null;
if (custSheetMergeBands && custSheetMergeBands.length > i) {
appointedMergeBand = custSheetMergeBands[i];
}
rst.push(writeSheet(pageData, pageData.items[i], paperSize, sharedStrList, stylesObj, appointedMergeBand, i === 0));
}
}
return rst;
}
function writeSheet(pageData, sheetData, paperSize, sharedStrList, stylesObj, appointedMergeBand, isFirstSheet){
let rst = [], xPos = [], yPos = [], yMultiPos = [], currentMergeBorder = null,
headerStr = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
let currentPageMergePos = null; //在 JV.PAGING_OPTION_INFINITY 场合应用
let private_pre_analyze_pos = function(){
let cell, pos;
let self_analyze_sheet_pos = function (theShtData, theXPos, theYPos) {
// theShtData.cells.sort(function(cell1, cell2) {
// let rst = 0;
// if (cell1[JV.PROP_AREA][JV.PROP_TOP] > cell2[JV.PROP_AREA][JV.PROP_TOP]) {
// rst = 1;
// } else if (cell1[JV.PROP_AREA][JV.PROP_TOP] < cell2[JV.PROP_AREA][JV.PROP_TOP]) {
// rst = -1;
// } else {
// if (cell1[JV.PROP_AREA][JV.PROP_LEFT] > cell2[JV.PROP_AREA][JV.PROP_LEFT]) {
// rst = 1;
// } else if (cell1[JV.PROP_AREA][JV.PROP_LEFT] < cell2[JV.PROP_AREA][JV.PROP_LEFT]) {
// rst = -1;
// }
// }
// return rst;
// });
for (let i = 0; i < theShtData.cells.length; i++) {
cell = theShtData.cells[i];
pos = cell[JV.PROP_AREA][JV.PROP_LEFT];
if (theXPos.indexOf(pos) < 0) theXPos.push(pos);
pos = cell[JV.PROP_AREA][JV.PROP_RIGHT];
if (theXPos.indexOf(pos) < 0) theXPos.push(pos);
pos = cell[JV.PROP_AREA][JV.PROP_TOP];
if (theYPos.indexOf(pos) < 0) theYPos.push(pos);
pos = cell[JV.PROP_AREA][JV.PROP_BOTTOM];
if (theYPos.indexOf(pos) < 0) theYPos.push(pos);
}
};
xPos.push(0);
if (sheetData) {
//current sheet data
yPos.push(0);
self_analyze_sheet_pos(sheetData, xPos, yPos);
xPos.sort(private_array_sort);
yPos.sort(private_array_sort);
} else {
//total data in one sheet
let marginBottomPos = Math.round( (pageData[JV.NODE_PAGE_INFO][JV.NODE_PAGE_SIZE][1] - parseFloat(pageData[JV.NODE_PAGE_INFO][JV.NODE_MARGINS][JV.PROP_BOTTOM]) / 2.54 ) * DPI);
for (let shtItemData of pageData.items) {
let tmpPos = [];
tmpPos.push(0);
self_analyze_sheet_pos(shtItemData, xPos, tmpPos);
tmpPos.sort(private_array_sort);
if (marginBottomPos - tmpPos[tmpPos.length - 1] > 10) {
//此逻辑是为了防止打印跨页(假设有些报表模板高度设置离底部margin还好远,导出excel后预览时会发现跨页现象(即下一页的某几行数据会挪到前一页来预览))
tmpPos.push(marginBottomPos - 10);
}
yMultiPos.push(tmpPos);
}
xPos.sort(private_array_sort);
yPos = yMultiPos[0];
}
};
let private_array_sort = function(i1, i2){
let rst = 0;
if (i1 > i2) {rst = 1} else
if (i1 < i2) rst = -1;
return rst;
};
let private_getCellIdxStr = function(idx){
let rst = 'A';
if (idx < 26) {
rst = headerStr[idx];
} else if (idx < 26*26+26) {
let ti = Math.floor(idx / 26), tj = idx % 26;
rst = headerStr[ti - 1] + headerStr[tj];
} else if (idx < 26*26*26+26) {
let ti = Math.floor(idx / (26*26)), tj = Math.floor((idx - ti * 26*26) / 26), tk = idx % 26;
rst = headerStr[ti - 1] + headerStr[tj-1] + headerStr[tk];
}
return rst;
};
let private_getSharedStrIdx = function(val) {
let strVal = val;
if (val === null || val === undefined || (typeof val === 'number' && isNaN(val))) {
strVal = "";
}
let rst = sharedStrList.indexOf(strVal);
if (rst < 0) {
sharedStrList.push(strVal);
rst = sharedStrList.length - 1;
}
return rst;
};
let private_getFontId = function(cell) {
let rst = 0, hasFont = false;
if (!(stylesObj.fonts)) {
stylesObj.fonts = [];
//for (let i = 0; i < sheetData.font_collection)
}
let sheetFont = null;
if (typeof cell[JV.PROP_FONT] === "string") {
sheetFont = pageData[JV.NODE_FONT_COLLECTION][cell[JV.PROP_FONT]];
} else {
sheetFont = cell[JV.PROP_FONT];
}
for (let i = 0; i < stylesObj.fonts.length; i++) {
let font = stylesObj.fonts[i];
if (sheetFont) {
// if (font[JV.FONT_PROPS[0]] === sheetFont[JV.FONT_PROPS[0]] && font.size === Math.round(sheetFont[JV.FONT_PROPS[1]] * 3 / 4)
if (font[JV.FONT_PROPS[0]] === sheetFont[JV.FONT_PROPS[0]] && font.size === Math.floor(sheetFont[JV.FONT_PROPS[1]] * 3 / 4)
&& font[JV.FONT_PROPS[3]] === sheetFont[JV.FONT_PROPS[3]] && font[JV.FONT_PROPS[4]] === sheetFont[JV.FONT_PROPS[4]]
&& font[JV.FONT_PROPS[5]] === sheetFont[JV.FONT_PROPS[5]] ) {
hasFont = true;
rst = i;
break;
}
} else {
break;
}
}
if (!hasFont) {
let font = {};
font[JV.FONT_PROPS[0]] = sheetFont[JV.FONT_PROPS[0]]; //font name
// font.size = Math.round(sheetFont[JV.FONT_PROPS[1]] * 3 / 4);
font.size = Math.floor(sheetFont[JV.FONT_PROPS[1]] * 3 / 4);
font.charset = 134;
font.colorIdx = "8";
font[JV.FONT_PROPS[3]] = sheetFont[JV.FONT_PROPS[3]]; //font bold
font[JV.FONT_PROPS[4]] = sheetFont[JV.FONT_PROPS[4]]; //font italic
font[JV.FONT_PROPS[5]] = sheetFont[JV.FONT_PROPS[5]]; //font underline
stylesObj.fonts.push(font);
rst = stylesObj.fonts.length - 1;
}
return rst;
};
let private_checkBorder = function(cell, border, sheetBorder) {
let rst = true, borderLineWidths = [], sheetBorderLineWidths = [];
borderLineWidths.push(border[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT]);
borderLineWidths.push(border[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT]);
borderLineWidths.push(border[JV.PROP_TOP][JV.PROP_LINE_WEIGHT]);
borderLineWidths.push(border[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT]);
if (sheetBorder[JV.PROP_LEFT] && sheetBorder[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT]) {
sheetBorderLineWidths.push(private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_LEFT, true));
} else {
sheetBorderLineWidths.push(0);
}
if (sheetBorder[JV.PROP_RIGHT] && sheetBorder[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT]) {
sheetBorderLineWidths.push(private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_RIGHT, true));
} else {
sheetBorderLineWidths.push(0);
}
if (sheetBorder[JV.PROP_TOP] && sheetBorder[JV.PROP_TOP][JV.PROP_LINE_WEIGHT]) {
sheetBorderLineWidths.push(private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_TOP, false));
} else {
sheetBorderLineWidths.push(0);
}
if (sheetBorder[JV.PROP_BOTTOM] && sheetBorder[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT]) {
sheetBorderLineWidths.push(private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_BOTTOM, false));
} else {
sheetBorderLineWidths.push(0);
}
for (let i = 0; i < 4; i++) {
if (borderLineWidths[i] != sheetBorderLineWidths[i]) {
rst = false;
break;
}
}
return rst;
};
let private_chkAndGetMergeLine = function(cell, sheetBorder, borderStr, needFurtherChk) {
let rst = 0,
// mergeBorder = (sheetData)?sheetData[JV.PROP_PAGE_MERGE_BORDER]:pageData[JV.BAND_PROP_MERGE_BAND],
mergeBorder = currentMergeBorder,
mergeBand = pageData[JV.BAND_PROP_MERGE_BAND]
;
if (appointedMergeBand !== null) {
mergeBand = appointedMergeBand;
}
if (sheetBorder[borderStr] && sheetBorder[borderStr][JV.PROP_LINE_WEIGHT] !== undefined) {
rst = sheetBorder[borderStr][JV.PROP_LINE_WEIGHT];
}
if (currentPageMergePos) {
let side = currentPageMergePos[borderStr];
if (side.indexOf(cell[JV.PROP_AREA][borderStr]) >= 0) {
if (needFurtherChk) {
let topSide = currentPageMergePos[JV.PROP_TOP];
let bottomSide = currentPageMergePos[JV.PROP_BOTTOM];
for (let i = 0; i < topSide.length; i++) {
if (cell[JV.PROP_AREA][JV.PROP_TOP] >= topSide[i]) {
if (cell[JV.PROP_AREA][JV.PROP_BOTTOM] <= bottomSide[i]) {
let destStyle = pageData[JV.NODE_STYLE_COLLECTION][mergeBand[JV.PROP_STYLE][JV.PROP_ID]];
rst = destStyle[borderStr][JV.PROP_LINE_WEIGHT];
break;
}
} else {
break;
}
}
} else {
let destStyle = pageData[JV.NODE_STYLE_COLLECTION][mergeBand[JV.PROP_STYLE][JV.PROP_ID]];
rst = destStyle[borderStr][JV.PROP_LINE_WEIGHT];
}
}
} else {
if (cell[JV.PROP_AREA][borderStr] === mergeBorder[borderStr]) {
let destStyle = pageData[JV.NODE_STYLE_COLLECTION][mergeBand[JV.PROP_STYLE][JV.PROP_ID]];
if (needFurtherChk) {
if (cell[JV.PROP_AREA][JV.PROP_TOP] >= mergeBorder[JV.PROP_TOP] &&
cell[JV.PROP_AREA][JV.PROP_BOTTOM] <= mergeBorder[JV.PROP_BOTTOM]) {
rst = destStyle[borderStr][JV.PROP_LINE_WEIGHT];
}
} else {
rst = destStyle[borderStr][JV.PROP_LINE_WEIGHT];
}
}
}
return parseInt(rst);
};
let private_getIniBorder = function() {
let rst = {};
rst[JV.PROP_LEFT] = {};
rst[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT] = 0;
rst[JV.PROP_RIGHT] = {};
rst[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT] = 0;
rst[JV.PROP_TOP] = {};
rst[JV.PROP_TOP][JV.PROP_LINE_WEIGHT] = 0;
rst[JV.PROP_BOTTOM] = {};
rst[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT] = 0;
return rst;
};
let private_getBorderId = function(cell) {
let rst = 0, hasBorder = false;
if (!(stylesObj.borders)) {
stylesObj.borders = [];
}
let sheetBorder = pageData[JV.NODE_STYLE_COLLECTION][cell.style];
let mergedBorder = private_getIniBorder();
mergedBorder[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT] = private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_LEFT, true);
mergedBorder[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT] = private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_RIGHT, true);
mergedBorder[JV.PROP_TOP][JV.PROP_LINE_WEIGHT] = private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_TOP, false);
mergedBorder[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT] = private_chkAndGetMergeLine(cell, sheetBorder, JV.PROP_BOTTOM, false);
for (let i = 0; i < stylesObj.borders.length; i++) {
let border = stylesObj.borders[i];
if (private_checkBorder(cell, border, mergedBorder)) {
hasBorder = true;
rst = i;
break;
}
}
if (!hasBorder) {
stylesObj.borders.push(mergedBorder);
rst = stylesObj.borders.length - 1;
}
return rst;
};
let private_checkControl = function(cellControl, sheetControl) {
let rst = true;
for (let i = 0; i < JV.CONTROL_PROPS.length; i++) {
if (cellControl[JV.CONTROL_PROPS[i]] != sheetControl[JV.CONTROL_PROPS[i]]) {
rst = false;
break;
}
}
return rst;
};
let private_getStyleId = function(cell) {
let rst = 1, hasStyle = false;
if (!(stylesObj.cellXfs)) stylesObj.cellXfs = [];
let fontId = private_getFontId(cell);
let fontAngle = 0;
if (typeof cell[JV.PROP_FONT] === "string") {
fontAngle = parseInt(pageData[JV.NODE_FONT_COLLECTION][cell[JV.PROP_FONT]].FontAngle);
} else {
fontAngle = parseInt(cell[JV.PROP_FONT].FontAngle);
}
let borderId = private_getBorderId(cell);
let cellControl = null;
if (typeof cell[JV.PROP_CONTROL] === "string") {
cellControl = pageData[JV.NODE_CONTROL_COLLECTION][cell[JV.PROP_CONTROL]];
} else {
cellControl = cell[JV.PROP_CONTROL];
}
for (let i = 0; i < stylesObj.cellXfs.length; i++) {
let sheetControl = stylesObj.cellXfs[i];
if (sheetControl.fontId === fontId && sheetControl.borderId === borderId) {
if (private_checkControl(cellControl, sheetControl)) {
rst = i;
hasStyle = true;
break;
}
}
}
if (!hasStyle) {
let sheetControl = {};
sheetControl.fontId = fontId;
sheetControl.borderId = borderId;
sheetControl.fontAngle = fontAngle;
for (let i = 0; i < JV.CONTROL_PROPS.length; i++) {
sheetControl[JV.CONTROL_PROPS[i]] = cellControl[JV.CONTROL_PROPS[i]];
}
stylesObj.cellXfs.push(sheetControl);
rst = stylesObj.cellXfs.length - 1;
}
return rst;
};
let private_setCols = function(){
//remark: 1 excel unit width = 2.117 mm
rst.push('');
let w = 0;
for (let i = 1; i < xPos.length - 1; i++) {
w = 1.0 * (xPos[i + 1] - xPos[i]) / DPI * 25.4 / 2.117;
w = Math.round(w * 1000) / 1000;
rst.push('');
}
rst.push('');
rst.push('');
};
let private_setMergedCells = function() {
let cell, idxR, idxL, idxT, idxB, cnt = 0;
rst.push('');
let startIdx = rst.length - 1;
let self_setMergedCells = function (theData, theYPos, offsetY) {
for (let i = 0; i < theData.cells.length; i++) {
cell = theData.cells[i];
idxR = xPos.indexOf(cell[JV.PROP_AREA][JV.PROP_RIGHT]);
idxL = xPos.indexOf(cell[JV.PROP_AREA][JV.PROP_LEFT]);
idxB = theYPos.indexOf(cell[JV.PROP_AREA][JV.PROP_BOTTOM]);
idxT = theYPos.indexOf(cell[JV.PROP_AREA][JV.PROP_TOP]);
if (idxR - idxL > 1 || idxB - idxT > 1) {
rst.push('');
cnt++;
}
}
}
if (sheetData) {
self_setMergedCells(sheetData, yPos, 0);
} else {
let osY = 0;
for (let i = 0; i < pageData.items.length; i++) {
let shtItemData = pageData.items[i];
let tmpPos = yMultiPos[i];
self_setMergedCells(shtItemData, tmpPos, osY);
osY += tmpPos.length - 2;
}
}
rst[startIdx] = '';
rst.push('');
};
let private_setSheetData = function(){
//remark: 1 excel unit height = 0.3612 mm
rst.push('');
let spanX = xPos.length - 2, cellIdx = 0, h = 0
;
let self_setDataEx = function (theShtData, theYPos, offsetY) {
let rows = [];
//1. build full set of blank rows/cells
for (let i = 1; i < theYPos.length - 1; i++) {
let rowObj = {};
h = (theYPos[i+1] - theYPos[i]) / DPI * 25.4 / 0.3612;
h = Math.round(h * 1000) / 1000;
rowObj.height = h;
rowObj.r = i + offsetY;
rowObj.items = [];
rows.push(rowObj);
for (let j = 1; j < xPos.length - 1; j++) {
let colIdxStr = private_getCellIdxStr(j - 1);
let cellObj = {};
cellObj.r = colIdxStr + (i + offsetY);
cellObj.s = 0;
cellObj.isBlank = true;
rows[i - 1].items.push(cellObj);
}
}
//2. then fill up the cell style-ids and values
let rowIdx1 = 0, colIdx1 = 0, rowIdx2 = 0, colIdx2 = 0, colIdxStr = '';
for (let cIdx = 0; cIdx < theShtData.cells.length; cIdx++) {
let styleIdx = private_getStyleId(theShtData.cells[cIdx]);
//colIdxStr = private_getCellIdxStr(j - 1);
rowIdx1 = theYPos.indexOf(theShtData.cells[cIdx][JV.PROP_AREA][JV.PROP_TOP]);
colIdx1 = xPos.indexOf(theShtData.cells[cIdx][JV.PROP_AREA][JV.PROP_LEFT]);
let cellObj = rows[rowIdx1 - 1].items[colIdx1 - 1];
cellObj.s = styleIdx;
cellObj.isBlank = false;
if (!(strUtil.isEmptyString(theShtData.cells[cIdx][JV.PROP_VALUE]))) {
let valIdx = private_getSharedStrIdx(theShtData.cells[cIdx][JV.PROP_VALUE]);
cellObj.v = valIdx;
}
rowIdx2 = theYPos.indexOf(theShtData.cells[cIdx][JV.PROP_AREA][JV.PROP_BOTTOM]);
colIdx2 = xPos.indexOf(theShtData.cells[cIdx][JV.PROP_AREA][JV.PROP_RIGHT]);
if ((rowIdx2 - rowIdx1 > 1) || (colIdx2 - colIdx1 > 1)) {
for (let i = 0; i < rowIdx2 - rowIdx1; i++) {
for (let j = 0; j < colIdx2 - colIdx1; j++) {
if (i === 0 && j === 0) continue;
cellObj = rows[rowIdx1 - 1 + i].items[colIdx1 - 1 + j];
cellObj.s = styleIdx;
cellObj.isBlank = true;
}
}
}
}
//3. then fill up rst
for (let i = 0; i < rows.length; i++) {
rst.push('');
for (let j = 0; j < rows[i].items.length; j++) {
let cellObj = rows[i].items[j];
if (cellObj.v === undefined) {
rst.push('');
} else {
rst.push('');
rst.push('' + cellObj.v + '');
rst.push('');
}
}
rst.push('
');
}
//4. maybe need to dispose the memory
//...
};
if (sheetData) {
//current sheet data
currentPageMergePos = sheetData[JV.PAGE_SPECIAL_MERGE_POS];
currentMergeBorder = sheetData[JV.PROP_PAGE_MERGE_BORDER];
self_setDataEx(sheetData, yPos, 0);
} else {
//total data in one sheet
let cnt = 0;
for (let i = 0; i < pageData.items.length; i++) {
let shtItemData = pageData.items[i];
currentPageMergePos = shtItemData[JV.PAGE_SPECIAL_MERGE_POS];
currentMergeBorder = shtItemData[JV.PROP_PAGE_MERGE_BORDER];
let tmpPos = yMultiPos[i];
cellIdx = 0;
self_setDataEx(shtItemData, tmpPos, cnt);
cnt += tmpPos.length - 2;
}
}
rst.push('');
};
private_pre_analyze_pos();
rst.push(dftHeadXml + '\r\n');
rst.push('');
let colStr = private_getCellIdxStr(xPos.length - 3);
rst.push('');
if (isFirstSheet) {
rst.push('');
} else {
rst.push('');
}
//rst.push('');
rst.push('');
rst.push('');
rst.push('');
private_setCols();
private_setSheetData();
private_setMergedCells();
rst.push('');
// rst.push('');
rst.push('');
let paperSizeIdx = JV.PAGES_SIZE_STR.indexOf(paperSize);
let pStr = '';
if (paperSizeIdx >= 0) {
pStr = 'paperSize="' + JV.PAGES_SIZE_IDX[paperSizeIdx] + '"';
}
let orientationStr = (pageData[JV.NODE_PAGE_INFO][JV.NODE_PAGE_SIZE][0] > pageData[JV.NODE_PAGE_INFO][JV.NODE_PAGE_SIZE][1])?'landscape':'portrait';
if (currentPageMergePos) {
orientationStr = (currentPageMergePos[JV.NODE_PAGE_SIZE][0] > currentPageMergePos[JV.NODE_PAGE_SIZE][1])?'landscape':'portrait';
}
rst.push('');
rst.push('');
rst.push('');
return rst;
}
function mergeProperties(orgObj, newObj) {
let orgPropArr = [], newPropArr = [];
for (let p in orgObj) {
orgPropArr.push(p);
}
for (let p in newObj) {
newPropArr.push(p);
}
for (let i = 0; i < newPropArr.length; i++) {
if (orgPropArr.indexOf(newPropArr[i]) < 0) {
orgObj[newPropArr[i]] = newObj[newPropArr[i]];
}
}
}
module.exports = {
exportExcel: function (pageData, paperSize, fName, options, custSheetNames, custSheetMergeBands, callback) {
let rptOptions = ({singlePage: false, fileName: 'report'});
if (options === 'true' || options === true) {
rptOptions.singlePage = true;
}
let isSinglePage = rptOptions.singlePage;
let sheets = [];
if (isSinglePage) {
sheets.push({sheetName: '全部页'});
} else {
if (custSheetNames && custSheetNames.length === pageData.items.length) {
for (let i = 0; i < pageData.items.length; i++) {
sheets.push({sheetName: custSheetNames[i]});
}
} else {
for (let i = 0; i < pageData.items.length; i++) {
sheets.push({sheetName: '第' + (i + 1) + '页'});
}
}
}
//1.
let file = '[Content_Types].xml';
let data = writeContentTypes(sheets, isSinglePage);
let zip = new JSZip();
zip.file(file, data.join(''), {compression: 'DEFLATE'});
//2.
let zip_rels = zip.folder('_rels');
file = '.rels';
data = writeRootRels();
zip_rels.file(file, data.join(''), {compression: 'DEFLATE'});
//3.
let zip_docProps = zip.folder('docProps');
file = 'app.xml';
data = writeApp(sheets, isSinglePage);
zip_docProps.file(file, data.join(''), {compression: 'DEFLATE'});
file = 'core.xml';
data = writeCore();
zip_docProps.file(file, data.join(''), {compression: 'DEFLATE'});
//4.
let zip_xl = zip.folder('xl');
file = 'workbook.xml';
data = writeXlWorkBook(sheets, isSinglePage);
zip_xl.file(file, data.join(''), {compression: 'DEFLATE'});
let zip_rels2 = zip_xl.folder('_rels');
file = 'workbook.xml.rels';
data = writeXlRels(sheets, isSinglePage);
zip_rels2.file(file, data.join(''), {compression: 'DEFLATE'});
//5.
let zip_theme = zip_xl.folder('theme');
file = 'theme1.xml';
data = writeTheme();
zip_theme.file(file, data, {compression: 'DEFLATE'});
//6.
let zip_worksheets = zip_xl.folder('worksheets');
let sharedStrList = [], stylesObj = {};
data = writeSheets(pageData, paperSize, sharedStrList, stylesObj, isSinglePage, custSheetMergeBands);
if (isSinglePage) {
for (let i = 0; i < 1; i++) {
file = 'sheet' + (i + 1) + '.xml';
zip_worksheets.file(file, data[i].join(''), {compression: 'DEFLATE'});
}
} else {
for (let i = 0; i < data.length; i++) {
file = 'sheet' + (i + 1) + '.xml';
zip_worksheets.file(file, data[i].join(''), {compression: 'DEFLATE'});
}
}
file = 'sharedStrings.xml';
data = writeSharedString(sharedStrList);
zip_xl.file(file, data.join(''), {compression: 'DEFLATE'});
file = 'styles.xml';
data = writeStyles(stylesObj);
zip_xl.file(file, data.join(''), {compression: 'DEFLATE'});
if (fName) {
// let newName = '' + (new Date()).valueOf();
let newName = uuidV1();
zip.generateNodeStream({type:'nodebuffer',streamFiles:true})
.pipe(fs.createWriteStream(__dirname.slice(0, __dirname.length - 21) + '/tmp/' + newName + '.xlsx'))
.on('finish', function () {
// JSZip generates a readable stream with a "end" event,
// but is piped here in a writable stream which emits a "finish" event.
console.log(newName + ".xlsx was written.");
if (callback) callback(newName);
}
);
} else {
//return zip.generateNodeStream({type:'nodebuffer',streamFiles:true});
return zip;
}
},
exportExcelInOneBook: function (pageDataArray, paperSize, fName, callback) {
let me = this, newPageData = {};
//1. 重新编排一下数据,把一份报表的pageData合并到一起作为一个Sheet输出(需要重新调整数据纵向坐标),多份报表数据就形成多个Sheet
// -- 简单来说,就是重新包装数据
try {
// 1.1 newPageData外围属性
let newContrl = {}, newFont = {}, newStyle = {}, custMergeBands = [];
for (let i = 0; i < pageDataArray.length; i++) {
mergeProperties(newContrl, pageDataArray[i][JV.NODE_CONTROL_COLLECTION]);
mergeProperties(newFont, pageDataArray[i][JV.NODE_FONT_COLLECTION]);
mergeProperties(newStyle, pageDataArray[i][JV.NODE_STYLE_COLLECTION]);
}
newPageData[JV.NODE_CONTROL_COLLECTION] = newContrl;
newPageData[JV.NODE_FONT_COLLECTION] = newFont;
newPageData[JV.NODE_STYLE_COLLECTION] = newStyle;
newPageData[JV.NODE_PAGE_INFO] = pageDataArray[0][JV.NODE_PAGE_INFO];
newPageData[JV.BAND_PROP_MERGE_BAND] = pageDataArray[0][JV.BAND_PROP_MERGE_BAND];
// 1.2 重新设置pageDataArray的各个cell的Top/Bottom坐标
let sheetNames = [], newPagePos = [];
for (let i = 0; i < pageDataArray.length; i++) {
let offsetY = 0;
let mergeBand = {};
custMergeBands.push(pageDataArray[i][JV.BAND_PROP_MERGE_BAND]);
//备注:不同的报表有可能有不同的边框,如封面表就是无边框的
mergeBand[JV.PROP_LEFT] = [];
mergeBand[JV.PROP_RIGHT] = [];
mergeBand[JV.PROP_TOP] = [];
mergeBand[JV.PROP_BOTTOM] = [];
newPagePos.push(mergeBand);
mergeBand[JV.PROP_LEFT].push(pageDataArray[i][JV.BAND_PROP_MERGE_BAND][JV.PROP_LEFT]);
mergeBand[JV.PROP_RIGHT].push(pageDataArray[i][JV.BAND_PROP_MERGE_BAND][JV.PROP_RIGHT]);
sheetNames.push(pageDataArray[i][JV.NODE_PAGE_INFO][JV.NODE_MAIN_INFO_RPT_NAME]);
for (let j = 0; j < pageDataArray[i].items.length; j++) {
let maxY = 0, minY = 100000;
if (pageDataArray[i].items[j][JV.PAGE_SPECIAL_MERGE_POS]) {
let pos = pageDataArray[i].items[j][JV.PAGE_SPECIAL_MERGE_POS][JV.PROP_TOP][0] + offsetY;
mergeBand[JV.PROP_TOP].push(pos);
pos = pageDataArray[i].items[j][JV.PAGE_SPECIAL_MERGE_POS][JV.PROP_BOTTOM][0] + offsetY;
mergeBand[JV.PROP_BOTTOM].push(pos);
} else if (pageDataArray[i].items[j][JV.PROP_PAGE_MERGE_BORDER]) {
let pos = pageDataArray[i].items[j][JV.PROP_PAGE_MERGE_BORDER][JV.PROP_TOP] + offsetY;
mergeBand[JV.PROP_TOP].push(pos);
pos = pageDataArray[i].items[j][JV.PROP_PAGE_MERGE_BORDER][JV.PROP_BOTTOM] + offsetY;
mergeBand[JV.PROP_BOTTOM].push(pos);
} else {
mergeBand[JV.PROP_TOP].push(pageDataArray[i][JV.BAND_PROP_MERGE_BAND][JV.PROP_TOP] + offsetY);
mergeBand[JV.PROP_BOTTOM].push(pageDataArray[i][JV.BAND_PROP_MERGE_BAND][JV.PROP_BOTTOM] + offsetY);
}
for (let k = 0; k < pageDataArray[i].items[j].cells.length; k++) {
if (maxY < pageDataArray[i].items[j].cells[k][JV.PROP_AREA][JV.PROP_BOTTOM]) {
maxY = pageDataArray[i].items[j].cells[k][JV.PROP_AREA][JV.PROP_BOTTOM];
}
if (minY > pageDataArray[i].items[j].cells[k][JV.PROP_AREA][JV.PROP_TOP]) {
minY = pageDataArray[i].items[j].cells[k][JV.PROP_AREA][JV.PROP_TOP];
}
pageDataArray[i].items[j].cells[k][JV.PROP_AREA][JV.PROP_BOTTOM] += offsetY;
pageDataArray[i].items[j].cells[k][JV.PROP_AREA][JV.PROP_TOP] += offsetY;
}
let bottomGap = Math.round( (pageDataArray[i][JV.NODE_PAGE_INFO][JV.NODE_PAGE_SIZE][1] - parseFloat(pageDataArray[i][JV.NODE_PAGE_INFO][JV.NODE_MARGINS][JV.PROP_BOTTOM]) / 2.54 ) * DPI) - maxY;
offsetY += (maxY - minY);
if (bottomGap > 10) {
offsetY += (bottomGap - 10);
}
}
}
//2. newPageData的items属性
newPageData.items = [];
for (let i = 0; i < pageDataArray.length; i++) {
let pageItem = {};
pageItem[JV.PROP_PAGE_SEQ] = i + 1;
pageItem[JV.PROP_CELLS] = [];
for (let j = 0; j < pageDataArray[i].items.length; j++) {
for (let k = 0; k < pageDataArray[i].items[j].cells.length; k++) {
pageItem[JV.PROP_CELLS].push(pageDataArray[i].items[j].cells[k]);
}
}
newPagePos[i][JV.NODE_PAGE_SIZE] = pageDataArray[i][JV.NODE_PAGE_INFO][JV.NODE_PAGE_SIZE];
pageItem[JV.PAGE_SPECIAL_MERGE_POS] = newPagePos[i];
newPageData.items.push(pageItem);
}
//3. everything is ok, then call me
me.exportExcel(newPageData, paperSize, fName, 'false', sheetNames, custMergeBands, callback);
// fsUtil.writeObjToFile(newPageData, 'D:/GitHome/ConstructionOperation/tmp/combinedHeader.js');
} catch (e) {
console.log(e);
}
}
}