2016-08-05 01:52:05 -04:00
|
|
|
// Copyright (c) 2014-2016 Thomas Fussell
|
|
|
|
// Copyright (c) 2010-2015 openpyxl
|
|
|
|
//
|
|
|
|
// 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, WRISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE
|
|
|
|
//
|
|
|
|
// @license: http://www.opensource.org/licenses/mit-license.php
|
|
|
|
// @author: see AUTHORS file
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
#include <iostream>
|
2016-10-30 15:48:40 -04:00
|
|
|
#include <memory>
|
2016-08-05 01:52:05 -04:00
|
|
|
#include <unordered_map>
|
|
|
|
#include <vector>
|
|
|
|
|
2016-09-16 22:59:13 -04:00
|
|
|
#include <detail/include_libstudxml.hpp>
|
2016-10-31 20:48:43 -04:00
|
|
|
#include <detail/zip.hpp>
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
namespace xlnt {
|
|
|
|
|
|
|
|
class path;
|
|
|
|
class relationship;
|
|
|
|
class workbook;
|
2016-10-29 10:23:04 -04:00
|
|
|
class worksheet;
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
|
2016-10-31 20:48:43 -04:00
|
|
|
class ZipFileReader;
|
|
|
|
|
2016-08-05 01:52:05 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Handles writing a workbook into an XLSX file.
|
|
|
|
/// </summary>
|
2016-10-30 15:48:40 -04:00
|
|
|
class xlsx_consumer
|
2016-08-05 01:52:05 -04:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
xlsx_consumer(workbook &destination);
|
|
|
|
|
|
|
|
void read(std::istream &source);
|
|
|
|
|
2016-10-10 07:28:49 -04:00
|
|
|
void read(std::istream &source, const std::string &password);
|
|
|
|
|
2016-08-05 01:52:05 -04:00
|
|
|
private:
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Ignore all remaining elements at the same depth in the current XML parser.
|
|
|
|
/// </summary>
|
|
|
|
void skip_remaining_elements();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Convenience method to dereference the pointer to the current parser to avoid
|
|
|
|
/// having to use "parser_->" constantly.
|
|
|
|
/// </summary>
|
|
|
|
xml::parser &parser();
|
|
|
|
|
2016-08-05 01:52:05 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Read all the files needed from the XLSX archive and initialize all of
|
|
|
|
/// the data in the workbook to match.
|
|
|
|
/// </summary>
|
|
|
|
void populate_workbook();
|
|
|
|
|
|
|
|
// Package Parts
|
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Parse content types ([Content_Types].xml) and package relationships (_rels/.rels).
|
|
|
|
/// </summary>
|
2016-08-05 01:52:05 -04:00
|
|
|
void read_manifest();
|
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Parse the core properties about the current package (usually docProps/core.xml).
|
|
|
|
/// </summary>
|
|
|
|
void read_core_properties();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Parse extra application-speicific package properties (usually docProps/app.xml).
|
|
|
|
void read_extended_properties();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Parse custom file properties. These aren't associated with a particular application
|
|
|
|
/// but extensions to OOXML can use this part to hold extra data about the package.
|
|
|
|
/// </summary>
|
|
|
|
void read_custom_file_properties();
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
// SpreadsheetML-Specific Package Parts
|
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Parse the main XML document about the workbook and then all child relationships
|
|
|
|
/// of the workbook (e.g. worksheets).
|
|
|
|
/// </summary>
|
|
|
|
void read_workbook();
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
// Workbook Relationship Target Parts
|
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
/// xl/calcChain.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_calculation_chain();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_connections();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_custom_property();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_custom_xml_mappings();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_external_workbook_references();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_metadata();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_pivot_table();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// xl/sharedStrings.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_shared_string_table();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_shared_workbook_revision_headers();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_shared_workbook();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_shared_workbook_user_data();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// xl/styles.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_stylesheet();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// xl/theme/theme1.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_theme();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_volatile_dependencies();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// xl/sheets/*.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_chartsheet(const std::string &title);
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// xl/sheets/*.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_dialogsheet(const std::string &title);
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// xl/sheets/*.xml
|
|
|
|
/// </summary>
|
|
|
|
void read_worksheet(const std::string &title);
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
// Sheet Relationship Target Parts
|
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
2016-10-29 10:23:04 -04:00
|
|
|
void read_comments(worksheet ws);
|
2016-10-11 23:16:14 -04:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_drawings();
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
// Unknown Parts
|
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_unknown_parts();
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
/// <summary>
|
2016-10-11 23:16:14 -04:00
|
|
|
///
|
|
|
|
/// </summary>
|
|
|
|
void read_unknown_relationships();
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// The ZIP file containing the files that make up the OOXML package.
|
2016-08-05 01:52:05 -04:00
|
|
|
/// </summary>
|
2016-10-31 20:48:43 -04:00
|
|
|
std::unique_ptr<ZipFileReader> archive_;
|
2016-08-05 01:52:05 -04:00
|
|
|
|
2016-10-11 23:16:14 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Map of sheet titles to relationship IDs.
|
|
|
|
/// </summary>
|
2016-08-06 10:40:17 -04:00
|
|
|
std::unordered_map<std::string, std::size_t> sheet_title_id_map_;
|
2016-10-11 23:16:14 -04:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Map of sheet titles to indices. Used to ensure sheets are maintained
|
|
|
|
/// in the correct order.
|
|
|
|
/// </summary>
|
2016-08-06 10:40:17 -04:00
|
|
|
std::unordered_map<std::string, std::size_t> sheet_title_index_map_;
|
2016-08-05 01:52:05 -04:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// A reference to the workbook which is being read.
|
|
|
|
/// </summary>
|
2016-10-11 23:16:14 -04:00
|
|
|
workbook &target_;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// This pointer is generally set by instantiating an xml::parser in a function
|
|
|
|
/// scope and then calling a read_*() method which uses xlsx_consumer::parser()
|
|
|
|
/// to access the object.
|
|
|
|
/// </summary>
|
|
|
|
xml::parser *parser_;
|
2016-08-05 01:52:05 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace detail
|
|
|
|
} // namespace xlnt
|