+static void es_interpret_csi(madtty_t *rt)
+{
+ static int csiparam[MAX_CSI_ES_PARAMS];
+ int param_count = 0;
+ const char *p = rt->esbuf + 1;
+ char verb = rt->esbuf[rt->esbuf_len - 1];
+
+ if (!strncmp(rt->esbuf, "[?", 2)) { /* private-mode CSI, ignore */
+ return;
+ }
+
+ /* parse numeric parameters */
+ while (isdigit((unsigned char)*p) || *p == ';') {
+ if (*p == ';') {
+ if (param_count >= MAX_CSI_ES_PARAMS) return; /* too long! */
+ csiparam[param_count++] = 0;
+ } else {
+ if (param_count == 0) csiparam[param_count++] = 0;
+ csiparam[param_count - 1] *= 10;
+ csiparam[param_count - 1] += *p - '0';
+ }
+
+ p++;
+ }
+
+ /* delegate handling depending on command character (verb) */
+ switch (verb) {
+ case 'h':
+ if (param_count == 1 && csiparam[0] == 4) /* insert mode */
+ rt->insert = true;
+ break;
+ case 'l':
+ if (param_count == 1 && csiparam[0] == 4) /* replace mode */
+ rt->insert = false;
+ break;
+ case 'm': /* it's a 'set attribute' sequence */
+ interpret_csi_SGR(rt, csiparam, param_count); break;
+ case 'J': /* it's an 'erase display' sequence */
+ interpret_csi_ED(rt, csiparam, param_count); break;
+ case 'H': case 'f': /* it's a 'move cursor' sequence */
+ interpret_csi_CUP(rt, csiparam, param_count); break;
+ case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': case 'G':
+ case 'e': case 'a': case 'd': case '`':
+ /* it is a 'relative move' */
+ interpret_csi_C(rt, verb, csiparam, param_count); break;
+ case 'K': /* erase line */
+ interpret_csi_EL(rt, csiparam, param_count); break;
+ case '@': /* insert characters */
+ interpret_csi_ICH(rt, csiparam, param_count); break;
+ case 'P': /* delete characters */
+ interpret_csi_DCH(rt, csiparam, param_count); break;
+ case 'L': /* insert lines */
+ interpret_csi_IL(rt, csiparam, param_count); break;
+ case 'M': /* delete lines */
+ interpret_csi_DL(rt, csiparam, param_count); break;
+ case 'X': /* erase chars */
+ interpret_csi_ECH(rt, csiparam, param_count); break;
+ case 'r': /* set scrolling region */
+ interpret_csi_DECSTBM(rt, csiparam, param_count); break;
+ case 's': /* save cursor location */
+ rt->curs_srow = rt->curs_col;
+ rt->curs_scol = rt->curs_row;
+ break;
+ case 'u': /* restore cursor location */
+ rt->curs_col = rt->curs_srow;
+ rt->curs_row = rt->curs_scol;
+ break;
+ default:
+ break;
+ }
+}