From d720026461c8b20977ef799c52bf5388c8dc67cd Mon Sep 17 00:00:00 2001 From: js324 Date: Wed, 29 Nov 2023 17:49:36 -0500 Subject: [PATCH] Add native and generic representation for filesystem::path format spec (#3729) --- include/fmt/std.h | 15 +++++++++++++-- test/std-test.cc | 9 +++++++-- 2 files changed, 20 insertions(+), 4 deletions(-) diff --git a/include/fmt/std.h b/include/fmt/std.h index 4799df1a317d..7b92cea5bf14 100644 --- a/include/fmt/std.h +++ b/include/fmt/std.h @@ -114,6 +114,7 @@ template struct formatter { format_specs specs_; detail::arg_ref width_ref_; bool debug_ = false; + char path_type_ = 'n'; public: FMT_CONSTEXPR void set_debug_format(bool set = true) { debug_ = set; } @@ -130,20 +131,30 @@ template struct formatter { debug_ = true; ++it; } + if (it != end && (*it == 'g' || *it == 'n')) { + path_type_ = *it++; + } return it; } template auto format(const std::filesystem::path& p, FormatContext& ctx) const { auto specs = specs_; + auto path_type = path_type_; + # ifdef _WIN32 + auto path_string = path_type == 'n' ? p.native() : p.generic_wstring(); + # else + auto path_string = path_type == 'n' ? p.native() : p.generic_string(); + # endif + detail::handle_dynamic_spec(specs.width, width_ref_, ctx); if (!debug_) { - auto s = detail::get_path_string(p, p.native()); + auto s = detail::get_path_string(p, path_string); return detail::write(ctx.out(), basic_string_view(s), specs); } auto quoted = basic_memory_buffer(); - detail::write_escaped_path(quoted, p, p.native()); + detail::write_escaped_path(quoted, p, path_string); return detail::write(ctx.out(), basic_string_view(quoted.data(), quoted.size()), specs); diff --git a/test/std-test.cc b/test/std-test.cc index dc1073b58fc8..7a81aaf9a5f5 100644 --- a/test/std-test.cc +++ b/test/std-test.cc @@ -25,15 +25,20 @@ TEST(std_test, path) { EXPECT_EQ(fmt::format("{}", path("foo\"bar")), "foo\"bar"); EXPECT_EQ(fmt::format("{:?}", path("foo\"bar")), "\"foo\\\"bar\""); + + EXPECT_EQ(fmt::format("{:n}", path("/usr/bin")), "/usr/bin"); + EXPECT_EQ(fmt::format("{:g}", path("/usr/bin")), "/usr/bin"); +# ifdef _WIN32 + EXPECT_EQ(fmt::format("{:n}", path("C:\\foo")), "C:\\foo"); + EXPECT_EQ(fmt::format("{:g}", path("C:\\foo")), "C:/foo"); -# ifdef _WIN32 EXPECT_EQ(fmt::format("{}", path( L"\x0428\x0447\x0443\x0447\x044B\x043D\x0448" L"\x0447\x044B\x043D\x0430")), "Шчучыншчына"); EXPECT_EQ(fmt::format("{}", path(L"\xd800")), "�"); EXPECT_EQ(fmt::format("{:?}", path(L"\xd800")), "\"\\ud800\""); -# endif +# endif } // Test ambiguity problem described in #2954.