2014-07-11 03:27:04 +04:00
|
|
|
#include "face_registry.hh"
|
2012-09-17 21:01:13 +04:00
|
|
|
|
|
|
|
#include "exception.hh"
|
2017-08-29 11:23:03 +03:00
|
|
|
#include "ranges.hh"
|
2017-10-09 17:12:42 +03:00
|
|
|
#include "string_utils.hh"
|
2012-09-17 21:01:13 +04:00
|
|
|
|
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2014-07-11 03:44:59 +04:00
|
|
|
static Face parse_face(StringView facedesc)
|
2012-09-17 21:01:13 +04:00
|
|
|
{
|
2015-12-12 12:00:52 +03:00
|
|
|
constexpr StringView invalid_face_error = "invalid face description, expected <fg>[,<bg>][+<attr>]";
|
2014-12-23 16:34:21 +03:00
|
|
|
auto bg_it = find(facedesc, ',');
|
|
|
|
auto attr_it = find(facedesc, '+');
|
2015-12-11 22:58:28 +03:00
|
|
|
if (bg_it != facedesc.end()
|
|
|
|
and (attr_it < bg_it or (bg_it + 1) == facedesc.end()))
|
2015-12-12 12:00:52 +03:00
|
|
|
throw runtime_error(invalid_face_error.str());
|
2015-12-11 22:58:28 +03:00
|
|
|
if (attr_it != facedesc.end()
|
|
|
|
and (attr_it + 1) == facedesc.end())
|
2015-12-12 12:00:52 +03:00
|
|
|
throw runtime_error(invalid_face_error.str());
|
2014-07-11 03:44:59 +04:00
|
|
|
Face res;
|
2015-10-05 22:32:51 +03:00
|
|
|
res.fg = attr_it != facedesc.begin() ?
|
|
|
|
str_to_color({facedesc.begin(), std::min(attr_it, bg_it)}) : Color::Default;
|
2014-07-11 03:44:59 +04:00
|
|
|
if (bg_it != facedesc.end())
|
2015-10-05 22:32:51 +03:00
|
|
|
res.bg = bg_it+1 != attr_it ? str_to_color({bg_it+1, attr_it}) : Color::Default;
|
2014-07-11 03:44:59 +04:00
|
|
|
if (attr_it != facedesc.end())
|
|
|
|
{
|
|
|
|
for (++attr_it; attr_it != facedesc.end(); ++attr_it)
|
|
|
|
{
|
|
|
|
switch (*attr_it)
|
|
|
|
{
|
2015-10-23 15:34:03 +03:00
|
|
|
case 'e': res.attributes |= Attribute::Exclusive; break;
|
2014-07-15 23:11:47 +04:00
|
|
|
case 'u': res.attributes |= Attribute::Underline; break;
|
|
|
|
case 'r': res.attributes |= Attribute::Reverse; break;
|
|
|
|
case 'b': res.attributes |= Attribute::Bold; break;
|
|
|
|
case 'B': res.attributes |= Attribute::Blink; break;
|
|
|
|
case 'd': res.attributes |= Attribute::Dim; break;
|
2015-09-27 16:24:42 +03:00
|
|
|
case 'i': res.attributes |= Attribute::Italic; break;
|
2018-04-06 17:56:53 +03:00
|
|
|
default: throw runtime_error(format("no such face attribute: '{}'", StringView{*attr_it}));
|
2014-07-11 03:44:59 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res;
|
2013-05-15 16:01:23 +04:00
|
|
|
}
|
2012-09-17 21:01:13 +04:00
|
|
|
|
2017-09-12 06:31:57 +03:00
|
|
|
String to_string(Attribute attributes)
|
2017-09-09 16:55:35 +03:00
|
|
|
{
|
|
|
|
if (attributes == Attribute::Normal)
|
|
|
|
return "";
|
|
|
|
|
|
|
|
struct Attr { Attribute attr; StringView name; }
|
|
|
|
attrs[] {
|
|
|
|
{ Attribute::Exclusive, "e" },
|
|
|
|
{ Attribute::Underline, "u" },
|
|
|
|
{ Attribute::Reverse, "r" },
|
|
|
|
{ Attribute::Blink, "B" },
|
|
|
|
{ Attribute::Bold, "b" },
|
|
|
|
{ Attribute::Dim, "d" },
|
|
|
|
{ Attribute::Italic, "i" },
|
|
|
|
};
|
|
|
|
|
|
|
|
auto filteredAttrs = attrs |
|
|
|
|
filter([=](const Attr& a) { return attributes & a.attr; }) |
|
|
|
|
transform([](const Attr& a) { return a.name; });
|
|
|
|
|
2018-01-18 01:00:54 +03:00
|
|
|
return accumulate(filteredAttrs, "+"_str, std::plus<>{});
|
2017-09-09 16:55:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
String to_string(Face face)
|
|
|
|
{
|
2017-09-12 06:31:57 +03:00
|
|
|
return format("{},{}{}", face.fg, face.bg, face.attributes);
|
2017-09-09 16:55:35 +03:00
|
|
|
}
|
|
|
|
|
2018-04-07 08:36:39 +03:00
|
|
|
Face FaceRegistry::operator[](StringView facedesc) const
|
2013-05-15 16:01:23 +04:00
|
|
|
{
|
2018-04-07 08:36:39 +03:00
|
|
|
auto it = m_faces.find(facedesc);
|
|
|
|
if (it != m_faces.end())
|
2014-08-20 02:10:56 +04:00
|
|
|
{
|
2017-03-07 03:30:54 +03:00
|
|
|
if (it->value.alias.empty())
|
|
|
|
return it->value.face;
|
2018-04-07 08:36:39 +03:00
|
|
|
return operator[](it->value.alias);
|
2014-08-20 02:10:56 +04:00
|
|
|
}
|
2018-04-07 08:36:39 +03:00
|
|
|
if (m_parent)
|
|
|
|
return (*m_parent)[facedesc];
|
2014-07-12 14:19:35 +04:00
|
|
|
return parse_face(facedesc);
|
2012-09-17 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2018-04-07 08:36:39 +03:00
|
|
|
void FaceRegistry::add_face(StringView name, StringView facedesc, bool override)
|
2012-09-17 21:01:13 +04:00
|
|
|
{
|
2018-04-07 08:36:39 +03:00
|
|
|
if (not override and m_faces.find(name) != m_faces.end())
|
|
|
|
throw runtime_error(format("face '{}' already defined", name));
|
2012-09-17 21:01:13 +04:00
|
|
|
|
2014-08-20 02:16:21 +04:00
|
|
|
if (name.empty() or is_color_name(name) or
|
|
|
|
std::any_of(name.begin(), name.end(),
|
|
|
|
[](char c){ return not isalnum(c); }))
|
2018-04-07 08:36:39 +03:00
|
|
|
throw runtime_error(format("invalid face name: '{}'", name));
|
2015-08-12 23:49:29 +03:00
|
|
|
|
|
|
|
if (name == facedesc)
|
|
|
|
throw runtime_error(format("cannot alias face '{}' to itself", name));
|
2012-09-17 21:01:13 +04:00
|
|
|
|
2018-04-07 08:36:39 +03:00
|
|
|
FaceOrAlias& face = m_faces[name];
|
|
|
|
auto it = m_faces.find(facedesc);
|
|
|
|
if (it != m_faces.end())
|
2014-08-20 02:10:56 +04:00
|
|
|
{
|
2018-04-07 08:36:39 +03:00
|
|
|
while (it != m_faces.end())
|
2014-08-20 02:10:56 +04:00
|
|
|
{
|
2017-03-07 03:30:54 +03:00
|
|
|
if (it->value.alias.empty())
|
2014-08-20 02:10:56 +04:00
|
|
|
break;
|
2017-03-07 03:30:54 +03:00
|
|
|
if (it->value.alias == name)
|
2014-08-20 02:10:56 +04:00
|
|
|
throw runtime_error("face cycle detected");
|
2018-04-07 08:36:39 +03:00
|
|
|
it = m_faces.find(it->value.alias);
|
2014-08-20 02:10:56 +04:00
|
|
|
}
|
|
|
|
|
2018-04-07 08:36:39 +03:00
|
|
|
face.alias = facedesc.str();
|
2014-08-20 02:10:56 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-04-07 08:36:39 +03:00
|
|
|
face.alias = "";
|
|
|
|
face.face = parse_face(facedesc);
|
2014-08-20 02:10:56 +04:00
|
|
|
}
|
2012-09-17 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2018-04-07 08:36:39 +03:00
|
|
|
void FaceRegistry::remove_face(StringView name)
|
2014-03-29 17:18:46 +04:00
|
|
|
{
|
2018-04-07 08:36:39 +03:00
|
|
|
m_faces.remove(name);
|
2014-03-29 17:18:46 +04:00
|
|
|
}
|
|
|
|
|
2014-07-11 03:27:04 +04:00
|
|
|
FaceRegistry::FaceRegistry()
|
2018-04-07 08:36:39 +03:00
|
|
|
: m_faces{
|
2017-09-01 13:32:38 +03:00
|
|
|
{ "Default", {Face{ Color::Default, Color::Default }} },
|
|
|
|
{ "PrimarySelection", {Face{ Color::White, Color::Blue }} },
|
|
|
|
{ "SecondarySelection", {Face{ Color::Black, Color::Blue }} },
|
|
|
|
{ "PrimaryCursor", {Face{ Color::Black, Color::White }} },
|
|
|
|
{ "SecondaryCursor", {Face{ Color::Black, Color::White }} },
|
2018-02-24 13:24:06 +03:00
|
|
|
{ "PrimaryCursorEol", {Face{ Color::Black, Color::Cyan }} },
|
|
|
|
{ "SecondaryCursorEol", {Face{ Color::Black, Color::Cyan }} },
|
2017-09-01 13:32:38 +03:00
|
|
|
{ "LineNumbers", {Face{ Color::Default, Color::Default }} },
|
|
|
|
{ "LineNumberCursor", {Face{ Color::Default, Color::Default, Attribute::Reverse }} },
|
|
|
|
{ "LineNumbersWrapped", {Face{ Color::Default, Color::Default, Attribute::Italic }} },
|
|
|
|
{ "MenuForeground", {Face{ Color::White, Color::Blue }} },
|
|
|
|
{ "MenuBackground", {Face{ Color::Blue, Color::White }} },
|
|
|
|
{ "MenuInfo", {Face{ Color::Cyan, Color::Default }} },
|
|
|
|
{ "Information", {Face{ Color::Black, Color::Yellow }} },
|
|
|
|
{ "Error", {Face{ Color::Black, Color::Red }} },
|
|
|
|
{ "StatusLine", {Face{ Color::Cyan, Color::Default }} },
|
|
|
|
{ "StatusLineMode", {Face{ Color::Yellow, Color::Default }} },
|
|
|
|
{ "StatusLineInfo", {Face{ Color::Blue, Color::Default }} },
|
|
|
|
{ "StatusLineValue", {Face{ Color::Green, Color::Default }} },
|
|
|
|
{ "StatusCursor", {Face{ Color::Black, Color::Cyan }} },
|
|
|
|
{ "Prompt", {Face{ Color::Yellow, Color::Default }} },
|
|
|
|
{ "MatchingChar", {Face{ Color::Default, Color::Default, Attribute::Bold }} },
|
|
|
|
{ "BufferPadding", {Face{ Color::Blue, Color::Default }} },
|
|
|
|
{ "Whitespace", {Face{ Color::Default, Color::Default }} },
|
2013-03-06 23:31:07 +04:00
|
|
|
}
|
|
|
|
{}
|
|
|
|
|
2012-09-17 21:01:13 +04:00
|
|
|
}
|