2016-06-10 13:40:50 -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
|
|
|
|
|
2016-08-18 07:34:18 -04:00
|
|
|
#include <list>
|
2016-06-10 13:40:50 -04:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
2016-08-16 00:23:49 -04:00
|
|
|
#include <detail/format_impl.hpp>
|
|
|
|
#include <detail/style_impl.hpp>
|
2016-11-03 19:26:11 -04:00
|
|
|
#include <xlnt/cell/cell.hpp>
|
2016-06-10 13:40:50 -04:00
|
|
|
#include <xlnt/styles/format.hpp>
|
|
|
|
#include <xlnt/styles/style.hpp>
|
2016-08-16 00:23:49 -04:00
|
|
|
#include <xlnt/utils/exceptions.hpp>
|
2016-11-03 19:26:11 -04:00
|
|
|
#include <xlnt/worksheet/worksheet.hpp>
|
|
|
|
#include <xlnt/workbook/workbook.hpp>
|
|
|
|
#include <xlnt/workbook/worksheet_iterator.hpp>
|
2016-06-10 13:40:50 -04:00
|
|
|
|
|
|
|
namespace xlnt {
|
|
|
|
namespace detail {
|
|
|
|
|
|
|
|
struct stylesheet
|
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
format create_format(bool default_format)
|
2016-06-10 13:40:50 -04:00
|
|
|
{
|
2016-11-02 22:04:51 -04:00
|
|
|
format_impls.push_back(format_impl());
|
|
|
|
auto &impl = format_impls.back();
|
2016-11-03 19:26:11 -04:00
|
|
|
|
2016-11-02 22:04:51 -04:00
|
|
|
impl.parent = this;
|
|
|
|
impl.id = format_impls.size() - 1;
|
2016-11-07 21:11:30 -05:00
|
|
|
|
2016-11-03 19:26:11 -04:00
|
|
|
impl.border_id = 0;
|
|
|
|
impl.fill_id = 0;
|
|
|
|
impl.font_id = 0;
|
|
|
|
impl.number_format_id = 0;
|
2016-11-02 22:04:51 -04:00
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
impl.references = default_format ? 1 : 0;
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
return format(&impl);
|
2016-06-10 13:40:50 -04:00
|
|
|
}
|
2016-08-16 00:23:49 -04:00
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
format get_format(std::size_t index)
|
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
auto iter = format_impls.begin();
|
2016-11-09 19:52:18 -05:00
|
|
|
std::advance(iter, static_cast<std::list<format_impl>::difference_type>(index));
|
|
|
|
return format(&*iter);
|
|
|
|
}
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
class style create_style(const std::string &name)
|
2016-06-10 13:40:50 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
auto &impl = style_impls.emplace(name, style_impl()).first->second;
|
|
|
|
|
2016-08-18 07:34:18 -04:00
|
|
|
impl.parent = this;
|
2016-11-07 21:11:30 -05:00
|
|
|
impl.name = name;
|
|
|
|
|
|
|
|
impl.border_id = 0;
|
|
|
|
impl.fill_id = 0;
|
|
|
|
impl.font_id = 0;
|
|
|
|
impl.number_format_id = 0;
|
|
|
|
|
|
|
|
style_names.push_back(name);
|
|
|
|
return xlnt::style(&impl);
|
2016-08-16 00:23:49 -04:00
|
|
|
}
|
2016-06-10 13:40:50 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
class style style(const std::string &name)
|
2016-08-16 00:23:49 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
if (!has_style(name)) throw key_not_found();
|
|
|
|
return xlnt::style(&style_impls[name]);
|
2016-08-16 00:23:49 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool has_style(const std::string &name)
|
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
return style_impls.count(name) > 0;
|
2016-08-16 00:23:49 -04:00
|
|
|
}
|
|
|
|
|
2016-08-18 07:34:18 -04:00
|
|
|
std::size_t next_custom_number_format_id() const
|
|
|
|
{
|
|
|
|
std::size_t id = 164;
|
|
|
|
|
|
|
|
for (const auto &nf : number_formats)
|
|
|
|
{
|
2016-11-02 22:04:51 -04:00
|
|
|
if (nf.get_id() >= id)
|
2016-08-18 07:34:18 -04:00
|
|
|
{
|
2016-11-02 22:04:51 -04:00
|
|
|
id = nf.get_id() + 1;
|
2016-08-18 07:34:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
2016-09-07 18:02:46 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
template<typename T, typename C>
|
|
|
|
std::size_t find_or_add(C &container, const T &item)
|
2016-09-10 10:05:06 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
std::size_t i = 0;
|
2016-09-10 10:05:06 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
for (auto iter = container.begin(); iter != container.end(); ++iter)
|
2016-09-10 10:05:06 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
if (*iter == item)
|
2016-09-10 10:05:06 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
return i;
|
2016-09-10 10:05:06 -04:00
|
|
|
}
|
2016-11-07 21:11:30 -05:00
|
|
|
|
|
|
|
++i;
|
2016-09-10 10:05:06 -04:00
|
|
|
}
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
container.emplace(container.end(), item);
|
2016-09-10 10:05:06 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
return container.size() - 1;
|
2016-09-10 10:05:06 -04:00
|
|
|
}
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
template<typename T>
|
|
|
|
std::unordered_map<std::size_t, std::size_t> garbage_collect(
|
|
|
|
const std::unordered_map<std::size_t, std::size_t> &reference_counts,
|
|
|
|
std::vector<T> &container)
|
2016-09-10 10:05:06 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
std::unordered_map<std::size_t, std::size_t> id_map;
|
|
|
|
std::size_t unreferenced = 0;
|
|
|
|
const auto original_size = container.size();
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < original_size; ++i)
|
2016-09-10 10:05:06 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
id_map[i] = i - unreferenced;
|
|
|
|
|
|
|
|
if (reference_counts.count(i) == 0 || reference_counts.at(i) == 0)
|
2016-09-10 10:05:06 -04:00
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
container.erase(container.begin() + static_cast<typename std::vector<T>::difference_type>(i - unreferenced));
|
2016-11-07 21:11:30 -05:00
|
|
|
unreferenced++;
|
2016-09-10 10:05:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
return id_map;
|
2016-09-10 10:05:06 -04:00
|
|
|
}
|
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
void garbage_collect()
|
2016-09-07 18:02:46 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
if (!garbage_collection_enabled) return;
|
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
auto format_iter = format_impls.begin();
|
|
|
|
|
|
|
|
while (format_iter != format_impls.end())
|
2016-11-07 21:11:30 -05:00
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
auto &impl = *format_iter;
|
|
|
|
|
|
|
|
if (impl.references != 0)
|
2016-11-07 21:11:30 -05:00
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
++format_iter;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
format_iter = format_impls.erase(format_iter);
|
2016-11-07 21:11:30 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t new_id = 0;
|
|
|
|
|
|
|
|
std::unordered_map<std::size_t, std::size_t> alignment_reference_counts;
|
|
|
|
std::unordered_map<std::size_t, std::size_t> border_reference_counts;
|
|
|
|
std::unordered_map<std::size_t, std::size_t> fill_reference_counts;
|
|
|
|
std::unordered_map<std::size_t, std::size_t> font_reference_counts;
|
|
|
|
std::unordered_map<std::size_t, std::size_t> number_format_reference_counts;
|
|
|
|
std::unordered_map<std::size_t, std::size_t> protection_reference_counts;
|
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
fill_reference_counts[0]++;
|
|
|
|
fill_reference_counts[1]++;
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
for (auto &impl : format_impls)
|
|
|
|
{
|
|
|
|
impl.id = new_id++;
|
|
|
|
|
|
|
|
if (impl.alignment_id.is_set())
|
|
|
|
{
|
|
|
|
alignment_reference_counts[impl.alignment_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.border_id.is_set())
|
|
|
|
{
|
|
|
|
border_reference_counts[impl.border_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.fill_id.is_set())
|
|
|
|
{
|
|
|
|
fill_reference_counts[impl.fill_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.font_id.is_set())
|
|
|
|
{
|
|
|
|
font_reference_counts[impl.font_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.number_format_id.is_set())
|
|
|
|
{
|
|
|
|
number_format_reference_counts[impl.number_format_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.protection_id.is_set())
|
|
|
|
{
|
|
|
|
protection_reference_counts[impl.protection_id.get()]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto &name_impl_pair : style_impls)
|
|
|
|
{
|
|
|
|
auto &impl = name_impl_pair.second;
|
|
|
|
|
|
|
|
if (impl.alignment_id.is_set())
|
|
|
|
{
|
|
|
|
alignment_reference_counts[impl.alignment_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.border_id.is_set())
|
|
|
|
{
|
|
|
|
border_reference_counts[impl.border_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.fill_id.is_set())
|
|
|
|
{
|
|
|
|
fill_reference_counts[impl.fill_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.font_id.is_set())
|
|
|
|
{
|
|
|
|
font_reference_counts[impl.font_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.number_format_id.is_set())
|
|
|
|
{
|
|
|
|
number_format_reference_counts[impl.number_format_id.get()]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.protection_id.is_set())
|
|
|
|
{
|
|
|
|
protection_reference_counts[impl.protection_id.get()]++;
|
|
|
|
}
|
|
|
|
}
|
2016-09-07 18:02:46 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
auto alignment_id_map = garbage_collect(alignment_reference_counts, alignments);
|
|
|
|
auto border_id_map = garbage_collect(border_reference_counts, borders);
|
|
|
|
auto fill_id_map = garbage_collect(fill_reference_counts, fills);
|
|
|
|
auto font_id_map = garbage_collect(font_reference_counts, fonts);
|
|
|
|
auto protection_id_map = garbage_collect(protection_reference_counts, protections);
|
|
|
|
|
|
|
|
for (auto &impl : format_impls)
|
2016-09-07 18:02:46 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
if (impl.alignment_id.is_set())
|
2016-09-07 18:02:46 -04:00
|
|
|
{
|
2016-11-07 21:11:30 -05:00
|
|
|
impl.alignment_id = alignment_id_map[impl.alignment_id.get()];
|
2016-09-07 18:02:46 -04:00
|
|
|
}
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
if (impl.border_id.is_set())
|
|
|
|
{
|
|
|
|
impl.border_id = border_id_map[impl.border_id.get()];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.fill_id.is_set())
|
|
|
|
{
|
|
|
|
impl.fill_id = fill_id_map[impl.fill_id.get()];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.font_id.is_set())
|
|
|
|
{
|
|
|
|
impl.font_id = font_id_map[impl.font_id.get()];
|
|
|
|
}
|
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
if (impl.protection_id.is_set())
|
2016-11-07 21:11:30 -05:00
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
impl.protection_id = protection_id_map[impl.protection_id.get()];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto &name_impl : style_impls)
|
|
|
|
{
|
|
|
|
auto &impl = name_impl.second;
|
|
|
|
|
|
|
|
if (impl.alignment_id.is_set())
|
|
|
|
{
|
|
|
|
impl.alignment_id = alignment_id_map[impl.alignment_id.get()];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.border_id.is_set())
|
|
|
|
{
|
|
|
|
impl.border_id = border_id_map[impl.border_id.get()];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.fill_id.is_set())
|
|
|
|
{
|
|
|
|
impl.fill_id = fill_id_map[impl.fill_id.get()];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.font_id.is_set())
|
|
|
|
{
|
|
|
|
impl.font_id = font_id_map[impl.font_id.get()];
|
2016-11-07 21:11:30 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (impl.protection_id.is_set())
|
|
|
|
{
|
|
|
|
impl.protection_id = protection_id_map[impl.protection_id.get()];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
format_impl *find_or_create(format_impl &pattern)
|
2016-11-07 21:11:30 -05:00
|
|
|
{
|
|
|
|
auto iter = format_impls.begin();
|
|
|
|
auto id = find_or_add(format_impls, pattern);
|
2016-11-09 19:52:18 -05:00
|
|
|
std::advance(iter, static_cast<std::list<format_impl>::difference_type>(id));
|
2016-11-07 21:11:30 -05:00
|
|
|
|
|
|
|
auto &result = *iter;
|
|
|
|
|
|
|
|
result.parent = this;
|
|
|
|
result.id = id;
|
|
|
|
|
|
|
|
if (id != pattern.id)
|
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
pattern.references -= pattern.references > 0 ? 1 : 0;
|
|
|
|
++result.references;
|
|
|
|
garbage_collect();
|
2016-09-07 18:02:46 -04:00
|
|
|
}
|
2016-11-07 21:11:30 -05:00
|
|
|
|
|
|
|
return &result;
|
|
|
|
}
|
|
|
|
|
2016-11-09 19:52:18 -05:00
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const std::string &style_name)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.style = style_name;
|
|
|
|
|
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const alignment &new_alignment, bool applied)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.alignment_id = find_or_add(alignments, new_alignment);
|
|
|
|
new_format.alignment_applied = applied;
|
2016-09-07 18:02:46 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
2016-09-07 18:02:46 -04:00
|
|
|
|
2016-11-07 21:11:30 -05:00
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const border &new_border, bool applied)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.border_id = find_or_add(borders, new_border);
|
|
|
|
new_format.border_applied = applied;
|
|
|
|
|
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
|
|
|
|
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const fill &new_fill, bool applied)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.fill_id = find_or_add(fills, new_fill);
|
|
|
|
new_format.fill_applied = applied;
|
|
|
|
|
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
|
|
|
|
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const font &new_font, bool applied)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.font_id = find_or_add(fonts, new_font);
|
|
|
|
new_format.font_applied = applied;
|
|
|
|
|
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
|
|
|
|
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const number_format &new_number_format, bool applied)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.number_format_id = find_or_add(number_formats, new_number_format);
|
|
|
|
new_format.number_format_applied = applied;
|
|
|
|
|
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
|
|
|
|
|
|
|
format_impl *find_or_create_with(format_impl *pattern, const protection &new_protection, bool applied)
|
|
|
|
{
|
|
|
|
format_impl new_format = *pattern;
|
|
|
|
new_format.protection_id = find_or_add(protections, new_protection);
|
|
|
|
new_format.protection_applied = applied;
|
|
|
|
|
|
|
|
return find_or_create(new_format);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::size_t style_index(const std::string &name) const
|
|
|
|
{
|
2016-11-09 19:52:18 -05:00
|
|
|
return static_cast<std::size_t>(std::distance(style_names.begin(),
|
|
|
|
std::find(style_names.begin(), style_names.end(), name)));
|
2016-09-07 18:02:46 -04:00
|
|
|
}
|
2016-09-10 10:05:06 -04:00
|
|
|
|
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
format_impls.clear();
|
|
|
|
|
|
|
|
style_impls.clear();
|
2016-11-07 21:11:30 -05:00
|
|
|
style_names.clear();
|
2016-09-10 10:05:06 -04:00
|
|
|
|
|
|
|
alignments.clear();
|
|
|
|
borders.clear();
|
|
|
|
fills.clear();
|
|
|
|
fonts.clear();
|
|
|
|
number_formats.clear();
|
|
|
|
protections.clear();
|
|
|
|
|
|
|
|
colors.clear();
|
|
|
|
}
|
2016-11-03 19:26:11 -04:00
|
|
|
|
|
|
|
workbook *parent;
|
2016-11-07 21:11:30 -05:00
|
|
|
|
|
|
|
bool garbage_collection_enabled = true;
|
2016-11-03 19:26:11 -04:00
|
|
|
|
2016-08-18 07:34:18 -04:00
|
|
|
std::list<format_impl> format_impls;
|
2016-11-07 21:11:30 -05:00
|
|
|
std::unordered_map<std::string, style_impl> style_impls;
|
|
|
|
std::vector<std::string> style_names;
|
2016-06-10 13:40:50 -04:00
|
|
|
|
2016-08-16 00:23:49 -04:00
|
|
|
std::vector<alignment> alignments;
|
2016-06-10 13:40:50 -04:00
|
|
|
std::vector<border> borders;
|
|
|
|
std::vector<fill> fills;
|
|
|
|
std::vector<font> fonts;
|
2016-11-02 22:04:51 -04:00
|
|
|
std::vector<number_format> number_formats;
|
2016-08-16 00:23:49 -04:00
|
|
|
std::vector<protection> protections;
|
2016-06-10 13:40:50 -04:00
|
|
|
|
|
|
|
std::vector<color> colors;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace detail
|
|
|
|
} // namespace xlnt
|