mirror of
https://github.com/tfussell/xlnt.git
synced 2024-03-22 13:11:17 +08:00
230 lines
5.5 KiB
C++
230 lines
5.5 KiB
C++
#pragma once
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <xlnt/cell/cell.hpp>
|
|
#include <xlnt/cell/comment.hpp>
|
|
#include <xlnt/common/exceptions.hpp>
|
|
#include <xlnt/common/datetime.hpp>
|
|
#include <xlnt/common/types.hpp>
|
|
#include <xlnt/common/relationship.hpp>
|
|
|
|
#include "comment_impl.hpp"
|
|
|
|
namespace {
|
|
|
|
// return s after checking encoding, size, and illegal characters
|
|
std::string check_string(std::string s)
|
|
{
|
|
if (s.size() == 0)
|
|
{
|
|
return s;
|
|
}
|
|
|
|
// check encoding?
|
|
|
|
if (s.size() > 32767)
|
|
{
|
|
s = s.substr(0, 32767); // max string length in Excel
|
|
}
|
|
|
|
for (unsigned char c : s)
|
|
{
|
|
if (c <= 8 || c == 11 || c == 12 || (c >= 14 && c <= 31))
|
|
{
|
|
throw xlnt::illegal_character_error(static_cast<char>(c));
|
|
}
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
std::pair<bool, long double> cast_numeric(const std::string &s)
|
|
{
|
|
const char *str = s.c_str();
|
|
char *str_end = nullptr;
|
|
auto result = std::strtold(str, &str_end);
|
|
if (str_end != str + s.size()) return{ false, 0 };
|
|
return{ true, result };
|
|
}
|
|
|
|
std::pair<bool, long double> cast_percentage(const std::string &s)
|
|
{
|
|
if (s.back() == '%')
|
|
{
|
|
auto number = cast_numeric(s.substr(0, s.size() - 1));
|
|
|
|
if (number.first)
|
|
{
|
|
return{ true, number.second / 100 };
|
|
}
|
|
}
|
|
|
|
return { false, 0 };
|
|
}
|
|
|
|
std::pair<bool, xlnt::time> cast_time(const std::string &s)
|
|
{
|
|
xlnt::time result;
|
|
|
|
try
|
|
{
|
|
auto last_colon = s.find_last_of(':');
|
|
if (last_colon == std::string::npos) return { false, result };
|
|
double seconds = std::stod(s.substr(last_colon + 1));
|
|
result.second = static_cast<int>(seconds);
|
|
result.microsecond = static_cast<int>((seconds - static_cast<double>(result.second)) * 1e6);
|
|
|
|
auto first_colon = s.find_first_of(':');
|
|
|
|
if (first_colon == last_colon)
|
|
{
|
|
auto decimal_pos = s.find('.');
|
|
if (decimal_pos != std::string::npos)
|
|
{
|
|
result.minute = std::stoi(s.substr(0, first_colon));
|
|
}
|
|
else
|
|
{
|
|
result.hour = std::stoi(s.substr(0, first_colon));
|
|
result.minute = result.second;
|
|
result.second = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
result.hour = std::stoi(s.substr(0, first_colon));
|
|
result.minute = std::stoi(s.substr(first_colon + 1, last_colon - first_colon - 1));
|
|
}
|
|
}
|
|
catch (std::invalid_argument)
|
|
{
|
|
return{ false, result };
|
|
}
|
|
|
|
return { true, result };
|
|
}
|
|
|
|
} // namespace
|
|
|
|
namespace xlnt {
|
|
|
|
class style;
|
|
|
|
namespace detail {
|
|
|
|
struct worksheet_impl;
|
|
|
|
struct cell_impl
|
|
{
|
|
cell_impl();
|
|
cell_impl(column_t column, row_t row);
|
|
cell_impl(worksheet_impl *parent, column_t column, row_t row);
|
|
cell_impl(const cell_impl &rhs);
|
|
cell_impl &operator=(const cell_impl &rhs);
|
|
|
|
style &get_style(bool create_if_null)
|
|
{
|
|
if(style_ == nullptr && create_if_null)
|
|
{
|
|
style_.reset(new style());
|
|
}
|
|
|
|
return *style_;
|
|
}
|
|
|
|
const style &get_style() const
|
|
{
|
|
if(style_ == nullptr)
|
|
{
|
|
throw std::runtime_error("call has_style to check if const cell has a style before accessing it");
|
|
}
|
|
|
|
return *style_;
|
|
}
|
|
|
|
void set_date(long double number, xlnt::number_format::format format_code)
|
|
{
|
|
is_date_ = true;
|
|
auto number_format = xlnt::number_format(format_code);
|
|
get_style(true).set_number_format(number_format);
|
|
value_numeric_ = number;
|
|
type_ = cell::type::numeric;
|
|
}
|
|
|
|
void set_string(const std::string &s, bool guess_types)
|
|
{
|
|
is_date_ = false;
|
|
value_string_ = check_string(s);
|
|
type_ = cell::type::string;
|
|
|
|
if (value_string_.size() > 1 && value_string_.front() == '=')
|
|
{
|
|
formula_ = value_string_;
|
|
type_ = cell::type::formula;
|
|
value_string_.clear();
|
|
}
|
|
else if(cell::ErrorCodes.find(s) != cell::ErrorCodes.end())
|
|
{
|
|
type_ = cell::type::error;
|
|
}
|
|
else if(guess_types)
|
|
{
|
|
auto percentage = cast_percentage(s);
|
|
|
|
if (percentage.first)
|
|
{
|
|
value_numeric_ = percentage.second;
|
|
type_ = cell::type::numeric;
|
|
get_style(true).get_number_format().set_format_code(xlnt::number_format::format::percentage);
|
|
}
|
|
else
|
|
{
|
|
auto time = cast_time(s);
|
|
|
|
if (time.first)
|
|
{
|
|
set_date(time.second.to_number(), xlnt::number_format::format::date_time6);
|
|
}
|
|
else
|
|
{
|
|
auto numeric = cast_numeric(s);
|
|
|
|
if (numeric.first)
|
|
{
|
|
value_numeric_ = numeric.second;
|
|
type_ = cell::type::numeric;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
cell::type type_;
|
|
|
|
worksheet_impl *parent_;
|
|
|
|
column_t column_;
|
|
row_t row_;
|
|
|
|
std::string value_string_;
|
|
long double value_numeric_;
|
|
|
|
std::string formula_;
|
|
|
|
bool has_hyperlink_;
|
|
relationship hyperlink_;
|
|
|
|
bool is_merged_;
|
|
bool is_date_;
|
|
|
|
std::size_t xf_index_;
|
|
|
|
std::unique_ptr<style> style_;
|
|
std::unique_ptr<comment_impl> comment_;
|
|
};
|
|
|
|
} // namespace detail
|
|
} // namespace xlnt
|