#include "filters.hh" #include "window.hh" #include "display_buffer.hh" #include "filter_registry.hh" #include namespace Kakoune { void colorize_regex(DisplayBuffer& display_buffer, const boost::regex& ex, Color color) { BufferIterator display_begin = display_buffer.begin()->begin(); BufferIterator display_end = display_buffer.back().end(); boost::regex_iterator re_it(display_begin, display_end, ex, boost::match_nosubs); boost::regex_iterator re_end; DisplayBuffer::iterator atom_it = display_buffer.begin(); for (; re_it != re_end; ++re_it) { BufferIterator begin = (*re_it)[0].first; BufferIterator end = (*re_it)[0].second; auto begin_atom_it = display_buffer.atom_containing(begin, atom_it); auto end_atom_it = display_buffer.atom_containing(end, atom_it); if (begin_atom_it == end_atom_it) { if (begin_atom_it->begin() != begin) begin_atom_it = ++display_buffer.split(begin_atom_it, begin); if (begin_atom_it->end() != end) begin_atom_it = display_buffer.split(begin_atom_it, end); begin_atom_it->fg_color() = color; } atom_it = begin_atom_it; } } void colorize_cplusplus(DisplayBuffer& display_buffer) { static boost::regex preprocessor("(\\`|(?<=\\n))\\h*#\\h*[^\\n]*(?=\\n)"); colorize_regex(display_buffer, preprocessor, Color::Magenta); static boost::regex comments("//[^\\n]*\\n"); colorize_regex(display_buffer, comments, Color::Cyan); static boost::regex strings("(?|\\<-?\\d+[fdiu]?|'\\\\?[^']?'"); colorize_regex(display_buffer, values, Color::Red); static boost::regex builtin_types("\\<(void|int|char|unsigned|float|bool|size_t)\\>"); colorize_regex(display_buffer, builtin_types, Color::Yellow); static boost::regex control_keywords("\\<(while|for|if|else|do|switch|case|default|goto|break|continue|return|using|try|catch|throw)\\>"); colorize_regex(display_buffer, control_keywords, Color::Blue); //static boost::regex operators("->|\\+|\\-|\\*|/|\\\\|\\&|\\|\\^|[<>=!+-]=|=|\\(|\\)|\\[|\\]|\\{|\\}|\\<(not|and|or|xor)\\>"); //colorize_regex(display_buffer, operators, Color::Green); static boost::regex types_keywords("\\<(const|auto|namespace|static|volatile|class|struct|enum|union|public|protected|private|template|typedef|virtual)\\>"); colorize_regex(display_buffer, types_keywords, Color::Green); } void expand_tabulations(DisplayBuffer& display_buffer) { const int tabstop = 8; for (auto atom_it = display_buffer.begin(); atom_it != display_buffer.end(); ++atom_it) { for (BufferIterator it = atom_it->begin(); it != atom_it->end(); ++it) { if (*it == '\t') { if (it != atom_it->begin()) atom_it = ++display_buffer.split(atom_it, it); if (it+1 != atom_it->end()) atom_it = display_buffer.split(atom_it, it+1); BufferCoord pos = it.buffer().line_and_column_at(it); int column = 0; for (auto line_it = it.buffer().iterator_at({pos.line, 0}); line_it != it; ++line_it) { assert(*line_it != '\n'); if (*line_it == '\t') column += tabstop - (column % tabstop); else ++column; } int count = tabstop - (column % tabstop); display_buffer.replace_atom_content(atom_it, std::string(count, ' ')); } } } } void show_line_numbers(DisplayBuffer& display_buffer) { const Buffer& buffer = display_buffer.front().begin().buffer(); BufferCoord coord = buffer.line_and_column_at(display_buffer.begin()->begin()); int last_line = buffer.line_and_column_at(display_buffer.back().end()-1).line; for (; coord.line <= last_line; ++coord.line) { BufferIterator line_start = buffer.iterator_at(coord); DisplayBuffer::iterator atom_it = display_buffer.atom_containing(line_start); if (atom_it != display_buffer.end()) { if (atom_it->begin() != line_start) { if (not atom_it->splitable()) continue; atom_it = ++display_buffer.split(atom_it, line_start); } atom_it = display_buffer.insert( atom_it, DisplayAtom(atom_it->coord(), atom_it->begin(), atom_it->begin(), Color::Black, Color::White)); char buffer[6]; snprintf(buffer, 6, "%3d ", coord.line + 1); display_buffer.replace_atom_content(atom_it, buffer); } } } template class SimpleFilterFactory { public: SimpleFilterFactory(const std::string& id) : m_id(id) {} FilterAndId operator()(Window& window, const FilterParameters& params) const { return FilterAndId(m_id, FilterFunc(filter_func)); } private: std::string m_id; }; class SelectionsHighlighter { public: SelectionsHighlighter(Window& window) : m_window(window) { } void operator()(DisplayBuffer& display_buffer) { SelectionList sorted_selections = m_window.selections(); std::sort(sorted_selections.begin(), sorted_selections.end(), [](const Selection& lhs, const Selection& rhs) { return lhs.begin() < rhs.begin(); }); auto atom_it = display_buffer.begin(); auto sel_it = sorted_selections.begin(); while (atom_it != display_buffer.end() and sel_it != sorted_selections.end()) { Selection& sel = *sel_it; DisplayAtom& atom = *atom_it; // [###------] if (atom.begin() >= sel.begin() and atom.begin() < sel.end() and atom.end() > sel.end()) { atom_it = display_buffer.split(atom_it, sel.end()); atom_it->attribute() |= Attributes::Underline; ++atom_it; ++sel_it; } // [---###---] else if (atom.begin() < sel.begin() and atom.end() > sel.end()) { atom_it = display_buffer.split(atom_it, sel.begin()); atom_it = display_buffer.split(++atom_it, sel.end()); atom_it->attribute() |= Attributes::Underline; ++atom_it; ++sel_it; } // [------###] else if (atom.begin() < sel.begin() and atom.end() > sel.begin()) { atom_it = ++display_buffer.split(atom_it, sel.begin()); atom_it->attribute() |= Attributes::Underline; ++atom_it; } // [#########] else if (atom.begin() >= sel.begin() and atom.end() <= sel.end()) { atom_it->attribute() |= Attributes::Underline; ++atom_it; } // [---------] else if (atom.begin() >= sel.end()) ++sel_it; // [---------] else if (atom.end() <= sel.begin()) ++atom_it; else assert(false); } } static FilterAndId create(Window& window, const FilterParameters& params) { return FilterAndId("highlight_selections", SelectionsHighlighter(window)); } private: const Window& m_window; }; void register_filters() { FilterRegistry& registry = FilterRegistry::instance(); registry.register_factory("highlight_selections", SelectionsHighlighter::create); registry.register_factory("expand_tabs", SimpleFilterFactory("expand_tabs")); registry.register_factory("number_lines", SimpleFilterFactory("number_lines")); registry.register_factory("hlcpp", SimpleFilterFactory("hlcpp")); } }