2014-07-11 01:27:04 +02:00
|
|
|
#include "face_registry.hh"
|
2012-09-17 19:01:13 +02:00
|
|
|
|
2014-12-23 14:34:21 +01:00
|
|
|
#include "containers.hh"
|
2012-09-17 19:01:13 +02:00
|
|
|
#include "exception.hh"
|
2014-12-23 14:54:09 +01:00
|
|
|
#include "containers.hh"
|
2012-09-17 19:01:13 +02:00
|
|
|
|
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2014-07-11 01:44:59 +02:00
|
|
|
static Face parse_face(StringView facedesc)
|
2012-09-17 19:01:13 +02:00
|
|
|
{
|
2014-12-23 14:34:21 +01:00
|
|
|
auto bg_it = find(facedesc, ',');
|
|
|
|
auto attr_it = find(facedesc, '+');
|
2014-07-11 01:44:59 +02:00
|
|
|
if (bg_it != facedesc.end() and attr_it < bg_it)
|
|
|
|
throw runtime_error("invalid face description, expected <fg>[,<bg>][+<attr>]");
|
|
|
|
Face res;
|
|
|
|
res.fg = str_to_color({facedesc.begin(), std::min(attr_it, bg_it)});
|
|
|
|
if (bg_it != facedesc.end())
|
|
|
|
res.bg = str_to_color({bg_it+1, attr_it});
|
|
|
|
if (attr_it != facedesc.end())
|
|
|
|
{
|
|
|
|
for (++attr_it; attr_it != facedesc.end(); ++attr_it)
|
|
|
|
{
|
|
|
|
switch (*attr_it)
|
|
|
|
{
|
2014-07-15 21:11:47 +02: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 15:24:42 +02:00
|
|
|
case 'i': res.attributes |= Attribute::Italic; break;
|
2015-06-01 22:15:59 +02:00
|
|
|
default: throw runtime_error(format("unknown face attribute '{}'", StringView{*attr_it}));
|
2014-07-11 01:44:59 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res;
|
2013-05-15 14:01:23 +02:00
|
|
|
}
|
2012-09-17 19:01:13 +02:00
|
|
|
|
2014-07-12 12:19:35 +02:00
|
|
|
Face FaceRegistry::operator[](const String& facedesc)
|
2013-05-15 14:01:23 +02:00
|
|
|
{
|
2014-07-11 01:27:04 +02:00
|
|
|
auto it = m_aliases.find(facedesc);
|
2014-08-20 00:10:56 +02:00
|
|
|
while (it != m_aliases.end())
|
|
|
|
{
|
|
|
|
if (it->second.alias.empty())
|
|
|
|
return it->second.face;
|
|
|
|
it = m_aliases.find(it->second.alias);
|
|
|
|
}
|
2014-07-12 12:19:35 +02:00
|
|
|
return parse_face(facedesc);
|
2012-09-17 19:01:13 +02:00
|
|
|
}
|
|
|
|
|
2014-07-11 01:27:04 +02:00
|
|
|
void FaceRegistry::register_alias(const String& name, const String& facedesc,
|
2014-08-20 00:10:56 +02:00
|
|
|
bool override)
|
2012-09-17 19:01:13 +02:00
|
|
|
{
|
2012-09-17 21:01:11 +02:00
|
|
|
if (not override and m_aliases.find(name) != m_aliases.end())
|
2015-06-01 22:15:59 +02:00
|
|
|
throw runtime_error(format("alias '{}' already defined", name));
|
2012-09-17 19:01:13 +02:00
|
|
|
|
2014-08-20 00:16:21 +02:00
|
|
|
if (name.empty() or is_color_name(name) or
|
|
|
|
std::any_of(name.begin(), name.end(),
|
|
|
|
[](char c){ return not isalnum(c); }))
|
2015-06-01 22:15:59 +02:00
|
|
|
throw runtime_error(format("invalid alias name: '{}'", name));
|
2015-08-12 22:49:29 +02:00
|
|
|
|
|
|
|
if (name == facedesc)
|
|
|
|
throw runtime_error(format("cannot alias face '{}' to itself", name));
|
2012-09-17 19:01:13 +02:00
|
|
|
|
2014-08-20 00:10:56 +02:00
|
|
|
FaceOrAlias& alias = m_aliases[name];
|
2014-07-11 01:27:04 +02:00
|
|
|
auto it = m_aliases.find(facedesc);
|
2014-08-20 00:10:56 +02:00
|
|
|
if (it != m_aliases.end())
|
|
|
|
{
|
|
|
|
while (it != m_aliases.end())
|
|
|
|
{
|
|
|
|
if (it->second.alias.empty())
|
|
|
|
break;
|
|
|
|
if (it->second.alias == name)
|
|
|
|
throw runtime_error("face cycle detected");
|
|
|
|
it = m_aliases.find(it->second.alias);
|
|
|
|
}
|
|
|
|
|
|
|
|
alias.alias = facedesc;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
alias.alias = "";
|
|
|
|
alias.face = parse_face(facedesc);
|
|
|
|
}
|
2012-09-17 19:01:13 +02:00
|
|
|
}
|
|
|
|
|
2014-07-11 01:27:04 +02:00
|
|
|
CandidateList FaceRegistry::complete_alias_name(StringView prefix,
|
2014-12-23 14:54:09 +01:00
|
|
|
ByteCount cursor_pos) const
|
2014-03-29 14:18:46 +01:00
|
|
|
{
|
2014-12-23 14:54:09 +01:00
|
|
|
using ValueType = std::pair<String, FaceOrAlias>;
|
|
|
|
return complete(prefix, cursor_pos,
|
|
|
|
transformed(m_aliases,
|
|
|
|
[](const ValueType& v){ return v.first; }));
|
2014-03-29 14:18:46 +01:00
|
|
|
}
|
|
|
|
|
2014-07-11 01:27:04 +02:00
|
|
|
FaceRegistry::FaceRegistry()
|
2013-03-06 20:31:07 +01:00
|
|
|
: m_aliases{
|
2015-06-04 14:49:28 +02:00
|
|
|
{ "Default", Face{ Color::Default, Color::Default } },
|
2015-04-25 11:47:39 +02:00
|
|
|
{ "PrimarySelection", Face{ Color::White, Color::Blue } },
|
|
|
|
{ "SecondarySelection", Face{ Color::Black, Color::Blue } },
|
|
|
|
{ "PrimaryCursor", Face{ Color::Black, Color::White } },
|
|
|
|
{ "SecondaryCursor", Face{ Color::Black, Color::White } },
|
|
|
|
{ "LineNumbers", Face{ Color::Default, Color::Default } },
|
|
|
|
{ "LineNumberCursor", Face{ Color::Default, Color::Default, Attribute::Reverse } },
|
|
|
|
{ "MenuForeground", Face{ Color::White, Color::Blue } },
|
|
|
|
{ "MenuBackground", Face{ Color::Blue, Color::White } },
|
2015-10-05 02:48:00 +02:00
|
|
|
{ "MenuInfo", Face{ Color::Cyan, Color::Default } },
|
2015-04-25 11:47:39 +02:00
|
|
|
{ "Information", Face{ Color::Black, Color::Yellow } },
|
|
|
|
{ "Error", Face{ Color::Black, Color::Red } },
|
|
|
|
{ "StatusLine", Face{ Color::Cyan, Color::Default } },
|
|
|
|
{ "StatusCursor", Face{ Color::Black, Color::Cyan } },
|
|
|
|
{ "Prompt", Face{ Color::Yellow, Color::Default } },
|
|
|
|
{ "MatchingChar", Face{ Color::Default, Color::Default, Attribute::Bold } },
|
|
|
|
{ "Search", Face{ Color::Default, Color::Default, Attribute::Underline } },
|
2013-03-06 20:31:07 +01:00
|
|
|
}
|
|
|
|
{}
|
|
|
|
|
2012-09-17 19:01:13 +02:00
|
|
|
}
|