include/o3tl/string_view.hxx   |  121 ++++++++
 o3tl/CppunitTest_o3tl_tests.mk |    1 
 o3tl/qa/test-string_view.cxx   |  591 +++++++++++++++++++++++++++++++++++++++++
 3 files changed, 713 insertions(+)

New commits:
commit b46894e7d063cc96609c8aaea76f4a7ffc5f170f
Author:     Stephan Bergmann <sberg...@redhat.com>
AuthorDate: Tue Sep 21 22:19:06 2021 +0200
Commit:     Stephan Bergmann <sberg...@redhat.com>
CommitDate: Tue Sep 21 23:32:02 2021 +0200

    Some more o3tl::starts/ends_with overloads
    
    ...including ones with a rest out parameter, modeled after their O(U)String
    counterparts and not present in any C++ standard (so intended to stay in 
o3tl,
    even when the C++20 wrappers are eventually removed again).  Plus test code.
    
    These additional overloads will be used in forthcoming commits changing more
    places to use std string_view.
    
    Change-Id: Ifd05e0fdcb93e8751da22b8e4a59796ad00fe581
    Reviewed-on: https://gerrit.libreoffice.org/c/core/+/122407
    Tested-by: Jenkins
    Reviewed-by: Stephan Bergmann <sberg...@redhat.com>

diff --git a/include/o3tl/string_view.hxx b/include/o3tl/string_view.hxx
index b66ba1175a57..4cb9790900f6 100644
--- a/include/o3tl/string_view.hxx
+++ b/include/o3tl/string_view.hxx
@@ -15,6 +15,7 @@
 #include <cstddef>
 #include <string>
 #include <string_view>
+#include <type_traits>
 
 #include <rtl/ustring.h>
 
@@ -107,6 +108,126 @@ constexpr bool ends_with(std::basic_string_view<charT, 
traits> sv, charT const*
     return ends_with(sv, std::basic_string_view<charT, traits>(x));
 #endif
 }
