mirror of
https://github.com/pocoproject/poco.git
synced 2024-12-13 10:32:57 +01:00
298 lines
5.5 KiB
C++
298 lines
5.5 KiB
C++
//
|
|
// HelpFormatter.cpp
|
|
//
|
|
// Library: Util
|
|
// Package: Options
|
|
// Module: HelpFormatter
|
|
//
|
|
// Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "Poco/Util/HelpFormatter.h"
|
|
#include "Poco/Util/OptionSet.h"
|
|
#include "Poco/Util/Option.h"
|
|
|
|
|
|
namespace Poco {
|
|
namespace Util {
|
|
|
|
|
|
const int HelpFormatter::TAB_WIDTH = 4;
|
|
const int HelpFormatter::LINE_WIDTH = 78;
|
|
|
|
|
|
HelpFormatter::HelpFormatter(const OptionSet& options):
|
|
_options(options),
|
|
_width(LINE_WIDTH),
|
|
_indent(0),
|
|
_unixStyle(true)
|
|
{
|
|
#if !defined(POCO_OS_FAMILY_UNIX)
|
|
_unixStyle = false;
|
|
#endif
|
|
_indent = calcIndent();
|
|
}
|
|
|
|
|
|
HelpFormatter::~HelpFormatter()
|
|
{
|
|
}
|
|
|
|
|
|
void HelpFormatter::setCommand(const std::string& command)
|
|
{
|
|
_command = command;
|
|
}
|
|
|
|
|
|
void HelpFormatter::setUsage(const std::string& usage)
|
|
{
|
|
_usage = usage;
|
|
}
|
|
|
|
|
|
void HelpFormatter::setHeader(const std::string& header)
|
|
{
|
|
_header = header;
|
|
}
|
|
|
|
|
|
void HelpFormatter::setFooter(const std::string& footer)
|
|
{
|
|
_footer = footer;
|
|
}
|
|
|
|
|
|
void HelpFormatter::format(std::ostream& ostr) const
|
|
{
|
|
ostr << "usage: " << _command;
|
|
if (!_usage.empty())
|
|
{
|
|
ostr << ' ';
|
|
formatText(ostr, _usage, (int) _command.length() + 1);
|
|
}
|
|
ostr << '\n';
|
|
if (!_header.empty())
|
|
{
|
|
formatText(ostr, _header, 0);
|
|
ostr << "\n\n";
|
|
}
|
|
formatOptions(ostr);
|
|
if (!_footer.empty())
|
|
{
|
|
ostr << '\n';
|
|
formatText(ostr, _footer, 0);
|
|
ostr << '\n';
|
|
}
|
|
}
|
|
|
|
|
|
void HelpFormatter::setWidth(int width)
|
|
{
|
|
poco_assert (width > 0);
|
|
|
|
_width = width;
|
|
}
|
|
|
|
|
|
void HelpFormatter::setIndent(int indent)
|
|
{
|
|
poco_assert (indent >= 0 && indent < _width);
|
|
|
|
_indent = indent;
|
|
}
|
|
|
|
|
|
void HelpFormatter::setAutoIndent()
|
|
{
|
|
_indent = calcIndent();
|
|
}
|
|
|
|
|
|
void HelpFormatter::setUnixStyle(bool flag)
|
|
{
|
|
_unixStyle = flag;
|
|
}
|
|
|
|
|
|
int HelpFormatter::calcIndent() const
|
|
{
|
|
int indent = 0;
|
|
for (const auto& opt: _options)
|
|
{
|
|
auto shortLen = opt.shortName().length();
|
|
auto fullLen = opt.fullName().length();
|
|
std::size_t n = 0;
|
|
if (_unixStyle && shortLen > 0)
|
|
{
|
|
n += shortLen + shortPrefix().length();
|
|
if (opt.takesArgument())
|
|
n += opt.argumentName().length() + (opt.argumentRequired() ? 0 : 2);
|
|
if (fullLen > 0) n += 2;
|
|
}
|
|
if (fullLen > 0)
|
|
{
|
|
n += fullLen + longPrefix().length();
|
|
if (opt.takesArgument())
|
|
n += 1 + opt.argumentName().length() + (opt.argumentRequired() ? 0 : 2);
|
|
}
|
|
n += 2;
|
|
if (n > indent)
|
|
indent = static_cast<int>(n);
|
|
}
|
|
return indent;
|
|
}
|
|
|
|
|
|
void HelpFormatter::formatOptions(std::ostream& ostr) const
|
|
{
|
|
int optWidth = calcIndent();
|
|
for (const auto& opt: _options)
|
|
{
|
|
formatOption(ostr, opt, optWidth);
|
|
if (_indent < optWidth)
|
|
{
|
|
ostr << '\n' << std::string(_indent, ' ');
|
|
formatText(ostr, opt.description(), _indent, _indent);
|
|
}
|
|
else
|
|
{
|
|
formatText(ostr, opt.description(), _indent, optWidth);
|
|
}
|
|
ostr << '\n';
|
|
}
|
|
}
|
|
|
|
|
|
void HelpFormatter::formatOption(std::ostream& ostr, const Option& option, int width) const
|
|
{
|
|
int shortLen = (int) option.shortName().length();
|
|
int fullLen = (int) option.fullName().length();
|
|
|
|
int n = 0;
|
|
if (_unixStyle && shortLen > 0)
|
|
{
|
|
ostr << shortPrefix() << option.shortName();
|
|
n += (int) shortPrefix().length() + (int) option.shortName().length();
|
|
if (option.takesArgument())
|
|
{
|
|
ostr << ' ';
|
|
if (!option.argumentRequired()) { ostr << '['; ++n; }
|
|
ostr << option.argumentName();
|
|
n += (int) option.argumentName().length();
|
|
if (!option.argumentRequired()) { ostr << ']'; ++n; }
|
|
}
|
|
if (fullLen > 0) { ostr << ", "; n += 2; }
|
|
}
|
|
if (fullLen > 0)
|
|
{
|
|
ostr << longPrefix() << option.fullName();
|
|
n += (int) longPrefix().length() + (int) option.fullName().length();
|
|
if (option.takesArgument())
|
|
{
|
|
if (!option.argumentRequired()) { ostr << '['; ++n; }
|
|
ostr << '=';
|
|
++n;
|
|
ostr << option.argumentName();
|
|
n += (int) option.argumentName().length();
|
|
if (!option.argumentRequired()) { ostr << ']'; ++n; }
|
|
}
|
|
}
|
|
while (n < width) { ostr << ' '; ++n; }
|
|
}
|
|
|
|
|
|
void HelpFormatter::formatText(std::ostream& ostr, const std::string& text, int indent) const
|
|
{
|
|
formatText(ostr, text, indent, indent);
|
|
}
|
|
|
|
|
|
void HelpFormatter::formatText(std::ostream& ostr, const std::string& text, int indent, int firstIndent) const
|
|
{
|
|
int pos = firstIndent;
|
|
int maxWordLen = _width - indent;
|
|
std::string word;
|
|
for (auto ch: text)
|
|
{
|
|
if (ch == '\n')
|
|
{
|
|
clearWord(ostr, pos, word, indent);
|
|
ostr << '\n';
|
|
pos = 0;
|
|
while (pos < indent) { ostr << ' '; ++pos; }
|
|
}
|
|
else if (ch == '\t')
|
|
{
|
|
clearWord(ostr, pos, word, indent);
|
|
if (pos < _width) ++pos;
|
|
while (pos < _width && pos % TAB_WIDTH != 0)
|
|
{
|
|
ostr << ' ';
|
|
++pos;
|
|
}
|
|
}
|
|
else if (ch == ' ')
|
|
{
|
|
clearWord(ostr, pos, word, indent);
|
|
if (pos < _width) { ostr << ' '; ++pos; }
|
|
}
|
|
else
|
|
{
|
|
if (word.length() == maxWordLen)
|
|
{
|
|
clearWord(ostr, pos, word, indent);
|
|
}
|
|
else word += ch;
|
|
}
|
|
}
|
|
clearWord(ostr, pos, word, indent);
|
|
}
|
|
|
|
|
|
void HelpFormatter::formatWord(std::ostream& ostr, int& pos, const std::string& word, int indent) const
|
|
{
|
|
if (pos + word.length() > _width)
|
|
{
|
|
ostr << '\n';
|
|
pos = 0;
|
|
while (pos < indent) { ostr << ' '; ++pos; }
|
|
}
|
|
ostr << word;
|
|
pos += (int) word.length();
|
|
}
|
|
|
|
|
|
void HelpFormatter::clearWord(std::ostream& ostr, int& pos, std::string& word, int indent) const
|
|
{
|
|
formatWord(ostr, pos, word, indent);
|
|
word.clear();
|
|
}
|
|
|
|
|
|
std::string HelpFormatter::shortPrefix() const
|
|
{
|
|
#if defined(POCO_OS_FAMILY_UNIX)
|
|
return "-";
|
|
#else
|
|
return _unixStyle ? "-" : "/";
|
|
#endif
|
|
}
|
|
|
|
|
|
std::string HelpFormatter::longPrefix() const
|
|
{
|
|
#if defined(POCO_OS_FAMILY_UNIX)
|
|
return "--";
|
|
#else
|
|
return _unixStyle ? "--" : "/";
|
|
#endif
|
|
}
|
|
|
|
|
|
} } // namespace Poco::Util
|