#ifndef face_hh_INCLUDED #define face_hh_INCLUDED #include "color.hh" #include "flags.hh" namespace Kakoune { enum class Attribute : int { Normal = 0, Underline = 1 << 1, Reverse = 1 << 2, Blink = 1 << 3, Bold = 1 << 4, Dim = 1 << 5, Italic = 1 << 6, FinalFg = 1 << 7, FinalBg = 1 << 8, FinalAttr = 1 << 9, Final = FinalFg | FinalBg | FinalAttr }; constexpr bool with_bit_ops(Meta::Type) { return true; } struct Face { Color fg = Color::Default; Color bg = Color::Default; Attribute attributes = Attribute::Normal; friend constexpr bool operator==(const Face& lhs, const Face& rhs) { return lhs.fg == rhs.fg and lhs.bg == rhs.bg and lhs.attributes == rhs.attributes; } friend constexpr bool operator!=(const Face& lhs, const Face& rhs) { return not (lhs == rhs); } friend constexpr size_t hash_value(const Face& val) { return hash_values(val.fg, val.bg, val.attributes); } }; inline Face merge_faces(const Face& base, const Face& face) { auto alpha_blend = [](Color base, Color color) { auto blend = [&](unsigned char Color::*field) { int blended = (base.*field * (255 - color.a) + color.*field * color.a) / 255; return static_cast(blended <= 255 ? blended : 255); }; int alpha = color.a + base.a * (255 - color.a) / 255; return Color{blend(&Color::r), blend(&Color::g), blend(&Color::b), static_cast(alpha <= 255 ? alpha : 255)}; }; auto choose = [&](Color Face::*color, Attribute final_attr) { if (face.attributes & final_attr) return face.*color; if (base.attributes & final_attr) return base.*color; if (face.*color == Color::Default) return base.*color; if ((base.*color).isRGB() and (face.*color).isRGB() and (face.*color).a != 255) return alpha_blend(base.*color, face.*color); return face.*color; }; return Face{ choose(&Face::fg, Attribute::FinalFg), choose(&Face::bg, Attribute::FinalBg), face.attributes & Attribute::FinalAttr ? face.attributes : base.attributes & Attribute::FinalAttr ? base.attributes : face.attributes | base.attributes }; } } #endif // face_hh_INCLUDED