2021-01-04 23:13:42 +00:00
|
|
|
#ifndef MALACHSCRIPT_INPUTWINDOW_HPP
|
|
|
|
#define MALACHSCRIPT_INPUTWINDOW_HPP
|
|
|
|
#include <ncurses.h>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace MalachScriptRepl {
|
|
|
|
class InputWindow {
|
|
|
|
private:
|
|
|
|
WINDOW* _window;
|
|
|
|
|
2021-01-08 15:14:02 +00:00
|
|
|
std::vector<std::string> _lines = {""};
|
2021-01-04 23:13:42 +00:00
|
|
|
int _row = 0;
|
|
|
|
int _col = 0;
|
|
|
|
int _rightPos = 0;
|
2021-01-05 14:51:59 +00:00
|
|
|
int _scroll = 0;
|
|
|
|
int _lineCount;
|
2021-01-04 23:13:42 +00:00
|
|
|
|
2021-01-08 15:14:02 +00:00
|
|
|
std::function<void(const std::vector<std::string>&)> _onChange;
|
2021-01-05 17:17:40 +00:00
|
|
|
const MalachScript::Diagnostics::Diagnostic* _diag;
|
2021-01-04 23:13:42 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
InputWindow(int height, int width, int y, int x) {
|
|
|
|
_window = newwin(height, width, y, x);
|
|
|
|
keypad(_window, true);
|
2021-01-05 14:51:59 +00:00
|
|
|
_lineCount = height;
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
|
2021-01-08 15:14:02 +00:00
|
|
|
inline void RegisterOnChange(std::function<void(const std::vector<std::string>&)> listener) {
|
2021-01-04 23:13:42 +00:00
|
|
|
_onChange = listener;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void Refresh() { wrefresh(_window); }
|
|
|
|
inline void Clear() { wclear(_window); }
|
|
|
|
|
|
|
|
[[nodiscard]] inline int GetInput() const { return wgetch(_window); }
|
|
|
|
inline std::pair<int, int> GetCursorPosition() {
|
|
|
|
int row, col;
|
|
|
|
getyx(_window, row, col);
|
|
|
|
return std::pair<int, int>(row, col);
|
|
|
|
}
|
|
|
|
|
2021-01-05 14:51:59 +00:00
|
|
|
inline void MoveCursorPosition(int row, int col) {
|
|
|
|
auto& line = _lines[row + _scroll];
|
|
|
|
if (col > (int)line.size()) {
|
|
|
|
col = line.size();
|
|
|
|
}
|
|
|
|
wmove(_window, row, col);
|
|
|
|
}
|
2021-01-04 23:13:42 +00:00
|
|
|
|
|
|
|
inline void MoveCursorLeft() {
|
|
|
|
if (_col > 0) {
|
|
|
|
wmove(_window, _row, --_col);
|
|
|
|
_rightPos = _col;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void MoveCursorRight() {
|
2021-01-05 14:51:59 +00:00
|
|
|
auto& line = _lines[_row + _scroll];
|
2021-01-04 23:13:42 +00:00
|
|
|
if (_col < (int)line.size()) {
|
|
|
|
wmove(_window, _row, ++_col);
|
|
|
|
_rightPos = _col;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void MoveCursorUp() {
|
|
|
|
if (_row > 0) {
|
|
|
|
_col = _rightPos;
|
2021-01-05 14:51:59 +00:00
|
|
|
if (_col > (int)_lines[_row - 1 + _scroll].size()) {
|
|
|
|
_col = (int)_lines[_row - 1 + _scroll].size();
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
wmove(_window, --_row, _col);
|
2021-01-05 14:51:59 +00:00
|
|
|
} else {
|
|
|
|
ScrollUp();
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void MoveCursorDown() {
|
2021-01-05 14:51:59 +00:00
|
|
|
if (_row < _lineCount - 1 && _row + _scroll < (int)_lines.size() - 1) {
|
2021-01-04 23:13:42 +00:00
|
|
|
_col = _rightPos;
|
2021-01-05 14:51:59 +00:00
|
|
|
if (_col > (int)_lines[_row + 1 + _scroll].size()) {
|
|
|
|
_col = (int)_lines[_row + 1 + _scroll].size();
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
wmove(_window, ++_row, _col);
|
2021-01-05 14:51:59 +00:00
|
|
|
} else {
|
|
|
|
ScrollDown();
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void Backspace() {
|
|
|
|
if (_col > 0) {
|
|
|
|
mvwdelch(_window, _row, --_col);
|
2021-01-05 14:51:59 +00:00
|
|
|
auto& line = _lines[_row + _scroll];
|
|
|
|
if (_col < (int)line.size()) {
|
|
|
|
line = line.erase(_col, 1);
|
|
|
|
_onChange(_lines);
|
|
|
|
}
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void Return() {
|
2021-01-08 15:14:02 +00:00
|
|
|
_lines.insert(_lines.begin() + _row + _scroll + 1, "");
|
2021-01-05 14:51:59 +00:00
|
|
|
if (_row >= _lineCount - 1) {
|
|
|
|
ScrollDown();
|
|
|
|
} else {
|
|
|
|
_row++;
|
|
|
|
_col = 0;
|
|
|
|
for (size_t i = _row - 1; i < _lines.size(); i++) {
|
|
|
|
wmove(_window, i, 0);
|
|
|
|
wclrtoeol(_window);
|
|
|
|
waddstr(_window, (char*)_lines[i].c_str());
|
|
|
|
}
|
|
|
|
wmove(_window, _row, _col);
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
2021-01-05 17:17:40 +00:00
|
|
|
_onChange(_lines);
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
inline void Tab() {
|
|
|
|
wmove(_window, _row, 0);
|
|
|
|
wclrtoeol(_window);
|
2021-01-05 14:51:59 +00:00
|
|
|
auto& line = _lines[_row + _scroll];
|
2021-01-04 23:13:42 +00:00
|
|
|
if ((int)line.length() <= _col) {
|
2021-01-08 15:14:02 +00:00
|
|
|
line += " ";
|
2021-01-04 23:13:42 +00:00
|
|
|
} else {
|
|
|
|
for (size_t i = 0; i < 4; i++) {
|
2021-01-08 15:14:02 +00:00
|
|
|
line.insert(line.begin() + _col, ' ');
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
waddstr(_window, (char*)line.c_str());
|
|
|
|
_col += 4;
|
|
|
|
_rightPos = _col;
|
|
|
|
wmove(_window, _row, _col);
|
|
|
|
_onChange(_lines);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void GoToEndOfLine() {
|
2021-01-05 14:51:59 +00:00
|
|
|
_col = _lines[_row + _scroll].size();
|
2021-01-04 23:13:42 +00:00
|
|
|
_rightPos = _col;
|
|
|
|
wmove(_window, _row, _col);
|
|
|
|
}
|
2021-01-05 14:51:59 +00:00
|
|
|
|
2021-01-04 23:13:42 +00:00
|
|
|
inline void GoToStartOfLine() {
|
2021-01-05 14:51:59 +00:00
|
|
|
_col = 0;
|
|
|
|
_rightPos = 0;
|
2021-01-04 23:13:42 +00:00
|
|
|
wmove(_window, _row, _col);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void Input(int c) {
|
|
|
|
wmove(_window, _row, 0);
|
2021-01-05 14:51:59 +00:00
|
|
|
auto& line = _lines[_row + _scroll];
|
2021-01-04 23:13:42 +00:00
|
|
|
if ((int)line.length() <= _col) {
|
2021-01-08 15:14:02 +00:00
|
|
|
line += (char)c;
|
2021-01-04 23:13:42 +00:00
|
|
|
} else {
|
|
|
|
line.insert(line.begin() + _col, c);
|
|
|
|
}
|
|
|
|
waddstr(_window, (char*)line.c_str());
|
|
|
|
_col++;
|
|
|
|
_rightPos = _col;
|
|
|
|
wmove(_window, _row, _col);
|
|
|
|
_onChange(_lines);
|
|
|
|
}
|
|
|
|
|
2021-01-05 14:51:59 +00:00
|
|
|
inline void ResetText() {
|
2021-01-08 15:14:02 +00:00
|
|
|
std::string script;
|
2021-01-05 17:17:40 +00:00
|
|
|
for (size_t i = 0; i < _lines.size(); i++) {
|
|
|
|
script += _lines[i];
|
|
|
|
if (i != _lines.size() - 1) {
|
|
|
|
script += '\n';
|
|
|
|
}
|
2021-01-05 14:51:59 +00:00
|
|
|
}
|
2021-01-05 17:17:40 +00:00
|
|
|
SetScriptWithDiagnostics(script, _diag);
|
2021-01-05 14:51:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void ScrollDown() {
|
|
|
|
if (_scroll + _row < (int)_lines.size() - 1) {
|
|
|
|
_scroll++;
|
|
|
|
ResetText();
|
|
|
|
|
|
|
|
_col = _rightPos;
|
|
|
|
if (_col > (int)_lines[_row + _scroll].size()) {
|
|
|
|
_col = (int)_lines[_row + _scroll].size();
|
|
|
|
}
|
|
|
|
wmove(_window, _row, _col);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inline void ScrollUp() {
|
|
|
|
if (_scroll <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_scroll--;
|
|
|
|
ResetText();
|
|
|
|
|
|
|
|
_col = _rightPos;
|
|
|
|
if (_col > (int)_lines[_row + _scroll].size()) {
|
|
|
|
_col = (int)_lines[_row + _scroll].size();
|
|
|
|
}
|
|
|
|
wmove(_window, _row, _col);
|
|
|
|
}
|
|
|
|
|
2021-01-08 15:14:02 +00:00
|
|
|
inline void SetScriptWithDiagnostics(std::string& script, const MalachScript::Diagnostics::Diagnostic* diag) {
|
2021-01-05 17:17:40 +00:00
|
|
|
_diag = diag;
|
2021-01-04 23:13:42 +00:00
|
|
|
auto yx = GetCursorPosition();
|
|
|
|
Clear();
|
2021-01-08 15:14:02 +00:00
|
|
|
script += " ";
|
2021-01-04 23:13:42 +00:00
|
|
|
|
2021-01-05 14:51:59 +00:00
|
|
|
size_t linenum = 0;
|
|
|
|
size_t index = 0;
|
|
|
|
size_t linestart = 0;
|
2021-01-05 17:17:40 +00:00
|
|
|
size_t lineend = 0;
|
2021-01-05 14:51:59 +00:00
|
|
|
|
|
|
|
std::istringstream f((char*)script.data());
|
|
|
|
std::string line;
|
|
|
|
while (std::getline(f, line)) {
|
|
|
|
if ((int)linenum == _scroll) {
|
|
|
|
linestart = index;
|
|
|
|
}
|
2021-01-05 17:17:40 +00:00
|
|
|
if ((int)linenum == _scroll + _lineCount - 1 || linenum == _lines.size() - 1) {
|
|
|
|
lineend = index + line.size() + 1;
|
|
|
|
break;
|
|
|
|
}
|
2021-01-05 14:51:59 +00:00
|
|
|
index += line.size() + 1;
|
|
|
|
linenum++;
|
|
|
|
}
|
2021-01-05 17:17:40 +00:00
|
|
|
if (lineend == 0) {
|
|
|
|
lineend = script.size();
|
|
|
|
}
|
2021-01-05 14:51:59 +00:00
|
|
|
|
2021-01-05 17:17:40 +00:00
|
|
|
if (diag != nullptr && diag->GetSpan().GetStart() >= linestart && diag->GetSpan().GetStart() <= lineend) {
|
2021-01-05 14:51:59 +00:00
|
|
|
auto start = diag->GetSpan().GetStart() - linestart;
|
2021-01-05 17:17:40 +00:00
|
|
|
auto end = diag->GetSpan().GetEnd() - linestart + 1;
|
|
|
|
if (diag->GetSpan().GetStart() > script.size()) {
|
|
|
|
start = script.size() - linestart - 1;
|
|
|
|
}
|
|
|
|
if (diag->GetSpan().GetEnd() >= script.size()) {
|
|
|
|
end = script.size() - linestart;
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
|
2021-01-05 17:17:40 +00:00
|
|
|
waddnstr(_window, (char*)script.substr(linestart, start).data(), start);
|
2021-01-04 23:13:42 +00:00
|
|
|
wattron(_window, COLOR_PAIR(1));
|
2021-01-05 17:17:40 +00:00
|
|
|
wattron(_window, A_UNDERLINE);
|
|
|
|
waddnstr(_window, (char*)script.substr(start + linestart, end - start).data(), end - start);
|
|
|
|
wattroff(_window, A_UNDERLINE);
|
2021-01-04 23:13:42 +00:00
|
|
|
wattroff(_window, COLOR_PAIR(1));
|
2021-01-05 17:17:40 +00:00
|
|
|
|
|
|
|
waddnstr(_window, (char*)script.substr(end + linestart, script.size() - end + linestart + 1).data(),
|
|
|
|
script.size() - end + linestart + 1);
|
2021-01-04 23:13:42 +00:00
|
|
|
|
|
|
|
if (start >= script.size() - 1) {
|
|
|
|
wattron(_window, COLOR_PAIR(1));
|
|
|
|
waddch(_window, ' ');
|
|
|
|
wattroff(_window, COLOR_PAIR(1));
|
|
|
|
}
|
|
|
|
} else {
|
2021-01-05 14:51:59 +00:00
|
|
|
waddstr(_window, (char*)script.substr(linestart).data());
|
2021-01-04 23:13:42 +00:00
|
|
|
}
|
|
|
|
MoveCursorPosition(yx.first, yx.second);
|
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // MALACHSCRIPT_INPUTWINDOW_HPP
|