Как заставить fmt::format работать с wchar_t?
Я хочу, чтобы fmt:: format возвращал wstring из STL EA, но когда я пытаюсь изменить этот код из документации - это прекрасно работает:
// Prints formatted error message.
void
vreport_error(const char *format, fmt::format_args args) {
fmt::print("Error: ");
fmt::vprint(format, args);
}
template <typename... Args>
void
report_error(const char *format, const Args &... args) {
vreport_error(format, fmt::make_format_args(args...));
}
использовать wchar_t вместо char:
// Prints formatted error message.
void vreport_error2(const wchar_t *format, fmt::format_args args);
template <typename... Args>
void
report_error2(const wchar_t *format, const Args &... args) {
vreport_error2(format, fmt::make_format_args(args...));
}
Я получил:
2>fmt\core.h(1219,1): error C2665: 'fmt::v5::basic_format_args<fmt::v5::wformat_context>::basic_format_args': none of the 4 overloads could convert all the argument types
2>fmt\core.h(1219,1): error C2665: : basic_format_args<wformat_context>(std::forward<Args>(args)...) {}
2>fmt\core.h(1219,1): error C2665: ^ (compiling source file ..\core\typedefs.cpp)
2>fmt\core.h(1207,1): message : could be 'fmt::v5::basic_format_args<fmt::v5::wformat_context>::basic_format_args(fmt::v5::basic_format_args<fmt::v5::wformat_context> &&)'
2>fmt\core.h(1207,1): message : };
2>fmt\core.h(1207,1): message : ^ (compiling source file ..\core\typedefs.cpp)
2>fmt\core.h(1207,1): message : or 'fmt::v5::basic_format_args<fmt::v5::wformat_context>::basic_format_args(const fmt::v5::basic_format_args<fmt::v5::wformat_context> &)'
2>fmt\core.h(1207,1): message : };
2>fmt\core.h(1207,1): message : ^ (compiling source file ..\core\typedefs.cpp)
2>fmt\core.h(1189,3): message : or 'fmt::v5::basic_format_args<fmt::v5::wformat_context>::basic_format_args(const fmt::v5::basic_format_arg<Context> *,int)'
2>fmt\core.h(1189,3): message : with
2>fmt\core.h(1189,3): message : [
2>fmt\core.h(1189,3): message : Context=fmt::v5::wformat_context
2>fmt\core.h(1189,3): message : ]
2>fmt\core.h(1189,3): message : basic_format_args(const format_arg* args, int count)
2>fmt\core.h(1189,3): message : ^ (compiling source file ..\core\typedefs.cpp)
2>fmt\core.h(1171,3): message : or 'fmt::v5::basic_format_args<fmt::v5::wformat_context>::basic_format_args(void)'
2>fmt\core.h(1171,3): message : basic_format_args() : types_(0) {}
2>fmt\core.h(1171,3): message : ^ (compiling source file ..\core\typedefs.cpp)
2>fmt\core.h(1219,1): message : while trying to match the argument list '(fmt::v5::format_args)'
2>fmt\core.h(1219,1): message : : basic_format_args<wformat_context>(std::forward<Args>(args)...) {}
2>fmt\core.h(1219,1): message : ^ (compiling source file ..\core\typedefs.cpp)
В качестве альтернативы, если я пытаюсь преобразовать в char, а затем преобразовать, я не могу заставить его работать более чем для одного аргумента:
wstring
fmt_msg2(const wchar_t *msg, fmt::format_args args) {
std::string s = utf8_from_utf16(msg).c_str();
std::string s_ = fmt::vformat(s, args);
return utf16_from_utf8(s_.data());
}
wstring s = fmt_msg2(L"hey {}", 24); // works
wstring s = fmt_msg2(L"hey {} {}", 24, 58); // error
1>client.cpp
1>client.cpp(757,44): error C2660: 'fmt_msg2': function does not take 3 arguments
1>client.cpp(757,44): error C2660: wstring s = fmt_msg2(L"hey {} {}", 24, 58);
1>client.cpp(757,44): error C2660: ^
1>typedefs.h(109,9): message : see declaration of 'fmt_msg2'
1>typedefs.h(109,9): message : wstring fmt_msg2(wchar_t cc *msg, fmt::format_args args);
1>typedefs.h(109,9): message : ^
Я также нашел упоминание о fmt::MemoryWriter, который может помочь мне отформатировать в wchar_t[], но похоже, что он больше не доступен?
Не уверен, что еще попробовать.
Обновить
Посмотрев на это свежим взглядом, я понял, что не оборачиваю fmt_msg2 - и теперь у меня есть это:
в файле typedefs.h:
ea::wstring vfmt_msg3(const ea::wstring &msg, fmt::wformat_args args);
template <typename... Args>
ea::wstring
fmt_msg3(const ea::wstring &format, const Args &... args) {
return vfmt_msg3(format, fmt::make_format_args<fmt::wformat_args>(args...));
}
в файле client.cpp:
ea::wstring
vfmt_msg3(const ea::wstring &msg, fmt::wformat_args args) {
std::wstring s_ = fmt::vformat(std::wstring(msg.data()), args);
return ea::wstring(s_.data());
}
wstring s = fmt_msg3(L"hey {} {}", 24, 58);
Но на Visual Studio 16.2.0 я получаю следующую ошибку:
core.h(687,39): error C2039: 'char_type': is not a member of 'fmt::v5::wformat_args'
core.h(687,39): error C2039: using char_type = typename Context::char_type;
core.h(687,39): error C2039: ^ (compiling source file client.cpp)
core.h(1216): message : see declaration of 'fmt::v5::wformat_args'
core.h(1216): message : struct wformat_args : basic_format_args<wformat_context> { (compiling source file client.cpp)
core.h(1095): message : see reference to class template instantiation 'fmt::v5::internal::value<Context>' being compiled
core.h(1095): message : with
core.h(1095): message : [
core.h(1095): message : Context=fmt::v5::wformat_args
core.h(1095): message : ]
core.h(1095): message : value_type data_[num_args + (!is_packed || num_args == 0 ? 1 : 0)]; (compiling source file client.cpp)
typedefs.h(119): message : see reference to class template instantiation 'fmt::v5::format_arg_store<fmt::v5::wformat_args,int,int>' being compiled
typedefs.h(119): message : fmt_msg3(const ea::wstring &format, const Args &... args) { (compiling source file client.cpp)
client.cpp(757): message : see reference to function template instantiation 'eastl::wstring fmt_msg3<int,int>(const eastl::wstring &,const int &,const int &)' being compiled
client.cpp(757): message : wstring s = fmt_msg3(L"hey {} {}", 24, 58);
1 ответ
Кажется, вам нужно использовать wformat_args
и явно указать wformat_context
за make_format_args
:
void vreport_error2(const wchar_t *format, fmt::wformat_args args) {
fmt::print("Error: ");
fmt::vprint(format, args);
}
template <typename... Args>
void
report_error2(const wchar_t *format, const Args &... args) {
vreport_error2(format, fmt::make_format_args<fmt::wformat_context>(args...));
}