2014-06-08 17:40:59 +01:00
|
|
|
/*
|
|
|
|
* libxlsxwriter
|
2014-06-26 00:49:41 +01:00
|
|
|
*
|
2024-02-26 18:47:32 +00:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2025-02-11 00:03:36 +00:00
|
|
|
* Copyright 2014-2025, John McNamara, jmcnamara@cpan.org.
|
2014-06-08 17:40:59 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @page workbook_page The Workbook object
|
|
|
|
*
|
|
|
|
* The Workbook is the main object exposed by the libxlsxwriter library. It
|
|
|
|
* represents the entire spreadsheet as you see it in Excel and internally it
|
|
|
|
* represents the Excel file as it is written on disk.
|
|
|
|
*
|
|
|
|
* See @ref workbook.h for full details of the functionality.
|
2014-06-26 00:49:41 +01:00
|
|
|
*
|
2014-06-08 17:40:59 +01:00
|
|
|
* @file workbook.h
|
|
|
|
*
|
|
|
|
* @brief Functions related to creating an Excel xlsx workbook.
|
|
|
|
*
|
|
|
|
* The Workbook is the main object exposed by the libxlsxwriter library. It
|
|
|
|
* represents the entire spreadsheet as you see it in Excel and internally it
|
|
|
|
* represents the Excel file as it is written on disk.
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* #include "xlsxwriter.h"
|
|
|
|
*
|
|
|
|
* int main() {
|
|
|
|
*
|
2016-01-04 19:48:16 +00:00
|
|
|
* lxw_workbook *workbook = workbook_new("filename.xlsx");
|
2014-06-08 17:40:59 +01:00
|
|
|
* lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL);
|
|
|
|
*
|
|
|
|
* worksheet_write_string(worksheet, 0, 0, "Hello Excel", NULL);
|
|
|
|
*
|
|
|
|
* return workbook_close(workbook);
|
|
|
|
* }
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* @image html workbook01.png
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#ifndef __LXW_WORKBOOK_H__
|
|
|
|
#define __LXW_WORKBOOK_H__
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2015-04-13 00:24:32 +01:00
|
|
|
#include <stdio.h>
|
2015-11-07 23:43:26 +00:00
|
|
|
#include <errno.h>
|
2014-06-08 17:40:59 +01:00
|
|
|
|
|
|
|
#include "worksheet.h"
|
2018-09-07 11:51:03 +01:00
|
|
|
#include "chartsheet.h"
|
2016-05-02 21:43:37 +01:00
|
|
|
#include "chart.h"
|
2014-06-08 17:40:59 +01:00
|
|
|
#include "shared_strings.h"
|
|
|
|
#include "hash_table.h"
|
|
|
|
#include "common.h"
|
|
|
|
|
2016-05-05 00:20:04 +01:00
|
|
|
#define LXW_DEFINED_NAME_LENGTH 128
|
|
|
|
|
|
|
|
/* Define the tree.h RB structs for the red-black head types. */
|
|
|
|
RB_HEAD(lxw_worksheet_names, lxw_worksheet_name);
|
2018-09-08 23:36:09 +01:00
|
|
|
RB_HEAD(lxw_chartsheet_names, lxw_chartsheet_name);
|
2019-12-24 19:16:13 +00:00
|
|
|
RB_HEAD(lxw_image_md5s, lxw_image_md5);
|
2016-05-05 00:20:04 +01:00
|
|
|
|
2015-03-08 23:25:43 +00:00
|
|
|
/* Define the queue.h structs for the workbook lists. */
|
2018-09-07 11:51:03 +01:00
|
|
|
STAILQ_HEAD(lxw_sheets, lxw_sheet);
|
2014-06-08 17:40:59 +01:00
|
|
|
STAILQ_HEAD(lxw_worksheets, lxw_worksheet);
|
2018-09-08 23:36:09 +01:00
|
|
|
STAILQ_HEAD(lxw_chartsheets, lxw_chartsheet);
|
2016-05-02 21:43:37 +01:00
|
|
|
STAILQ_HEAD(lxw_charts, lxw_chart);
|
2015-04-06 23:46:39 +01:00
|
|
|
TAILQ_HEAD(lxw_defined_names, lxw_defined_name);
|
2015-03-08 23:25:43 +00:00
|
|
|
|
2019-11-16 20:32:37 +00:00
|
|
|
/* Struct to hold the 2 sheet types. */
|
2018-09-07 11:51:03 +01:00
|
|
|
typedef struct lxw_sheet {
|
|
|
|
uint8_t is_chartsheet;
|
|
|
|
|
|
|
|
union {
|
|
|
|
lxw_worksheet *worksheet;
|
|
|
|
lxw_chartsheet *chartsheet;
|
|
|
|
} u;
|
|
|
|
|
|
|
|
STAILQ_ENTRY (lxw_sheet) list_pointers;
|
|
|
|
} lxw_sheet;
|
|
|
|
|
2016-05-05 00:20:04 +01:00
|
|
|
/* Struct to represent a worksheet name/pointer pair. */
|
|
|
|
typedef struct lxw_worksheet_name {
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name;
|
2016-05-05 00:20:04 +01:00
|
|
|
lxw_worksheet *worksheet;
|
|
|
|
|
|
|
|
RB_ENTRY (lxw_worksheet_name) tree_pointers;
|
|
|
|
} lxw_worksheet_name;
|
|
|
|
|
2018-09-08 23:36:09 +01:00
|
|
|
/* Struct to represent a chartsheet name/pointer pair. */
|
|
|
|
typedef struct lxw_chartsheet_name {
|
|
|
|
const char *name;
|
|
|
|
lxw_chartsheet *chartsheet;
|
|
|
|
|
|
|
|
RB_ENTRY (lxw_chartsheet_name) tree_pointers;
|
|
|
|
} lxw_chartsheet_name;
|
|
|
|
|
2019-12-24 19:16:13 +00:00
|
|
|
/* Struct to represent an image MD5/ID pair. */
|
|
|
|
typedef struct lxw_image_md5 {
|
|
|
|
uint32_t id;
|
2019-12-26 15:52:01 +00:00
|
|
|
char *md5;
|
2019-12-24 19:16:13 +00:00
|
|
|
|
|
|
|
RB_ENTRY (lxw_image_md5) tree_pointers;
|
|
|
|
} lxw_image_md5;
|
|
|
|
|
2016-05-05 00:20:04 +01:00
|
|
|
/* Wrapper around RB_GENERATE_STATIC from tree.h to avoid unused function
|
|
|
|
* warnings and to avoid portability issues with the _unused attribute. */
|
2018-09-08 23:36:09 +01:00
|
|
|
#define LXW_RB_GENERATE_WORKSHEET_NAMES(name, type, field, cmp) \
|
|
|
|
RB_GENERATE_INSERT_COLOR(name, type, field, static) \
|
|
|
|
RB_GENERATE_REMOVE_COLOR(name, type, field, static) \
|
|
|
|
RB_GENERATE_INSERT(name, type, field, cmp, static) \
|
|
|
|
RB_GENERATE_REMOVE(name, type, field, static) \
|
|
|
|
RB_GENERATE_FIND(name, type, field, cmp, static) \
|
|
|
|
RB_GENERATE_NEXT(name, type, field, static) \
|
|
|
|
RB_GENERATE_MINMAX(name, type, field, static) \
|
|
|
|
/* Add unused struct to allow adding a semicolon */ \
|
|
|
|
struct lxw_rb_generate_worksheet_names{int unused;}
|
|
|
|
|
|
|
|
#define LXW_RB_GENERATE_CHARTSHEET_NAMES(name, type, field, cmp) \
|
|
|
|
RB_GENERATE_INSERT_COLOR(name, type, field, static) \
|
|
|
|
RB_GENERATE_REMOVE_COLOR(name, type, field, static) \
|
|
|
|
RB_GENERATE_INSERT(name, type, field, cmp, static) \
|
|
|
|
RB_GENERATE_REMOVE(name, type, field, static) \
|
|
|
|
RB_GENERATE_FIND(name, type, field, cmp, static) \
|
|
|
|
RB_GENERATE_NEXT(name, type, field, static) \
|
|
|
|
RB_GENERATE_MINMAX(name, type, field, static) \
|
|
|
|
/* Add unused struct to allow adding a semicolon */ \
|
|
|
|
struct lxw_rb_generate_charsheet_names{int unused;}
|
2015-03-08 23:25:43 +00:00
|
|
|
|
2019-12-24 19:16:13 +00:00
|
|
|
#define LXW_RB_GENERATE_IMAGE_MD5S(name, type, field, cmp) \
|
|
|
|
RB_GENERATE_INSERT_COLOR(name, type, field, static) \
|
|
|
|
RB_GENERATE_REMOVE_COLOR(name, type, field, static) \
|
|
|
|
RB_GENERATE_INSERT(name, type, field, cmp, static) \
|
|
|
|
RB_GENERATE_REMOVE(name, type, field, static) \
|
|
|
|
RB_GENERATE_FIND(name, type, field, cmp, static) \
|
|
|
|
RB_GENERATE_NEXT(name, type, field, static) \
|
|
|
|
RB_GENERATE_MINMAX(name, type, field, static) \
|
|
|
|
/* Add unused struct to allow adding a semicolon */ \
|
|
|
|
struct lxw_rb_generate_image_md5s{int unused;}
|
|
|
|
|
2015-04-14 16:11:47 +01:00
|
|
|
/**
|
|
|
|
* @brief Macro to loop over all the worksheets in a workbook.
|
|
|
|
*
|
|
|
|
* This macro allows you to loop over all the worksheets that have been
|
|
|
|
* added to a workbook. You must provide a lxw_worksheet pointer and
|
|
|
|
* a pointer to the lxw_workbook:
|
|
|
|
*
|
|
|
|
* @code
|
2016-01-04 19:48:16 +00:00
|
|
|
* lxw_workbook *workbook = workbook_new("test.xlsx");
|
2015-04-14 16:11:47 +01:00
|
|
|
*
|
|
|
|
* lxw_worksheet *worksheet; // Generic worksheet pointer.
|
|
|
|
*
|
|
|
|
* // Worksheet objects used in the program.
|
|
|
|
* lxw_worksheet *worksheet1 = workbook_add_worksheet(workbook, NULL);
|
|
|
|
* lxw_worksheet *worksheet2 = workbook_add_worksheet(workbook, NULL);
|
|
|
|
* lxw_worksheet *worksheet3 = workbook_add_worksheet(workbook, NULL);
|
|
|
|
*
|
|
|
|
* // Iterate over the 3 worksheets and perform the same operation on each.
|
2015-04-15 00:49:18 +01:00
|
|
|
* LXW_FOREACH_WORKSHEET(worksheet, workbook) {
|
2015-04-14 16:11:47 +01:00
|
|
|
* worksheet_write_string(worksheet, 0, 0, "Hello", NULL);
|
|
|
|
* }
|
|
|
|
* @endcode
|
|
|
|
*/
|
|
|
|
#define LXW_FOREACH_WORKSHEET(worksheet, workbook) \
|
|
|
|
STAILQ_FOREACH((worksheet), (workbook)->worksheets, list_pointers)
|
|
|
|
|
2015-03-08 23:25:43 +00:00
|
|
|
/* Struct to represent a defined name. */
|
|
|
|
typedef struct lxw_defined_name {
|
|
|
|
int16_t index;
|
|
|
|
uint8_t hidden;
|
|
|
|
char name[LXW_DEFINED_NAME_LENGTH];
|
2015-04-12 23:53:03 +01:00
|
|
|
char app_name[LXW_DEFINED_NAME_LENGTH];
|
2015-04-18 14:06:14 +01:00
|
|
|
char formula[LXW_DEFINED_NAME_LENGTH];
|
2015-04-06 23:46:39 +01:00
|
|
|
char normalised_name[LXW_DEFINED_NAME_LENGTH];
|
|
|
|
char normalised_sheetname[LXW_DEFINED_NAME_LENGTH];
|
2015-03-08 23:25:43 +00:00
|
|
|
|
|
|
|
/* List pointers for queue.h. */
|
2015-04-06 23:46:39 +01:00
|
|
|
TAILQ_ENTRY (lxw_defined_name) list_pointers;
|
2015-03-08 23:25:43 +00:00
|
|
|
} lxw_defined_name;
|
2014-06-08 17:40:59 +01:00
|
|
|
|
2015-12-13 22:49:13 +00:00
|
|
|
/**
|
2019-11-16 17:45:30 +00:00
|
|
|
* Workbook document properties. Set any unused fields to NULL or 0.
|
2015-12-13 22:49:13 +00:00
|
|
|
*/
|
|
|
|
typedef struct lxw_doc_properties {
|
|
|
|
/** The title of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *title;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The subject of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *subject;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The author of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *author;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The manager field of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *manager;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The company field of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *company;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The category of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *category;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The keywords of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *keywords;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The comment field of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *comments;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
|
|
|
/** The status of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *status;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
2019-11-16 17:45:30 +00:00
|
|
|
/** The hyperlink base URL of the Excel Document. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *hyperlink_base;
|
2015-12-13 22:49:13 +00:00
|
|
|
|
2019-11-16 17:45:30 +00:00
|
|
|
/** The file creation date/time shown in Excel. This defaults to the
|
|
|
|
* current time and date if set to 0. If you wish to create files that are
|
|
|
|
* binary equivalent (for the same input data) then you should set this
|
|
|
|
* creation date/time to a known value. */
|
2015-12-13 22:49:13 +00:00
|
|
|
time_t created;
|
|
|
|
|
|
|
|
} lxw_doc_properties;
|
|
|
|
|
2014-06-26 00:49:41 +01:00
|
|
|
/**
|
|
|
|
* @brief Workbook options.
|
|
|
|
*
|
2015-04-08 00:25:17 +01:00
|
|
|
* Optional parameters when creating a new Workbook object via
|
2016-01-04 19:48:16 +00:00
|
|
|
* workbook_new_opt().
|
2014-06-26 00:49:41 +01:00
|
|
|
*
|
2016-07-10 23:12:01 +01:00
|
|
|
* The following properties are supported:
|
|
|
|
*
|
2019-06-08 16:20:04 +01:00
|
|
|
* - `constant_memory`: This option reduces the amount of data stored in
|
|
|
|
* memory so that large files can be written efficiently. This option is off
|
2021-03-23 19:15:54 +00:00
|
|
|
* by default. See the notes below for limitations when this mode is on.
|
2016-07-10 23:12:01 +01:00
|
|
|
*
|
2019-06-08 16:20:04 +01:00
|
|
|
* - `tmpdir`: libxlsxwriter stores workbook data in temporary files prior to
|
|
|
|
* assembling the final XLSX file. The temporary files are created in the
|
2016-07-10 23:12:01 +01:00
|
|
|
* system's temp directory. If the default temporary directory isn't
|
|
|
|
* accessible to your application, or doesn't contain enough space, you can
|
2018-11-10 15:01:32 +00:00
|
|
|
* specify an alternative location using the `tmpdir` option.
|
2019-06-08 16:20:04 +01:00
|
|
|
*
|
|
|
|
* - `use_zip64`: Make the zip library use ZIP64 extensions when writing very
|
|
|
|
* large xlsx files to allow the zip container, or individual XML files
|
|
|
|
* within it, to be greater than 4 GB. See [ZIP64 on Wikipedia][zip64_wiki]
|
|
|
|
* for more information. This option is off by default.
|
|
|
|
*
|
|
|
|
* [zip64_wiki]: https://en.wikipedia.org/wiki/Zip_(file_format)#ZIP64
|
2022-11-10 13:15:28 +04:00
|
|
|
|
|
|
|
* - `output_buffer`: Output to a buffer instead of a file. The buffer must be
|
|
|
|
* freed manually by calling free(). This option can only be used if filename
|
|
|
|
* is NULL.
|
|
|
|
*
|
|
|
|
* - `output_buffer_size`: Used with output_buffer to get the size of the
|
|
|
|
* created buffer. This option can only be used if filename is NULL.
|
2019-06-08 16:20:04 +01:00
|
|
|
*
|
2021-03-23 19:15:54 +00:00
|
|
|
* @note In `constant_memory` mode each row of in-memory data is written to
|
|
|
|
* disk and then freed when a new row is started via one of the
|
|
|
|
* `worksheet_write_*()` functions. Therefore, once this option is active data
|
|
|
|
* should be written in sequential row by row order. For this reason
|
|
|
|
* `worksheet_merge_range()` and some other row based functionality doesn't
|
|
|
|
* work in this mode. See @ref ww_mem_constant for more details.
|
|
|
|
*
|
|
|
|
* @note Also, in `constant_memory` mode the library uses temp file storage
|
|
|
|
* for worksheet data. This can lead to an issue on OSes that map the `/tmp`
|
|
|
|
* directory into memory since it is possible to consume the "system" memory
|
|
|
|
* even though the "process" memory remains constant. In these cases you
|
|
|
|
* should use an alternative temp file location by using the `tmpdir` option
|
|
|
|
* shown above. See @ref ww_mem_temp for more details.
|
2014-06-26 00:49:41 +01:00
|
|
|
*/
|
|
|
|
typedef struct lxw_workbook_options {
|
2019-06-08 16:20:04 +01:00
|
|
|
/** Optimize the workbook to use constant memory for worksheets. */
|
2014-06-26 00:49:41 +01:00
|
|
|
uint8_t constant_memory;
|
2016-07-10 23:12:01 +01:00
|
|
|
|
2016-07-11 21:42:18 +01:00
|
|
|
/** Directory to use for the temporary files created by libxlsxwriter. */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char *tmpdir;
|
2019-06-08 16:20:04 +01:00
|
|
|
|
|
|
|
/** Allow ZIP64 extensions when creating the xlsx file zip container. */
|
|
|
|
uint8_t use_zip64;
|
|
|
|
|
2022-11-10 13:15:28 +04:00
|
|
|
/** Output buffer to use instead of writing to a file */
|
2023-09-24 20:40:07 +01:00
|
|
|
const char **output_buffer;
|
2022-11-10 13:15:28 +04:00
|
|
|
|
|
|
|
/** Used with output_buffer to get the size of the created buffer */
|
|
|
|
size_t *output_buffer_size;
|
2014-06-26 00:49:41 +01:00
|
|
|
} lxw_workbook_options;
|
|
|
|
|
2014-06-08 17:40:59 +01:00
|
|
|
/**
|
|
|
|
* @brief Struct to represent an Excel workbook.
|
|
|
|
*
|
|
|
|
* The members of the lxw_workbook struct aren't modified directly. Instead
|
|
|
|
* the workbook properties are set by calling the functions shown in
|
|
|
|
* workbook.h.
|
|
|
|
*/
|
|
|
|
typedef struct lxw_workbook {
|
|
|
|
|
|
|
|
FILE *file;
|
2018-09-07 11:51:03 +01:00
|
|
|
struct lxw_sheets *sheets;
|
2014-06-08 17:40:59 +01:00
|
|
|
struct lxw_worksheets *worksheets;
|
2018-09-08 23:36:09 +01:00
|
|
|
struct lxw_chartsheets *chartsheets;
|
2016-05-05 00:20:04 +01:00
|
|
|
struct lxw_worksheet_names *worksheet_names;
|
2018-09-08 23:36:09 +01:00
|
|
|
struct lxw_chartsheet_names *chartsheet_names;
|
2019-12-24 19:16:13 +00:00
|
|
|
struct lxw_image_md5s *image_md5s;
|
2024-07-30 00:33:14 +01:00
|
|
|
struct lxw_image_md5s *embedded_image_md5s;
|
2020-08-16 16:03:56 +01:00
|
|
|
struct lxw_image_md5s *header_image_md5s;
|
2021-05-13 22:59:01 +01:00
|
|
|
struct lxw_image_md5s *background_md5s;
|
2016-05-02 21:43:37 +01:00
|
|
|
struct lxw_charts *charts;
|
2016-05-19 21:21:05 +01:00
|
|
|
struct lxw_charts *ordered_charts;
|
2014-06-08 17:40:59 +01:00
|
|
|
struct lxw_formats *formats;
|
2015-03-08 23:25:43 +00:00
|
|
|
struct lxw_defined_names *defined_names;
|
2014-06-08 17:40:59 +01:00
|
|
|
lxw_sst *sst;
|
|
|
|
lxw_doc_properties *properties;
|
2016-06-10 00:03:38 +01:00
|
|
|
struct lxw_custom_properties *custom_properties;
|
|
|
|
|
2015-11-08 20:52:45 +00:00
|
|
|
char *filename;
|
2014-06-26 00:49:41 +01:00
|
|
|
lxw_workbook_options options;
|
2014-06-08 17:40:59 +01:00
|
|
|
|
|
|
|
uint16_t num_sheets;
|
2018-09-07 11:51:03 +01:00
|
|
|
uint16_t num_worksheets;
|
2018-09-08 23:36:09 +01:00
|
|
|
uint16_t num_chartsheets;
|
2014-06-08 17:40:59 +01:00
|
|
|
uint16_t first_sheet;
|
2015-12-13 13:46:04 +00:00
|
|
|
uint16_t active_sheet;
|
2014-06-08 17:40:59 +01:00
|
|
|
uint16_t num_xf_formats;
|
2020-08-21 00:59:57 +01:00
|
|
|
uint16_t num_dxf_formats;
|
2014-06-08 17:40:59 +01:00
|
|
|
uint16_t num_format_count;
|
2015-12-28 11:22:17 +00:00
|
|
|
uint16_t drawing_count;
|
2019-12-27 20:10:11 +00:00
|
|
|
uint16_t comment_count;
|
2024-07-30 23:30:33 +01:00
|
|
|
uint32_t num_embedded_images;
|
2025-02-14 11:15:56 +00:00
|
|
|
uint16_t window_width;
|
|
|
|
uint16_t window_height;
|
2014-06-08 17:40:59 +01:00
|
|
|
|
|
|
|
uint16_t font_count;
|
|
|
|
uint16_t border_count;
|
|
|
|
uint16_t fill_count;
|
2014-06-25 00:59:04 +01:00
|
|
|
uint8_t optimize;
|
2019-12-15 11:48:58 +00:00
|
|
|
uint16_t max_url_length;
|
2021-03-27 20:36:13 +00:00
|
|
|
uint8_t read_only;
|
2014-06-08 17:40:59 +01:00
|
|
|
|
2015-12-30 00:35:03 +00:00
|
|
|
uint8_t has_png;
|
2015-12-31 08:28:23 +00:00
|
|
|
uint8_t has_jpeg;
|
2015-12-30 00:35:03 +00:00
|
|
|
uint8_t has_bmp;
|
2021-05-08 00:25:14 +01:00
|
|
|
uint8_t has_gif;
|
2019-12-27 20:10:11 +00:00
|
|
|
uint8_t has_vml;
|
2020-01-08 23:16:56 +00:00
|
|
|
uint8_t has_comments;
|
2021-04-23 07:25:50 +01:00
|
|
|
uint8_t has_metadata;
|
2024-07-30 00:33:14 +01:00
|
|
|
uint8_t has_embedded_images;
|
|
|
|
uint8_t has_dynamic_functions;
|
2024-07-30 23:30:33 +01:00
|
|
|
uint8_t has_embedded_image_descriptions;
|
2015-12-30 00:35:03 +00:00
|
|
|
|
2015-03-01 12:36:59 +00:00
|
|
|
lxw_hash_table *used_xf_formats;
|
2020-08-21 00:59:57 +01:00
|
|
|
lxw_hash_table *used_dxf_formats;
|
2014-06-08 17:40:59 +01:00
|
|
|
|
2019-06-16 15:46:09 +01:00
|
|
|
char *vba_project;
|
2023-09-20 20:37:38 +02:00
|
|
|
char *vba_project_signature;
|
2019-06-16 15:46:09 +01:00
|
|
|
char *vba_codename;
|
|
|
|
|
2019-12-10 00:10:13 +00:00
|
|
|
lxw_format *default_url_format;
|
|
|
|
|
2014-06-08 17:40:59 +01:00
|
|
|
} lxw_workbook;
|
|
|
|
|
|
|
|
|
|
|
|
/* *INDENT-OFF* */
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
/* *INDENT-ON* */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Create a new workbook object.
|
|
|
|
*
|
|
|
|
* @param filename The name of the new Excel file to create.
|
|
|
|
*
|
|
|
|
* @return A lxw_workbook instance.
|
|
|
|
*
|
2016-01-04 19:48:16 +00:00
|
|
|
* The `%workbook_new()` constructor is used to create a new Excel workbook
|
2014-06-08 17:40:59 +01:00
|
|
|
* with a given filename:
|
|
|
|
*
|
|
|
|
* @code
|
2016-01-04 19:48:16 +00:00
|
|
|
* lxw_workbook *workbook = workbook_new("filename.xlsx");
|
2014-06-08 17:40:59 +01:00
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* When specifying a filename it is recommended that you use an `.xlsx`
|
|
|
|
* extension or Excel will generate a warning when opening the file.
|
|
|
|
*
|
|
|
|
*/
|
2016-01-04 19:48:16 +00:00
|
|
|
lxw_workbook *workbook_new(const char *filename);
|
2014-06-08 17:40:59 +01:00
|
|
|
|
2014-06-26 00:49:41 +01:00
|
|
|
/**
|
|
|
|
* @brief Create a new workbook object, and set the workbook options.
|
|
|
|
*
|
|
|
|
* @param filename The name of the new Excel file to create.
|
|
|
|
* @param options Workbook options.
|
|
|
|
*
|
|
|
|
* @return A lxw_workbook instance.
|
|
|
|
*
|
2016-12-25 10:50:29 +00:00
|
|
|
* This function is the same as the `workbook_new()` constructor but allows
|
2014-06-26 00:49:41 +01:00
|
|
|
* additional options to be set.
|
|
|
|
*
|
|
|
|
* @code
|
2019-06-08 16:20:04 +01:00
|
|
|
* lxw_workbook_options options = {.constant_memory = LXW_TRUE,
|
|
|
|
* .tmpdir = "C:\\Temp",
|
2022-11-10 13:15:28 +04:00
|
|
|
* .use_zip64 = LXW_FALSE,
|
|
|
|
* .output_buffer = NULL,
|
|
|
|
* .output_buffer_size = NULL};
|
2014-06-26 00:49:41 +01:00
|
|
|
*
|
2016-01-04 19:48:16 +00:00
|
|
|
* lxw_workbook *workbook = workbook_new_opt("filename.xlsx", &options);
|
2014-06-26 00:49:41 +01:00
|
|
|
* @endcode
|
|
|
|
*
|
2016-07-11 21:42:18 +01:00
|
|
|
* The options that can be set via #lxw_workbook_options are:
|
|
|
|
*
|
2019-06-08 16:20:04 +01:00
|
|
|
* - `constant_memory`: This option reduces the amount of data stored in
|
|
|
|
* memory so that large files can be written efficiently. This option is off
|
|
|
|
* by default. See the note below for limitations when this mode is on.
|
2016-07-11 21:42:18 +01:00
|
|
|
*
|
2019-06-08 16:20:04 +01:00
|
|
|
* - `tmpdir`: libxlsxwriter stores workbook data in temporary files prior to
|
|
|
|
* assembling the final XLSX file. The temporary files are created in the
|
2016-07-11 21:42:18 +01:00
|
|
|
* system's temp directory. If the default temporary directory isn't
|
|
|
|
* accessible to your application, or doesn't contain enough space, you can
|
2019-06-08 16:20:04 +01:00
|
|
|
* specify an alternative location using the `tmpdir` option.
|
2014-06-26 00:49:41 +01:00
|
|
|
*
|
2019-06-08 16:20:04 +01:00
|
|
|
* - `use_zip64`: Make the zip library use ZIP64 extensions when writing very
|
|
|
|
* large xlsx files to allow the zip container, or individual XML files
|
|
|
|
* within it, to be greater than 4 GB. See [ZIP64 on Wikipedia][zip64_wiki]
|
|
|
|
* for more information. This option is off by default.
|
|
|
|
*
|
|
|
|
* [zip64_wiki]: https://en.wikipedia.org/wiki/Zip_(file_format)#ZIP64
|
|
|
|
*
|
2022-12-30 15:15:28 +00:00
|
|
|
* - `output_buffer`: Output to a memory buffer instead of a file. The buffer
|
|
|
|
* must be freed manually by calling `free()`. This option can only be used if
|
|
|
|
* filename is NULL.
|
|
|
|
*
|
|
|
|
* - `output_buffer_size`: Used with output_buffer to get the size of the
|
|
|
|
* created buffer. This option can only be used if filename is `NULL`.
|
|
|
|
*
|
2021-03-23 19:15:54 +00:00
|
|
|
* @note In `constant_memory` mode each row of in-memory data is written to
|
|
|
|
* disk and then freed when a new row is started via one of the
|
|
|
|
* `worksheet_write_*()` functions. Therefore, once this option is active data
|
|
|
|
* should be written in sequential row by row order. For this reason
|
|
|
|
* `worksheet_merge_range()` and some other row based functionality doesn't
|
|
|
|
* work in this mode. See @ref ww_mem_constant for more details.
|
|
|
|
*
|
|
|
|
* @note Also, in `constant_memory` mode the library uses temp file storage
|
|
|
|
* for worksheet data. This can lead to an issue on OSes that map the `/tmp`
|
|
|
|
* directory into memory since it is possible to consume the "system" memory
|
|
|
|
* even though the "process" memory remains constant. In these cases you
|
|
|
|
* should use an alternative temp file location by using the `tmpdir` option
|
|
|
|
* shown above. See @ref ww_mem_temp for more details.
|
2014-06-26 00:49:41 +01:00
|
|
|
*/
|
2016-01-04 19:48:16 +00:00
|
|
|
lxw_workbook *workbook_new_opt(const char *filename,
|
|
|
|
lxw_workbook_options *options);
|
|
|
|
|
2014-06-08 17:40:59 +01:00
|
|
|
/**
|
2016-12-26 12:51:13 +00:00
|
|
|
* @brief Add a new worksheet to a workbook.
|
2014-06-08 17:40:59 +01:00
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param sheetname Optional worksheet name, defaults to Sheet1, etc.
|
|
|
|
*
|
2016-05-23 00:22:26 +01:00
|
|
|
* @return A lxw_worksheet object.
|
2014-06-08 17:40:59 +01:00
|
|
|
*
|
2018-09-11 21:22:00 +01:00
|
|
|
* The `%workbook_add_worksheet()` function adds a new worksheet to a workbook.
|
2014-06-08 17:40:59 +01:00
|
|
|
*
|
|
|
|
* At least one worksheet should be added to a new workbook: The @ref
|
|
|
|
* worksheet.h "Worksheet" object is used to write data and configure a
|
|
|
|
* worksheet in the workbook.
|
|
|
|
*
|
|
|
|
* The `sheetname` parameter is optional. If it is `NULL` the default
|
|
|
|
* Excel convention will be followed, i.e. Sheet1, Sheet2, etc.:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* worksheet = workbook_add_worksheet(workbook, NULL ); // Sheet1
|
|
|
|
* worksheet = workbook_add_worksheet(workbook, "Foglio2"); // Foglio2
|
|
|
|
* worksheet = workbook_add_worksheet(workbook, "Data"); // Data
|
|
|
|
* worksheet = workbook_add_worksheet(workbook, NULL ); // Sheet4
|
|
|
|
*
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* @image html workbook02.png
|
|
|
|
*
|
2019-06-07 00:00:36 +01:00
|
|
|
* The worksheet name must be a valid Excel worksheet name, i.e:
|
2014-06-08 17:40:59 +01:00
|
|
|
*
|
2024-05-08 19:59:22 +01:00
|
|
|
* - The name cannot be blank.
|
2019-06-07 00:00:36 +01:00
|
|
|
* - The name is less than or equal to 31 UTF-8 characters.
|
|
|
|
* - The name doesn't contain any of the characters: ` [ ] : * ? / \ `
|
|
|
|
* - The name doesn't start or end with an apostrophe.
|
2019-06-07 21:28:08 +01:00
|
|
|
* - The name isn't already in use. (Case insensitive).
|
2014-06-08 17:40:59 +01:00
|
|
|
*
|
2019-06-07 00:00:36 +01:00
|
|
|
* If any of these errors are encountered the function will return NULL.
|
|
|
|
* You can check for valid name using the `workbook_validate_sheet_name()`
|
|
|
|
* function.
|
2020-03-21 17:40:08 +00:00
|
|
|
*
|
|
|
|
* @note You should also avoid using the worksheet name "History" (case
|
|
|
|
* insensitive) which is reserved in English language versions of
|
|
|
|
* Excel. Non-English versions may have restrictions on the equivalent word.
|
2014-06-08 17:40:59 +01:00
|
|
|
*/
|
|
|
|
lxw_worksheet *workbook_add_worksheet(lxw_workbook *workbook,
|
|
|
|
const char *sheetname);
|
|
|
|
|
2018-09-11 21:22:00 +01:00
|
|
|
/**
|
|
|
|
* @brief Add a new chartsheet to a workbook.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param sheetname Optional chartsheet name, defaults to Chart1, etc.
|
|
|
|
*
|
|
|
|
* @return A lxw_chartsheet object.
|
|
|
|
*
|
|
|
|
* The `%workbook_add_chartsheet()` function adds a new chartsheet to a
|
|
|
|
* workbook. The @ref chartsheet.h "Chartsheet" object is like a worksheet
|
|
|
|
* except it displays a chart instead of cell data.
|
|
|
|
*
|
|
|
|
* @image html chartsheet.png
|
|
|
|
*
|
|
|
|
* The `sheetname` parameter is optional. If it is `NULL` the default
|
|
|
|
* Excel convention will be followed, i.e. Chart1, Chart2, etc.:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* chartsheet = workbook_add_chartsheet(workbook, NULL ); // Chart1
|
|
|
|
* chartsheet = workbook_add_chartsheet(workbook, "My Chart"); // My Chart
|
|
|
|
* chartsheet = workbook_add_chartsheet(workbook, NULL ); // Chart3
|
|
|
|
*
|
|
|
|
* @endcode
|
|
|
|
*
|
2019-06-07 00:00:36 +01:00
|
|
|
* The chartsheet name must be a valid Excel worksheet name, i.e.:
|
2018-09-11 21:22:00 +01:00
|
|
|
*
|
2024-05-08 19:59:22 +01:00
|
|
|
* - The name cannot be blank.
|
2019-06-07 00:00:36 +01:00
|
|
|
* - The name is less than or equal to 31 UTF-8 characters.
|
|
|
|
* - The name doesn't contain any of the characters: ` [ ] : * ? / \ `
|
|
|
|
* - The name doesn't start or end with an apostrophe.
|
2019-06-07 21:28:08 +01:00
|
|
|
* - The name isn't already in use. (Case insensitive).
|
2018-09-11 21:22:00 +01:00
|
|
|
*
|
2019-06-07 00:00:36 +01:00
|
|
|
* If any of these errors are encountered the function will return NULL.
|
|
|
|
* You can check for valid name using the `workbook_validate_sheet_name()`
|
|
|
|
* function.
|
2018-09-11 21:22:00 +01:00
|
|
|
*
|
2020-03-21 17:40:08 +00:00
|
|
|
* @note You should also avoid using the worksheet name "History" (case
|
|
|
|
* insensitive) which is reserved in English language versions of
|
|
|
|
* Excel. Non-English versions may have restrictions on the equivalent word.
|
|
|
|
*
|
2018-09-11 21:22:00 +01:00
|
|
|
* At least one worksheet should be added to a new workbook when creating a
|
|
|
|
* chartsheet in order to provide data for the chart. The @ref worksheet.h
|
|
|
|
* "Worksheet" object is used to write data and configure a worksheet in the
|
|
|
|
* workbook.
|
|
|
|
*/
|
|
|
|
lxw_chartsheet *workbook_add_chartsheet(lxw_workbook *workbook,
|
2018-09-08 23:36:09 +01:00
|
|
|
const char *sheetname);
|
|
|
|
|
2014-06-08 17:40:59 +01:00
|
|
|
/**
|
|
|
|
* @brief Create a new @ref format.h "Format" object to formats cells in
|
|
|
|
* worksheets.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
*
|
|
|
|
* @return A lxw_format instance.
|
|
|
|
*
|
|
|
|
* The `workbook_add_format()` function can be used to create new @ref
|
|
|
|
* format.h "Format" objects which are used to apply formatting to a cell.
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* // Create the Format.
|
|
|
|
* lxw_format *format = workbook_add_format(workbook);
|
|
|
|
*
|
|
|
|
* // Set some of the format properties.
|
|
|
|
* format_set_bold(format);
|
|
|
|
* format_set_font_color(format, LXW_COLOR_RED);
|
|
|
|
*
|
|
|
|
* // Use the format to change the text format in a cell.
|
|
|
|
* worksheet_write_string(worksheet, 0, 0, "Hello", format);
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* See @ref format.h "the Format object" and @ref working_with_formats
|
|
|
|
* sections for more details about Format properties and how to set them.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
lxw_format *workbook_add_format(lxw_workbook *workbook);
|
|
|
|
|
2016-05-23 00:22:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Create a new chart to be added to a worksheet:
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
2016-12-30 23:11:30 +00:00
|
|
|
* @param chart_type The type of chart to be created. See #lxw_chart_type.
|
2016-05-23 00:22:26 +01:00
|
|
|
*
|
|
|
|
* @return A lxw_chart object.
|
|
|
|
*
|
|
|
|
* The `%workbook_add_chart()` function creates a new chart object that can
|
|
|
|
* be added to a worksheet:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* // Create a chart object.
|
|
|
|
* lxw_chart *chart = workbook_add_chart(workbook, LXW_CHART_COLUMN);
|
|
|
|
*
|
|
|
|
* // Add data series to the chart.
|
|
|
|
* chart_add_series(chart, NULL, "Sheet1!$A$1:$A$5");
|
|
|
|
* chart_add_series(chart, NULL, "Sheet1!$B$1:$B$5");
|
|
|
|
* chart_add_series(chart, NULL, "Sheet1!$C$1:$C$5");
|
|
|
|
*
|
|
|
|
* // Insert the chart into the worksheet
|
|
|
|
* worksheet_insert_chart(worksheet, CELL("B7"), chart);
|
|
|
|
* @endcode
|
|
|
|
*
|
2016-12-30 23:11:30 +00:00
|
|
|
* The available chart types are defined in #lxw_chart_type. The types of
|
2016-05-23 00:22:26 +01:00
|
|
|
* charts that are supported are:
|
|
|
|
*
|
|
|
|
* | Chart type | Description |
|
|
|
|
* | :--------------------------------------- | :------------------------------------ |
|
|
|
|
* | #LXW_CHART_AREA | Area chart. |
|
|
|
|
* | #LXW_CHART_AREA_STACKED | Area chart - stacked. |
|
|
|
|
* | #LXW_CHART_AREA_STACKED_PERCENT | Area chart - percentage stacked. |
|
|
|
|
* | #LXW_CHART_BAR | Bar chart. |
|
|
|
|
* | #LXW_CHART_BAR_STACKED | Bar chart - stacked. |
|
|
|
|
* | #LXW_CHART_BAR_STACKED_PERCENT | Bar chart - percentage stacked. |
|
|
|
|
* | #LXW_CHART_COLUMN | Column chart. |
|
|
|
|
* | #LXW_CHART_COLUMN_STACKED | Column chart - stacked. |
|
|
|
|
* | #LXW_CHART_COLUMN_STACKED_PERCENT | Column chart - percentage stacked. |
|
|
|
|
* | #LXW_CHART_DOUGHNUT | Doughnut chart. |
|
|
|
|
* | #LXW_CHART_LINE | Line chart. |
|
2020-05-31 12:59:19 +01:00
|
|
|
* | #LXW_CHART_LINE_STACKED | Line chart - stacked. |
|
|
|
|
* | #LXW_CHART_LINE_STACKED_PERCENT | Line chart - percentage stacked. |
|
2016-05-23 00:22:26 +01:00
|
|
|
* | #LXW_CHART_PIE | Pie chart. |
|
|
|
|
* | #LXW_CHART_SCATTER | Scatter chart. |
|
|
|
|
* | #LXW_CHART_SCATTER_STRAIGHT | Scatter chart - straight. |
|
|
|
|
* | #LXW_CHART_SCATTER_STRAIGHT_WITH_MARKERS | Scatter chart - straight with markers. |
|
|
|
|
* | #LXW_CHART_SCATTER_SMOOTH | Scatter chart - smooth. |
|
|
|
|
* | #LXW_CHART_SCATTER_SMOOTH_WITH_MARKERS | Scatter chart - smooth with markers. |
|
|
|
|
* | #LXW_CHART_RADAR | Radar chart. |
|
|
|
|
* | #LXW_CHART_RADAR_WITH_MARKERS | Radar chart - with markers. |
|
|
|
|
* | #LXW_CHART_RADAR_FILLED | Radar chart - filled. |
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* See @ref chart.h for details.
|
|
|
|
*/
|
2016-05-23 22:42:27 +01:00
|
|
|
lxw_chart *workbook_add_chart(lxw_workbook *workbook, uint8_t chart_type);
|
2016-05-23 00:22:26 +01:00
|
|
|
|
2014-06-08 17:40:59 +01:00
|
|
|
/**
|
|
|
|
* @brief Close the Workbook object and write the XLSX file.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
*
|
2016-05-30 21:16:25 +01:00
|
|
|
* @return A #lxw_error.
|
2014-06-08 17:40:59 +01:00
|
|
|
*
|
|
|
|
* The `%workbook_close()` function closes a Workbook object, writes the Excel
|
|
|
|
* file to disk, frees any memory allocated internally to the Workbook and
|
|
|
|
* frees the object itself.
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_close(workbook);
|
|
|
|
* @endcode
|
|
|
|
*
|
2016-05-30 21:16:25 +01:00
|
|
|
* The `%workbook_close()` function returns any #lxw_error error codes
|
2014-06-08 17:40:59 +01:00
|
|
|
* encountered when creating the Excel file. The error code can be returned
|
|
|
|
* from the program main or the calling function:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* return workbook_close(workbook);
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_close(lxw_workbook *workbook);
|
2014-06-08 17:40:59 +01:00
|
|
|
|
2015-12-13 22:49:13 +00:00
|
|
|
/**
|
|
|
|
* @brief Set the document properties such as Title, Author etc.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param properties Document properties to set.
|
|
|
|
*
|
2016-07-05 00:16:22 +01:00
|
|
|
* @return A #lxw_error.
|
2016-06-11 14:40:09 +01:00
|
|
|
*
|
2016-12-25 10:50:29 +00:00
|
|
|
* The `%workbook_set_properties` function can be used to set the document
|
2015-12-13 22:49:13 +00:00
|
|
|
* properties of the Excel file created by `libxlsxwriter`. These properties
|
|
|
|
* are visible when you use the `Office Button -> Prepare -> Properties`
|
|
|
|
* option in Excel and are also available to external applications that read
|
|
|
|
* or index windows files.
|
|
|
|
*
|
|
|
|
* The properties that can be set are:
|
|
|
|
*
|
|
|
|
* - `title`
|
|
|
|
* - `subject`
|
|
|
|
* - `author`
|
|
|
|
* - `manager`
|
|
|
|
* - `company`
|
|
|
|
* - `category`
|
|
|
|
* - `keywords`
|
|
|
|
* - `comments`
|
|
|
|
* - `hyperlink_base`
|
2019-11-17 19:25:58 +00:00
|
|
|
* - `created`
|
2015-12-13 22:49:13 +00:00
|
|
|
*
|
|
|
|
* The properties are specified via a `lxw_doc_properties` struct. All the
|
2019-11-17 19:25:58 +00:00
|
|
|
* fields are all optional. An example of how to create and pass the
|
|
|
|
* properties is:
|
2015-12-13 22:49:13 +00:00
|
|
|
*
|
|
|
|
* @code
|
2015-12-15 21:30:08 +00:00
|
|
|
* // Create a properties structure and set some of the fields.
|
|
|
|
* lxw_doc_properties properties = {
|
|
|
|
* .title = "This is an example spreadsheet",
|
|
|
|
* .subject = "With document properties",
|
|
|
|
* .author = "John McNamara",
|
|
|
|
* .manager = "Dr. Heinz Doofenshmirtz",
|
|
|
|
* .company = "of Wolves",
|
|
|
|
* .category = "Example spreadsheets",
|
|
|
|
* .keywords = "Sample, Example, Properties",
|
|
|
|
* .comments = "Created with libxlsxwriter",
|
|
|
|
* .status = "Quo",
|
|
|
|
* };
|
|
|
|
*
|
|
|
|
* // Set the properties in the workbook.
|
|
|
|
* workbook_set_properties(workbook, &properties);
|
2015-12-13 22:49:13 +00:00
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* @image html doc_properties.png
|
|
|
|
*
|
2019-11-17 19:25:58 +00:00
|
|
|
* The `created` parameter sets the file creation date/time shown in
|
|
|
|
* Excel. This defaults to the current time and date if set to 0. If you wish
|
|
|
|
* to create files that are binary equivalent (for the same input data) then
|
|
|
|
* you should set this creation date/time to a known value using a `time_t`
|
|
|
|
* value.
|
|
|
|
*
|
2015-12-13 22:49:13 +00:00
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_set_properties(lxw_workbook *workbook,
|
|
|
|
lxw_doc_properties *properties);
|
2015-12-13 22:49:13 +00:00
|
|
|
|
2016-06-11 14:40:09 +01:00
|
|
|
/**
|
|
|
|
* @brief Set a custom document text property.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name The name of the custom property.
|
|
|
|
* @param value The value of the custom property.
|
|
|
|
*
|
2016-07-05 00:16:22 +01:00
|
|
|
* @return A #lxw_error.
|
2016-06-11 14:40:09 +01:00
|
|
|
*
|
2016-12-25 10:50:29 +00:00
|
|
|
* The `%workbook_set_custom_property_string()` function can be used to set one
|
2016-06-11 14:40:09 +01:00
|
|
|
* or more custom document text properties not covered by the standard
|
2016-12-25 10:50:29 +00:00
|
|
|
* properties in the `workbook_set_properties()` function above.
|
2016-06-11 14:40:09 +01:00
|
|
|
*
|
|
|
|
* For example:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_set_custom_property_string(workbook, "Checked by", "Eve");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* @image html custom_properties.png
|
|
|
|
*
|
|
|
|
* There are 4 `workbook_set_custom_property_string_*()` functions for each
|
|
|
|
* of the custom property types supported by Excel:
|
|
|
|
*
|
|
|
|
* - text/string: `workbook_set_custom_property_string()`
|
|
|
|
* - number: `workbook_set_custom_property_number()`
|
|
|
|
* - datetime: `workbook_set_custom_property_datetime()`
|
|
|
|
* - boolean: `workbook_set_custom_property_boolean()`
|
|
|
|
*
|
|
|
|
* **Note**: the name and value parameters are limited to 255 characters
|
|
|
|
* by Excel.
|
|
|
|
*
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_set_custom_property_string(lxw_workbook *workbook,
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name,
|
|
|
|
const char *value);
|
2016-06-11 14:40:09 +01:00
|
|
|
/**
|
|
|
|
* @brief Set a custom document number property.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name The name of the custom property.
|
|
|
|
* @param value The value of the custom property.
|
|
|
|
*
|
2016-07-05 00:16:22 +01:00
|
|
|
* @return A #lxw_error.
|
2016-06-11 14:40:09 +01:00
|
|
|
*
|
|
|
|
* Set a custom document number property.
|
|
|
|
* See `workbook_set_custom_property_string()` above for details.
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_set_custom_property_number(workbook, "Document number", 12345);
|
|
|
|
* @endcode
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_set_custom_property_number(lxw_workbook *workbook,
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name, double value);
|
2016-06-11 01:03:30 +01:00
|
|
|
|
2016-06-11 14:40:09 +01:00
|
|
|
/* Undocumented since the user can use workbook_set_custom_property_number().
|
|
|
|
* Only implemented for file format completeness and testing.
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_set_custom_property_integer(lxw_workbook *workbook,
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name,
|
|
|
|
int32_t value);
|
2016-06-11 01:03:30 +01:00
|
|
|
|
2016-06-11 14:40:09 +01:00
|
|
|
/**
|
|
|
|
* @brief Set a custom document boolean property.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name The name of the custom property.
|
|
|
|
* @param value The value of the custom property.
|
|
|
|
*
|
2016-07-05 00:16:22 +01:00
|
|
|
* @return A #lxw_error.
|
2016-06-11 14:40:09 +01:00
|
|
|
*
|
|
|
|
* Set a custom document boolean property.
|
|
|
|
* See `workbook_set_custom_property_string()` above for details.
|
|
|
|
*
|
|
|
|
* @code
|
2016-06-23 21:35:23 +01:00
|
|
|
* workbook_set_custom_property_boolean(workbook, "Has Review", 1);
|
2016-06-11 14:40:09 +01:00
|
|
|
* @endcode
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_set_custom_property_boolean(lxw_workbook *workbook,
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name,
|
|
|
|
uint8_t value);
|
2016-06-11 14:40:09 +01:00
|
|
|
/**
|
|
|
|
* @brief Set a custom document date or time property.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name The name of the custom property.
|
|
|
|
* @param datetime The value of the custom property.
|
|
|
|
*
|
2016-07-05 00:16:22 +01:00
|
|
|
* @return A #lxw_error.
|
2016-06-11 14:40:09 +01:00
|
|
|
*
|
|
|
|
* Set a custom date or time number property.
|
|
|
|
* See `workbook_set_custom_property_string()` above for details.
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* lxw_datetime datetime = {2016, 12, 1, 11, 55, 0.0};
|
|
|
|
*
|
|
|
|
* workbook_set_custom_property_datetime(workbook, "Date completed", &datetime);
|
|
|
|
* @endcode
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_set_custom_property_datetime(lxw_workbook *workbook,
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name,
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_datetime *datetime);
|
2016-06-11 01:03:30 +01:00
|
|
|
|
2015-04-06 23:46:39 +01:00
|
|
|
/**
|
|
|
|
* @brief Create a defined name in the workbook to use as a variable.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name The defined name.
|
|
|
|
* @param formula The cell or range that the defined name refers to.
|
|
|
|
*
|
2016-07-05 00:16:22 +01:00
|
|
|
* @return A #lxw_error.
|
2015-04-06 23:46:39 +01:00
|
|
|
*
|
2016-12-25 10:50:29 +00:00
|
|
|
* This function is used to defined a name that can be used to represent a
|
2015-04-06 23:46:39 +01:00
|
|
|
* value, a single cell or a range of cells in a workbook: These defined names
|
|
|
|
* can then be used in formulas:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_define_name(workbook, "Exchange_rate", "=0.96");
|
|
|
|
* worksheet_write_formula(worksheet, 2, 1, "=Exchange_rate", NULL);
|
|
|
|
*
|
|
|
|
* @endcode
|
2015-04-06 23:55:40 +01:00
|
|
|
*
|
2015-04-06 23:46:39 +01:00
|
|
|
* @image html defined_name.png
|
|
|
|
*
|
|
|
|
* As in Excel a name defined like this is "global" to the workbook and can be
|
|
|
|
* referred to from any worksheet:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* // Global workbook name.
|
|
|
|
* workbook_define_name(workbook, "Sales", "=Sheet1!$G$1:$H$10");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* It is also possible to define a local/worksheet name by prefixing it with
|
|
|
|
* the sheet name using the syntax `'sheetname!definedname'`:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* // Local worksheet name.
|
|
|
|
* workbook_define_name(workbook, "Sheet2!Sales", "=Sheet2!$G$1:$G$10");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* If the sheet name contains spaces or special characters you must follow the
|
|
|
|
* Excel convention and enclose it in single quotes:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_define_name(workbook, "'New Data'!Sales", "=Sheet2!$G$1:$G$10");
|
|
|
|
* @endcode
|
|
|
|
*
|
2015-04-06 23:55:40 +01:00
|
|
|
* The rules for names in Excel are explained in the
|
2024-05-16 09:09:14 +01:00
|
|
|
* [Microsoft Office documentation](https://support.microsoft.com/en-us/office/define-and-use-names-in-formulas-4d0f13ac-53b7-422e-afd2-abd7ff379c64).
|
2015-04-06 23:46:39 +01:00
|
|
|
*
|
|
|
|
*/
|
2016-07-03 13:53:11 +01:00
|
|
|
lxw_error workbook_define_name(lxw_workbook *workbook, const char *name,
|
|
|
|
const char *formula);
|
2015-04-06 23:46:39 +01:00
|
|
|
|
2019-12-11 00:39:11 +00:00
|
|
|
/**
|
|
|
|
* @brief Get the default URL format used with `worksheet_write_url()`.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @return A lxw_format instance that has hyperlink properties set.
|
|
|
|
*
|
|
|
|
* This function returns a lxw_format instance that is used for the default
|
|
|
|
* blue underline hyperlink in the `worksheet_write_url()` function when a
|
|
|
|
* format isn't specified:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* lxw_format *url_format = workbook_get_default_url_format(workbook);
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* The format is the hyperlink style defined by Excel for the default theme.
|
|
|
|
* This format is only ever required when overwriting a string URL with
|
|
|
|
* data of a different type. See the example below.
|
|
|
|
*/
|
|
|
|
lxw_format *workbook_get_default_url_format(lxw_workbook *workbook);
|
|
|
|
|
2016-12-26 12:51:13 +00:00
|
|
|
/**
|
|
|
|
* @brief Get a worksheet object from its name.
|
|
|
|
*
|
2017-06-25 20:26:56 +01:00
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name Worksheet name.
|
2016-12-26 12:51:13 +00:00
|
|
|
*
|
|
|
|
* @return A lxw_worksheet object.
|
|
|
|
*
|
|
|
|
* This function returns a lxw_worksheet object reference based on its name:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* worksheet = workbook_get_worksheet_by_name(workbook, "Sheet1");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
*/
|
2016-05-07 15:16:24 +01:00
|
|
|
lxw_worksheet *workbook_get_worksheet_by_name(lxw_workbook *workbook,
|
2016-07-04 21:47:42 +01:00
|
|
|
const char *name);
|
2016-05-07 15:16:24 +01:00
|
|
|
|
2016-12-25 10:50:29 +00:00
|
|
|
/**
|
2018-09-08 23:36:09 +01:00
|
|
|
* @brief Get a chartsheet object from its name.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name chartsheet name.
|
|
|
|
*
|
|
|
|
* @return A lxw_chartsheet object.
|
|
|
|
*
|
|
|
|
* This function returns a lxw_chartsheet object reference based on its name:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* chartsheet = workbook_get_chartsheet_by_name(workbook, "Chart1");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
lxw_chartsheet *workbook_get_chartsheet_by_name(lxw_workbook *workbook,
|
|
|
|
const char *name);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Validate a worksheet or chartsheet name.
|
2016-12-26 12:51:13 +00:00
|
|
|
*
|
2016-12-25 10:50:29 +00:00
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
2018-09-08 23:36:09 +01:00
|
|
|
* @param sheetname Sheet name to validate.
|
2016-12-25 10:50:29 +00:00
|
|
|
*
|
|
|
|
* @return A #lxw_error.
|
|
|
|
*
|
2018-09-08 23:36:09 +01:00
|
|
|
* This function is used to validate a worksheet or chartsheet name according
|
|
|
|
* to the rules used by Excel:
|
2016-12-25 10:50:29 +00:00
|
|
|
*
|
2024-05-08 19:59:22 +01:00
|
|
|
* - The name cannot be blank.
|
2016-12-25 10:50:29 +00:00
|
|
|
* - The name is less than or equal to 31 UTF-8 characters.
|
2016-12-26 12:51:13 +00:00
|
|
|
* - The name doesn't contain any of the characters: ` [ ] : * ? / \ `
|
2019-06-07 00:00:36 +01:00
|
|
|
* - The name doesn't start or end with an apostrophe.
|
2019-06-07 21:28:08 +01:00
|
|
|
* - The name isn't already in use. (Case insensitive, see the note below).
|
2016-12-25 10:50:29 +00:00
|
|
|
*
|
|
|
|
* @code
|
2018-09-08 23:36:09 +01:00
|
|
|
* lxw_error err = workbook_validate_sheet_name(workbook, "Foglio");
|
2016-12-25 10:50:29 +00:00
|
|
|
* @endcode
|
|
|
|
*
|
2018-09-08 23:36:09 +01:00
|
|
|
* This function is called by `workbook_add_worksheet()` and
|
|
|
|
* `workbook_add_chartsheet()` but it can be explicitly called by the user
|
|
|
|
* beforehand to ensure that the sheet name is valid.
|
2016-12-25 10:50:29 +00:00
|
|
|
*
|
2020-03-21 17:40:08 +00:00
|
|
|
* @note You should also avoid using the worksheet name "History" (case
|
|
|
|
* insensitive) which is reserved in English language versions of
|
|
|
|
* Excel. Non-English versions may have restrictions on the equivalent word.
|
|
|
|
*
|
2019-06-07 21:28:08 +01:00
|
|
|
* @note This function does an ASCII lowercase string comparison to determine
|
|
|
|
* if the sheet name is already in use. It doesn't take UTF-8 characters into
|
|
|
|
* account. Thus it would flag "Café" and "café" as a duplicate (just like
|
|
|
|
* Excel) but it wouldn't catch "CAFÉ". If you need a full UTF-8 case
|
|
|
|
* insensitive check you should use a third party library to implement it.
|
|
|
|
*
|
2016-12-25 10:50:29 +00:00
|
|
|
*/
|
2018-09-08 23:36:09 +01:00
|
|
|
lxw_error workbook_validate_sheet_name(lxw_workbook *workbook,
|
|
|
|
const char *sheetname);
|
2016-12-23 19:45:16 +00:00
|
|
|
|
2019-06-16 15:46:09 +01:00
|
|
|
/**
|
|
|
|
* @brief Add a vbaProject binary to the Excel workbook.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param filename The path/filename of the vbaProject.bin file.
|
|
|
|
*
|
|
|
|
* The `%workbook_add_vba_project()` function can be used to add macros or
|
|
|
|
* functions to a workbook using a binary VBA project file that has been
|
|
|
|
* extracted from an existing Excel xlsm file:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_add_vba_project(workbook, "vbaProject.bin");
|
|
|
|
* @endcode
|
|
|
|
*
|
2023-09-20 20:37:38 +02:00
|
|
|
* Only one `vbaProject.bin` file can be added per workbook. The name doesn't
|
2019-06-19 00:21:04 +01:00
|
|
|
* have to be `vbaProject.bin`. Any suitable path/name for an existing VBA bin
|
|
|
|
* file will do.
|
|
|
|
*
|
|
|
|
* Once you add a VBA project had been add to an libxlsxwriter workbook you
|
|
|
|
* should ensure that the file extension is `.xlsm` to prevent Excel from
|
|
|
|
* giving a warning when it opens the file:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* lxw_workbook *workbook = new_workbook("macro.xlsm");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* See also @ref working_with_macros
|
2019-06-16 15:46:09 +01:00
|
|
|
*
|
|
|
|
* @return A #lxw_error.
|
|
|
|
*/
|
|
|
|
lxw_error workbook_add_vba_project(lxw_workbook *workbook,
|
|
|
|
const char *filename);
|
|
|
|
|
2023-09-20 20:37:38 +02:00
|
|
|
/**
|
|
|
|
* @brief Add a vbaProject binary and a vbaProjectSignature binary to the Excel
|
|
|
|
* workbook.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param vba_project The path/filename of the vbaProject.bin file.
|
|
|
|
* @param signature The path/filename of the vbaProjectSignature.bin file.
|
|
|
|
*
|
|
|
|
* The `%workbook_add_signed_vba_project()` function can be used to add digitally
|
|
|
|
* signed macros or functions to a workbook. The function adds a binary VBA project
|
|
|
|
* file and a binary VBA project signature file that have been extracted from an
|
|
|
|
* existing Excel xlsm file with digitally signed macros:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_add_signed_vba_project(workbook, "vbaProject.bin", "vbaProjectSignature.bin");
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* Only one `vbaProject.bin` file can be added per workbook. The name doesn't
|
|
|
|
* have to be `vbaProject.bin`. Any suitable path/name for an existing VBA bin
|
|
|
|
* file will do. The same applies for `vbaProjectSignature.bin`.
|
|
|
|
*
|
|
|
|
* See also @ref working_with_macros
|
|
|
|
*
|
|
|
|
* @return A #lxw_error.
|
|
|
|
*/
|
|
|
|
lxw_error workbook_add_signed_vba_project(lxw_workbook *workbook,
|
|
|
|
const char *vba_project,
|
|
|
|
const char *signature);
|
|
|
|
|
2019-06-16 15:46:09 +01:00
|
|
|
/**
|
|
|
|
* @brief Set the VBA name for the workbook.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param name Name of the workbook used by VBA.
|
|
|
|
*
|
|
|
|
* The `workbook_set_vba_name()` function can be used to set the VBA name for
|
|
|
|
* the workbook. This is sometimes required when a vbaProject macro included
|
2019-06-19 00:21:04 +01:00
|
|
|
* via `workbook_add_vba_project()` refers to the workbook by a name other
|
|
|
|
* than `ThisWorkbook`.
|
2019-06-16 15:46:09 +01:00
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_set_vba_name(workbook, "MyWorkbook");
|
|
|
|
* @endcode
|
|
|
|
*
|
2019-06-19 00:21:04 +01:00
|
|
|
* If an Excel VBA name for the workbook isn't specified then libxlsxwriter
|
|
|
|
* will use `ThisWorkbook`.
|
|
|
|
*
|
|
|
|
* See also @ref working_with_macros
|
2019-06-16 15:46:09 +01:00
|
|
|
*
|
|
|
|
* @return A #lxw_error.
|
|
|
|
*/
|
|
|
|
lxw_error workbook_set_vba_name(lxw_workbook *workbook, const char *name);
|
|
|
|
|
2021-03-27 20:36:13 +00:00
|
|
|
/**
|
|
|
|
* @brief Add a recommendation to open the file in "read-only" mode.
|
|
|
|
*
|
2021-03-30 14:08:58 +01:00
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
2021-03-27 20:36:13 +00:00
|
|
|
*
|
|
|
|
* This function can be used to set the Excel "Read-only Recommended" option
|
|
|
|
* that is available when saving a file. This presents the user of the file
|
|
|
|
* with an option to open it in "read-only" mode. This means that any changes
|
|
|
|
* to the file can't be saved back to the same file and must be saved to a new
|
|
|
|
* file. It can be set as follows:
|
|
|
|
*
|
|
|
|
* @code
|
|
|
|
* workbook_read_only_recommended(workbook);
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* Which will raise a dialog like the following when opening the file:
|
|
|
|
*
|
|
|
|
* @image html read_only.png
|
|
|
|
*/
|
|
|
|
void workbook_read_only_recommended(lxw_workbook *workbook);
|
|
|
|
|
2025-02-14 11:15:56 +00:00
|
|
|
/**
|
|
|
|
* @brief Set the size of a workbook window.
|
|
|
|
*
|
|
|
|
* @param workbook Pointer to a lxw_workbook instance.
|
|
|
|
* @param width Width of the window in pixels.
|
|
|
|
* @param height Height of the window in pixels.
|
|
|
|
*
|
|
|
|
* Set the size of a workbook window. This is generally only useful on macOS
|
|
|
|
* since Microsoft Windows uses the window size from the last time an Excel file
|
|
|
|
* was opened/saved. The default size is 1073 x 644 pixels.
|
|
|
|
*
|
|
|
|
* The resulting pixel sizes may not exactly match the target screen and
|
|
|
|
* resolution since it is based on the original Excel for Windows sizes. Some
|
|
|
|
* trial and error may be required to get an exact size.
|
|
|
|
*/
|
|
|
|
void workbook_set_size(lxw_workbook *workbook,
|
|
|
|
uint16_t width, uint16_t height);
|
|
|
|
|
2016-01-04 20:05:22 +00:00
|
|
|
void lxw_workbook_free(lxw_workbook *workbook);
|
|
|
|
void lxw_workbook_assemble_xml_file(lxw_workbook *workbook);
|
|
|
|
void lxw_workbook_set_default_xf_indices(lxw_workbook *workbook);
|
2019-12-11 00:39:11 +00:00
|
|
|
void workbook_unset_default_url_format(lxw_workbook *workbook);
|
2014-06-08 17:40:59 +01:00
|
|
|
|
|
|
|
/* Declarations required for unit testing. */
|
|
|
|
#ifdef TESTING
|
|
|
|
|
|
|
|
STATIC void _workbook_xml_declaration(lxw_workbook *self);
|
|
|
|
STATIC void _write_workbook(lxw_workbook *self);
|
|
|
|
STATIC void _write_file_version(lxw_workbook *self);
|
|
|
|
STATIC void _write_workbook_pr(lxw_workbook *self);
|
|
|
|
STATIC void _write_book_views(lxw_workbook *self);
|
|
|
|
STATIC void _write_workbook_view(lxw_workbook *self);
|
|
|
|
STATIC void _write_sheet(lxw_workbook *self,
|
|
|
|
const char *name, uint32_t sheet_id, uint8_t hidden);
|
|
|
|
STATIC void _write_sheets(lxw_workbook *self);
|
|
|
|
STATIC void _write_calc_pr(lxw_workbook *self);
|
|
|
|
|
2015-03-08 23:25:43 +00:00
|
|
|
STATIC void _write_defined_name(lxw_workbook *self,
|
|
|
|
lxw_defined_name *define_name);
|
|
|
|
STATIC void _write_defined_names(lxw_workbook *self);
|
|
|
|
|
2016-07-03 13:53:11 +01:00
|
|
|
STATIC lxw_error _store_defined_name(lxw_workbook *self, const char *name,
|
|
|
|
const char *app_name,
|
|
|
|
const char *formula, int16_t index,
|
|
|
|
uint8_t hidden);
|
2015-04-06 23:46:39 +01:00
|
|
|
|
2014-06-08 17:40:59 +01:00
|
|
|
#endif /* TESTING */
|
|
|
|
|
|
|
|
/* *INDENT-OFF* */
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* *INDENT-ON* */
|
|
|
|
|
|
|
|
#endif /* __LXW_WORKBOOK_H__ */
|