+// The following overloads prevent deduction failures that would occur with 
their template
+// counterparts, when x is of a type that is implicitly convertible to 
basic_string_view (like
+// OString or OUString, and we only bother to provide overloads for the char 
and char16_t cases, not
+// also for char32_t and wchar_t, nor for C++20 char8_t):
+constexpr bool starts_with(std::string_view sv, std::string_view x) noexcept
+{
+    return starts_with<char>(sv, x);
+}
+constexpr bool starts_with(std::u16string_view sv, std::u16string_view x) 
noexcept
+{
+    return starts_with<char16_t>(sv, x);
+}
+constexpr bool ends_with(std::string_view sv, std::string_view x) noexcept
+{
+    return ends_with<char>(sv, x);
+}
+constexpr bool ends_with(std::u16string_view sv, std::u16string_view x) 
noexcept
+{
+    return ends_with<char16_t>(sv, x);
+}
+
+// Variants of C++20 std::basic_string_view::starts_with and
+// std::basic_string_view::ends_with that have a rest out parameter, similar 
to our OString and
+// OUString startsWith and endsWith member functions:
+template <typename charT, typename traits = std::char_traits<charT>>
+constexpr bool starts_with(std::basic_string_view<charT, traits> sv,
+                           std::basic_string_view<charT, traits> x,
+                           std::basic_string_view<charT, traits>* rest) 
noexcept
+{
+    assert(rest != nullptr);
+    auto const found = starts_with(sv, x);
+    if (found)
+    {
+        *rest = sv.substr(x.length());
+    }
+    return found;
+}
+template <typename charT, typename traits = std::char_traits<charT>>
+constexpr bool starts_with(std::basic_string_view<charT, traits> sv, charT x,
+                           std::basic_string_view<charT, traits>* rest) 
noexcept
+{
+    assert(rest != nullptr);
+    auto const found = starts_with(sv, x);
+    if (found)
+    {
+        *rest = sv.substr(1);
+    }
+    return found;
+}
+template <typename charT, typename traits = std::char_traits<charT>>
+constexpr bool starts_with(std::basic_string_view<charT, traits> sv, charT 
const* x,
+                           std::basic_string_view<charT, traits>* rest)
+{
+    assert(rest != nullptr);
+    auto const found = starts_with(sv, x);
+    if (found)
+    {
+        *rest = sv.substr(traits::length(x));
+    }
+    return found;
+}
+template <typename charT, typename traits = std::char_traits<charT>>
+constexpr bool ends_with(std::basic_string_view<charT, traits> sv,
+                         std::basic_string_view<charT, traits> x,
+                         std::basic_string_view<charT, traits>* rest) noexcept
+{
+    assert(rest != nullptr);
+    auto const found = ends_with(sv, x);
+    if (found)
+    {
+        *rest = sv.substr(0, sv.length() - x.length());
+    }
+    return found;
+}
+template <typename charT, typename traits = std::char_traits<charT>>
+constexpr bool ends_with(std::basic_string_view<charT, traits> sv, charT x,
+                         std::basic_string_view<charT, traits>* rest) noexcept
+{
+    assert(rest != nullptr);
+    auto const found = ends_with(sv, x);
+    if (found)
+    {
+        *rest = sv.substr(0, sv.length() - 1);
+    }
+    return found;
+}
+template <typename charT, typename traits = std::char_traits<charT>>
+constexpr bool ends_with(std::basic_string_view<charT, traits> sv, charT 
const* x,
+                         std::basic_string_view<charT, traits>* rest)
+{
+    assert(rest != nullptr);
+    auto const found = ends_with(sv, x);
+    if (found)
+    {
+        *rest = sv.substr(0, sv.length() - traits::length(x));
+    }
+    return found;
+}
+// The following overloads prevent deduction failures that would occur with 
their template
+// counterparts, when x is of a type that is implicitly convertible to 
basic_string_view (like
+// OString or OUString, and we only bother to provide overloads for the char 
and char16_t cases, not
+// also for char32_t and wchar_t, nor for C++20 char8_t):
+constexpr bool starts_with(std::string_view sv, std::string_view x, 
std::string_view* rest) noexcept
+{
+    return starts_with<char>(sv, x, rest);
+}
+constexpr bool starts_with(std::u16string_view sv, std::u16string_view x,
+                           std::u16string_view* rest) noexcept
+{
+    return starts_with<char16_t>(sv, x, rest);
+}
+constexpr bool ends_with(std::string_view sv, std::string_view x, 
std::string_view* rest) noexcept
+{
+    return ends_with<char>(sv, x, rest);
+}
+constexpr bool ends_with(std::u16string_view sv, std::u16string_view x,
+                         std::u16string_view* rest) noexcept
+{
+    return ends_with<char16_t>(sv, x, rest);
+}
 }
 
 /* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s 
cinkeys+=0=break: */
diff --git a/o3tl/CppunitTest_o3tl_tests.mk b/o3tl/CppunitTest_o3tl_tests.mk
index 6866778f6c1a..1cbb1c86bd91 100644
--- a/o3tl/CppunitTest_o3tl_tests.mk
+++ b/o3tl/CppunitTest_o3tl_tests.mk
@@ -33,6 +33,7 @@ $(eval $(call 
gb_CppunitTest_add_exception_objects,o3tl_tests,\
        o3tl/qa/test-safeint \
        o3tl/qa/test-sorted_vector \
        o3tl/qa/test-span \
+       o3tl/qa/test-string_view \
        o3tl/qa/test-temporary \
        o3tl/qa/test-typed_flags \
        o3tl/qa/test-unit_conversion \
diff --git a/o3tl/qa/test-string_view.cxx b/o3tl/qa/test-string_view.cxx
new file mode 100644
index 000000000000..61145613693b
--- /dev/null
+++ b/o3tl/qa/test-string_view.cxx
@@ -0,0 +1,591 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; 
fill-column: 100 -*- */
+/*
+ * This file is part of the LibreOffice project.
+ *
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
+ */
+
+#include <sal/config.h>
+
+#include <iostream>
+#include <string_view>
+
+#include <cppunit/TestAssert.h>
+#include <cppunit/TestFixture.h>
+#include <cppunit/extensions/HelperMacros.h>
+
+#include <o3tl/string_view.hxx>
+#include <rtl/string.hxx>
+#include <rtl/ustring.hxx>
+
+namespace CppUnit
+{
+template <> struct assertion_traits<std::u16string_view>
+{
+    static bool equal(std::u16string_view x, std::u16string_view y) { return x 
== y; }
+
+    static std::string toString(std::u16string_view x)
+    {
+        return OUStringToOString(x, RTL_TEXTENCODING_UTF8).getStr();
+    }
+};
+}
+
+namespace
+{
+OString ostringEmpty() { return {}; } // avoid loplugin:stringview
+OString ostringDoof() { return "doof"; } // avoid loplugin:stringview
+OString ostringFoo() { return "foo"; } // avoid loplugin:stringview
+OString ostringFoobars() { return "foobars"; } // avoid loplugin:stringview
+OString ostringFood() { return "food"; } // avoid loplugin:stringview
+OString ostringOof() { return "oof"; } // avoid loplugin:stringview
+OString ostringSraboof() { return "sraboof"; } // avoid loplugin:stringview
+OUString oustringEmpty() { return {}; } // avoid loplugin:stringview
+OUString oustringDoof() { return "doof"; } // avoid loplugin:stringview
+OUString oustringFoo() { return "foo"; } // avoid loplugin:stringview
+OUString oustringFoobars() { return "foobars"; } // avoid loplugin:stringview
+OUString oustringFood() { return "food"; } // avoid loplugin:stringview
+OUString oustringOof() { return "oof"; } // avoid loplugin:stringview
+OUString oustringSraboof() { return "sraboof"; } // avoid loplugin:stringview
+
+class Test : public CppUnit::TestFixture
+{
+private:
+    CPPUNIT_TEST_SUITE(Test);
+    CPPUNIT_TEST(testStartsWith);
+    CPPUNIT_TEST(testStartsWithRest);
+    CPPUNIT_TEST(testEndsWith);
+    CPPUNIT_TEST(testEndsWithRest);
+    CPPUNIT_TEST_SUITE_END();
+
+    void testStartsWith()
+    {
+        using namespace std::string_view_literals;
+        CPPUNIT_ASSERT(o3tl::starts_with(""sv, ""sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with(""sv, "foo"sv));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ""sv));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, "foo"sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "food"sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "foobars"sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with(""sv, 'f'));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, 'f'));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, 'g'));
+        CPPUNIT_ASSERT(o3tl::starts_with(""sv, ""));
+        CPPUNIT_ASSERT(!o3tl::starts_with(""sv, "foo"));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ""));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, "foo"));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "food"));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "foobars"));
+        CPPUNIT_ASSERT(o3tl::starts_with(""sv, ostringEmpty()));
+        CPPUNIT_ASSERT(!o3tl::starts_with(""sv, ostringFoo()));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ostringEmpty()));
+        CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ostringFoo()));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, ostringFood()));
+        CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, ostringFoobars()));
+        CPPUNIT_ASSERT(o3tl::starts_with(u""sv, u""sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, u"foo"sv));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u""sv));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u"foo"sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"food"sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"foobars"sv));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, u'f'));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u'f'));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u'g'));
+        CPPUNIT_ASSERT(o3tl::starts_with(u""sv, u""));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, u"foo"));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u""));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u"foo"));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"food"));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"foobars"));
+        CPPUNIT_ASSERT(o3tl::starts_with(u""sv, oustringEmpty()));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, oustringFoo()));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, oustringEmpty()));
+        CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, oustringFoo()));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, oustringFood()));
+        CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, oustringFoobars()));
+    }
+
+    void testStartsWithRest()
+    {
+        using namespace std::string_view_literals;
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(""sv, ""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(""sv, "foo"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL("foobar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, "foo"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL("bar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "food"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "foobars"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(""sv, 'f', &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, 'f', &rest));
+            CPPUNIT_ASSERT_EQUAL("oobar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, 'g', &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(""sv, "", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(""sv, "foo", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, "", &rest));
+            CPPUNIT_ASSERT_EQUAL("foobar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, "foo", &rest));
+            CPPUNIT_ASSERT_EQUAL("bar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "food", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, "foobars", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(""sv, ostringEmpty(), &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(""sv, ostringFoo(), &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ostringEmpty(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL("foobar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with("foobar"sv, ostringFoo(), &rest));
+            CPPUNIT_ASSERT_EQUAL("bar"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, ostringFood(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with("foobar"sv, ostringFoobars(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u""sv, u""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, u"foo"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u"foobar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u"foo"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u"bar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"food"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"foobars"sv, 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, u'f', &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u'f', &rest));
+            CPPUNIT_ASSERT_EQUAL(u"oobar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u'g', &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u""sv, u"", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, u"foo", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u"", &rest));
+            CPPUNIT_ASSERT_EQUAL(u"foobar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, u"foo", &rest));
+            CPPUNIT_ASSERT_EQUAL(u"bar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"food", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, u"foobars", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u""sv, oustringEmpty(), &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u""sv, oustringFoo(), &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, oustringEmpty(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u"foobar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::starts_with(u"foobar"sv, oustringFoo(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u"bar"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, oustringFood(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::starts_with(u"foobar"sv, oustringFoobars(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+    }
+
+    void testEndsWith()
+    {
+        using namespace std::string_view_literals;
+        CPPUNIT_ASSERT(o3tl::ends_with(""sv, ""sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with(""sv, "oof"sv));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ""sv));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, "oof"sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "doof"sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "sraboof"sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with(""sv, 'f'));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, 'f'));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, 'g'));
+        CPPUNIT_ASSERT(o3tl::ends_with(""sv, ""));
+        CPPUNIT_ASSERT(!o3tl::ends_with(""sv, "oof"));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ""));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, "oof"));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "doof"));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "sraboof"));
+        CPPUNIT_ASSERT(o3tl::ends_with(""sv, ostringEmpty()));
+        CPPUNIT_ASSERT(!o3tl::ends_with(""sv, ostringOof()));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ostringEmpty()));
+        CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ostringOof()));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, ostringDoof()));
+        CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, ostringSraboof()));
+        CPPUNIT_ASSERT(o3tl::ends_with(u""sv, u""sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, u"oof"sv));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u""sv));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u"oof"sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"doof"sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"sraboof"sv));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, u'f'));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u'f'));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u'g'));
+        CPPUNIT_ASSERT(o3tl::ends_with(u""sv, u""));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, u"oof"));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u""));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u"oof"));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"doof"));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"sraboof"));
+        CPPUNIT_ASSERT(o3tl::ends_with(u""sv, oustringEmpty()));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, oustringOof()));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, oustringEmpty()));
+        CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, oustringOof()));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, oustringDoof()));
+        CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, oustringSraboof()));
+    }
+
+    void testEndsWithRest()
+    {
+        using namespace std::string_view_literals;
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(""sv, ""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(""sv, "oof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL("raboof"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, "oof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL("rab"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "doof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "sraboof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(""sv, 'f', &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, 'f', &rest));
+            CPPUNIT_ASSERT_EQUAL("raboo"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, 'g', &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(""sv, "", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(""sv, "oof", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, "", &rest));
+            CPPUNIT_ASSERT_EQUAL("raboof"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, "oof", &rest));
+            CPPUNIT_ASSERT_EQUAL("rab"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "doof", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, "sraboof", &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(""sv, ostringEmpty(), &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(""sv, ostringOof(), &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ostringEmpty(), &rest));
+            CPPUNIT_ASSERT_EQUAL("raboof"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with("raboof"sv, ostringOof(), &rest));
+            CPPUNIT_ASSERT_EQUAL("rab"sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, ostringDoof(), &rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with("raboof"sv, ostringSraboof(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u""sv, u""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, u"oof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u""sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u"raboof"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u"oof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u"rab"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"doof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"sraboof"sv, &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, u'f', &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u'f', &rest));
+            CPPUNIT_ASSERT_EQUAL(u"raboo"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u'g', &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u""sv, u"", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, u"oof", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u"", &rest));
+            CPPUNIT_ASSERT_EQUAL(u"raboof"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, u"oof", &rest));
+            CPPUNIT_ASSERT_EQUAL(u"rab"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"doof", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, u"sraboof", &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u""sv, oustringEmpty(), &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u""sv, oustringOof(), &rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, oustringEmpty(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u"raboof"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(o3tl::ends_with(u"raboof"sv, oustringOof(), &rest));
+            CPPUNIT_ASSERT_EQUAL(u"rab"sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, oustringDoof(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+        {
+            std::u16string_view rest;
+            CPPUNIT_ASSERT(!o3tl::ends_with(u"raboof"sv, oustringSraboof(), 
&rest));
+            CPPUNIT_ASSERT_EQUAL(u""sv, rest);
+        }
+    }
+};
+
+CPPUNIT_TEST_SUITE_REGISTRATION(Test);
+}
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s 
cinkeys+=0=break: */

Reply via email to