mirror of
https://github.com/fmtlib/fmt.git
synced 2025-01-15 19:51:19 +00:00
Update docs
This commit is contained in:
parent
0e01e46c11
commit
3baaa8d899
132
doc/api.rst
132
doc/api.rst
@ -108,7 +108,7 @@ with the same format specifiers. The ``format_as`` function should take an
|
|||||||
object of your type and return an object of a formattable type. It should be
|
object of your type and return an object of a formattable type. It should be
|
||||||
defined in the same namespace as your type.
|
defined in the same namespace as your type.
|
||||||
|
|
||||||
Example (https://godbolt.org/z/r7vvGE1v7)::
|
Example (https://godbolt.org/z/nvME4arz8)::
|
||||||
|
|
||||||
#include <fmt/format.h>
|
#include <fmt/format.h>
|
||||||
|
|
||||||
@ -123,67 +123,13 @@ Example (https://godbolt.org/z/r7vvGE1v7)::
|
|||||||
fmt::print("{}\n", kevin_namespacy::film::se7en); // prints "7"
|
fmt::print("{}\n", kevin_namespacy::film::se7en); // prints "7"
|
||||||
}
|
}
|
||||||
|
|
||||||
Using the specialization API is more complex but gives you full control over
|
Using specialization is more complex but gives you full control over parsing and
|
||||||
parsing and formatting. To use this method specialize the ``formatter`` struct
|
formatting. To use this method specialize the ``formatter`` struct template for
|
||||||
template for your type and implement ``parse`` and ``format`` methods.
|
your type and implement ``parse`` and ``format`` methods.
|
||||||
For example::
|
|
||||||
|
|
||||||
#include <fmt/core.h>
|
The recommended way of defining a formatter is by reusing an existing one via
|
||||||
|
inheritance or composition. This way you can support standard format specifiers
|
||||||
struct point {
|
without implementing them yourself. For example::
|
||||||
double x, y;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <> struct fmt::formatter<point> {
|
|
||||||
// Presentation format: 'f' - fixed, 'e' - exponential.
|
|
||||||
char presentation = 'f';
|
|
||||||
|
|
||||||
// Parses format specifications of the form ['f' | 'e'].
|
|
||||||
constexpr auto parse(format_parse_context& ctx) -> format_parse_context::iterator {
|
|
||||||
// [ctx.begin(), ctx.end()) is a character range that contains a part of
|
|
||||||
// the format string starting from the format specifications to be parsed,
|
|
||||||
// e.g. in
|
|
||||||
//
|
|
||||||
// fmt::format("{:f} - point of interest", point{1, 2});
|
|
||||||
//
|
|
||||||
// the range will contain "f} - point of interest". The formatter should
|
|
||||||
// parse specifiers until '}' or the end of the range. In this example
|
|
||||||
// the formatter should parse the 'f' specifier and return an iterator
|
|
||||||
// pointing to '}'.
|
|
||||||
|
|
||||||
// Please also note that this character range may be empty, in case of
|
|
||||||
// the "{}" format string, so therefore you should check ctx.begin()
|
|
||||||
// for equality with ctx.end().
|
|
||||||
|
|
||||||
// Parse the presentation format and store it in the formatter:
|
|
||||||
auto it = ctx.begin(), end = ctx.end();
|
|
||||||
if (it != end && (*it == 'f' || *it == 'e')) presentation = *it++;
|
|
||||||
|
|
||||||
// Check if reached the end of the range:
|
|
||||||
if (it != end && *it != '}') throw_format_error("invalid format");
|
|
||||||
|
|
||||||
// Return an iterator past the end of the parsed range:
|
|
||||||
return it;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Formats the point p using the parsed format specification (presentation)
|
|
||||||
// stored in this formatter.
|
|
||||||
auto format(const point& p, format_context& ctx) const -> format_context::iterator {
|
|
||||||
// ctx.out() is an output iterator to write to.
|
|
||||||
return presentation == 'f'
|
|
||||||
? fmt::format_to(ctx.out(), "({:.1f}, {:.1f})", p.x, p.y)
|
|
||||||
: fmt::format_to(ctx.out(), "({:.1e}, {:.1e})", p.x, p.y);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Then you can pass objects of type ``point`` to any formatting function::
|
|
||||||
|
|
||||||
point p = {1, 2};
|
|
||||||
std::string s = fmt::format("{:f}", p);
|
|
||||||
// s == "(1.0, 2.0)"
|
|
||||||
|
|
||||||
You can also reuse existing formatters via inheritance or composition, for
|
|
||||||
example::
|
|
||||||
|
|
||||||
// color.h:
|
// color.h:
|
||||||
#include <fmt/core.h>
|
#include <fmt/core.h>
|
||||||
@ -211,9 +157,9 @@ example::
|
|||||||
}
|
}
|
||||||
|
|
||||||
Note that ``formatter<string_view>::format`` is defined in ``fmt/format.h`` so
|
Note that ``formatter<string_view>::format`` is defined in ``fmt/format.h`` so
|
||||||
it has to be included in the source file.
|
it has to be included in the source file. Since ``parse`` is inherited from
|
||||||
Since ``parse`` is inherited from ``formatter<string_view>`` it will recognize
|
``formatter<string_view>`` it will recognize all string format specifications,
|
||||||
all string format specifications, for example
|
for example
|
||||||
|
|
||||||
.. code-block:: c++
|
.. code-block:: c++
|
||||||
|
|
||||||
@ -221,6 +167,64 @@ all string format specifications, for example
|
|||||||
|
|
||||||
will return ``" blue"``.
|
will return ``" blue"``.
|
||||||
|
|
||||||
|
The experimental ``nested_formatter`` provides an easy way applying a formatter
|
||||||
|
to one or more subobjects.
|
||||||
|
|
||||||
|
For example::
|
||||||
|
|
||||||
|
#include <fmt/format.h>
|
||||||
|
|
||||||
|
struct point {
|
||||||
|
double x, y;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct fmt::formatter<point> : nested_formatter<double> {
|
||||||
|
auto format(point p, format_context& ctx) const {
|
||||||
|
return write_padded(ctx, [=](auto out) {
|
||||||
|
return format_to(out, "({}, {})", nested(p.x), nested(p.y));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
fmt::print("[{:>20.2f}]", point{1, 2});
|
||||||
|
}
|
||||||
|
|
||||||
|
prints::
|
||||||
|
|
||||||
|
[ (1.00, 2.00)]
|
||||||
|
|
||||||
|
Notice that fill, align and width are applied to the whole object which is the
|
||||||
|
recommended behavior while the remaining specifiers apply to elements.
|
||||||
|
|
||||||
|
In general the formatter has the following form::
|
||||||
|
|
||||||
|
template <> struct fmt::formatter<T> {
|
||||||
|
// Parses format specifiers and stores them in the formatter.
|
||||||
|
//
|
||||||
|
// [ctx.begin(), ctx.end()) is a, possibly empty, character range that
|
||||||
|
// contains a part of the format string starting from the format
|
||||||
|
// specifications to be parsed, e.g. in
|
||||||
|
//
|
||||||
|
// fmt::format("{:f} continued", ...);
|
||||||
|
//
|
||||||
|
// the range will contain "f} continued". The formatter should parse
|
||||||
|
// specifiers until '}' or the end of the range. In this example the
|
||||||
|
// formatter should parse the 'f' specifier and return an iterator
|
||||||
|
// pointing to '}'.
|
||||||
|
constexpr auto parse(format_parse_context& ctx)
|
||||||
|
-> format_parse_context::iterator;
|
||||||
|
|
||||||
|
// Formats value using the parsed format specification stored in this
|
||||||
|
// formatter and writes the output to ctx.out().
|
||||||
|
auto format(const T& value, format_context& ctx) const
|
||||||
|
-> format_context::iterator;
|
||||||
|
};
|
||||||
|
|
||||||
|
It is recommended to at least support fill, align and width that apply to the
|
||||||
|
whole object and have the same semantics as in standard formatters.
|
||||||
|
|
||||||
You can also write a formatter for a hierarchy of classes::
|
You can also write a formatter for a hierarchy of classes::
|
||||||
|
|
||||||
// demo.h:
|
// demo.h:
|
||||||
|
Loading…
Reference in New Issue
Block a user