pptxgenjs
Version:
JavaScript library that creates PowerPoint presentations
785 lines (708 loc) • 287 kB
JavaScript
/*\
|*| :: 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' : "•",
'CHECK' : "✓",
'STAR' : "★",
'TRIANGLE': "▶"
};
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