13 : out_(std::move(out)) {}
31 unsigned char TerminalInputParser::Current() {
32 return pending_[position_];
35 bool TerminalInputParser::Eat() {
37 return position_ < (int)pending_.size();
40 void TerminalInputParser::Send(TerminalInputParser::Output output) {
41 switch (output.type) {
60 out_->Send(
Event::Mouse(std::move(pending_), output.mouse));
64 case CURSOR_REPORTING:
73 TerminalInputParser::Output TerminalInputParser::Parse() {
113 TerminalInputParser::Output TerminalInputParser::ParseUTF8() {
114 unsigned char head =
static_cast<unsigned char>(Current());
115 unsigned char selector = 0b1000
'0000;
117 // The non code-point part of the first byte.
118 unsigned char mask = selector;
120 // Find the first zero in the first byte.
122 for (int i = 0; i < 8; ++i) {
124 if (head & selector) {
132 // Accumulate the value of the first byte.
133 uint32_t value = head & ~mask;
135 // Invalid UTF8, with more than 5 bytes.
136 if (first_zero == 1 || first_zero >= 5)
140 for (int i = 2; i <= first_zero; ++i) {
144 // Invalid continuation byte.
145 head = static_cast<unsigned char>(Current());
146 if ((head & 0b1100'0000) != 0b1000
'0000)
149 value += head & 0b0011'1111;
154 if (value <= 0b000
'0000'0111
'1111) {
156 } else if (value <= 0b000'0111
'1111'1111) {
158 }
else if (value <= 0b1111
'1111'1111
'1111) {
160 } else if (value <= 0b1'0000
'1111'1111
'1111'1111) {
166 if (extra_byte != position_)
172 TerminalInputParser::Output TerminalInputParser::ParseESC() {
189 TerminalInputParser::Output TerminalInputParser::ParseDCS() {
195 if (Current() !=
'\x1B')
201 if (Current() !=
'\\')
208 TerminalInputParser::Output TerminalInputParser::ParseCSI() {
209 bool altered =
false;
211 std::vector<int> arguments;
216 if (Current() ==
'<') {
221 if (Current() >=
'0' && Current() <=
'9') {
223 argument += int(Current() -
'0');
227 if (Current() ==
';') {
228 arguments.push_back(argument);
233 if (Current() >=
' ' && Current() <=
'~' && Current() !=
'<') {
234 arguments.push_back(argument);
238 return ParseMouse(altered,
true, std::move(arguments));
240 return ParseMouse(altered,
false, std::move(arguments));
242 return ParseCursorReporting(std::move(arguments));
249 if (Current() ==
'\x1B')
254 TerminalInputParser::Output TerminalInputParser::ParseOSC() {
259 if (Current() !=
'\x1B')
263 if (Current() !=
'\\')
269 TerminalInputParser::Output TerminalInputParser::ParseMouse(
272 std::vector<int> arguments) {
273 if (arguments.size() != 3)
278 Output output(MOUSE);
280 ((arguments[0] & 64) >> 4));
281 output.mouse.motion = Mouse::Motion(pressed);
282 output.mouse.shift = bool(arguments[0] & 4);
283 output.mouse.meta = bool(arguments[0] & 8);
284 output.mouse.x = arguments[1];
285 output.mouse.y = arguments[2];
289 TerminalInputParser::Output TerminalInputParser::ParseCursorReporting(
290 std::vector<int> arguments) {
291 if (arguments.size() != 2)
293 Output output(CURSOR_REPORTING);
294 output.cursor.y = arguments[0];
295 output.cursor.x = arguments[1];