Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
F
fmt
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Libraries
fmt
Commits
d4ddaaf2
Commit
d4ddaaf2
authored
Jul 20, 2016
by
Victor Zverovich
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Make PrintfFormatter public (#335, #360)
parent
fa0f870a
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
41 additions
and
22 deletions
+41
-22
doc/api.rst
doc/api.rst
+9
-3
fmt/format.cc
fmt/format.cc
+2
-2
fmt/printf.h
fmt/printf.h
+29
-16
test/custom-formatter-test.cc
test/custom-formatter-test.cc
+1
-1
No files found.
doc/api.rst
View file @
d4ddaaf2
...
...
@@ -125,12 +125,15 @@ custom argument formatter class::
.. doxygenclass:: fmt::ArgFormatter
:members:
Printf formatting
functions
-----------------
----------
Printf formatting
-----------------
The header ``fmt/printf.h`` provides ``printf``-like formatting functionality.
The following functions use `printf format string syntax
<http://pubs.opengroup.org/onlinepubs/009695399/functions/fprintf.html>`_ with
a POSIX extension for positional arguments.
the POSIX extension for positional arguments. Unlike their standard
counterparts, the ``fmt`` functions are type-safe and throw an exception if an
argument type doesn't match its format specification.
.. doxygenfunction:: printf(CStringRef, ArgList)
...
...
@@ -138,6 +141,9 @@ a POSIX extension for positional arguments.
.. doxygenfunction:: sprintf(CStringRef, ArgList)
.. doxygenclass:: fmt::PrintfFormatter
:members:
Write API
=========
...
...
fmt/format.cc
View file @
d4ddaaf2
...
...
@@ -521,7 +521,7 @@ template void internal::FixedBuffer<char>::grow(std::size_t);
template
void
internal
::
ArgMap
<
char
>
::
init
(
const
ArgList
&
args
);
template
void
internal
::
PrintfFormatter
<
char
>
::
format
(
template
void
PrintfFormatter
<
char
>
::
format
(
BasicWriter
<
char
>
&
writer
,
CStringRef
format
);
template
int
internal
::
CharTraits
<
char
>
::
format_float
(
...
...
@@ -538,7 +538,7 @@ template void internal::FixedBuffer<wchar_t>::grow(std::size_t);
template
void
internal
::
ArgMap
<
wchar_t
>
::
init
(
const
ArgList
&
args
);
template
void
internal
::
PrintfFormatter
<
wchar_t
>
::
format
(
template
void
PrintfFormatter
<
wchar_t
>
::
format
(
BasicWriter
<
wchar_t
>
&
writer
,
WCStringRef
format
);
template
int
internal
::
CharTraits
<
wchar_t
>
::
format_float
(
...
...
fmt/printf.h
View file @
d4ddaaf2
...
...
@@ -248,23 +248,34 @@ class PrintfArgFormatter
PrintfArgFormatter
(
BasicWriter
<
Char
>
&
w
,
FormatSpec
&
s
)
:
BasicPrintfArgFormatter
<
PrintfArgFormatter
<
Char
>
,
Char
>
(
w
,
s
)
{}
};
}
// namespace internal
// A printf formatter.
template
<
typename
Char
,
typename
ArgFormatter
=
PrintfArgFormatter
<
Char
>
>
class
PrintfFormatter
:
private
FormatterBase
{
/** This template formats data and writes the output to a writer. */
template
<
typename
Char
,
typename
ArgFormatter
=
internal
::
PrintfArgFormatter
<
Char
>
>
class
PrintfFormatter
:
private
internal
::
FormatterBase
{
private:
void
parse_flags
(
FormatSpec
&
spec
,
const
Char
*&
s
);
// Returns the argument with specified index or, if arg_index is equal
// to the maximum unsigned value, the next argument.
Arg
get_arg
(
const
Char
*
s
,
internal
::
Arg
get_arg
(
const
Char
*
s
,
unsigned
arg_index
=
(
std
::
numeric_limits
<
unsigned
>::
max
)());
// Parses argument index, flags and width and returns the argument index.
unsigned
parse_header
(
const
Char
*&
s
,
FormatSpec
&
spec
);
public:
/**
\rst
Constructs a ``PrintfFormatter`` object. References to the arguments
are stored in the formatter object so make sure they have appropriate
lifetimes.
\endrst
*/
explicit
PrintfFormatter
(
const
ArgList
&
args
)
:
FormatterBase
(
args
)
{}
FMT_API
void
format
(
BasicWriter
<
Char
>
&
writer
,
BasicCStringRef
<
Char
>
format_str
);
};
...
...
@@ -296,10 +307,11 @@ void PrintfFormatter<Char, AF>::parse_flags(FormatSpec &spec, const Char *&s) {
}
template
<
typename
Char
,
typename
AF
>
Arg
PrintfFormatter
<
Char
,
AF
>::
get_arg
(
const
Char
*
s
,
unsigned
arg_index
)
{
internal
::
Arg
PrintfFormatter
<
Char
,
AF
>::
get_arg
(
const
Char
*
s
,
unsigned
arg_index
)
{
(
void
)
s
;
const
char
*
error
=
0
;
Arg
arg
=
arg_index
==
std
::
numeric_limits
<
unsigned
>::
max
()
?
internal
::
Arg
arg
=
arg_index
==
std
::
numeric_limits
<
unsigned
>::
max
()
?
next_arg
(
error
)
:
FormatterBase
::
get_arg
(
arg_index
-
1
,
error
);
if
(
error
)
FMT_THROW
(
FormatError
(
!*
s
?
"invalid format string"
:
error
));
...
...
@@ -314,7 +326,7 @@ unsigned PrintfFormatter<Char, AF>::parse_header(
if
(
c
>=
'0'
&&
c
<=
'9'
)
{
// Parse an argument index (if followed by '$') or a width possibly
// preceded with '0' flag(s).
unsigned
value
=
parse_nonnegative_int
(
s
);
unsigned
value
=
internal
::
parse_nonnegative_int
(
s
);
if
(
*
s
==
'$'
)
{
// value is an argument index
++
s
;
arg_index
=
value
;
...
...
@@ -332,10 +344,10 @@ unsigned PrintfFormatter<Char, AF>::parse_header(
parse_flags
(
spec
,
s
);
// Parse width.
if
(
*
s
>=
'0'
&&
*
s
<=
'9'
)
{
spec
.
width_
=
parse_nonnegative_int
(
s
);
spec
.
width_
=
internal
::
parse_nonnegative_int
(
s
);
}
else
if
(
*
s
==
'*'
)
{
++
s
;
spec
.
width_
=
WidthHandler
(
spec
).
visit
(
get_arg
(
s
));
spec
.
width_
=
internal
::
WidthHandler
(
spec
).
visit
(
get_arg
(
s
));
}
return
arg_index
;
}
...
...
@@ -365,16 +377,17 @@ void PrintfFormatter<Char, AF>::format(
if
(
*
s
==
'.'
)
{
++
s
;
if
(
'0'
<=
*
s
&&
*
s
<=
'9'
)
{
spec
.
precision_
=
static_cast
<
int
>
(
parse_nonnegative_int
(
s
));
spec
.
precision_
=
static_cast
<
int
>
(
internal
::
parse_nonnegative_int
(
s
));
}
else
if
(
*
s
==
'*'
)
{
++
s
;
spec
.
precision_
=
PrecisionHandler
().
visit
(
get_arg
(
s
));
spec
.
precision_
=
internal
::
PrecisionHandler
().
visit
(
get_arg
(
s
));
}
}
using
internal
::
Arg
;
Arg
arg
=
get_arg
(
s
,
arg_index
);
if
(
spec
.
flag
(
HASH_FLAG
)
&&
IsZeroInt
().
visit
(
arg
))
spec
.
flags_
&=
~
to_unsigned
<
int
>
(
HASH_FLAG
);
if
(
spec
.
flag
(
HASH_FLAG
)
&&
internal
::
IsZeroInt
().
visit
(
arg
))
spec
.
flags_
&=
~
internal
::
to_unsigned
<
int
>
(
HASH_FLAG
);
if
(
spec
.
fill_
==
'0'
)
{
if
(
arg
.
type
<=
Arg
::
LAST_NUMERIC_TYPE
)
spec
.
align_
=
ALIGN_NUMERIC
;
...
...
@@ -383,6 +396,7 @@ void PrintfFormatter<Char, AF>::format(
}
// Parse length and convert the argument to the required type.
using
internal
::
ArgConverter
;
switch
(
*
s
++
)
{
case
'h'
:
if
(
*
s
==
'h'
)
...
...
@@ -426,7 +440,7 @@ void PrintfFormatter<Char, AF>::format(
break
;
case
'c'
:
// TODO: handle wchar_t
CharConverter
(
arg
).
visit
(
arg
);
internal
::
CharConverter
(
arg
).
visit
(
arg
);
break
;
}
}
...
...
@@ -438,11 +452,10 @@ void PrintfFormatter<Char, AF>::format(
}
write
(
writer
,
start
,
s
);
}
}
// namespace internal
template
<
typename
Char
>
void
printf
(
BasicWriter
<
Char
>
&
w
,
BasicCStringRef
<
Char
>
format
,
ArgList
args
)
{
internal
::
PrintfFormatter
<
Char
>
(
args
).
format
(
w
,
format
);
PrintfFormatter
<
Char
>
(
args
).
format
(
w
,
format
);
}
/**
...
...
test/custom-formatter-test.cc
View file @
d4ddaaf2
...
...
@@ -53,7 +53,7 @@ FMT_VARIADIC(std::string, custom_format, const char *)
std
::
string
custom_sprintf
(
const
char
*
fstr
,
fmt
::
ArgList
args
){
fmt
::
MemoryWriter
writer
;
fmt
::
internal
::
PrintfFormatter
<
char
,
CustomPAF
>
pfer
(
args
);
fmt
::
PrintfFormatter
<
char
,
CustomPAF
>
pfer
(
args
);
pfer
.
format
(
writer
,
fstr
);
return
writer
.
str
();
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment