UNPKG

pptxgenjs

Version:

JavaScript library that creates PowerPoint presentations

785 lines (708 loc) 287 kB
/*\ |*| :: pptxgen.js :: |*| |*| JavaScript framework that creates PowerPoint (pptx) presentations |*| https://github.com/gitbrent/PptxGenJS |*| |*| This framework is released under the MIT Public License (MIT) |*| |*| PptxGenJS (C) 2015-2018 Brent Ely -- https://github.com/gitbrent |*| |*| Some code derived from the OfficeGen project: |*| github.com/Ziv-Barber/officegen/ (Copyright 2013 Ziv Barber) |*| |*| Permission is hereby granted, free of charge, to any person obtaining a copy |*| of this software and associated documentation files (the "Software"), to deal |*| in the Software without restriction, including without limitation the rights |*| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |*| copies of the Software, and to permit persons to whom the Software is |*| furnished to do so, subject to the following conditions: |*| |*| The above copyright notice and this permission notice shall be included in all |*| copies or substantial portions of the Software. |*| |*| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |*| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |*| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |*| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |*| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |*| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |*| SOFTWARE. \*/ /* PPTX Units are "DXA" (except for font sizing) ....: There are 1440 DXA per inch. 1 inch is 72 points. 1 DXA is 1/20th's of a point (20 DXA is 1 point). ....: There is also something called EMU's (914400 EMUs is 1 inch, 12700 EMUs is 1pt). SEE: https://startbigthinksmall.wordpress.com/2010/01/04/points-inches-and-emus-measuring-units-in-office-open-xml/ | OBJECT LAYOUTS: 16x9 (10" x 5.625"), 16x10 (10" x 6.25"), 4x3 (10" x 7.5"), Wide (13.33" x 7.5") and Custom (any size) | REFS: * "Structure of a PresentationML document (Open XML SDK)" * @see: https://msdn.microsoft.com/en-us/library/office/gg278335.aspx * TableStyleId enumeration * @see: https://msdn.microsoft.com/en-us/library/office/hh273476(v=office.14).aspx */ // Polyfill for IE11 (https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isInteger) Number.isInteger = Number.isInteger || function(value) { return typeof value === "number" && isFinite(value) && Math.floor(value) === value; }; // Detect Node.js (NODEJS is ultimately used to determine how to save: either `fs` or web-based, so using fs-detection is perfect) var NODEJS = false; { // NOTE: `NODEJS` determines which network library to use, so using fs-detection is apropos. if ( typeof module !== 'undefined' && module.exports && typeof require === 'function' ) { try { require.resolve('fs'); NODEJS = true; } catch (ex) { NODEJS = false; } } } // [Node.js] <script> includes if ( NODEJS ) { var gObjPptxColors = require('../dist/pptxgen.colors.js'); var gObjPptxShapes = require('../dist/pptxgen.shapes.js'); } var PptxGenJS = function(){ // APP var APP_VER = "2.3.0"; var APP_BLD = "20180912"; // CONSTANTS var MASTER_OBJECTS = { 'chart' : { name:'chart' }, 'image' : { name:'image' }, 'line' : { name:'line' }, 'rect' : { name:'rect' }, 'text' : { name:'text' }, 'placeholder': { name:'placeholder' } }; var LAYOUTS = { 'LAYOUT_4x3' : { name: 'screen4x3', width: 9144000, height: 6858000 }, 'LAYOUT_16x9' : { name: 'screen16x9', width: 9144000, height: 5143500 }, 'LAYOUT_16x10': { name: 'screen16x10', width: 9144000, height: 5715000 }, 'LAYOUT_WIDE' : { name: 'custom', width: 12192000, height: 6858000 }, 'LAYOUT_USER' : { name: 'custom', width: 12192000, height: 6858000 } }; var BASE_SHAPES = { 'RECTANGLE': { 'displayName': 'Rectangle', 'name': 'rect', 'avLst': {} }, 'LINE' : { 'displayName': 'Line', 'name': 'line', 'avLst': {} } }; var PLACEHOLDER_TYPES = { 'title': { name: 'title' }, 'body' : { name: 'body' }, 'image': { name: 'pic' }, 'chart': { name: 'chart' }, 'table': { name: 'tbl' }, 'media': { name: 'media' } }; // NOTE: 20170304: BULLET_TYPES: Only default is used so far. I'd like to combine the two pieces of code that use these before implementing these as options // Since we close <p> within the text object bullets, its slightly more difficult than combining into a func and calling to get the paraProp // and i'm not sure if anyone will even use these... so, skipping for now. var BULLET_TYPES = { 'DEFAULT' : "&#x2022;", 'CHECK' : "&#x2713;", 'STAR' : "&#x2605;", 'TRIANGLE': "&#x25B6;" }; var CHART_TYPES = { 'AREA' : { 'displayName':'Area Chart', 'name':'area' }, 'BAR' : { 'displayName':'Bar Chart' , 'name':'bar' }, 'BUBBLE' : { 'displayName':'Bubble Chart', 'name':'bubble' }, 'DOUGHNUT': { 'displayName':'Doughnut Chart', 'name':'doughnut' }, 'LINE' : { 'displayName':'Line Chart', 'name':'line' }, 'PIE' : { 'displayName':'Pie Chart' , 'name':'pie' }, 'RADAR' : { 'displayName':'Radar Chart', 'name':'radar' }, 'SCATTER' : { 'displayName':'Scatter Chart', 'name':'scatter' } }; var PIECHART_COLORS = ['5DA5DA','FAA43A','60BD68','F17CB0','B2912F','B276B2','DECF3F','F15854','A7A7A7', '5DA5DA','FAA43A','60BD68','F17CB0','B2912F','B276B2','DECF3F','F15854','A7A7A7']; var BARCHART_COLORS = ['C0504D','4F81BD','9BBB59','8064A2','4BACC6','F79646','628FC6','C86360', 'C0504D','4F81BD','9BBB59','8064A2','4BACC6','F79646','628FC6','C86360']; // IMAGES (base64) { var IMG_BROKEN = ''; var IMG_PLAYBTN = ''; } // var CRLF = '\r\n'; // AKA: Chr(13) & Chr(10) var EMU = 914400; // One (1) inch (OfficeXML measures in EMU (English Metric Units)) var ONEPT = 12700; // One (1) point (pt) var LAYOUT_IDX_SERIES_BASE = 2147483649; var LETTERS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'.split(''); var JSZIP_OUTPUT_TYPES = ['arraybuffer', 'base64', 'binarystring', 'blob', 'nodebuffer', 'uint8array']; /** @see https://stuk.github.io/jszip/documentation/api_jszip/generate_async.html */ var REGEX_HEX_COLOR = /^[0-9a-fA-F]{6}$/; var SLDNUMFLDID = '{F7021451-1387-4CA6-816F-3879F97B5CBC}'; // var DEF_CELL_BORDER = { color:"666666" }; var DEF_CELL_MARGIN_PT = [3, 3, 3, 3]; // TRBL-style var DEF_FONT_COLOR = '000000'; var DEF_FONT_SIZE = 12; var DEF_FONT_TITLE_SIZE = 18; var DEF_SLIDE_BKGD = 'FFFFFF'; var DEF_SLIDE_MARGIN_IN = [0.5, 0.5, 0.5, 0.5]; // TRBL-style var DEF_CHART_GRIDLINE = { color: "888888", style: "solid", size: 1 }; var DEF_SHAPE_SHADOW = { type: 'outer', blur: 3, offset: (23000 / 12700), angle: 90, color: '000000', opacity: 0.35, rotateWithShape: true }; var DEF_TEXT_SHADOW = { type: 'outer', blur: 8, offset: 4, angle: 270, color: '000000', opacity: 0.75 }; var AXIS_ID_VALUE_PRIMARY = '2094734552'; var AXIS_ID_VALUE_SECONDARY = '2094734553'; var AXIS_ID_CATEGORY_PRIMARY = '2094734554'; var AXIS_ID_CATEGORY_SECONDARY = '2094734555'; // A: Create internal pptx object var gObjPptx = {}; // B: Set Presentation Property Defaults { // Presentation props/metadata gObjPptx.author = 'PptxGenJS'; gObjPptx.company = 'PptxGenJS'; gObjPptx.revision = '1'; gObjPptx.subject = 'PptxGenJS Presentation'; gObjPptx.title = 'PptxGenJS Presentation'; // PptxGenJS props gObjPptx.isBrowser = false; gObjPptx.fileName = 'Presentation'; gObjPptx.fileExtn = '.pptx'; gObjPptx.pptLayout = LAYOUTS['LAYOUT_16x9']; gObjPptx.rtlMode = false; gObjPptx.saveCallback = null; // PptxGenJS data /** @type {object} master slide layout object */ gObjPptx.masterSlide = { slide: {}, data : [], rels : [], slideNumberObj: null }; /** @type {Number} global counter for included charts (used for index in their filenames) */ gObjPptx.chartCounter = 0; /** @type {Number} global counter for included images (used for index in their filenames) */ gObjPptx.imageCounter = 0; /** @type {object[]} this Presentation's Slide objects */ gObjPptx.slides = []; /** @type {object[]} slide layout definition objects, used for generating slide layout files */ gObjPptx.slideLayouts = [{ name : 'BLANK', slide: {}, data : [], rels : [], margin: DEF_SLIDE_MARGIN_IN, slideNumberObj: null }]; } // C: Expose shape library to clients this.charts = CHART_TYPES; this.colors = ( typeof gObjPptxColors !== 'undefined' ? gObjPptxColors : {} ); this.shapes = ( typeof gObjPptxShapes !== 'undefined' ? gObjPptxShapes : BASE_SHAPES ); // Declare only after `this.colors` is initialized var SCHEME_COLOR_NAMES = Object.keys(this.colors).map(function(clrKey){return this.colors[clrKey]}.bind(this)); // D: Fall back to base shapes if shapes file was not linked gObjPptxShapes = ( gObjPptxShapes || this.shapes ); /* =============================================================================================== | ##### # # ###### # # ###### ##### ## ##### #### ##### #### # # ## # # # # # # # # # # # # # #### ##### # # # ##### # # # # # # # # # #### # # # # # # # ##### ###### # # # ##### # # # # # ## # # # # # # # # # # # # ##### ###### # # ###### # # # # # #### # # #### | ================================================================================================== */ var gObjPptxGenerators = { /** * Adds a background image or color to a slide definition. * @param {String|Object} bkg color string or an object with image definition * @param {Object} target slide object that the background is set to */ addBackgroundDefinition: function addBackgroundDefinition(bkg, target) { if ( typeof bkg === 'object' && (bkg.src || bkg.path || bkg.data) ) { // Allow the use of only the data key (`path` isnt reqd) bkg.src = bkg.src || bkg.path || null; if (!bkg.src) bkg.src = 'preencoded.png'; var targetRels = target.rels; var strImgExtn = bkg.src.split('.').pop() || 'png'; if ( strImgExtn == 'jpg' ) strImgExtn = 'jpeg'; // base64-encoded jpg's come out as "[...]", so correct exttnesion to avoid content warnings at PPT startup var intRels = targetRels.length + 1; targetRels.push({ path: bkg.src, type: 'image/'+strImgExtn, extn: strImgExtn, data: (bkg.data || ''), rId: intRels, Target: '../media/image' + (++gObjPptx.imageCounter) + '.' + strImgExtn }); target.slide.bkgdImgRid = intRels; } else if ( bkg && typeof bkg === 'string' ) { target.slide.back = bkg; } }, /** * Adds a text object to a slide definition. * @param {String} text * @param {Object} opt * @param {Object} target slide object that the text should be added to * @since: 1.0.0 */ addTextDefinition: function addTextDefinition(text, opt, target, isPlaceholder) { var opt = ( opt && typeof opt === 'object' ? opt : {} ); var resultObject = {}; var text = ( text || '' ); if ( Array.isArray(text) && text.length == 0 ) text = ''; // STEP 2: Set some options // Placeholders should inherit their colors or override them, so don't default them if ( !opt.placeholder ) { opt.color = ( opt.color || target.slide.color || DEF_FONT_COLOR ); // Set color (options > inherit from Slide > default to black) } // ROBUST: Convert attr values that will likely be passed by users to valid OOXML values if ( opt.valign ) opt.valign = opt.valign.toLowerCase().replace(/^c.*/i,'ctr').replace(/^m.*/i,'ctr').replace(/^t.*/i,'t').replace(/^b.*/i,'b'); if ( opt.align ) opt.align = opt.align.toLowerCase().replace(/^c.*/i,'center').replace(/^m.*/i,'center').replace(/^l.*/i,'left').replace(/^r.*/i,'right'); // ROBUST: Set rational values for some shadow props if needed correctShadowOptions(opt.shadow); // STEP 3: Set props resultObject.type = isPlaceholder ? 'placeholder' : 'text'; resultObject.text = text; // STEP 4: Set options resultObject.options = opt; if ( opt.shape && opt.shape.name == 'line' ) { opt.line = (opt.line || '333333' ); opt.lineSize = (opt.lineSize || 1 ); } resultObject.options.bodyProp = {}; resultObject.options.bodyProp.autoFit = (opt.autoFit || false); // If true, shape will collapse to text size (Fit To Shape) resultObject.options.bodyProp.anchor = (opt.valign || (!opt.placeholder ? 'ctr' : null)); // VALS: [t,ctr,b] resultObject.options.bodyProp.rot = (opt.rotate || null); // VALS: degree * 60,000 resultObject.options.bodyProp.vert = (opt.vert || null); // VALS: [eaVert,horz,mongolianVert,vert,vert270,wordArtVert,wordArtVertRtl] resultObject.options.lineSpacing = (opt.lineSpacing && !isNaN(opt.lineSpacing) ? opt.lineSpacing : null); if ( (opt.inset && !isNaN(Number(opt.inset))) || opt.inset == 0 ) { resultObject.options.bodyProp.lIns = inch2Emu(opt.inset); resultObject.options.bodyProp.rIns = inch2Emu(opt.inset); resultObject.options.bodyProp.tIns = inch2Emu(opt.inset); resultObject.options.bodyProp.bIns = inch2Emu(opt.inset); } target.data.push(resultObject); createHyperlinkRels(text || '', target.rels); return resultObject; }, /** * Adds Notes to a slide. * @param {String} notes * @param {Object} opt (*unused*) * @param {Object} target slide object * @since 2.3.0 */ addNotesDefinition: function addNotesDefinition(notes, opt, target) { var opt = ( opt && typeof opt === 'object' ? opt : {} ); var resultObject = {}; resultObject.type = 'notes'; resultObject.text = notes; target.data.push(resultObject); return resultObject; }, /** * Adds a placeholder object to a slide definition. * @param {String} text * @param {Object} opt * @param {Object} target slide object that the placeholder should be added to */ addPlaceholderDefinition: function addPlaceholderDefinition(text, opt, target) { return gObjPptxGenerators.addTextDefinition(text, opt, target, true); }, /** * Adds a shape object to a slide definition. * @param {Object} shape shape const object (pptx.shapes) * @param {Object} opt * @param {Object} target slide object that the shape should be added to * @return {Object} shape object */ addShapeDefinition: function addShapeDefinition(shape, opt, target) { var resultObject = {}; var options = ( typeof opt === 'object' ? opt : {} ); if ( !shape || typeof shape !== 'object' ) { console.error("Missing/Invalid shape parameter! Example: `addShape(pptx.shapes.LINE, {x:1, y:1, w:1, h:1});` "); return; } resultObject.type = 'text'; resultObject.options = options; options.shape = shape; options.x = ( options.x || (options.x == 0 ? 0 : 1) ); options.y = ( options.y || (options.y == 0 ? 0 : 1) ); options.w = ( options.w || (options.w == 0 ? 0 : 1) ); options.h = ( options.h || (options.h == 0 ? 0 : 1) ); options.line = ( options.line || (shape.name == 'line' ? '333333' : null) ); options.lineSize = ( options.lineSize || (shape.name == 'line' ? 1 : null) ); if ( ['dash','dashDot','lgDash','lgDashDot','lgDashDotDot','solid','sysDash','sysDot'].indexOf(options.lineDash || '') < 0 ) options.lineDash = 'solid'; target.data.push(resultObject); return resultObject; }, /** * Adds an image object to a slide definition. * This method can be called with only two args (opt, target) - this is supposed to be the only way in future. * @param {Object} objImage - object containing `path`/`data`, `x`, `y`, etc. * @param {Object} target - slide that the image should be added to (if not specified as the 2nd arg) * @return {Object} image object */ addImageDefinition: function addImageDefinition(objImage, target) { var resultObject = {}; // FIRST: Set vars for this image (object param replaces positional args in 1.1.0) var intPosX = (objImage.x || 0); var intPosY = (objImage.y || 0); var intWidth = (objImage.w || 0); var intHeight = (objImage.h || 0); var sizing = objImage.sizing || null; var objHyperlink = (objImage.hyperlink || ''); var strImageData = (objImage.data || ''); var strImagePath = (objImage.path || ''); var imageRelId = target.rels.length + 1; // REALITY-CHECK: if ( !strImagePath && !strImageData ) { console.error("ERROR: `addImage()` requires either 'data' or 'path' parameter!"); return null; } else if ( strImageData && strImageData.toLowerCase().indexOf('base64,') == -1 ) { console.error("ERROR: Image `data` value lacks a base64 header! Ex: 'image/png;base64,NMP[...]')"); return null; } // STEP 1: Set extension var strImgExtn = strImagePath.split('.').pop() || 'png'; // However, pre-encoded images can be whatever mime-type they want (and good for them!) if ( strImageData && /image\/(\w+)\;/.exec(strImageData) && /image\/(\w+)\;/.exec(strImageData).length > 0 ) { strImgExtn = /image\/(\w+)\;/.exec(strImageData)[1]; } // STEP 2: Set type/path resultObject.type = 'image'; resultObject.image = (strImagePath || 'preencoded.png'); // STEP 3: Set image properties & options // FIXME: Measure actual image when no intWidth/intHeight params passed // ....: This is an async process: we need to make getSizeFromImage use callback, then set H/W... // if ( !intWidth || !intHeight ) { var imgObj = getSizeFromImage(strImagePath); var imgObj = { width:1, height:1 }; resultObject.options = { x: (intPosX || 0), y: (intPosY || 0), cx: (intWidth || imgObj.width), cy: (intHeight || imgObj.height), rounding: (objImage.rounding || false), sizing: sizing, placeholder: objImage.placeholder }; // STEP 4: Add this image to this Slide Rels (rId/rels count spans all slides! Count all images to get next rId) target.rels.push({ path: (strImagePath || 'preencoded'+strImgExtn), type: 'image/'+strImgExtn, extn: strImgExtn, data: (strImageData || ''), rId: imageRelId, Target: '../media/image'+ (++gObjPptx.imageCounter) +'.'+ strImgExtn }); resultObject.imageRid = imageRelId; // STEP 5: (Issue#77) Hyperlink support if ( typeof objHyperlink === 'object' ) { if ( !objHyperlink.url && !objHyperlink.slide ) console.log("ERROR: 'hyperlink requires either: `url` or `slide`'"); else { var intRelId = imageRelId + 1; target.rels.push({ type: 'hyperlink', data: (objHyperlink.slide ? 'slide' : 'dummy'), rId: intRelId, Target: objHyperlink.url || objHyperlink.slide }); objHyperlink.rId = intRelId; resultObject.hyperlink = objHyperlink; } } target.data.push(resultObject); return resultObject; }, /** * Generate the chart based on input data. * OOXML Chart Spec: ISO/IEC 29500-1:2016(E) * * @param {object} type should belong to: 'column', 'pie' * @param {object} data a JSON object with follow the following format * @param {object} opt * @param {object} target slide object that the chart should be added to * @return {Object} chart object * { * title: 'eSurvey chart', * data: [ * { * name: 'Income', * labels: ['2005', '2006', '2007', '2008', '2009'], * values: [23.5, 26.2, 30.1, 29.5, 24.6] * }, * { * name: 'Expense', * labels: ['2005', '2006', '2007', '2008', '2009'], * values: [18.1, 22.8, 23.9, 25.1, 25] * } * ] * } */ addChartDefinition: function addChartDefinition(type, data, opt, target) { var targetRels = target.rels; var chartId = (++gObjPptx.chartCounter); var chartRelId = target.rels.length + 1; var resultObject = {}; // DESIGN: `type` can an object (ex: `pptx.charts.DOUGHNUT`) or an array of chart objects // EX: addChartDefinition([ { type:pptx.charts.BAR, data:{name:'', labels:[], values[]} }, {<etc>} ]) // Multi-Type Charts var tmpOpt; var tmpData = [], options; if (Array.isArray(type)) { // For multi-type charts there needs to be data for each type, // as well as a single data source for non-series operations. // The data is indexed below to keep the data in order when segmented // into types. type.forEach(function(obj) { tmpData = tmpData.concat(obj.data) }); tmpOpt = data || opt; } else { tmpData = data; tmpOpt = opt; } tmpData.forEach(function(item,i){ item.index = i; }); options = ( tmpOpt && typeof tmpOpt === 'object' ? tmpOpt : {} ); // STEP 1: TODO: check for reqd fields, correct type, etc // `type` exists in CHART_TYPES // Array.isArray(data) /* if ( Array.isArray(rel.data) && rel.data.length > 0 && typeof rel.data[0] === 'object' && rel.data[0].labels && Array.isArray(rel.data[0].labels) && rel.data[0].values && Array.isArray(rel.data[0].values) ) { obj = rel.data[0]; } else { console.warn("USAGE: addChart( 'pie', [ {name:'Sales', labels:['Jan','Feb'], values:[10,20]} ], {x:1, y:1} )"); return; } */ // STEP 2: Set default options/decode user options // A: Core options.type = type; options.x = (typeof options.x !== 'undefined' && options.x != null && !isNaN(options.x) ? options.x : 1); options.y = (typeof options.y !== 'undefined' && options.y != null && !isNaN(options.y) ? options.y : 1); options.w = (options.w || '50%'); options.h = (options.h || '50%'); // B: Options: misc if ( ['bar','col'].indexOf(options.barDir || '') < 0 ) options.barDir = 'col'; // IMPORTANT: 'bestFit' will cause issues with PPT-Online in some cases, so defualt to 'ctr'! if ( ['bestFit','b','ctr','inBase','inEnd','l','outEnd','r','t'].indexOf(options.dataLabelPosition || '') < 0 ) options.dataLabelPosition = (options.type.name == 'pie' || options.type.name == 'doughnut' ? 'bestFit' : 'ctr'); if ( ['b','l','r','t','tr'].indexOf(options.legendPos || '') < 0 ) options.legendPos = 'r'; // barGrouping: "21.2.3.17 ST_Grouping (Grouping)" if ( ['clustered','standard','stacked','percentStacked'].indexOf(options.barGrouping || '') < 0 ) options.barGrouping = 'standard'; if ( options.barGrouping.indexOf('tacked') > -1 ) { options.dataLabelPosition = 'ctr'; // IMPORTANT: PPT-Online will not open Presentation when 'outEnd' etc is used on stacked! if (!options.barGapWidthPct) options.barGapWidthPct = 50; } // lineDataSymbol: http://www.datypic.com/sc/ooxml/a-val-32.html // Spec has [plus,star,x] however neither PPT2013 nor PPT-Online support them if ( ['circle','dash','diamond','dot','none','square','triangle'].indexOf(options.lineDataSymbol || '') < 0 ) options.lineDataSymbol = 'circle'; if ( ['gap','span'].indexOf(options.displayBlanksAs || '') < 0 ) options.displayBlanksAs = 'span'; if ( ['standard','marker','filled'].indexOf(options.radarStyle || '') < 0 ) options.radarStyle = 'standard'; options.lineDataSymbolSize = ( options.lineDataSymbolSize && !isNaN(options.lineDataSymbolSize ) ? options.lineDataSymbolSize : 6 ); options.lineDataSymbolLineSize = ( options.lineDataSymbolLineSize && !isNaN(options.lineDataSymbolLineSize ) ? options.lineDataSymbolLineSize * ONEPT : 0.75 * ONEPT ); // `layout` allows the override of PPT defaults to maximize space if ( options.layout ) { ['x', 'y', 'w', 'h'].forEach(function(key) { var val = options.layout[key]; if (isNaN(Number(val)) || val < 0 || val > 1) { console.warn('Warning: chart.layout.' + key + ' can only be 0-1'); delete options.layout[key]; // remove invalid value so that default will be used } }); } // Set gridline defaults options.catGridLine = options.catGridLine || (type.name == 'scatter' ? { color:'D9D9D9', pt:1 } : 'none'); options.valGridLine = options.valGridLine || (type.name == 'scatter' ? { color:'D9D9D9', pt:1 } : {}); correctGridLineOptions(options.catGridLine); correctGridLineOptions(options.valGridLine); correctShadowOptions(options.shadow); // C: Options: plotArea options.showDataTable = (options.showDataTable == true || options.showDataTable == false ? options.showDataTable : false); options.showDataTableHorzBorder = (options.showDataTableHorzBorder == true || options.showDataTableHorzBorder == false ? options.showDataTableHorzBorder : true ); options.showDataTableVertBorder = (options.showDataTableVertBorder == true || options.showDataTableVertBorder == false ? options.showDataTableVertBorder : true ); options.showDataTableOutline = (options.showDataTableOutline == true || options.showDataTableOutline == false ? options.showDataTableOutline : true ); options.showDataTableKeys = (options.showDataTableKeys == true || options.showDataTableKeys == false ? options.showDataTableKeys : true ); options.showLabel = (options.showLabel == true || options.showLabel == false ? options.showLabel : false); options.showLegend = (options.showLegend == true || options.showLegend == false ? options.showLegend : false); options.showPercent = (options.showPercent == true || options.showPercent == false ? options.showPercent : true ); options.showTitle = (options.showTitle == true || options.showTitle == false ? options.showTitle : false); options.showValue = (options.showValue == true || options.showValue == false ? options.showValue : false); options.catAxisLineShow = (typeof options.catAxisLineShow !== 'undefined' ? options.catAxisLineShow : true); options.valAxisLineShow = (typeof options.valAxisLineShow !== 'undefined' ? options.valAxisLineShow : true); // D: Options: chart options.barGapWidthPct = (!isNaN(options.barGapWidthPct) && options.barGapWidthPct >= 0 && options.barGapWidthPct <= 1000 ? options.barGapWidthPct : 150); options.chartColors = ( Array.isArray(options.chartColors) ? options.chartColors : (options.type.name == 'pie' || options.type.name == 'doughnut' ? PIECHART_COLORS : BARCHART_COLORS) ); options.chartColorsOpacity = ( options.chartColorsOpacity && !isNaN(options.chartColorsOpacity) ? options.chartColorsOpacity : null ); // options.border = ( options.border && typeof options.border === 'object' ? options.border : null ); if ( options.border && (!options.border.pt || isNaN(options.border.pt)) ) options.border.pt = 1; if ( options.border && (!options.border.color || typeof options.border.color !== 'string' || options.border.color.length != 6) ) options.border.color = '363636'; // options.dataBorder = ( options.dataBorder && typeof options.dataBorder === 'object' ? options.dataBorder : null ); if ( options.dataBorder && (!options.dataBorder.pt || isNaN(options.dataBorder.pt)) ) options.dataBorder.pt = 0.75; if ( options.dataBorder && (!options.dataBorder.color || typeof options.dataBorder.color !== 'string' || options.dataBorder.color.length != 6) ) options.dataBorder.color = 'F9F9F9'; // if ( !options.dataLabelFormatCode && options.type.name === 'scatter' ) options.dataLabelFormatCode = "General"; options.dataLabelFormatCode = options.dataLabelFormatCode && typeof options.dataLabelFormatCode === 'string' ? options.dataLabelFormatCode : (options.type.name == 'pie' || options.type.name == 'doughnut') ? '0%' : '#,##0'; // options.lineSize = ( typeof options.lineSize === 'number' ? options.lineSize : 2 ); options.valAxisMajorUnit = ( typeof options.valAxisMajorUnit === 'number' ? options.valAxisMajorUnit : null ); // STEP 4: Set props resultObject.type = 'chart'; resultObject.options = options; // STEP 5: Add this chart to this Slide Rels (rId/rels count spans all slides! Count all images to get next rId) targetRels.push({ rId: chartRelId, data: tmpData, opts: options, type: 'chart', globalId: chartId, fileName:'chart'+ chartId +'.xml', Target: '/ppt/charts/chart'+ chartId +'.xml' }); resultObject.chartRid = chartRelId; target.data.push(resultObject); return resultObject; }, /* ===== */ /** * Transforms a slide definition to a slide object that is then passed to the XML transformation process. * The following object is expected as a slide definition: * { * bkgd: 'FF00FF', * objects: [{ * text: { * text: 'Hello World', * x: 1, * y: 1 * } * }] * } * @param {Object} slideDef slide definition * @param {Object} target empty slide object that should be updated by the passed definition */ createSlideObject: function createSlideObject(slideDef, target) { // STEP 1: Add background if ( slideDef.bkgd ) { gObjPptxGenerators.addBackgroundDefinition(slideDef.bkgd, target); } // STEP 2: Add all Slide Master objects in the order they were given (Issue#53) if ( slideDef.objects && Array.isArray(slideDef.objects) && slideDef.objects.length > 0 ) { slideDef.objects.forEach(function(object,idx){ var key = Object.keys(object)[0]; if ( MASTER_OBJECTS[key] && key == 'chart' ) gObjPptxGenerators.addChartDefinition(CHART_TYPES[(object.chart.type||'').toUpperCase()], object.chart.data, object.chart.opts, target); else if ( MASTER_OBJECTS[key] && key == 'image' ) gObjPptxGenerators.addImageDefinition(object[key], target); else if ( MASTER_OBJECTS[key] && key == 'line' ) gObjPptxGenerators.addShapeDefinition(gObjPptxShapes.LINE, object[key], target); else if ( MASTER_OBJECTS[key] && key == 'rect' ) gObjPptxGenerators.addShapeDefinition(gObjPptxShapes.RECTANGLE, object[key], target); else if ( MASTER_OBJECTS[key] && key == 'text' ) gObjPptxGenerators.addTextDefinition(object[key].text, object[key].options, target, false); else if ( MASTER_OBJECTS[key] && key == 'placeholder' ) { // TODO: 20180820: Check for existing `name`? object[key].options.placeholderName = object[key].options.name; delete object[key].options.name; // remap name for earier handling internally object[key].options.placeholderType = object[key].options.type; delete object[key].options.type; // remap name for earier handling internally object[key].options.placeholderIdx = (100+idx); gObjPptxGenerators.addPlaceholderDefinition(object[key].text, object[key].options, target); } }); } // STEP 3: Add Slide Numbers (NOTE: Do this last so numbers are not covered by objects!) if ( slideDef.slideNumber && typeof slideDef.slideNumber === 'object' ) { target.slideNumberObj = slideDef.slideNumber; }; }, /** * Transforms a slide object to resulting XML string. * @param {Object} slideObject slide object created within gObjPptxGenerators.createSlideObject * @return {String} XML string with <p:cSld> as the root */ slideObjectToXml: function slideObjectToXml(slideObject) { var strSlideXml = slideObject.name ? '<p:cSld name="'+ slideObject.name +'">' : '<p:cSld>'; var intTableNum = 1; // STEP 1: Add background if ( slideObject.slide.back ) { strSlideXml += genXmlColorSelection(false, slideObject.slide.back); } // STEP 2: Add background image (using Strech) (if any) if ( slideObject.slide.bkgdImgRid ) { // FIXME: We should be doing this in the slideLayout... strSlideXml += '<p:bg>' + '<p:bgPr><a:blipFill dpi="0" rotWithShape="1">' + '<a:blip r:embed="rId'+ slideObject.slide.bkgdImgRid +'"><a:lum/></a:blip>' + '<a:srcRect/><a:stretch><a:fillRect/></a:stretch></a:blipFill>' + '<a:effectLst/></p:bgPr>' + '</p:bg>'; } // STEP 3: Continue slide by starting spTree node strSlideXml += '<p:spTree>'; strSlideXml += '<p:nvGrpSpPr><p:cNvPr id="1" name=""/><p:cNvGrpSpPr/><p:nvPr/></p:nvGrpSpPr>'; strSlideXml += '<p:grpSpPr><a:xfrm><a:off x="0" y="0"/><a:ext cx="0" cy="0"/>'; strSlideXml += '<a:chOff x="0" y="0"/><a:chExt cx="0" cy="0"/></a:xfrm></p:grpSpPr>'; // STEP 4: Loop over all Slide.data objects and add them to this slide =============================== slideObject.data.forEach(function(slideItemObj, idx) { var x = 0, y = 0, cx = getSmartParseNumber('75%','X'), cy = 0, placeholderObj; var locationAttr = "", shapeType = null; if ( slideObject.layoutObj && slideObject.layoutObj.data && slideItemObj.options && slideItemObj.options.placeholder ) { placeholderObj = slideObject.layoutObj.data.filter(function(layoutObj){ return layoutObj.options.placeholderName == slideItemObj.options.placeholder})[0]; } // A: Set option vars slideItemObj.options = slideItemObj.options || {}; if ( slideItemObj.options.w || slideItemObj.options.w == 0 ) slideItemObj.options.cx = slideItemObj.options.w; if ( slideItemObj.options.h || slideItemObj.options.h == 0 ) slideItemObj.options.cy = slideItemObj.options.h; // if ( slideItemObj.options.x || slideItemObj.options.x == 0 ) x = getSmartParseNumber( slideItemObj.options.x , 'X' ); if ( slideItemObj.options.y || slideItemObj.options.y == 0 ) y = getSmartParseNumber( slideItemObj.options.y , 'Y' ); if ( slideItemObj.options.cx || slideItemObj.options.cx == 0 ) cx = getSmartParseNumber( slideItemObj.options.cx, 'X' ); if ( slideItemObj.options.cy || slideItemObj.options.cy == 0 ) cy = getSmartParseNumber( slideItemObj.options.cy, 'Y' ); // If using a placeholder then inherit it's position if ( placeholderObj ) { if ( placeholderObj.options.x || placeholderObj.options.x == 0 ) x = getSmartParseNumber( placeholderObj.options.x , 'X' ); if ( placeholderObj.options.y || placeholderObj.options.y == 0 ) y = getSmartParseNumber( placeholderObj.options.y , 'Y' ); if ( placeholderObj.options.cx || placeholderObj.options.cx == 0 ) cx = getSmartParseNumber( placeholderObj.options.cx, 'X' ); if ( placeholderObj.options.cy || placeholderObj.options.cy == 0 ) cy = getSmartParseNumber( placeholderObj.options.cy, 'Y' ); } // if ( slideItemObj.options.shape ) shapeType = getShapeInfo( slideItemObj.options.shape ); // if ( slideItemObj.options.flipH ) locationAttr += ' flipH="1"'; if ( slideItemObj.options.flipV ) locationAttr += ' flipV="1"'; if ( slideItemObj.options.rotate ) locationAttr += ' rot="' + convertRotationDegrees(slideItemObj.options.rotate)+ '"'; // B: Add OBJECT to current Slide ---------------------------- switch ( slideItemObj.type ) { case 'table': // FIRST: Ensure we have rows - otherwise, bail! if ( !slideItemObj.arrTabRows || (Array.isArray(slideItemObj.arrTabRows) && slideItemObj.arrTabRows.length == 0) ) break; // Set table vars var objTableGrid = {}; var arrTabRows = slideItemObj.arrTabRows; var objTabOpts = slideItemObj.options; var intColCnt = 0, intColW = 0; // Calc number of columns // NOTE: Cells may have a colspan, so merely taking the length of the [0] (or any other) row is not // ....: sufficient to determine column count. Therefore, check each cell for a colspan and total cols as reqd arrTabRows[0].forEach(function(cell,idx){ var cellOpts = cell.options || cell.opts || null; intColCnt += ( cellOpts && cellOpts.colspan ? Number(cellOpts.colspan) : 1 ); }); // STEP 1: Start Table XML ============================= // NOTE: Non-numeric cNvPr id values will trigger "presentation needs repair" type warning in MS-PPT-2013 var strXml = '<p:graphicFrame>' + ' <p:nvGraphicFramePr>' + ' <p:cNvPr id="'+ (intTableNum*slideObject.numb + 1) +'" name="Table '+ (intTableNum*slideObject.numb) +'"/>' + ' <p:cNvGraphicFramePr><a:graphicFrameLocks noGrp="1"/></p:cNvGraphicFramePr>' + ' <p:nvPr><p:extLst><p:ext uri="{D42A27DB-BD31-4B8C-83A1-F6EECF244321}"><p14:modId xmlns:p14="http://schemas.microsoft.com/office/powerpoint/2010/main" val="1579011935"/></p:ext></p:extLst></p:nvPr>' + ' </p:nvGraphicFramePr>' + ' <p:xfrm>' + ' <a:off x="'+ (x || EMU) +'" y="'+ (y || EMU) +'"/>' + ' <a:ext cx="'+ (cx || EMU) +'" cy="'+ (cy || EMU) +'"/>' + ' </p:xfrm>' + ' <a:graphic>' + ' <a:graphicData uri="http://schemas.openxmlformats.org/drawingml/2006/table">' + ' <a:tbl>' + ' <a:tblPr/>'; // + ' <a:tblPr bandRow="1"/>'; // FIXME: Support banded rows, first/last row, etc. // NOTE: Bandi