1378 lines
30 KiB
C
1378 lines
30 KiB
C
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
#include <ctype.h>
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <libgen.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/types.h>
|
|
#include <linux/limits.h>
|
|
|
|
#include "config.h"
|
|
|
|
int cat_mode = 0;
|
|
struct termios newt, oldt;
|
|
int rows, cols;
|
|
editor_t editor[9];
|
|
int editors = 0;
|
|
editor_t *cur_editor;
|
|
|
|
void draw_status_bar(void);
|
|
void refresh_screen(void);
|
|
void move_xy(int key);
|
|
void insert_char(int c);
|
|
void insert_new_line(void);
|
|
void shift_new_line(void);
|
|
void del_char(void);
|
|
void init_editor(char *filename);
|
|
char *prompt_editor(char *prompt, void (*callback)(char *, int));
|
|
void find_callback(char *query, int key);
|
|
void find_editor(void);
|
|
int readch(void);
|
|
void save_file(void);
|
|
void draw_rows(void);
|
|
void update_row(row_t *row);
|
|
void insert_row(int at, char *s, size_t len);
|
|
void del_row(int at);
|
|
char *export_buffer(int *buflen);
|
|
void replace_home(char *path);
|
|
int is_symbol(int c);
|
|
int is_separator(int c);
|
|
void update_highlight(row_t *row);
|
|
void select_syntax(void);
|
|
void die(const char *s);
|
|
void cleanup(void);
|
|
void wpprintw(const char *fmt, ...);
|
|
void move_cursor(int row, int col);
|
|
int get_window_size(int *row, int *col);
|
|
void bprintf(const char *fmt, ...);
|
|
|
|
/*
|
|
* Draw status bar
|
|
* Like neovim's lualine
|
|
*/
|
|
void draw_status_bar(void)
|
|
{
|
|
move_cursor(rows - 1, 1);
|
|
/* Reverse and bold */
|
|
bprintf("\033[7m\033[1m");
|
|
|
|
int mode_len;
|
|
switch (cur_editor->mode) {
|
|
case NORMAL:
|
|
bprintf("%s NORMAL ", BLUE_BG);
|
|
mode_len = 8;
|
|
break;
|
|
case INSERT:
|
|
bprintf("%s INSERT ", GREEN_BG);
|
|
mode_len = 8;
|
|
break;
|
|
case COMMAND:
|
|
bprintf("%s COMMAND ", PEACH_BG);
|
|
mode_len = 9;
|
|
break;
|
|
case VISUAL:
|
|
bprintf("%s VISUAL ", MAUVE_BG);
|
|
mode_len = 8;
|
|
break;
|
|
}
|
|
|
|
/* Reset */
|
|
bprintf("\033[22m");
|
|
|
|
char git[80], file[80], info[80], lines[80], coord[80];
|
|
int git_len = snprintf(git, sizeof(git), " %s | %s ", "master", "+1");
|
|
int file_len = snprintf(file, sizeof(file), " %s %s",
|
|
strlen(cur_editor->filename) > 0 ? cur_editor->filename : "[No Name]", cur_editor->dirty ? "[+]" : "");
|
|
int info_len = snprintf(info, sizeof(info), " %s ", cur_editor->syntax ? cur_editor->syntax->filetype : "");
|
|
int lines_len;
|
|
if (cur_editor->y == 0 || cur_editor->rows == cur_editor->y + 1) {
|
|
lines_len = snprintf(lines, sizeof(lines), " %s ", cur_editor->y == 0 ? "Top" : "Bot");
|
|
} else {
|
|
lines_len = snprintf(lines, sizeof(lines), " %d%% ", ((cur_editor->y + 1) * 100 / cur_editor->rows));
|
|
}
|
|
int coord_len = snprintf(coord, sizeof(coord), " %d:%d ", cur_editor->y + 1, cur_editor->x + 1);
|
|
|
|
bprintf(SURFACE_1_BG); /* background */
|
|
/* text */
|
|
switch (cur_editor->mode) {
|
|
case NORMAL:
|
|
bprintf("%s", BLUE_FG);
|
|
break;
|
|
case INSERT:
|
|
bprintf("%s", GREEN_FG);
|
|
break;
|
|
case COMMAND:
|
|
bprintf("%s", PEACH_FG);
|
|
break;
|
|
case VISUAL:
|
|
bprintf("%s", MAUVE_FG);
|
|
break;
|
|
}
|
|
bprintf("%s%s%s%s", git, BLACK_BG, WHITE_FG, file);
|
|
|
|
while (file_len < cols) {
|
|
if (cols - mode_len - git_len - file_len == info_len + lines_len + coord_len) {
|
|
bprintf("%s%s", info, SURFACE_1_BG);
|
|
if (cur_editor->mode == NORMAL) {
|
|
bprintf(BLUE_FG);
|
|
} else if (cur_editor->mode == INSERT) {
|
|
bprintf(GREEN_FG);
|
|
} else if (cur_editor->mode == COMMAND) {
|
|
bprintf(PEACH_FG);
|
|
} else if (cur_editor->mode == VISUAL) {
|
|
bprintf(MAUVE_FG);
|
|
}
|
|
bprintf("%s", lines);
|
|
if (cur_editor->mode == NORMAL) {
|
|
bprintf(BLUE_BG);
|
|
} else if (cur_editor->mode == INSERT) {
|
|
bprintf(GREEN_BG);
|
|
} else if (cur_editor->mode == COMMAND) {
|
|
bprintf(PEACH_BG);
|
|
} else if (cur_editor->mode == VISUAL) {
|
|
bprintf(MAUVE_BG);
|
|
}
|
|
bprintf("%s\033[1m%s\033[22m", BLACK_FG, coord);
|
|
break;
|
|
} else {
|
|
bprintf(" ");
|
|
file_len++;
|
|
}
|
|
}
|
|
bprintf("\033[m");
|
|
}
|
|
|
|
void refresh_screen(void)
|
|
{
|
|
/* Scroll */
|
|
cur_editor->rx = 0;
|
|
if (cur_editor->y < cur_editor->rows) {
|
|
for (int i = 0; i < cur_editor->x; i++) {
|
|
if (cur_editor->row[cur_editor->y].chars[i] == '\t') {
|
|
cur_editor->rx += (TAB_SIZE - 1) - (cur_editor->rx % TAB_SIZE);
|
|
}
|
|
cur_editor->rx++;
|
|
}
|
|
}
|
|
if (cur_editor->y < cur_editor->rowoff) {
|
|
cur_editor->rowoff = cur_editor->y;
|
|
}
|
|
if (cur_editor->y >= cur_editor->rowoff + rows - 3) {
|
|
cur_editor->rowoff = cur_editor->y - rows + 3;
|
|
}
|
|
if (cur_editor->rx < cur_editor->coloff) {
|
|
cur_editor->coloff = cur_editor->rx;
|
|
}
|
|
|
|
if (!cat_mode) {
|
|
if (cur_editor->rx >= cur_editor->coloff + cols) {
|
|
cur_editor->coloff = cur_editor->rx - cols + 1;
|
|
}
|
|
bprintf("\033H\033[2 q");
|
|
}
|
|
|
|
draw_rows();
|
|
if (!cat_mode) {
|
|
draw_status_bar();
|
|
move_cursor((cur_editor->y - cur_editor->rowoff) + 1,
|
|
(cur_editor->rx - cur_editor->coloff) + 1);
|
|
}
|
|
}
|
|
|
|
void move_xy(int key)
|
|
{
|
|
row_t *row = cur_editor->y >= cur_editor->rows ? NULL : &cur_editor->row[cur_editor->y];
|
|
switch (key) {
|
|
case ARROW_LEFT:
|
|
if (cur_editor->x != 0) {
|
|
cur_editor->x--;
|
|
}
|
|
break;
|
|
case ARROW_RIGHT:
|
|
if (row && cur_editor->x < row->size) {
|
|
cur_editor->x++;
|
|
}
|
|
break;
|
|
case ARROW_UP:
|
|
if (cur_editor->y != 0) {
|
|
cur_editor->y--;
|
|
}
|
|
break;
|
|
case ARROW_DOWN:
|
|
if (cur_editor->y < cur_editor->rows) {
|
|
cur_editor->y++;
|
|
}
|
|
break;
|
|
}
|
|
/* Shifted row */
|
|
row = cur_editor->y >= cur_editor->rows ? NULL : &cur_editor->row[cur_editor->y];
|
|
int rowlen = row ? row->size : 0;
|
|
if (cur_editor->x > rowlen) {
|
|
cur_editor->x = rowlen;
|
|
}
|
|
}
|
|
|
|
void insert_char(int c)
|
|
{
|
|
row_t *row = &cur_editor->row[cur_editor->y];
|
|
if (cur_editor->y == cur_editor->rows) {
|
|
insert_row(cur_editor->rows, "", 0);
|
|
}
|
|
if (cur_editor->x < 0 || cur_editor->x > row->size) {
|
|
cur_editor->x = row->size;
|
|
}
|
|
/* Insert char in row */
|
|
row->chars = realloc(row->chars, row->size + 2);
|
|
memmove(&row->chars[cur_editor->x + 1], &row->chars[cur_editor->x], row->size - cur_editor->x + 1);
|
|
row->size++;
|
|
row->chars[cur_editor->x] = c;
|
|
update_row(row);
|
|
cur_editor->dirty++;
|
|
cur_editor->x++;
|
|
}
|
|
|
|
void insert_new_line(void)
|
|
{
|
|
if (cur_editor->x == 0) {
|
|
insert_row(cur_editor->y, "", 0);
|
|
} else {
|
|
row_t *row = &cur_editor->row[cur_editor->y];
|
|
insert_row(cur_editor->y + 1, &row->chars[cur_editor->x], row->size - cur_editor->x);
|
|
row = &cur_editor->row[cur_editor->y];
|
|
row->size = cur_editor->x;
|
|
row->chars[row->size] = '\0';
|
|
update_row(row);
|
|
}
|
|
cur_editor->y++;
|
|
cur_editor->x = 0;
|
|
}
|
|
|
|
/*
|
|
* 'O' in vim
|
|
*/
|
|
void shift_previous_line(void)
|
|
{
|
|
insert_row(cur_editor->y, "", 0);
|
|
cur_editor->x = 0;
|
|
}
|
|
|
|
/*
|
|
* 'o' in vim
|
|
*/
|
|
void shift_new_line(void)
|
|
{
|
|
insert_row(cur_editor->y + 1, "", 0);
|
|
cur_editor->y++;
|
|
cur_editor->x = 0;
|
|
}
|
|
|
|
void del_char(void)
|
|
{
|
|
if (cur_editor->y == cur_editor->rows) return;
|
|
if (cur_editor->x == 0 && cur_editor->y == 0) return;
|
|
|
|
row_t *row = &cur_editor->row[cur_editor->y];
|
|
if (cur_editor->x > 0) {
|
|
/* Delete char in row */
|
|
if (cur_editor->x - 1 < 0 || cur_editor->x - 1 >= row->size) return;
|
|
memmove(&row->chars[cur_editor->x - 1], &row->chars[cur_editor->x], row->size - cur_editor->x + 1);
|
|
row->size--;
|
|
update_row(row);
|
|
cur_editor->dirty++;
|
|
cur_editor->x--;
|
|
} else {
|
|
row_t *prev_row = &cur_editor->row[cur_editor->y - 1];
|
|
cur_editor->x = prev_row->size;
|
|
/* Append string to previous row */
|
|
prev_row->chars = realloc(prev_row->chars, prev_row->size + row->size + 1);
|
|
memcpy(&prev_row->chars[prev_row->size], row->chars, row->size);
|
|
prev_row->size += row->size;
|
|
prev_row->chars[prev_row->size] = '\0';
|
|
update_row(prev_row);
|
|
cur_editor->dirty++;
|
|
|
|
del_row(cur_editor->y);
|
|
cur_editor->y--;
|
|
}
|
|
}
|
|
|
|
void init_editor(char *filename)
|
|
{
|
|
if (editors > 8) {
|
|
wpprintw("%sOnly 9 tabs are allowed", ERROR);
|
|
readch();
|
|
wpprintw("");
|
|
return;
|
|
}
|
|
cur_editor = &editor[editors++];
|
|
cur_editor->x = 0;
|
|
cur_editor->y = 0;
|
|
cur_editor->rx = 0;
|
|
cur_editor->rowoff = 0;
|
|
cur_editor->coloff = 0;
|
|
cur_editor->rows = 0;
|
|
cur_editor->row = NULL;
|
|
cur_editor->dirty = 0;
|
|
cur_editor->mode = NORMAL;
|
|
cur_editor->syntax = NULL;
|
|
char cwd[PATH_MAX];
|
|
getcwd(cwd, PATH_MAX);
|
|
strcpy(cur_editor->cwd, cwd);
|
|
|
|
if (filename) {
|
|
strcpy(cur_editor->filename, basename(filename));
|
|
|
|
FILE *fp = fopen(filename, "r");
|
|
if (!fp) {
|
|
die("fopen");
|
|
}
|
|
char *line = NULL;
|
|
size_t linecap = 0;
|
|
ssize_t len;
|
|
while ((len = getline(&line, &linecap, fp)) != -1) {
|
|
/* remove new line and carriage return at end of line */
|
|
while (len > 0 && (line[len - 1] == '\n' || line[len - 1] == '\r')) {
|
|
len--;
|
|
}
|
|
insert_row(cur_editor->rows, line, len);
|
|
}
|
|
|
|
select_syntax();
|
|
|
|
free(line);
|
|
fclose(fp);
|
|
/* Reset dirtiness as nothing is modified yet */
|
|
cur_editor->dirty = 0;
|
|
} else {
|
|
strcpy(cur_editor->filename, "");
|
|
}
|
|
}
|
|
|
|
char *prompt_editor(char *prompt, void (*callback)(char *, int))
|
|
{
|
|
size_t bufsize = 128;
|
|
char *buf = malloc(bufsize);
|
|
size_t buflen = 0;
|
|
int prompt_len = strlen(prompt);
|
|
buf[0] = '\0';
|
|
while (1) {
|
|
wpprintw(prompt, buf);
|
|
move_cursor(rows, prompt_len - 1 + strlen(buf));
|
|
int c = readch();
|
|
if (c == DEL_KEY || c == CTRL_KEY('h') || c == BACKSPACE) {
|
|
if (buflen != 0) {
|
|
buf[--buflen] = '\0';
|
|
}
|
|
} else if (c == '\033') {
|
|
wpprintw("");
|
|
cur_editor->mode = NORMAL;
|
|
if (callback) callback(buf, c);
|
|
free(buf);
|
|
return NULL;
|
|
} else if (c == '\r') {
|
|
if (buflen != 0) {
|
|
wpprintw("");
|
|
if (callback) callback(buf, c);
|
|
return buf;
|
|
}
|
|
} else if (!iscntrl(c) && c < 128) {
|
|
if (buflen == bufsize - 1) {
|
|
bufsize *= 2;
|
|
buf = realloc(buf, bufsize);
|
|
}
|
|
buf[buflen++] = c;
|
|
buf[buflen] = '\0';
|
|
}
|
|
|
|
if (callback) callback(buf, c);
|
|
}
|
|
}
|
|
|
|
void find_callback(char *query, int key)
|
|
{
|
|
static int last_match = -1;
|
|
static int direction = 1;
|
|
|
|
static int saved_hl_line;
|
|
static char *saved_hl = NULL;
|
|
if (saved_hl) {
|
|
memcpy(cur_editor->row[saved_hl_line].hl, saved_hl, cur_editor->row[saved_hl_line].render_size);
|
|
free(saved_hl);
|
|
saved_hl = NULL;
|
|
}
|
|
|
|
if (key == '\r' || key == '\033') {
|
|
last_match = -1;
|
|
direction = 1;
|
|
return;
|
|
} else if (key == CTRL_KEY('n')) {
|
|
direction = 1;
|
|
} else if (key == CTRL_KEY('p')) {
|
|
direction = -1;
|
|
} else {
|
|
last_match = -1;
|
|
direction = 1;
|
|
}
|
|
|
|
if (last_match == -1) direction = 1;
|
|
int current = last_match;
|
|
for (int i = 0; i < cur_editor->rows; i++) {
|
|
current += direction;
|
|
|
|
if (current == -1) current = cur_editor->rows - 1;
|
|
else if (current == cur_editor->rows) current = 0;
|
|
|
|
row_t *row = &cur_editor->row[current];
|
|
char *match = strstr(row->render, query);
|
|
if (match) {
|
|
last_match = current;
|
|
cur_editor->y = current;
|
|
int cur_rx = 0;
|
|
int x;
|
|
for (x = 0; x < row->size; x++) {
|
|
if (row->chars[x] == '\t') {
|
|
cur_rx += (TAB_SIZE - 1) - (cur_rx % TAB_SIZE);
|
|
}
|
|
cur_rx++;
|
|
if (cur_rx > match - row->render) {
|
|
cur_editor->x = x;
|
|
break;
|
|
}
|
|
}
|
|
cur_editor->rowoff = cur_editor->rows;
|
|
|
|
saved_hl_line = current;
|
|
saved_hl = malloc(row->render_size);
|
|
memcpy(saved_hl, row->hl, row->render_size);
|
|
|
|
memset(&row->hl[match - row->render], MATCH, strlen(query));
|
|
memset(&row->hl[match - row->render + strlen(query)], RESET, row->render_size - (match - row->render + strlen(query)));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void find_editor(void)
|
|
{
|
|
int tmp_x = cur_editor->x;
|
|
int tmp_y = cur_editor->y;
|
|
int tmp_coloff = cur_editor->coloff;
|
|
int tmp_rowoff = cur_editor->rowoff;
|
|
char *query = prompt_editor("/%s", find_callback);
|
|
if (query) {
|
|
free(query);
|
|
} else {
|
|
cur_editor->x = tmp_x;
|
|
cur_editor->y = tmp_y;
|
|
cur_editor->coloff = tmp_coloff;
|
|
cur_editor->rowoff = tmp_rowoff;
|
|
}
|
|
}
|
|
|
|
int readch(void)
|
|
{
|
|
int nread;
|
|
char c;
|
|
while ((nread = read(STDIN_FILENO, &c, 1)) != 1) {
|
|
if (nread == -1 && errno != EAGAIN) {
|
|
die("read");
|
|
}
|
|
}
|
|
|
|
if (c == '\033') {
|
|
char seq[3];
|
|
|
|
if (read(STDIN_FILENO, &seq[0], 1) != 1)
|
|
return '\033';
|
|
if (read(STDIN_FILENO, &seq[1], 1) != 1)
|
|
return '\033';
|
|
|
|
if (seq[0] == '[') {
|
|
if (seq[1] >= '0' && seq[1] <= '9') {
|
|
if (read(STDIN_FILENO, &seq[2], 1) != 1)
|
|
return '\033';
|
|
if (seq[2] == '~') {
|
|
switch (seq[1]) {
|
|
case '1': return HOME_KEY;
|
|
case '3': return DEL_KEY;
|
|
case '4': return END_KEY;
|
|
case '5': return PAGE_UP;
|
|
case '6': return PAGE_DOWN;
|
|
case '7': return HOME_KEY;
|
|
case '8': return END_KEY;
|
|
}
|
|
}
|
|
} else {
|
|
switch (seq[1]) {
|
|
case 'A': return ARROW_UP;
|
|
case 'B': return ARROW_DOWN;
|
|
case 'C': return ARROW_RIGHT;
|
|
case 'D': return ARROW_LEFT;
|
|
case 'F': return END_KEY;
|
|
case 'H': return HOME_KEY;
|
|
}
|
|
}
|
|
} else if (seq[0] == 'O') {
|
|
switch (seq[1]) {
|
|
case 'F': return END_KEY;
|
|
case 'H': return HOME_KEY;
|
|
}
|
|
}
|
|
return '\033';
|
|
} else {
|
|
return c;
|
|
}
|
|
}
|
|
|
|
void save_file(void)
|
|
{
|
|
if (strlen(cur_editor->filename) == 0) {
|
|
char *new_filename = prompt_editor("Save as: %s", NULL);
|
|
if (new_filename == NULL) {
|
|
wpprintw("Save aborted");
|
|
return;
|
|
}
|
|
strcpy(cur_editor->filename, new_filename);
|
|
select_syntax();
|
|
}
|
|
int fd = open(cur_editor->filename, O_RDWR | O_CREAT, 0644);
|
|
if (fd != -1) {
|
|
int len;
|
|
char *buf = export_buffer(&len);
|
|
if (ftruncate(fd, len) != -1) {
|
|
if (write(fd, buf, len) == len) {
|
|
close(fd);
|
|
free(buf);
|
|
cur_editor->dirty = 0;
|
|
wpprintw("\"%s\" %dL, %dB written", cur_editor->filename, cur_editor->rows, len);
|
|
return;
|
|
}
|
|
free(buf);
|
|
}
|
|
close(fd);
|
|
}
|
|
wpprintw("Error saving: %s", strerror(errno));
|
|
}
|
|
|
|
void draw_rows(void)
|
|
{
|
|
for (int y = 0; y < cur_editor->rows; y++) {
|
|
if (!cat_mode && y > rows - 2) return;
|
|
if (!cat_mode) move_cursor(y + 1, 1);
|
|
int filerow = y + cur_editor->rowoff;
|
|
if (filerow >= cur_editor->rows) {
|
|
if (cur_editor->rows == 0 && y == rows / 2) {
|
|
char welcome[11];
|
|
snprintf(welcome, sizeof(welcome), "VIP v%s", VERSION);
|
|
/* Length of welcome message must be 10 */
|
|
int padding = (cols - 10) / 2;
|
|
while (padding--)
|
|
bprintf(" ");
|
|
bprintf(welcome);
|
|
}
|
|
} else {
|
|
int len = cur_editor->row[filerow].render_size - cur_editor->coloff;
|
|
if (len < 0) len = 0;
|
|
if (!cat_mode && len > cols) len = cols;
|
|
char *c = &cur_editor->row[filerow].render[cur_editor->coloff];
|
|
unsigned char *hl = &cur_editor->row[filerow].hl[cur_editor->coloff];
|
|
|
|
for (int j = 0; j < len; j++) {
|
|
if (iscntrl(c[j])) {
|
|
bprintf("%s%c\033[m", OVERLAY_2_BG, '@' + c[j]);
|
|
} else if (hl[j] == NORMAL) {
|
|
bprintf("%s%c", WHITE_BG, c[j]);
|
|
} else {
|
|
switch (hl[j]) {
|
|
case NUMBER:
|
|
bprintf(PEACH_BG);
|
|
break;
|
|
|
|
case STRING:
|
|
bprintf(GREEN_BG);
|
|
break;
|
|
|
|
case CHAR:
|
|
bprintf(TEAL_BG);
|
|
break;
|
|
|
|
case ESCAPE:
|
|
bprintf(PINK_BG);
|
|
break;
|
|
|
|
case SYMBOL:
|
|
bprintf(SKY_BG);
|
|
break;
|
|
|
|
case TERMINATOR:
|
|
bprintf(OVERLAY_2_BG);
|
|
break;
|
|
|
|
case COMMENT:
|
|
case MLCOMMENT:
|
|
bprintf("\033[3m");
|
|
bprintf(OVERLAY_2_BG);
|
|
break;
|
|
|
|
case KW:
|
|
bprintf(MAUVE_BG);
|
|
break;
|
|
|
|
case KW_TYPE:
|
|
bprintf(YELLOW_BG);
|
|
break;
|
|
|
|
case KW_FN:
|
|
bprintf(BLUE_BG);
|
|
break;
|
|
|
|
case KW_BRACKET:
|
|
bprintf(RED_BG);
|
|
break;
|
|
|
|
case MATCH:
|
|
bprintf(BLACK_BG);
|
|
bprintf(SKY_FG);
|
|
break;
|
|
|
|
case RESET:
|
|
bprintf(BLACK_BG);
|
|
bprintf(SKY_FG);
|
|
break;
|
|
|
|
default:
|
|
bprintf(WHITE_BG);
|
|
}
|
|
bprintf("%c", c[j]);
|
|
if (hl[j] == COMMENT || hl[j] == MLCOMMENT) {
|
|
bprintf("\033[23m");
|
|
}
|
|
}
|
|
}
|
|
bprintf(WHITE_BG);
|
|
}
|
|
|
|
bprintf("\033[K\n");
|
|
}
|
|
}
|
|
|
|
void update_row(row_t *row)
|
|
{
|
|
int tabs = 0;
|
|
for (int j = 0; j < row->size; j++)
|
|
if (row->chars[j] == '\t') tabs++;
|
|
free(row->render);
|
|
row->render = malloc(row->size + tabs * (TAB_SIZE - 1) + 1);
|
|
int idx = 0;
|
|
for (int j = 0; j < row->size; j++) {
|
|
if (row->chars[j] == '\t') {
|
|
row->render[idx++] = ' ';
|
|
while (idx % TAB_SIZE != 0) {
|
|
row->render[idx++] = ' ';
|
|
}
|
|
}
|
|
else {
|
|
row->render[idx++] = row->chars[j];
|
|
}
|
|
}
|
|
row->render[idx] = '\0';
|
|
row->render_size = idx;
|
|
update_highlight(row);
|
|
}
|
|
|
|
void insert_row(int at, char *s, size_t len)
|
|
{
|
|
if (at < 0 || at > cur_editor->rows) return;
|
|
|
|
cur_editor->row = realloc(cur_editor->row, sizeof(row_t) * (cur_editor->rows + 1));
|
|
memmove(&cur_editor->row[at + 1], &cur_editor->row[at], sizeof(row_t) * (cur_editor->rows - at));
|
|
for (int j = at + 1; j <= cur_editor->rows; j++) {
|
|
cur_editor->row[j].idx++;
|
|
}
|
|
|
|
cur_editor->row[at].idx = at;
|
|
cur_editor->row[at].size = len;
|
|
cur_editor->row[at].chars = malloc(len + 1);
|
|
memcpy(cur_editor->row[at].chars, s, len);
|
|
cur_editor->row[at].chars[len] = '\0';
|
|
cur_editor->row[at].hl = NULL;
|
|
cur_editor->row[at].opened_comment = 0;
|
|
cur_editor->row[at].render_size = 0;
|
|
cur_editor->row[at].render = NULL;
|
|
update_row(&cur_editor->row[at]);
|
|
|
|
cur_editor->rows++;
|
|
cur_editor->dirty++;
|
|
}
|
|
|
|
void del_row(int at)
|
|
{
|
|
if (at < 0 || at >= cur_editor->rows) return;
|
|
|
|
/* Free row contents */
|
|
free(cur_editor->row[at].render);
|
|
free(cur_editor->row[at].chars);
|
|
free(cur_editor->row[at].hl);
|
|
|
|
memmove(&cur_editor->row[at], &cur_editor->row[at + 1], sizeof(row_t) * (cur_editor->rows - at - 1));
|
|
for (int j = at; j < cur_editor->rows - 1; j++) {
|
|
cur_editor->row[j].idx--;
|
|
}
|
|
cur_editor->rows--;
|
|
cur_editor->dirty++;
|
|
}
|
|
|
|
char *export_buffer(int *buflen)
|
|
{
|
|
int total_len = 0;
|
|
for (int j = 0; j < cur_editor->rows; j++) {
|
|
total_len += cur_editor->row[j].size + 1;
|
|
}
|
|
*buflen = total_len;
|
|
char *buf = malloc(total_len);
|
|
char *p = buf;
|
|
for (int j = 0; j < cur_editor->rows; j++) {
|
|
memcpy(p, cur_editor->row[j].chars, cur_editor->row[j].size);
|
|
p += cur_editor->row[j].size;
|
|
*p = '\n';
|
|
p++;
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
void replace_home(char *path)
|
|
{
|
|
char *home = getenv("HOME");
|
|
if (home == NULL) {
|
|
wpprintw("$HOME not defined");
|
|
return;
|
|
}
|
|
/* replace ~ with home */
|
|
snprintf(path, strlen(path) + strlen(home), "%s%s", home, path + 1);
|
|
return;
|
|
}
|
|
|
|
int is_separator(int c)
|
|
{
|
|
return isspace(c) || c == '\0' || strchr(",.()+-/*=~%<>[];", c) != NULL;
|
|
}
|
|
|
|
int is_symbol(int c)
|
|
{
|
|
return strchr("+-/*=~%><:?&|.", c) != NULL;
|
|
}
|
|
|
|
void update_highlight(row_t *row)
|
|
{
|
|
row->hl = realloc(row->hl, row->render_size);
|
|
memset(row->hl, NORMAL, row->render_size);
|
|
|
|
if (cur_editor->syntax == NULL) return;
|
|
|
|
char **keywords = cur_editor->syntax->keywords;
|
|
|
|
char *scs = cur_editor->syntax->singleline_comment_start;
|
|
char *mcs = cur_editor->syntax->multiline_comment_start;
|
|
char *mce = cur_editor->syntax->multiline_comment_end;
|
|
|
|
int scs_len = scs ? strlen(scs) : 0;
|
|
int mcs_len = mcs ? strlen(mcs) : 0;
|
|
int mce_len = mce ? strlen(mce) : 0;
|
|
|
|
int prev_sep = 1;
|
|
int in_string = 0;
|
|
int in_char = 0;
|
|
int in_include = 0;
|
|
int in_escape = 0;
|
|
int in_comment = row->idx > 0 && cur_editor->row[row->idx - 1].opened_comment;
|
|
|
|
int i = 0;
|
|
while (i < row->render_size) {
|
|
char c = row->render[i];
|
|
unsigned char prev_hl = (i > 0) ? row->hl[i - 1] : NORMAL;
|
|
|
|
if (scs_len && !in_string && !in_comment) {
|
|
if (!strncmp(&row->render[i], scs, scs_len)) {
|
|
memset(&row->hl[i], COMMENT, row->render_size - i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (mcs_len && mce_len && !in_string) {
|
|
if (in_comment) {
|
|
row->hl[i] = MLCOMMENT;
|
|
if (!strncmp(&row->render[i], mce, mce_len)) {
|
|
memset(&row->hl[i], MLCOMMENT, mce_len);
|
|
i += mce_len;
|
|
in_comment = 0;
|
|
prev_sep = 1;
|
|
continue;
|
|
} else {
|
|
i++;
|
|
continue;
|
|
}
|
|
} else if (!strncmp(&row->render[i], mcs, mcs_len)) {
|
|
memset(&row->hl[i], MLCOMMENT, mcs_len);
|
|
i += mcs_len;
|
|
in_comment = 1;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (in_escape) {
|
|
if ((c > 47 && c < 58) || c == 'n' || c == 't' || c == 'r') {
|
|
row->hl[i] = ESCAPE;
|
|
i++;
|
|
prev_sep = 0;
|
|
continue;
|
|
} else {
|
|
in_escape = 0;
|
|
}
|
|
} else {
|
|
if (c == '\\') {
|
|
in_escape = 1;
|
|
row->hl[i] = ESCAPE;
|
|
i++;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (in_string) {
|
|
row->hl[i] = STRING;
|
|
if (c == '\\' && i + 1 < row->render_size) {
|
|
row->hl[i + 1] = STRING;
|
|
i += 2;
|
|
continue;
|
|
}
|
|
if (c == in_string) in_string = 0;
|
|
i++;
|
|
prev_sep = 1;
|
|
continue;
|
|
} else {
|
|
if (c == '"') {
|
|
in_string = c;
|
|
row->hl[i] = STRING;
|
|
i++;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (in_char) {
|
|
row->hl[i] = CHAR;
|
|
if (c == '\\' && i + 1 < row->render_size) {
|
|
row->hl[i + 1] = CHAR;
|
|
i += 2;
|
|
continue;
|
|
}
|
|
if (c == in_char) in_char = 0;
|
|
i++;
|
|
prev_sep = 1;
|
|
continue;
|
|
} else {
|
|
if (c == '\'') {
|
|
in_char = c;
|
|
row->hl[i] = CHAR;
|
|
i++;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (in_include) {
|
|
row->hl[i] = STRING;
|
|
if (c == '>') in_include = 0;
|
|
i++;
|
|
prev_sep = 1;
|
|
continue;
|
|
} else {
|
|
if (c == '<' && (row->render[i-1] == 'e' || (row->render[i-1] == ' ' &&
|
|
row->render[i-2] == 'e'))) {
|
|
in_include = 1;
|
|
row->hl[i] = STRING;
|
|
i++;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if ((isdigit(c) && (prev_sep || prev_hl == NUMBER)) ||
|
|
(c == '.' && prev_hl == NUMBER) ||
|
|
(c >= 'A' && c <= 'Z') ||
|
|
(c == '_' && prev_hl == NUMBER)
|
|
) {
|
|
row->hl[i] = NUMBER;
|
|
i++;
|
|
prev_sep = 0;
|
|
continue;
|
|
}
|
|
|
|
if (c == '(' || c == ')' || c == '[' || c == ']' || c == '{' || c == '}') {
|
|
row->hl[i] = KW_BRACKET;
|
|
prev_sep = 1;
|
|
i++;
|
|
continue;
|
|
}
|
|
|
|
if (is_symbol(c)) {
|
|
row->hl[i] = SYMBOL;
|
|
prev_sep = 1;
|
|
i++;
|
|
continue;
|
|
}
|
|
|
|
if (c == ';' || c == ',') {
|
|
row->hl[i] = TERMINATOR;
|
|
prev_sep = 1;
|
|
i++;
|
|
continue;
|
|
}
|
|
|
|
if (prev_sep) {
|
|
int j;
|
|
for (j = 0; keywords[j]; j++) {
|
|
int klen = strlen(keywords[j]);
|
|
int type_keyword = keywords[j][klen - 1] == '|';
|
|
if (type_keyword) klen--;
|
|
|
|
if (!strncmp(&row->render[i], keywords[j], klen) &&
|
|
is_separator(row->render[i + klen])) {
|
|
memset(&row->hl[i], type_keyword ? KW_TYPE : KW, klen);
|
|
i += klen;
|
|
break;
|
|
}
|
|
}
|
|
/* Matched a keyword */
|
|
if (keywords[j] != NULL) {
|
|
prev_sep = 0;
|
|
continue;
|
|
}
|
|
/* Check for function */
|
|
int word_len = 0;
|
|
while (!is_separator(row->render[i])) {
|
|
word_len++;
|
|
i++;
|
|
}
|
|
if (row->render[i] == '(') {
|
|
memset(&row->hl[i - word_len], KW_FN, word_len);
|
|
prev_sep = 1;
|
|
} else {
|
|
prev_sep = 0;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
prev_sep = is_separator(row->render[i]);
|
|
i++;
|
|
}
|
|
int changed = (row->opened_comment != in_comment);
|
|
row->opened_comment = in_comment;
|
|
if (changed && row->idx + 1 < cur_editor->rows)
|
|
update_highlight(&cur_editor->row[row->idx + 1]);
|
|
}
|
|
|
|
void select_syntax(void)
|
|
{
|
|
if (strlen(cur_editor->filename) == 0) return;
|
|
cur_editor->syntax = NULL;
|
|
char *ext = strrchr(cur_editor->filename, '.');
|
|
for (uint8_t i = 0; i < LANGS_LEN; i++) {
|
|
language_t *lang = &langs[i];
|
|
for (int j = 0; lang->extensions[j] != NULL; j++) {
|
|
int is_ext = lang->extensions[j][0] == '.';
|
|
if ((is_ext && ext && !strcmp(ext, lang->extensions[j])) ||
|
|
(!is_ext && strstr(cur_editor->filename, lang->extensions[j]))) {
|
|
cur_editor->syntax = lang;
|
|
|
|
for (int row = 0; row < cur_editor->rows; row++) {
|
|
update_highlight(&cur_editor->row[row]);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void die(const char *s)
|
|
{
|
|
perror(s);
|
|
exit(1);
|
|
}
|
|
|
|
void handle_sigwinch(int ignore)
|
|
{
|
|
get_window_size(&rows, &cols);
|
|
if (cols < 80 || rows < 24) {
|
|
die("vip: Terminal size needs to be at least 80x24");
|
|
}
|
|
refresh_screen();
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
if (argc > 2 && !strcmp(argv[1], "-c")) {
|
|
cat_mode = 1;
|
|
} else {
|
|
struct sigaction sa;
|
|
sa.sa_handler = handle_sigwinch;
|
|
sa.sa_flags = SA_RESTART;
|
|
sigemptyset(&sa.sa_mask);
|
|
|
|
if (sigaction(SIGWINCH, &sa, NULL) == -1) {
|
|
perror("sigaction");
|
|
exit(1);
|
|
}
|
|
|
|
if (tcgetattr(STDIN_FILENO, &oldt) == -1) {
|
|
die("tcgetattr");
|
|
}
|
|
if (get_window_size(&rows, &cols) == -1) {
|
|
die("get_window_size");
|
|
}
|
|
bprintf("\033[?1049h\033[2J\033[2q");
|
|
|
|
newt = oldt;
|
|
/* Disable canonical mode and echo */
|
|
newt.c_iflag &= ~(BRKINT | ICRNL | INPCK | ISTRIP | IXON);
|
|
newt.c_cflag |= (CS8);
|
|
newt.c_lflag &= ~(ECHO | ICANON | IEXTEN | ISIG);
|
|
newt.c_cc[VMIN] = 0;
|
|
newt.c_cc[VTIME] = 1;
|
|
if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &newt) == -1) {
|
|
die("tcsetattr");
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < argc; i++) {
|
|
/* Create tabs for each arg */
|
|
if (argv[i][0] != '-') {
|
|
init_editor(i == 0 ? NULL : argv[i]);
|
|
}
|
|
}
|
|
|
|
if (cat_mode) {
|
|
refresh_screen();
|
|
return 0;
|
|
}
|
|
|
|
while (1) {
|
|
refresh_screen();
|
|
int c = readch();
|
|
switch (c) {
|
|
case '\r':
|
|
insert_new_line();
|
|
break;
|
|
|
|
case HOME_KEY:
|
|
cur_editor->x = 0;
|
|
break;
|
|
|
|
case END_KEY:
|
|
if (cur_editor->y < cur_editor->rows) {
|
|
cur_editor->x = cur_editor->row[cur_editor->y].size;
|
|
}
|
|
break;
|
|
|
|
case BACKSPACE:
|
|
case CTRL_KEY('h'):
|
|
case DEL_KEY:
|
|
if (cur_editor->mode == INSERT) {
|
|
if (c == DEL_KEY)
|
|
move_xy(ARROW_RIGHT);
|
|
del_char();
|
|
} else if (cur_editor->mode == NORMAL) {
|
|
move_xy(ARROW_LEFT);
|
|
}
|
|
break;
|
|
|
|
case PAGE_UP:
|
|
case PAGE_DOWN:
|
|
if (c == PAGE_UP) {
|
|
cur_editor->y = cur_editor->rowoff;
|
|
} else if (c == PAGE_DOWN) {
|
|
cur_editor->y = cur_editor->rowoff + rows - 1;
|
|
if (cur_editor->y > cur_editor->rows) cur_editor->y = cur_editor->rows;
|
|
}
|
|
int times = rows;
|
|
while (times--)
|
|
move_xy(c == PAGE_UP ? ARROW_UP : ARROW_DOWN);
|
|
break;
|
|
|
|
case ARROW_UP:
|
|
case ARROW_DOWN:
|
|
case ARROW_LEFT:
|
|
case ARROW_RIGHT:
|
|
if (cur_editor->mode == INSERT) {
|
|
move_xy(c);
|
|
}
|
|
break;
|
|
|
|
case CTRL_KEY('l'):
|
|
case '\033':
|
|
if (cur_editor->mode == INSERT) {
|
|
move_xy(ARROW_LEFT);
|
|
}
|
|
cur_editor->mode = NORMAL;
|
|
break;
|
|
|
|
case 'i': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
cur_editor->mode = INSERT;
|
|
break;
|
|
}
|
|
|
|
case 'v': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
cur_editor->mode = VISUAL;
|
|
break;
|
|
}
|
|
|
|
case 'g': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
c = readch();
|
|
if (c == 'g') {
|
|
cur_editor->y = 0;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 'G': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
cur_editor->y = cur_editor->rows - 1;
|
|
break;
|
|
}
|
|
|
|
case 'd': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
c = readch();
|
|
if (c == 'd') {
|
|
del_row(cur_editor->y);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ':': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
cur_editor->mode = COMMAND;
|
|
draw_status_bar();
|
|
char *cmd = prompt_editor(":%s", NULL);
|
|
if (cmd == NULL) {
|
|
cur_editor->mode = NORMAL;
|
|
break;
|
|
}
|
|
|
|
if (cmd[0] == 'q') {
|
|
if (cmd[1] == '!') {
|
|
cleanup();
|
|
exit(0);
|
|
break;
|
|
} else {
|
|
if (cur_editor->dirty) {
|
|
wpprintw("%sNo write since last change for buffer \"%s\"", ERROR, cur_editor->filename);
|
|
cur_editor->mode = NORMAL;
|
|
break;
|
|
}
|
|
cleanup();
|
|
exit(0);
|
|
break;
|
|
|
|
}
|
|
} else if (cmd[0] == 'w') {
|
|
save_file();
|
|
cur_editor->mode = NORMAL;
|
|
break;
|
|
}
|
|
|
|
}
|
|
case '/': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
cur_editor->mode = COMMAND;
|
|
find_editor();
|
|
break;
|
|
}
|
|
|
|
case 'k': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
move_xy(ARROW_UP);
|
|
break;
|
|
}
|
|
|
|
case 'j': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
move_xy(ARROW_DOWN);
|
|
break;
|
|
}
|
|
|
|
case 'l': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
move_xy(ARROW_RIGHT);
|
|
break;
|
|
}
|
|
|
|
case 'h': /* PASSTHROUGH */
|
|
if (cur_editor->mode != INSERT) {
|
|
move_xy(ARROW_LEFT);
|
|
break;
|
|
}
|
|
|
|
case 'O': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
shift_previous_line();
|
|
cur_editor->mode = INSERT;
|
|
break;
|
|
}
|
|
|
|
case 'o': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
shift_new_line();
|
|
cur_editor->mode = INSERT;
|
|
break;
|
|
}
|
|
|
|
case '0': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
cur_editor->x = 0;
|
|
break;
|
|
}
|
|
|
|
case '$': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
cur_editor->x = cur_editor->row[cur_editor->y].size;
|
|
break;
|
|
}
|
|
|
|
case ' ': /* PASSTHROUGH */
|
|
if (cur_editor->mode == NORMAL) {
|
|
c = readch();
|
|
if (c == 'p') {
|
|
c = readch();
|
|
if (c == 'v') {
|
|
pid_t pid = fork();
|
|
if (pid == 0) {
|
|
/* Child process */
|
|
execlp("ccc", "ccc", cur_editor->cwd, "-p", NULL);
|
|
_exit(1); /* Exit if exec fails */
|
|
} else if (pid > 0) {
|
|
/* Parent process */
|
|
waitpid(pid, NULL, 0);
|
|
char fpath[PATH_MAX];
|
|
strcpy(fpath, "~/.cache/ccc/opened_file");
|
|
replace_home(fpath);
|
|
FILE *f = fopen(fpath, "r");
|
|
char opened_file[PATH_MAX];
|
|
fread(opened_file, sizeof(char), PATH_MAX, f);
|
|
opened_file[strcspn(opened_file, "\n")] = 0;
|
|
|
|
init_editor(opened_file);
|
|
} else {
|
|
/* Fork failed */
|
|
}
|
|
}
|
|
} else if (c == 't') {
|
|
c = readch();
|
|
if (c > '0' && c <= '9') {
|
|
cur_editor = &editor[c - '0'];
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
if (cur_editor->mode == INSERT) {
|
|
insert_char(c);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
cleanup();
|
|
return 0;
|
|
}
|
|
|
|
void cleanup(void)
|
|
{
|
|
/* Restore old terminal settings */
|
|
if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &oldt) == -1) {
|
|
die("tcsetattr");
|
|
}
|
|
bprintf("\033[2J\033[?1049l");
|
|
}
|
|
|
|
/*
|
|
* Print line to the panel
|
|
*/
|
|
void wpprintw(const char *fmt, ...)
|
|
{
|
|
char buffer[cols];
|
|
va_list args;
|
|
|
|
va_start(args, fmt);
|
|
vsnprintf(buffer, sizeof(buffer), fmt, args);
|
|
va_end(args);
|
|
|
|
move_cursor(rows, 1);
|
|
/* Clear line and print formatted string */
|
|
bprintf("\033[K%s\033[0m", buffer);
|
|
}
|
|
|
|
void move_cursor(int row, int col)
|
|
{
|
|
bprintf("\033[%d;%dH", row, col);
|
|
}
|
|
|
|
int get_window_size(int *row, int *col)
|
|
{
|
|
struct winsize ws;
|
|
if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) == -1 || ws.ws_col == 0) {
|
|
/* Can't get window size */
|
|
bprintf("\033[999C\033[999B");
|
|
char buf[32];
|
|
unsigned int i = 0;
|
|
bprintf("\033[6n");
|
|
while (i < sizeof(buf) - 1) {
|
|
if (read(STDIN_FILENO, &buf[i], 1) != 1) {
|
|
break;
|
|
}
|
|
if (buf[i] == 'R') {
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
buf[i] = '\0';
|
|
if (buf[0] != '\033' || buf[1] != '[') {
|
|
return -1;
|
|
}
|
|
if (sscanf(&buf[2], "%d;%d", row, col) != 2) {
|
|
return -1;
|
|
}
|
|
return 0;
|
|
} else {
|
|
*col = ws.ws_col;
|
|
*row = ws.ws_row;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* printf, but write to STDOUT_FILENO
|
|
*/
|
|
void bprintf(const char *fmt, ...)
|
|
{
|
|
char buffer[512];
|
|
va_list args;
|
|
|
|
va_start(args, fmt);
|
|
vsnprintf(buffer, sizeof(buffer), fmt, args);
|
|
va_end(args);
|
|
|
|
if (cat_mode) {
|
|
printf("%s", buffer);
|
|
} else {
|
|
write(STDOUT_FILENO, buffer, strlen(buffer));
|
|
}
|
|
}
|