2016-02-17 16:05:41 +01:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include "smallstringiterator.h"
|
|
|
|
|
|
2017-08-17 15:20:36 +02:00
|
|
|
#include <QString>
|
2016-02-17 16:05:41 +01:00
|
|
|
|
|
|
|
|
#include <cstring>
|
2017-07-05 12:11:02 +02:00
|
|
|
#include <string>
|
2016-02-17 16:05:41 +01:00
|
|
|
|
|
|
|
|
namespace Utils {
|
|
|
|
|
|
2018-01-17 11:49:06 +01:00
|
|
|
template <typename String>
|
|
|
|
|
using enable_if_has_char_data_pointer = typename std::enable_if_t<
|
|
|
|
|
std::is_same<
|
|
|
|
|
std::remove_const_t<
|
|
|
|
|
std::remove_pointer_t<
|
|
|
|
|
std::result_of_t<
|
|
|
|
|
decltype(&String::data)(String)
|
|
|
|
|
>
|
|
|
|
|
>
|
|
|
|
|
>, char>::value
|
|
|
|
|
, int>;
|
|
|
|
|
|
2016-02-17 16:05:41 +01:00
|
|
|
class SmallStringView
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
using const_iterator = Internal::SmallStringIterator<std::random_access_iterator_tag, const char>;
|
|
|
|
|
using const_reverse_iterator = std::reverse_iterator<const_iterator>;
|
|
|
|
|
using size_type = std::size_t;
|
|
|
|
|
|
|
|
|
|
template<size_type Size>
|
|
|
|
|
constexpr
|
|
|
|
|
SmallStringView(const char(&string)[Size]) noexcept
|
|
|
|
|
: m_pointer(string),
|
|
|
|
|
m_size(Size - 1)
|
|
|
|
|
{
|
|
|
|
|
static_assert(Size >= 1, "Invalid string literal! Length is zero!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename Type,
|
2017-01-11 17:04:23 +01:00
|
|
|
typename = std::enable_if_t<std::is_pointer<Type>::value>
|
2016-02-17 16:05:41 +01:00
|
|
|
>
|
|
|
|
|
SmallStringView(Type characterPointer) noexcept
|
|
|
|
|
: m_pointer(characterPointer),
|
|
|
|
|
m_size(std::strlen(characterPointer))
|
|
|
|
|
{
|
|
|
|
|
static_assert(!std::is_array<Type>::value, "Input type is array and not char pointer!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constexpr
|
|
|
|
|
SmallStringView(const char *const string, const size_type size) noexcept
|
|
|
|
|
: m_pointer(string),
|
|
|
|
|
m_size(size)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-26 17:10:17 +02:00
|
|
|
SmallStringView(const const_iterator begin, const const_iterator end) noexcept
|
|
|
|
|
: m_pointer(begin.data()),
|
|
|
|
|
m_size(std::size_t(end - begin))
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-17 11:49:06 +01:00
|
|
|
template<typename String,
|
|
|
|
|
typename Utils::enable_if_has_char_data_pointer<String> = 0>
|
|
|
|
|
SmallStringView(const String &string) noexcept
|
2017-08-29 12:54:10 +02:00
|
|
|
: m_pointer(string.data()),
|
|
|
|
|
m_size(string.size())
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-17 16:05:41 +01:00
|
|
|
static
|
|
|
|
|
SmallStringView fromUtf8(const char *const characterPointer)
|
|
|
|
|
{
|
|
|
|
|
return SmallStringView(characterPointer, std::strlen(characterPointer));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constexpr
|
2017-08-29 12:54:10 +02:00
|
|
|
const char *data() const noexcept
|
2016-02-17 16:05:41 +01:00
|
|
|
{
|
|
|
|
|
return m_pointer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constexpr
|
2017-08-29 12:54:10 +02:00
|
|
|
size_type size() const noexcept
|
2016-02-17 16:05:41 +01:00
|
|
|
{
|
|
|
|
|
return m_size;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-31 17:24:47 +02:00
|
|
|
constexpr
|
2017-08-29 12:54:10 +02:00
|
|
|
size_type isEmpty() const noexcept
|
2017-07-31 17:24:47 +02:00
|
|
|
{
|
|
|
|
|
return m_size == 0;
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-21 11:34:41 +02:00
|
|
|
constexpr
|
|
|
|
|
size_type empty() const noexcept
|
|
|
|
|
{
|
|
|
|
|
return m_size == 0;
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-16 17:48:53 +01:00
|
|
|
constexpr
|
|
|
|
|
SmallStringView mid(size_type position) const noexcept
|
|
|
|
|
{
|
|
|
|
|
return SmallStringView(data() + position, size() - position);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constexpr
|
|
|
|
|
SmallStringView mid(size_type position, size_type length) const noexcept
|
|
|
|
|
{
|
|
|
|
|
return SmallStringView(data() + position, length);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-29 13:02:28 +02:00
|
|
|
constexpr
|
2016-02-17 16:05:41 +01:00
|
|
|
const_iterator begin() const noexcept
|
|
|
|
|
{
|
|
|
|
|
return data();
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-29 13:02:28 +02:00
|
|
|
constexpr
|
2016-02-17 16:05:41 +01:00
|
|
|
const_iterator end() const noexcept
|
|
|
|
|
{
|
|
|
|
|
return data() + size();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const_reverse_iterator rbegin() const noexcept
|
|
|
|
|
{
|
2017-09-19 19:01:05 +02:00
|
|
|
return const_reverse_iterator(end());
|
2016-02-17 16:05:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const_reverse_iterator rend() const noexcept
|
|
|
|
|
{
|
2017-09-19 19:01:05 +02:00
|
|
|
return const_reverse_iterator(begin());
|
2016-02-17 16:05:41 +01:00
|
|
|
}
|
|
|
|
|
|
2017-07-03 11:12:00 +02:00
|
|
|
operator std::string() const
|
|
|
|
|
{
|
|
|
|
|
return std::string(data(), size());
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-17 15:20:36 +02:00
|
|
|
explicit operator QString() const
|
|
|
|
|
{
|
|
|
|
|
return QString::fromUtf8(data(), int(size()));
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-31 17:27:46 +02:00
|
|
|
bool startsWith(SmallStringView subStringToSearch) const noexcept
|
|
|
|
|
{
|
|
|
|
|
if (size() >= subStringToSearch.size())
|
|
|
|
|
return !std::memcmp(m_pointer, subStringToSearch.data(), subStringToSearch.size());
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool startsWith(char characterToSearch) const noexcept
|
|
|
|
|
{
|
|
|
|
|
return m_pointer[0] == characterToSearch;
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-17 16:05:41 +01:00
|
|
|
private:
|
|
|
|
|
const char *m_pointer;
|
|
|
|
|
size_type m_size;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
inline
|
2017-08-29 12:54:10 +02:00
|
|
|
bool operator==(SmallStringView first, SmallStringView second) noexcept
|
2016-02-17 16:05:41 +01:00
|
|
|
{
|
2017-08-03 12:10:49 +02:00
|
|
|
return first.size() == second.size() && std::memcmp(first.data(), second.data(), first.size()) == 0;
|
2016-02-17 16:05:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline
|
2017-08-29 12:54:10 +02:00
|
|
|
bool operator!=(SmallStringView first, SmallStringView second) noexcept
|
2016-02-17 16:05:41 +01:00
|
|
|
{
|
|
|
|
|
return !(first == second);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-21 10:21:47 +02:00
|
|
|
inline
|
2017-08-29 12:54:10 +02:00
|
|
|
int compare(SmallStringView first, SmallStringView second) noexcept
|
2017-08-21 10:21:47 +02:00
|
|
|
{
|
|
|
|
|
int sizeDifference = int(first.size() - second.size());
|
|
|
|
|
|
|
|
|
|
if (sizeDifference == 0)
|
|
|
|
|
return std::memcmp(first.data(), second.data(), first.size());
|
|
|
|
|
|
|
|
|
|
return sizeDifference;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-29 12:54:10 +02:00
|
|
|
inline
|
|
|
|
|
bool operator<(SmallStringView first, SmallStringView second) noexcept
|
|
|
|
|
{
|
|
|
|
|
return compare(first, second) < 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline
|
|
|
|
|
bool operator>(SmallStringView first, SmallStringView second) noexcept
|
|
|
|
|
{
|
|
|
|
|
return second < first;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-21 10:21:47 +02:00
|
|
|
namespace Internal {
|
|
|
|
|
inline
|
|
|
|
|
int reverse_memcmp(const char *first, const char *second, size_t n)
|
|
|
|
|
{
|
|
|
|
|
|
2017-09-21 11:43:59 +02:00
|
|
|
const char *currentFirst = first + n - 1;
|
|
|
|
|
const char *currentSecond = second + n - 1;
|
2017-08-21 10:21:47 +02:00
|
|
|
|
|
|
|
|
while (n > 0)
|
|
|
|
|
{
|
|
|
|
|
// If the current characters differ, return an appropriately signed
|
|
|
|
|
// value; otherwise, keep searching backwards
|
2017-09-21 11:43:59 +02:00
|
|
|
int difference = *currentFirst - *currentSecond;
|
|
|
|
|
if (difference != 0)
|
|
|
|
|
return difference;
|
2017-08-21 10:21:47 +02:00
|
|
|
|
|
|
|
|
--currentFirst;
|
|
|
|
|
--currentSecond;
|
|
|
|
|
--n;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline
|
2017-08-29 12:54:10 +02:00
|
|
|
int reverseCompare(SmallStringView first, SmallStringView second) noexcept
|
2017-08-21 10:21:47 +02:00
|
|
|
{
|
|
|
|
|
int sizeDifference = int(first.size() - second.size());
|
|
|
|
|
|
|
|
|
|
if (sizeDifference == 0)
|
|
|
|
|
return Internal::reverse_memcmp(first.data(), second.data(), first.size());
|
|
|
|
|
|
|
|
|
|
return sizeDifference;
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-17 16:05:41 +01:00
|
|
|
} // namespace Utils
|
2017-08-17 15:19:46 +02:00
|
|
|
|
|
|
|
|
#ifdef __cpp_user_defined_literals
|
|
|
|
|
inline
|
|
|
|
|
constexpr Utils::SmallStringView operator""_sv(const char *const string, size_t size)
|
|
|
|
|
{
|
|
|
|
|
return Utils::SmallStringView(string, size);
|
|
|
|
|
}
|
|
|
|
|
#endif
|