/**
* 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 + '">' + '' + '' + borderDirection.toLowerCase() + '>');
}
};
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(sheetBorder[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(sheetBorder[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(sheetBorder[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(sheetBorder[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 - 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('');
};
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 - 2, 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 - 3);
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);
}
}