Files
ncmpcpp/src/regex_filter.h
2020-12-19 17:51:49 +01:00

210 lines
5.1 KiB
C++

/***************************************************************************
* Copyright (C) 2008-2021 by Andrzej Rybczak *
* andrzej@rybczak.net *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#ifndef NCMPCPP_REGEX_FILTER_H
#define NCMPCPP_REGEX_FILTER_H
#include "config.h"
#ifdef BOOST_REGEX_ICU
# include <boost/regex/icu.hpp>
# include <unicode/errorcode.h>
# include <unicode/translit.h>
#else
# include <boost/regex.hpp>
#endif // BOOST_REGEX_ICU
#include <cassert>
#include <iostream>
#include "utility/functional.h"
namespace {
#ifdef BOOST_REGEX_ICU
struct StripDiacritics
{
static void convert(icu::UnicodeString &s)
{
if (m_converter == nullptr)
{
icu::ErrorCode result;
m_converter = icu::Transliterator::createInstance(
"NFD; [:M:] Remove; NFC", UTRANS_FORWARD, result);
if (result.isFailure())
throw std::runtime_error(
"instantiation of transliterator instance failed with "
+ std::string(result.errorName()));
}
m_converter->transliterate(s);
}
private:
static icu::Transliterator *m_converter;
};
icu::Transliterator *StripDiacritics::m_converter;
#endif // BOOST_REGEX_ICU
}
namespace Regex {
typedef
#ifdef BOOST_REGEX_ICU
boost::u32regex
#else
boost::regex
#endif // BOOST_REGEX_ICU
Regex;
template <typename StringT>
inline Regex make(StringT &&s,
boost::regex_constants::syntax_option_type flags)
{
return
#ifdef BOOST_REGEX_ICU
boost::make_u32regex
#else
boost::regex
#endif // BOOST_REGEX_ICU
(std::forward<StringT>(s), flags);
}
template <typename CharT>
inline bool search(const std::basic_string<CharT> &s,
const Regex &rx,
bool ignore_diacritics)
{
try {
#ifdef BOOST_REGEX_ICU
if (ignore_diacritics)
{
auto us = icu::UnicodeString::fromUTF8(
icu::StringPiece(convertString<char, CharT>::apply(s)));
StripDiacritics::convert(us);
return boost::u32regex_search(us, rx);
}
else
return boost::u32regex_search(s, rx);
#else
return boost::regex_search(s, rx);
#endif // BOOST_REGEX_ICU
} catch (std::out_of_range &e) {
// Invalid UTF-8 sequence, ignore the string.
std::cerr << "Regex::search: error while processing \""
<< s
<< "\": "
<< e.what()
<< "\n";
return false;
}
}
template <typename T>
struct Filter
{
typedef NC::Menu<T> MenuT;
typedef typename NC::Menu<T>::Item Item;
typedef std::function<bool(const Regex &, const T &)> FilterFunction;
Filter() { }
template <typename FilterT>
Filter(const std::string &constraint_,
boost::regex_constants::syntax_option_type flags,
FilterT &&filter)
: m_rx(make(constraint_, flags))
, m_constraint(constraint_)
, m_filter(std::forward<FilterT>(filter))
{ }
void clear()
{
m_filter = nullptr;
}
const std::string &constraint() const {
return m_constraint;
}
bool operator()(const Item &item) const {
assert(defined());
return m_filter(m_rx, item.value());
}
bool defined() const
{
return m_filter.operator bool();
}
private:
Regex m_rx;
std::string m_constraint;
FilterFunction m_filter;
};
template <typename T> struct ItemFilter
{
typedef NC::Menu<T> MenuT;
typedef typename NC::Menu<T>::Item Item;
typedef std::function<bool(const Regex &, const Item &)> FilterFunction;
ItemFilter() { }
template <typename FilterT>
ItemFilter(const std::string &constraint_,
boost::regex_constants::syntax_option_type flags,
FilterT &&filter)
: m_rx(make(constraint_, flags))
, m_constraint(constraint_)
, m_filter(std::forward<FilterT>(filter))
{ }
void clear()
{
m_filter = nullptr;
}
const std::string &constraint() const {
return m_constraint;
}
bool operator()(const Item &item) {
return m_filter(m_rx, item);
}
bool defined() const
{
return m_filter.operator bool();
}
private:
Regex m_rx;
std::string m_constraint;
FilterFunction m_filter;
};
}
#endif // NCMPCPP_REGEX_FILTER_H