gnucash/libgnucash/app-utils/gnc-optiondb.cpp
John Ralls 76172af239 Implement saving and loading OptionDB items to/from scheme and
key-value string representations.
2021-07-10 17:24:57 -07:00

590 lines
21 KiB
C++

/********************************************************************\
* gnc-optiondb.cpp -- Collection of GncOption objects *
* Copyright (C) 2019 John Ralls <jralls@ceridwen.us> *
* *
* This program is free software; you can redistribute it and/or *
* modify it under the terms of the GNU General Public License as *
* published by the Free Software Foundation; either version 2 of *
* the License, or (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License*
* along with this program; if not, contact: *
* *
* Free Software Foundation Voice: +1-617-542-5942 *
* 51 Franklin Street, Fifth Floor Fax: +1-617-542-2652 *
* Boston, MA 02110-1301, USA gnu@gnu.org *
* *
\********************************************************************/
#include "gnc-optiondb.hpp"
#include <limits>
#include <sstream>
auto constexpr stream_max = std::numeric_limits<std::streamsize>::max();
GncOptionDB::GncOptionDB() : m_default_section{std::nullopt} {}
GncOptionDB::GncOptionDB(QofBook* book) : GncOptionDB() {}
void
GncOptionDB::save_to_book(QofBook* book, bool do_clear) const
{
}
void
GncOptionDB::register_option(const char* section, GncOption&& option)
{
auto db_section = find_section(section);
if (db_section)
{
db_section->get().second.emplace_back(std::move(option));
return;
}
m_sections.emplace_back(std::make_pair(std::string{section},
GncOptionVec{}));
auto new_section = std::prev(m_sections.end());
new_section->second.emplace_back(std::move(option));
}
void
GncOptionDB::unregister_option(const char* section, const char* name)
{
auto db_section = find_section(section);
if (db_section)
{
db_section->get().second.erase(
std::remove_if(
db_section->get().second.begin(), db_section->get().second.end(),
[name](const GncOption& option) -> bool
{
return option.get_name() == std::string{name};
}));
}
}
void
GncOptionDB::set_default_section(const char* section)
{
m_default_section = find_section(section);
}
const GncOptionSection* const
GncOptionDB::get_default_section() const noexcept
{
if (m_default_section)
return &(m_default_section.value().get());
return nullptr;
}
void
GncOptionDB::set_ui_item(const char* section, const char* name,
GncOptionUIItem* ui_item)
{
auto option = find_option(section, name);
if (!option) return;
option->get().set_ui_item(ui_item);
}
GncOptionUIItem* const
GncOptionDB::get_ui_item(const char* section, const char* name)
{
auto option = find_option(section, name);
if (!option) return nullptr;
return option->get().get_ui_item();
}
GncOptionUIType
GncOptionDB::get_ui_type(const char* section, const char* name)
{
auto option = find_option(section, name);
if (!option) return GncOptionUIType::INTERNAL;
return option->get().get_ui_type();
}
void
GncOptionDB::set_ui_from_option(const char* section, const char* name,
std::function<void(GncOption&)> func)
{
auto option = find_option(section, name);
if (!option) return;
func(option->get());
}
void
GncOptionDB::set_option_from_ui(const char* section, const char* name,
std::function<void(GncOption&)> func)
{
auto option = find_option(section, name);
if (!option) return;
func(option->get());
}
std::optional<std::reference_wrapper<GncOptionSection>>
GncOptionDB::find_section(const std::string& section)
{
auto db_section = std::find_if(
m_sections.begin(), m_sections.end(),
[&section](GncOptionSection sect) -> bool
{
return section.compare(0, classifier_size_max, sect.first) == 0;
});
if (db_section == m_sections.end())
return std::nullopt;
return *db_section;
}
std::optional<std::reference_wrapper<GncOption>>
GncOptionDB::find_option(const std::string& section, const std::string& name) const
{
auto db_section = const_cast<GncOptionDB*>(this)->find_section(section);
if (!db_section)
return std::nullopt;
auto db_opt = std::find_if(
db_section->get().second.begin(), db_section->get().second.end(),
[&name](GncOption& option) -> bool
{
return name.compare(0, classifier_size_max - 1,
option.get_name()) == 0;
});
if (db_opt == db_section->get().second.end())
return std::nullopt;
return *db_opt;
}
std::string
GncOptionDB::lookup_string_option(const char* section, const char* name)
{
static const std::string empty_string{};
auto db_opt = find_option(section, name);
if (!db_opt)
return empty_string;
return db_opt->get().get_value<std::string>();
}
void
GncOptionDB::make_internal(const char* section, const char* name)
{
auto db_opt = find_option(section, name);
if (db_opt)
db_opt->get().make_internal();
}
std::ostream&
GncOptionDB::save_option_scheme(std::ostream& oss,
const char* option_prolog,
const std::string& section,
const std::string& name) const noexcept
{
auto db_opt = find_option(section, name);
if (!db_opt || !db_opt->get().is_changed())
return oss;
oss << scheme_tags[0] << option_prolog << "\n";
oss << scheme_tags[1] << '"' << section.substr(0, classifier_size_max) << "\"\n";
oss << scheme_tags[1] << '"' << name.substr(0, classifier_size_max) << '"';
oss << scheme_tags[2] << "\n" << scheme_tags[3];
db_opt->get().to_scheme(oss);
oss << scheme_tags[4] << "\n\n";
return oss;
}
std::istream&
GncOptionDB::load_option_scheme(std::istream& iss) noexcept
{
std::string section{classifier_size_max};
std::string name{classifier_size_max};
iss.ignore(strlen(scheme_tags[0]) + 7, '\n'); //throw away the scheme noise;
iss >> section; // Whitespace automatically discarded.
iss >> name; // Ditto
if (section.size() > 2)
section = section.substr(1, section.size() - 2); //Trim the quotes
if (name.size() > 2 + strlen(scheme_tags[2]))
name = name.substr(1, name.size() - (2 + strlen(scheme_tags[2])));
auto option = find_option(section.c_str(), name.c_str());
if (!option)
{
std::cerr << "Option " << section << ":" << name << " not found." << std::endl;
iss.ignore(stream_max, '\n'); // No option, discard the line
iss.ignore(stream_max, '\n'); // And the trailing newline
return iss;
}
iss.ignore(strlen(scheme_tags[2]) +1, '\n');
iss.ignore(strlen(scheme_tags[3]));
option->get().from_scheme(iss);
iss.ignore(strlen(scheme_tags[4]) + 2, '\n'); //discard the noise at the end.
return iss;
}
std::ostream&
GncOptionDB::save_option_key_value(std::ostream& oss,
const char* section,
const char* name) const noexcept
{
auto db_opt = find_option(section, name);
if (!db_opt || !db_opt->get().is_changed())
return oss;
oss << section << ":" << name << "=" << db_opt->get() << ";";
return oss;
}
std::istream&
GncOptionDB::load_option_key_value(std::istream& iss)
{
char section[classifier_size_max], name[classifier_size_max];
iss.getline(section, classifier_size_max, ':');
iss.getline(name, classifier_size_max, '=');
if (!iss)
throw std::invalid_argument("Section or name delimiter not found or values too long");
auto option = find_option(section, name);
if (!option)
iss.ignore(stream_max, ';');
else
{
std::string value;
std::getline(iss, value, ';');
std::istringstream item_iss{value};
item_iss >> option->get();
}
return iss;
}
GncOptionDBPtr
gnc_option_db_new(void)
{
return GncOptionDBPtr{new GncOptionDB};
}
void
gnc_register_string_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::STRING};
db->register_option(section, std::move(option));
}
void
gnc_register_text_option(const GncOptionDBPtr& db, const char* section, const char* name,
const char* key, const char* doc_string,
std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::TEXT};
db->register_option(section, std::move(option));
}
void
gnc_register_font_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::FONT};
db->register_option(section, std::move(option));
}
void
gnc_register_budget_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, GncBudget *value)
{
GncOption option{section, name, key, doc_string, QOF_INSTANCE(value),
GncOptionUIType::BUDGET};
db->register_option(section, std::move(option));
}
void
gnc_register_color_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::COLOR};
db->register_option(section, std::move(option));
}
void
gnc_register_commodity_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, gnc_commodity *value)
{
GncOption option{section, name, key, doc_string, QOF_INSTANCE(value),
GncOptionUIType::COMMODITY};
db->register_option(section, std::move(option));
}
void
gnc_register_simple_boolean_option(const GncOptionDBPtr& db,
const char* section, const char* name,
const char* key, const char* doc_string,
bool value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::INTERNAL};
db->register_option(section, std::move(option));
}
void
gnc_register_complex_boolean_option(const GncOptionDBPtr& db,
const char* section, const char* name,
const char* key, const char* doc_string,
bool value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::BOOLEAN};
db->register_option(section, std::move(option));
}
void
gnc_register_pixmap_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::PIXMAP};
db->register_option(section, std::move(option));
}
void
gnc_register_account_list_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string,
const GncOptionAccountList& value)
{
GncOption option{GncOptionAccountValue{section, name, key, doc_string,
GncOptionUIType::ACCOUNT_LIST, value}};
db->register_option(section, std::move(option));
}
void
gnc_register_account_list_limited_option(const GncOptionDBPtr& db,
const char* section, const char* name,
const char* key,
const char* doc_string,
const GncOptionAccountList& value,
GncOptionAccountTypeList&& allowed)
{
try
{
GncOption option{GncOptionAccountValue{section, name, key, doc_string,
GncOptionUIType::ACCOUNT_LIST, value, std::move(allowed)}};
db->register_option(section, std::move(option));
}
catch (const std::invalid_argument& err)
{
std::cerr << "Account List Limited Option, value failed validation, option not registered.\n";
}
}
using AccountPair = std::pair<GncOptionAccountList&,
const GncOptionAccountTypeList&>;
static void
find_children(Account* account, void* data)
{
auto datapair =
(AccountPair*)data;
GncOptionAccountList& list = datapair->first;
const GncOptionAccountTypeList& types = datapair->second;
if (std::find(types.begin(), types.end(),
xaccAccountGetType(account)) != types.end())
list.push_back(account);
}
GncOptionAccountList
gnc_account_list_from_types(QofBook *book,
const GncOptionAccountTypeList& types)
{
GncOptionAccountList list;
AccountPair funcdata{list, types};
Account* base_acct = gnc_book_get_root_account(book);
gnc_account_foreach_descendant(base_acct, (AccountCb)find_children,
&funcdata);
return list;
}
void
gnc_register_account_sel_limited_option(const GncOptionDBPtr& db,
const char* section, const char* name,
const char* key, const char* doc_string,
const GncOptionAccountList& value,
GncOptionAccountTypeList&& allowed)
{
try
{
GncOption option{GncOptionAccountValue{section, name, key, doc_string,
GncOptionUIType::ACCOUNT_SEL, value, std::move(allowed)}};
db->register_option(section, std::move(option));
}
catch (const std::invalid_argument& err)
{
std::cerr <<"Account Sel Limited Option, value failed validation, option not registerd.\n";
}
}
void
gnc_register_multichoice_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string,
GncMultiChoiceOptionChoices&& choices)
{
GncOption option{GncOptionMultichoiceValue{section, name, key, doc_string,
std::get<0>(choices.at(0)).c_str(), std::move(choices)}};
db->register_option(section, std::move(option));
}
void
gnc_register_list_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, const char* value,
GncMultiChoiceOptionChoices&& list)
{
GncOption option{GncOptionMultichoiceValue{section, name, key, doc_string,
value, std::move(list), GncOptionUIType::LIST}};
db->register_option(section, std::move(option));
}
/* Only balance-forecast.scm, hello-world.scm, and net-charts.scm
* use decimals and fractional steps and they can be worked around. */
void
gnc_register_number_range_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, int value, int min,
int max, int step)
{
GncOption option{GncOptionRangeValue<int>{section, name, key, doc_string,
value, min, max, step}};
db->register_option(section, std::move(option));
}
void
gnc_register_number_plot_size_option(const GncOptionDBPtr& db,
const char* section, const char* name,
const char* key, const char* doc_string,
int value)
{
GncOption option{GncOptionRangeValue<int>{section, name, key, doc_string,
value, 100, 20000, 5}};
db->register_option(section, std::move(option));
}
void
gnc_register_query_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, QofQuery* value)
{
GncOption option{section, name, key, doc_string, QOF_INSTANCE(value),
GncOptionUIType::INTERNAL};
db->register_option(section, std::move(option));
}
void
gnc_register_internal_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::INTERNAL};
db->register_option(section, std::move(option));
}
void
gnc_register_invoice_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, GncInvoice* value)
{
GncOption option{section, name, key, doc_string, QOF_INSTANCE(value),
GncOptionUIType::INVOICE};
db->register_option(section, std::move(option));
}
void
gnc_register_owner_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, GncOwner* value)
{
GncOption option{section, name, key, doc_string, QOF_INSTANCE(value),
GncOptionUIType::OWNER};
db->register_option(section, std::move(option));
}
void
gnc_register_taxtable_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, GncTaxTable* value)
{
GncOption option{section, name, key, doc_string, QOF_INSTANCE(value),
GncOptionUIType::TAX_TABLE};
db->register_option(section, std::move(option));
}
void
gnc_register_counter_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, int value)
{
GncOption option{GncOptionRangeValue<int>{section, name, key, doc_string,
value, 0, 999999999, 1}};
db->register_option(section, std::move(option));
}
void
gnc_register_counter_format_option(const GncOptionDBPtr& db,
const char* section, const char* name,
const char* key, const char* doc_string,
std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::STRING};
db->register_option(section, std::move(option));
}
void
gnc_register_dateformat_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, std::string value)
{
GncOption option{section, name, key, doc_string, value,
GncOptionUIType::DATE_FORMAT};
db->register_option(section, std::move(option));
}
void
gnc_register_currency_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string, gnc_commodity *value)
{
GncOption option{GncOptionValidatedValue<QofInstance*>{
section, name, key, doc_string, QOF_INSTANCE(value),
[](QofInstance* new_value) -> bool
{
return GNC_IS_COMMODITY (new_value) &&
gnc_commodity_is_currency(GNC_COMMODITY(new_value));
},
GncOptionUIType::CURRENCY
}};
db->register_option(section, std::move(option));
}
void
gnc_register_date_interval_option(const GncOptionDBPtr& db, const char* section,
const char* name, const char* key,
const char* doc_string,
RelativeDatePeriod period)
{
GncOption option{GncOptionDateValue(section, name, key, doc_string, period)};
db->register_option(section, std::move(option));
}