/** * Created by Tony on 2017/4/1. */ var JV = require('../rpt_component/Jpc_ValueDefine'); var fs = require('fs'); var JSZip = require("jszip"); var strUtil = require('../../../public/stringUtil'); var jpcCmnHelper = require('../rpt_component/helper/Jpc_Helper_Common'); var DPI = jpcCmnHelper.getScreenDPI()[0]; const dftHeadXml = ''; function writeContentTypes(sheets) { var rst = []; rst.push(dftHeadXml + '\r\n'); rst.push(''); //... rst.push(''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); for (var i = 0; i < sheets.length; i++) { rst.push('') } rst.push(''); rst.push(''); return rst; } function writeRootRels(){ var rst = []; rst.push(dftHeadXml + '\r\n'); rst.push(''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); return rst; } function writeApp(sheets) { var rst = []; rst.push(dftHeadXml + '\r\n'); rst.push(''); rst.push('Microsoft Excel'); rst.push('0'); rst.push('false'); rst.push(''); rst.push(''); rst.push('工作表'); rst.push('' + sheets.length + ''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); for (var 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(''); rst.push(''); return rst; } function writeCore() { var rst = []; p_fillZero = function(val){ var rst = val; if (val < 10) { rst = '0' + val; } return rst; }; rst.push(dftHeadXml + '\r\n'); rst.push(''); rst.push('SmartCost'); rst.push('SmartCost'); var dt = new Date(), 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){ var rst = []; rst.push(dftHeadXml + '\r\n'); rst.push(''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); for (var i = 0; i < sheets.length; i++) { rst.push('') } rst.push(''); rst.push(''); //rst.push(''); rst.push(''); return rst; } function writeXlRels(sheets){ var rst = [], idx = 1; rst.push(dftHeadXml + '\r\n'); rst.push(''); for (var i = 0; i < sheets.length; i++) { rst.push('') idx++; } rst.push(''); idx++; rst.push(''); idx++; rst.push(''); //rst.push(''); rst.push(''); return rst; } function writeTheme(){ var rst = fs.readFileSync(__dirname + '/excel_base_files/theme1.xml', 'utf8', 'r'); return rst; } function writeStyles(stylesObj){ var rst = []; rst.push(dftHeadXml + '\r\n'); rst.push(''); //1. push fonts rst.push('') for (var i = 0; i < stylesObj.fonts.length; i++) { var font = stylesObj.fonts[i]; rst.push(''); if (strUtil.convertStrToBoolean(font[JV.FONT_PROPS[3]])) { 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('') private_setBorder = function(border, borderDirection) { if (border[borderDirection][JV.PROP_LINE_WEIGHT] == 0) { rst.push('<' + borderDirection.toLowerCase() + '/>'); } else { var bW = 'thin'; if (border[borderDirection][JV.PROP_LINE_WEIGHT] = 2) bW = 'medium'; if (border[borderDirection][JV.PROP_LINE_WEIGHT] > 2) bW = 'thick'; rst.push('<' + borderDirection.toLowerCase() + ' style="' + bW + '">' + '' + ''); } }; for (var i = 0; i < stylesObj.borders.length; i++) { var 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 (var i = 0; i < stylesObj.cellXfs.length; i++) { var excelStyle = stylesObj.cellXfs[i]; rst.push(''); var 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){ var rst = []; if (sharedStrList && sharedStrList.length > 0) { rst.push(dftHeadXml + '\r\n'); rst.push(''); for (var i = 0; i < sharedStrList.length; i++) { rst.push('' + sharedStrList[i] + ''); } rst.push(''); } return rst; } function writeSheets(pageData, sharedStrList, stylesObj){ var rst = []; private_pushDftFont = function(){ var font = {}; if (!(stylesObj.fonts)) { stylesObj.fonts = []; } font[JV.FONT_PROPS[0]] = "宋体"; //font name font.size = 11; font.charset = 134; font.colorIdx = "8"; stylesObj.fonts.push(font); }; private_pushDftFont(); for (var i = 0; i < pageData.items.length; i++) { rst.push(writeSheet(pageData, pageData.items[i], sharedStrList, stylesObj)); } return rst; } function writeSheet(pageData, sheetData, sharedStrList, stylesObj){ var rst = [], xPos = [], yPos = [], headerStr = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'; xPos.push(0); yPos.push(0); private_pre_analyze_pos = function(){ var cell, pos; sheetData.cells.sort(function(cell1, cell2) { var 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 (var i = 0; i < sheetData.cells.length; i++) { cell = sheetData.cells[i]; pos = cell[JV.PROP_AREA][JV.PROP_LEFT]; if (xPos.indexOf(pos) < 0) xPos.push(pos); pos = cell[JV.PROP_AREA][JV.PROP_RIGHT]; if (xPos.indexOf(pos) < 0) xPos.push(pos); pos = cell[JV.PROP_AREA][JV.PROP_TOP]; if (yPos.indexOf(pos) < 0) yPos.push(pos); pos = cell[JV.PROP_AREA][JV.PROP_BOTTOM]; if (yPos.indexOf(pos) < 0) yPos.push(pos); } xPos.sort(private_array_sort); yPos.sort(private_array_sort); }; private_array_sort = function(i1, i2){ var rst = 0; if (i1 > i2) {rst = 1} else if (i1 < i2) rst = -1; return rst; }; private_getCellIdxStr = function(idx){ var rst = 'A'; if (idx < 26) { rst = headerStr[idx]; } else if (idx < 26*26+26) { var ti = Math.floor(idx / 26), tj = idx % 26; rst = headerStr[ti - 1] + headerStr[tj]; } else if (idx < 26*26*26+26) { var 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; }; private_getSharedStrIdx = function(val) { var rst = sharedStrList.indexOf(val); if (rst < 0) { sharedStrList.push(val); rst = sharedStrList.length - 1; } return rst; }; private_getFontId = function(cell) { var rst = 0, hasFont = false; if (!(stylesObj.fonts)) { stylesObj.fonts = []; //for (var i = 0; i < sheetData.font_collection) } var sheetFont = pageData.font_collection[cell.font]; for (var i = 0; i < stylesObj.fonts.length; i++) { var 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) && font[JV.FONT_PROPS[3]] == sheetFont[JV.FONT_PROPS[3]]) { hasFont = true; rst = i; break; } } else { break; } } if (!hasFont) { var 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.charset = 134; font.colorIdx = "8"; font[JV.FONT_PROPS[3]] = sheetFont[JV.FONT_PROPS[3]]; //font bold stylesObj.fonts.push(font); rst = stylesObj.fonts.length - 1; } return rst; }; private_checkBorder = function(border, sheetBorder) { var 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(parseInt(border[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT])); } else { sheetBorderLineWidths.push(0); } if (sheetBorder[JV.PROP_RIGHT] && sheetBorder[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT]) { sheetBorderLineWidths.push(parseInt(border[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT])); } else { sheetBorderLineWidths.push(0); } if (sheetBorder[JV.PROP_TOP] && sheetBorder[JV.PROP_TOP][JV.PROP_LINE_WEIGHT]) { sheetBorderLineWidths.push(parseInt(border[JV.PROP_TOP][JV.PROP_LINE_WEIGHT])); } else { sheetBorderLineWidths.push(0); } if (sheetBorder[JV.PROP_BOTTOM] && sheetBorder[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT]) { sheetBorderLineWidths.push(parseInt(border[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT])); } else { sheetBorderLineWidths.push(0); } for (var i = 0; i < 4; i++) { if (borderLineWidths[i] != sheetBorderLineWidths[i]) { rst = false; break; } } return rst; }; private_getBorderId = function(cell) { var rst = 0, hasBorder = false; if (!(stylesObj.borders)) { stylesObj.borders = []; } var sheetBorder = pageData.style_collection[cell.style]; for (var i = 0; i < stylesObj.borders.length; i++) { var border = stylesObj.borders[i]; if (private_checkBorder(border, sheetBorder)) { hasBorder = true; rst = i; break; } } if (!hasBorder) { var 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; if (sheetBorder && sheetBorder[JV.PROP_LEFT]) { border[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT] = parseInt(sheetBorder[JV.PROP_LEFT][JV.PROP_LINE_WEIGHT]); } if (sheetBorder && sheetBorder[JV.PROP_RIGHT]) { border[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT] = parseInt(sheetBorder[JV.PROP_RIGHT][JV.PROP_LINE_WEIGHT]); } if (sheetBorder && sheetBorder[JV.PROP_TOP]) { border[JV.PROP_TOP][JV.PROP_LINE_WEIGHT] = parseInt(sheetBorder[JV.PROP_TOP][JV.PROP_LINE_WEIGHT]); } if (sheetBorder && sheetBorder[JV.PROP_BOTTOM]) { border[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT] = parseInt(sheetBorder[JV.PROP_BOTTOM][JV.PROP_LINE_WEIGHT]); } stylesObj.borders.push(border); rst = stylesObj.borders.length - 1; } return rst; }; private_checkControl = function(cellControl, sheetControl) { var rst = true; for (var i = 0; i < JV.CONTROL_PROPS.length; i++) { if (cellControl[JV.CONTROL_PROPS[i]] != sheetControl[JV.CONTROL_PROPS[i]]) { rst = false; break; } } return rst; }; private_getStyleId = function(cell) { var rst = 1, hasStyle = false; if (!(stylesObj.cellXfs)) stylesObj.cellXfs = []; var fontId = private_getFontId(cell); var borderId = private_getBorderId(cell); var cellControl = pageData[JV.NODE_CONTROL_COLLECTION][cell[JV.PROP_CONTROL]]; for (var i = 0; i < stylesObj.cellXfs.length; i++) { var sheetControl = stylesObj.cellXfs[i]; if (sheetControl.fontId == fontId && sheetControl.borderId == borderId) { if (private_checkControl(cellControl, sheetControl)) { rst = i; hasStyle = true; break; } } } if (!hasStyle) { var sheetControl = {}; sheetControl.fontId = fontId; sheetControl.borderId = borderId; for (var 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; }; private_setCols = function(){ //remark: 1 excel width = 2.117 mm rst.push(''); var w = 0; for (var i = 1; i < xPos.length; i++) { w = 1.0 * (xPos[i] - xPos[i - 1]) / DPI * 25.4 / 2.117; w = Math.round(w * 1000) / 1000; rst.push(''); } rst.push(''); }; private_setMergedCells = function() { var cell, idxR, idxL, idxT, idxB, cnt = 0; rst.push(''); var startIdx = rst.length - 1; for (var i = 0; i < sheetData.cells.length; i++) { cell = sheetData.cells[i]; idxR = xPos.indexOf(cell[JV.PROP_AREA][JV.PROP_RIGHT]); idxL = xPos.indexOf(cell[JV.PROP_AREA][JV.PROP_LEFT]); idxB = yPos.indexOf(cell[JV.PROP_AREA][JV.PROP_BOTTOM]); idxT = yPos.indexOf(cell[JV.PROP_AREA][JV.PROP_TOP]); if (idxR - idxL > 1 || idxB - idxT > 1) { rst.push(''); cnt++; } } rst[startIdx] = ''; rst.push(''); }; private_setSheetData = function(){ //remark: 1 excel height = 0.3612 mm rst.push(''); var spanX = xPos.length - 1, cellIdx = 0, h = 0, hasMoreCols = true, nextColIdx = -1, nextRowIdx = yPos.indexOf(sheetData.cells[cellIdx][JV.PROP_AREA][JV.PROP_TOP]); for (var i = 1; i < yPos.length - 1; i++) { h = 1.0 * (yPos[i+1] - yPos[i]) / DPI * 25.4 / 0.3612; h = Math.round(h * 1000) / 1000; rst.push(''); //then put the cells of this row var colIdxStr = ''; hasMoreCols = true; while (nextRowIdx < i) { if (cellIdx >= sheetData.cells.length || nextRowIdx > i) { break; } else { cellIdx++; nextRowIdx = yPos.indexOf(sheetData.cells[cellIdx][JV.PROP_AREA][JV.PROP_TOP]); } } if (nextRowIdx > i) { hasMoreCols = false; } nextColIdx = xPos.indexOf(sheetData.cells[cellIdx][JV.PROP_AREA][JV.PROP_LEFT]); for (var j = 1; j < xPos.length; j++) { colIdxStr = private_getCellIdxStr(j - 1); if (hasMoreCols) { if (nextColIdx == j) { var styleIdx = private_getStyleId(sheetData.cells[cellIdx]); if (strUtil.isEmptyString(sheetData.cells[cellIdx][JV.PROP_VALUE])) { rst.push(''); //should setup the right style instead! } else { var valIdx = private_getSharedStrIdx(sheetData.cells[cellIdx][JV.PROP_VALUE]); rst.push(''); rst.push('' + valIdx + ''); rst.push(''); } cellIdx++; if (cellIdx < sheetData.cells.length) { nextRowIdx = yPos.indexOf(sheetData.cells[cellIdx][JV.PROP_AREA][JV.PROP_TOP]); if (nextRowIdx > i) { hasMoreCols = false; } else { nextColIdx = xPos.indexOf(sheetData.cells[cellIdx][JV.PROP_AREA][JV.PROP_LEFT]); } } else { hasMoreCols = false; } } else if (nextColIdx < 0) { //impossible! console.log('has abnormal case!'); hasMoreCols = false; } else { rst.push(''); } } else { rst.push(''); } } rst.push(''); } //sheetData.cells.length rst.push(''); }; private_pre_analyze_pos(); rst.push(dftHeadXml + '\r\n'); rst.push(''); var colStr = private_getCellIdxStr(xPos.length - 2); rst.push(''); rst.push(''); rst.push(''); rst.push(''); rst.push(''); private_setCols(); private_setSheetData(); private_setMergedCells(); rst.push(''); rst.push(''); //rst.push(''); rst.push(''); rst.push(''); //rst.push(''); return rst; } module.exports = { exportExcel: function (pageData, options) { var rptOptions = (options || {singlePage: false, fileName: 'report'}); var sheets = []; for (var i = 0; i < pageData.items.length; i++) { sheets.push({sheetName: '第' + (i + 1) + '页'}); } //1. var file = '[Content_Types].xml'; var data = writeContentTypes(sheets); var zip = new JSZip(); zip.file(file, data.join(''), {compression: 'DEFLATE'}); //2. var zip_rels = zip.folder('_rels'); file = '.rels'; data = writeRootRels(); zip_rels.file(file, data.join(''), {compression: 'DEFLATE'}); //3. var zip_docProps = zip.folder('docProps'); file = 'app.xml'; data = writeApp(sheets); zip_docProps.file(file, data.join(''), {compression: 'DEFLATE'}); file = 'core.xml'; data = writeCore(); zip_docProps.file(file, data.join(''), {compression: 'DEFLATE'}); //4. var zip_xl = zip.folder('xl'); file = 'workbook.xml'; data = writeXlWorkBook(sheets); zip_xl.file(file, data.join(''), {compression: 'DEFLATE'}); var zip_rels2 = zip_xl.folder('_rels'); file = 'workbook.xml.rels'; data = writeXlRels(sheets); zip_rels2.file(file, data.join(''), {compression: 'DEFLATE'}); //5. var zip_theme = zip_xl.folder('theme'); file = 'theme1.xml'; data = writeTheme(); zip_theme.file(file, data, {compression: 'DEFLATE'}); //6. var zip_worksheets = zip_xl.folder('worksheets'); var sharedStrList = [], stylesObj = {}; data = writeSheets(pageData, sharedStrList, stylesObj); for (var 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'}); zip.generateNodeStream({type:'nodebuffer',streamFiles:true}) //.pipe(fs.createWriteStream('../../../tmp/outExcel.xlsx')) .pipe(fs.createWriteStream('../../../tmp/outExcel.zip')) .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("outExcel.xlsx was written."); } ); } ,testWriteContentTypes: function(sheets) { return writeContentTypes(sheets); } ,testWriteRootRels: function() { return writeRootRels(); } ,testWriteApp: function(sheets) { return writeApp(sheets); } ,testWriteCore: function() { return writeCore(); } ,testWriteXlWorkBook: function(sheets) { return writeXlWorkBook(sheets); } ,testWriteXlRels: function(sheets) { return writeXlRels(sheets); } ,testWriteSheets: function(pageData, sharedStrList, stylesObj){ return writeSheets(pageData, sharedStrList, stylesObj); } ,testWriteSharedString: function(sharedStrList){ return writeSharedString(sharedStrList); } ,testWriteTheme: function() { return writeTheme(); } ,testWriteStyles: function(stylesObj) { return writeStyles(stylesObj); } }