2015-12-24 17:10:02 -05:00
|
|
|
// Copyright (c) 2014-2016 Thomas Fussell
|
2015-12-24 15:51:11 -05:00
|
|
|
// 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
|
2016-05-14 15:19:08 -04:00
|
|
|
|
2015-11-01 12:31:29 -05:00
|
|
|
#include <xlnt/serialization/shared_strings_serializer.hpp>
|
2016-05-14 15:19:08 -04:00
|
|
|
#include <xlnt/cell/text.hpp>
|
2015-11-01 12:31:29 -05:00
|
|
|
#include <xlnt/serialization/xml_document.hpp>
|
|
|
|
#include <xlnt/serialization/xml_node.hpp>
|
2015-10-29 13:46:56 -04:00
|
|
|
|
2016-06-12 01:07:07 +08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
std::size_t string_to_size_t(const std::string &s)
|
|
|
|
{
|
|
|
|
#if ULLONG_MAX == SIZE_MAX
|
|
|
|
return std::stoull(s);
|
|
|
|
#else
|
|
|
|
return std::stoul(s);
|
|
|
|
#endif
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-10-29 13:46:56 -04:00
|
|
|
namespace xlnt {
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
xml_document shared_strings_serializer::write_shared_strings(const std::vector<text> &strings)
|
2015-10-30 18:54:04 -04:00
|
|
|
{
|
|
|
|
xml_document xml;
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2015-10-30 18:54:04 -04:00
|
|
|
auto root_node = xml.add_child("sst");
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2015-10-30 18:54:04 -04:00
|
|
|
xml.add_namespace("", "http://schemas.openxmlformats.org/spreadsheetml/2006/main");
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2016-05-15 11:51:32 -04:00
|
|
|
root_node.add_attribute("count", std::to_string(strings.size()));
|
2015-11-10 18:58:54 -05:00
|
|
|
root_node.add_attribute("uniqueCount", std::to_string(strings.size()));
|
2015-11-01 09:43:01 -05:00
|
|
|
|
|
|
|
for (const auto &string : strings)
|
2015-10-30 18:54:04 -04:00
|
|
|
{
|
2016-05-14 15:19:08 -04:00
|
|
|
if (string.get_runs().size() == 1 && !string.get_runs().at(0).has_formatting())
|
|
|
|
{
|
|
|
|
root_node.add_child("si").add_child("t").set_text(string.get_plain_string());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-05-15 11:51:32 -04:00
|
|
|
auto string_item_node = root_node.add_child("si");
|
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
for (const auto &run : string.get_runs())
|
|
|
|
{
|
|
|
|
auto rich_text_run_node = string_item_node.add_child("r");
|
|
|
|
|
|
|
|
if (run.has_formatting())
|
|
|
|
{
|
|
|
|
auto run_properties_node = rich_text_run_node.add_child("rPr");
|
|
|
|
|
2016-05-15 11:51:32 -04:00
|
|
|
if (run.has_size())
|
|
|
|
{
|
|
|
|
run_properties_node.add_child("sz").add_attribute("val", std::to_string(run.get_size()));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (run.has_color())
|
|
|
|
{
|
|
|
|
run_properties_node.add_child("color").add_attribute("rgb", run.get_color());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (run.has_font())
|
|
|
|
{
|
|
|
|
run_properties_node.add_child("rFont").add_attribute("val", run.get_font());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (run.has_family())
|
|
|
|
{
|
|
|
|
run_properties_node.add_child("family").add_attribute("val", std::to_string(run.get_family()));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (run.has_scheme())
|
|
|
|
{
|
|
|
|
run_properties_node.add_child("scheme").add_attribute("val", run.get_scheme());
|
|
|
|
}
|
2016-05-14 15:19:08 -04:00
|
|
|
}
|
2016-05-15 11:51:32 -04:00
|
|
|
|
|
|
|
auto text_node = rich_text_run_node.add_child("t");
|
|
|
|
text_node.set_text(run.get_string());
|
2016-05-14 15:19:08 -04:00
|
|
|
}
|
|
|
|
}
|
2015-10-30 18:54:04 -04:00
|
|
|
}
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2015-10-30 18:54:04 -04:00
|
|
|
return xml;
|
|
|
|
}
|
2015-10-29 13:46:56 -04:00
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
bool shared_strings_serializer::read_shared_strings(const xml_document &xml, std::vector<text> &strings)
|
2015-10-29 13:46:56 -04:00
|
|
|
{
|
|
|
|
strings.clear();
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2015-10-30 18:54:04 -04:00
|
|
|
auto root_node = xml.get_child("sst");
|
2016-03-07 13:42:33 +08:00
|
|
|
std::size_t unique_count = 0;
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2015-12-22 14:39:41 -05:00
|
|
|
if (root_node.has_attribute("uniqueCount"))
|
|
|
|
{
|
2016-06-12 01:07:07 +08:00
|
|
|
unique_count = string_to_size_t(root_node.get_attribute("uniqueCount"));
|
2015-12-22 14:39:41 -05:00
|
|
|
}
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
for (const auto &string_item_node : root_node.get_children())
|
2015-10-29 13:46:56 -04:00
|
|
|
{
|
2016-05-14 15:19:08 -04:00
|
|
|
if (string_item_node.get_name() != "si")
|
2015-10-30 18:54:04 -04:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
if (string_item_node.has_child("t"))
|
2015-10-30 18:54:04 -04:00
|
|
|
{
|
2016-05-14 15:19:08 -04:00
|
|
|
text t;
|
|
|
|
t.set_plain_string(string_item_node.get_child("t").get_text());
|
|
|
|
strings.push_back(t);
|
2015-10-30 18:54:04 -04:00
|
|
|
}
|
2016-05-14 15:19:08 -04:00
|
|
|
else if (string_item_node.has_child("r")) // possible multiple text entities.
|
2015-10-30 18:54:04 -04:00
|
|
|
{
|
2016-05-14 15:19:08 -04:00
|
|
|
text t;
|
|
|
|
|
|
|
|
for (const auto& rich_text_run_node : string_item_node.get_children())
|
2016-05-14 16:24:21 +08:00
|
|
|
{
|
2016-05-14 15:19:08 -04:00
|
|
|
if (rich_text_run_node.get_name() == "r" && rich_text_run_node.has_child("t"))
|
2016-05-14 16:24:21 +08:00
|
|
|
{
|
2016-05-14 15:19:08 -04:00
|
|
|
text_run run;
|
2016-05-15 11:51:32 -04:00
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
run.set_string(rich_text_run_node.get_child("t").get_text());
|
2016-05-15 11:51:32 -04:00
|
|
|
|
|
|
|
if (rich_text_run_node.has_child("rPr"))
|
|
|
|
{
|
|
|
|
auto run_properties_node = rich_text_run_node.get_child("rPr");
|
|
|
|
|
|
|
|
if (run_properties_node.has_child("sz"))
|
|
|
|
{
|
2016-06-12 01:07:07 +08:00
|
|
|
run.set_size(string_to_size_t(run_properties_node.get_child("sz").get_attribute("val")));
|
2016-05-15 11:51:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (run_properties_node.has_child("rFont"))
|
|
|
|
{
|
|
|
|
run.set_font(run_properties_node.get_child("rFont").get_attribute("val"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (run_properties_node.has_child("color"))
|
|
|
|
{
|
|
|
|
run.set_color(run_properties_node.get_child("color").get_attribute("rgb"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (run_properties_node.has_child("family"))
|
|
|
|
{
|
2016-06-12 01:07:07 +08:00
|
|
|
run.set_family(string_to_size_t(run_properties_node.get_child("family").get_attribute("val")));
|
2016-05-15 11:51:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (run_properties_node.has_child("scheme"))
|
|
|
|
{
|
|
|
|
run.set_scheme(run_properties_node.get_child("scheme").get_attribute("val"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-14 15:19:08 -04:00
|
|
|
t.add_run(run);
|
2016-05-14 16:24:21 +08:00
|
|
|
}
|
|
|
|
}
|
2016-05-14 15:19:08 -04:00
|
|
|
|
|
|
|
strings.push_back(t);
|
2015-10-30 18:54:04 -04:00
|
|
|
}
|
2015-10-29 13:46:56 -04:00
|
|
|
}
|
2015-11-01 09:43:01 -05:00
|
|
|
|
|
|
|
if (unique_count != strings.size())
|
2015-10-29 13:46:56 -04:00
|
|
|
{
|
|
|
|
throw std::runtime_error("counts don't match");
|
|
|
|
}
|
2015-11-01 09:43:01 -05:00
|
|
|
|
2015-10-29 13:46:56 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace xlnt
|