Compare commits
68 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| d9c5a6696e | |||
| 5c2571eba7 | |||
| 9afd030b87 | |||
| 2f198e611e | |||
| e079726ced | |||
| 3c79e368fa | |||
| bd68b809cd | |||
| f200a7bfcd | |||
| 58daeac6ad | |||
| 3ef6bab265 | |||
| 3800751bcf | |||
| 87272a669b | |||
| b77748f7c0 | |||
| c70b502006 | |||
| e14d620887 | |||
| db38266849 | |||
| 6d1b7f8e56 | |||
| 64647f77b0 | |||
| 2c3b2ae0f0 | |||
| a605e47458 | |||
| 40ee1e8d7b | |||
| 4464159301 | |||
| dc9fb58a41 | |||
| bbd682cec7 | |||
| ace25c5c65 | |||
| 09881706f2 | |||
| 2579bb01c8 | |||
| a528b3fc3e | |||
| faa70af9b7 | |||
| e1abaae6aa | |||
| 329b92f382 | |||
| 12553f301b | |||
| 75b19042b9 | |||
| 1a184b1a08 | |||
| acde895fb7 | |||
| 78d522ba98 | |||
| feb6667a24 | |||
|
|
66c79e0762 | ||
| a51b98c31f | |||
| 707362574c | |||
| 56db8bd8d2 | |||
| a32ef2ff53 | |||
| 0c0c3d9ce5 | |||
| 7245003769 | |||
| 542b1d90a0 | |||
| a359f4e6c4 | |||
| a03dd0c433 | |||
| 561faf537c | |||
| 3a36b35c1f | |||
|
|
b1cb2532f6 | ||
| 9f33cdc0e4 | |||
| afcc33329f | |||
| 128235283c | |||
| af3c0eddc4 | |||
| f8f01f7dd6 | |||
| 8206ab0a40 | |||
| af8dcb847e | |||
| 9434a34116 | |||
| 3b2b60466c | |||
| 52e1f2dd8b | |||
| 8bb9228339 | |||
| e937e3398c | |||
| f147c866ab | |||
| 1227d0abf4 | |||
| d3591331a5 | |||
| 5eafc1a34b | |||
| a0103dd5aa | |||
| 3782880062 |
@@ -2,7 +2,7 @@ cmake_minimum_required(VERSION 3.15)
|
|||||||
project(ke C) # Specify C language explicitly
|
project(ke C) # Specify C language explicitly
|
||||||
|
|
||||||
set(CMAKE_C_STANDARD 99)
|
set(CMAKE_C_STANDARD 99)
|
||||||
set(KE_VERSION "1.1.1")
|
set(KE_VERSION "1.5.6")
|
||||||
|
|
||||||
set(CMAKE_C_FLAGS "-Wall -Wextra -pedantic -Wshadow -Werror -std=c99 -g")
|
set(CMAKE_C_FLAGS "-Wall -Wextra -pedantic -Wshadow -Werror -std=c99 -g")
|
||||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -D_DEFAULT_SOURCE -D_XOPEN_SOURCE")
|
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -D_DEFAULT_SOURCE -D_XOPEN_SOURCE")
|
||||||
|
|||||||
16
Makefile
16
Makefile
@@ -1,8 +1,9 @@
|
|||||||
TARGET := ke
|
TARGET := ke
|
||||||
KE_VERSION := devel
|
KE_VERSION := devel
|
||||||
|
DEST := $(HOME)/.local/bin/$(TARGET)
|
||||||
|
|
||||||
CFLAGS := -Wall -Wextra -pedantic -Wshadow -Werror -std=c99 -g
|
CFLAGS := -Wall -Wextra -pedantic -Wshadow -Werror -std=c99 -g
|
||||||
|
CFLAGS += -Wno-unused-result
|
||||||
CFLAGS += -D_DEFAULT_SOURCE -D_XOPEN_SOURCE
|
CFLAGS += -D_DEFAULT_SOURCE -D_XOPEN_SOURCE
|
||||||
CFLAGS += -fsanitize=address -fno-omit-frame-pointer
|
CFLAGS += -fsanitize=address -fno-omit-frame-pointer
|
||||||
|
|
||||||
@@ -13,9 +14,20 @@ all: $(TARGET) test.txt
|
|||||||
$(TARGET): main.c
|
$(TARGET): main.c
|
||||||
$(CC) $(CFLAGS) -o $(TARGET) $(LDFLAGS) main.c
|
$(CC) $(CFLAGS) -o $(TARGET) $(LDFLAGS) main.c
|
||||||
|
|
||||||
|
.PHONY: install
|
||||||
|
#install: $(TARGET)
|
||||||
|
install:
|
||||||
|
cp $(TARGET) $(DEST)
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
rm -f $(TARGET)
|
rm -f $(TARGET)
|
||||||
|
rm -f asan.log*
|
||||||
|
|
||||||
.PHONY: test.txt
|
.PHONY: test.txt
|
||||||
test.txt:
|
test.txt:
|
||||||
cp test.txt.bak $@
|
cp test.txt.bak $@
|
||||||
|
|
||||||
|
.PHONY: gdb
|
||||||
|
gdb:
|
||||||
|
@test -f $(TARGET).pid || (echo "error: $(TARGET).pid not found" >&2; exit 1)
|
||||||
|
@gdb -p $$(cat $(TARGET).pid | tr -d ' \t\n\r') ./$(TARGET)
|
||||||
|
|||||||
@@ -16,3 +16,9 @@ To get verbose ASAN messages:
|
|||||||
export LSAN_OPTIONS=verbosity=1:log_threads=1
|
export LSAN_OPTIONS=verbosity=1:log_threads=1
|
||||||
|
|
||||||
Released under an ISC license.
|
Released under an ISC license.
|
||||||
|
|
||||||
|
Started by following along with kilo:
|
||||||
|
https://viewsourcecode.org/snaptoken/kilo/
|
||||||
|
|
||||||
|
E.g., in the devlogs
|
||||||
|
https://log.wntrmute.dev/2020/02/20200207
|
||||||
|
|||||||
17
TODO
17
TODO
@@ -1,15 +1,18 @@
|
|||||||
[X] goto-line
|
[X] goto-line
|
||||||
[X] text-corruption bug
|
[X] text-corruption bug
|
||||||
[ ] alt-modifiers
|
[ ] alt-modifiers
|
||||||
[ ] refresh-screen
|
[x] refresh-screen
|
||||||
[ ] functions -> keymapping?
|
[ ] functions -> keymapping? (what did this even mean)
|
||||||
[X] rendering: need to skip over control characters like we do with tabs
|
[X] rendering: need to skip over control characters like we do with tabs
|
||||||
[X] control-d -> delete
|
[X] control-d -> delete
|
||||||
[ ] control-g -> exit buf prompt
|
[x] control-g -> exit buf prompt
|
||||||
[ ] load-file prompt on dirty buffer
|
[x] load-file prompt on dirty buffer
|
||||||
[ ] multiple files
|
[ ] multiple files
|
||||||
|
[x] utf-8
|
||||||
|
[x] kill ring
|
||||||
|
[x] mark/region
|
||||||
|
|
||||||
What would it take for ke to be your daily drives?
|
What would it take for ke to be your daily drives?
|
||||||
+ C-u
|
[x] C-u (repeat actions)
|
||||||
+ Alt nav (backspace, delete, f, b, etc)
|
[x] Alt nav (backspace, delete, f, b, etc)
|
||||||
|
[-] undo tree (C-k u/U)
|
||||||
|
|||||||
40
default.nix
Normal file
40
default.nix
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
{
|
||||||
|
lib,
|
||||||
|
stdenv,
|
||||||
|
cmake,
|
||||||
|
installShellFiles,
|
||||||
|
...
|
||||||
|
}:
|
||||||
|
let
|
||||||
|
cmakeContent = builtins.readFile ./CMakeLists.txt;
|
||||||
|
cmakeLines = lib.splitString "\n" cmakeContent;
|
||||||
|
versionLine = lib.findFirst (l: builtins.match ".*set\\(KE_VERSION \".+\"\\).*" l != null) (throw "KE_VERSION not found in CMakeLists.txt") cmakeLines;
|
||||||
|
version = builtins.head (builtins.match ".*set\\(KE_VERSION \"(.+)\"\\).*" versionLine);
|
||||||
|
in
|
||||||
|
stdenv.mkDerivation {
|
||||||
|
pname = "ke";
|
||||||
|
inherit version;
|
||||||
|
|
||||||
|
src = lib.cleanSource ./.;
|
||||||
|
|
||||||
|
nativeBuildInputs = [
|
||||||
|
cmake
|
||||||
|
installShellFiles
|
||||||
|
];
|
||||||
|
|
||||||
|
cmakeFlags = [
|
||||||
|
"-DENABLE_ASAN=on"
|
||||||
|
"-DCMAKE_BUILD_TYPE=Debug"
|
||||||
|
];
|
||||||
|
|
||||||
|
installPhase = ''
|
||||||
|
runHook preInstall
|
||||||
|
|
||||||
|
mkdir -p $out/bin
|
||||||
|
cp ke $out/bin/
|
||||||
|
|
||||||
|
installManPage ../ke.1
|
||||||
|
|
||||||
|
runHook postInstall
|
||||||
|
'';
|
||||||
|
}
|
||||||
27
flake.lock
generated
Normal file
27
flake.lock
generated
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"nodes": {
|
||||||
|
"nixpkgs": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1763835633,
|
||||||
|
"narHash": "sha256-HzxeGVID5MChuCPESuC0dlQL1/scDKu+MmzoVBJxulM=",
|
||||||
|
"owner": "nixos",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"rev": "050e09e091117c3d7328c7b2b7b577492c43c134",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nixos",
|
||||||
|
"ref": "nixos-unstable",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"root": {
|
||||||
|
"inputs": {
|
||||||
|
"nixpkgs": "nixpkgs"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"root": "root",
|
||||||
|
"version": 7
|
||||||
|
}
|
||||||
18
flake.nix
Normal file
18
flake.nix
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
{
|
||||||
|
description = "Kyle's Text Editor";
|
||||||
|
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
|
||||||
|
};
|
||||||
|
|
||||||
|
outputs =
|
||||||
|
{ self, nixpkgs }:
|
||||||
|
let
|
||||||
|
pkgs = import nixpkgs { system = "x86_64-linux"; };
|
||||||
|
in
|
||||||
|
{
|
||||||
|
packages.x86_64-linux = {
|
||||||
|
default = pkgs.callPackage ./default.nix { };
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
64
ke.1
64
ke.1
@@ -19,32 +19,84 @@ is
|
|||||||
K-command mode is entered using C-k. This is taken from Wordstar and just
|
K-command mode is entered using C-k. This is taken from Wordstar and just
|
||||||
so happens to be blessed with starting with a most excellent letter of
|
so happens to be blessed with starting with a most excellent letter of
|
||||||
grandeur. Many commands work with and without control; for example,
|
grandeur. Many commands work with and without control; for example,
|
||||||
saving a file can be done with either C-k s or C-k C-s.
|
saving a file can be done with either C-k s or C-k C-s. Other commands work
|
||||||
.Pp
|
with ESC or CTRL.
|
||||||
|
.Sh K-COMMANDS
|
||||||
|
k-command mode can be exited with ESC or C-g.
|
||||||
.Bl -tag -width xxxxxxxxxxxx -offset indent
|
.Bl -tag -width xxxxxxxxxxxx -offset indent
|
||||||
.It C-k BACKSPACE
|
.It C-k BACKSPACE
|
||||||
Delete from the cursor to the beginning of the line.
|
Delete from the cursor to the beginning of the line.
|
||||||
|
.It C-k SPACE
|
||||||
|
Toggle the mark.
|
||||||
|
.It C-k -
|
||||||
|
If the mark is set, unindent the region.
|
||||||
|
.It C-k =
|
||||||
|
If the mark is set, indent the region.
|
||||||
|
.It C-k c
|
||||||
|
Clear (flush) the kill ring.
|
||||||
.It C-k d
|
.It C-k d
|
||||||
Delete from the cursor to the end of the line.
|
Delete from the cursor to the end of the line.
|
||||||
.It C-k C-d
|
.It C-k C-d
|
||||||
Delete the entire link.
|
Delete the entire line.
|
||||||
.It C-k e
|
.It C-k e
|
||||||
Edit a new file. Also C-k C-e.
|
Edit a new file. Also C-k C-e.
|
||||||
.It C-k f
|
.It C-k f
|
||||||
Incremental find.
|
Incremental find.
|
||||||
.It C-k g
|
.It C-k g
|
||||||
Go to a specific line. Also C-k C-g.
|
Go to a specific line.
|
||||||
|
.It C-k j
|
||||||
|
Jump to the mark.
|
||||||
|
.It C-k l
|
||||||
|
List the number of lines of code in a saved file.
|
||||||
.It C-k m
|
.It C-k m
|
||||||
Run make(1).
|
Run make(1).
|
||||||
.It C-k q
|
.It C-k q
|
||||||
exit the editor. Also C-k C-q.
|
Exit the editor. If the file has unsaved changes,
|
||||||
|
a warning will be printed; a second C-k q will exit.
|
||||||
|
.It C-k C-q
|
||||||
|
Immediately exit the editor.
|
||||||
|
.It C-k C-r
|
||||||
|
Reload the current buffer from disk.
|
||||||
.It C-k s
|
.It C-k s
|
||||||
save the file, prompting for a filename if needed. Also C-k C-s.
|
Save the file, prompting for a filename if needed. Also C-k C-s.
|
||||||
|
.It C-k u
|
||||||
|
Undo changes.
|
||||||
|
.It C-k U
|
||||||
|
Redo changes.
|
||||||
.It C-k x
|
.It C-k x
|
||||||
save the file and exit. Also C-k C-x.
|
save the file and exit. Also C-k C-x.
|
||||||
|
.It C-k y
|
||||||
|
Yank the kill ring.
|
||||||
.It C-k \[char92]
|
.It C-k \[char92]
|
||||||
Dump core.
|
Dump core.
|
||||||
.El
|
.El
|
||||||
|
.Sh OTHER KEYBINDINGS
|
||||||
|
.Bl -tag -width xxxxxxxxxxxx -offset indent
|
||||||
|
.It C-g
|
||||||
|
In general, C-g cancels an operation.
|
||||||
|
.It C-l
|
||||||
|
Refresh the display.
|
||||||
|
.It C-s
|
||||||
|
Incremental find.
|
||||||
|
.It C-u
|
||||||
|
Universal argument. C-u followed by numbers will repeat an
|
||||||
|
operation n times.
|
||||||
|
.It C-w
|
||||||
|
Kill the region if the mark is set.
|
||||||
|
.It C-y
|
||||||
|
Yank the kill ring.
|
||||||
|
.It ESC BACKSPACE
|
||||||
|
Delete the previous word.
|
||||||
|
.It ESC b
|
||||||
|
Move to the previous word.
|
||||||
|
.It ESC d
|
||||||
|
Delete the next word.
|
||||||
|
.It ESC f
|
||||||
|
Move to the next word.
|
||||||
|
.It ESC w
|
||||||
|
Save the region (if the mark is set) to the kill ring.
|
||||||
|
.It
|
||||||
|
.El
|
||||||
.Sh FIND
|
.Sh FIND
|
||||||
The find operation is an incremental search. The up or left arrow keys will
|
The find operation is an incremental search. The up or left arrow keys will
|
||||||
go to the previous result, while the down or right arrow keys will go to
|
go to the previous result, while the down or right arrow keys will go to
|
||||||
|
|||||||
162
scratch.c
Normal file
162
scratch.c
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
/*
|
||||||
|
* scratch.c - ideas in progress
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define REFLOW_MARGIN 72
|
||||||
|
|
||||||
|
void
|
||||||
|
reflow_region(void)
|
||||||
|
{
|
||||||
|
int start_row, end_row, i, col, wlen, this_len;
|
||||||
|
struct erow *row;
|
||||||
|
struct abuf buf = ABUF_INIT;
|
||||||
|
struct abuf out = ABUF_INIT;
|
||||||
|
int in_paragraph = 0;
|
||||||
|
int indent_len = 0;
|
||||||
|
char indent[REFLOW_MARGIN + 1];
|
||||||
|
char word[REFLOW_MARGIN + 1];
|
||||||
|
char *e = NULL;
|
||||||
|
char *p = NULL;
|
||||||
|
char *s = NULL;
|
||||||
|
|
||||||
|
if (editor.mark_set) {
|
||||||
|
if (editor.mark_cury < editor.cury ||
|
||||||
|
(editor.mark_cury == editor.cury &&
|
||||||
|
editor.mark_curx < editor.curx)) {
|
||||||
|
start_row = editor.mark_cury;
|
||||||
|
end_row = editor.cury;
|
||||||
|
} else {
|
||||||
|
start_row = editor.cury;
|
||||||
|
end_row = editor.mark_cury;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
start_row = end_row = editor.cury;
|
||||||
|
while (start_row > 0 && editor.row[start_row - 1].size > 0) {
|
||||||
|
start_row--;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (end_row < editor.nrows - 1 &&
|
||||||
|
editor.row[end_row + 1].size > 0) {
|
||||||
|
end_row++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (start_row >= editor.nrows) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (end_row >= editor.nrows) {
|
||||||
|
end_row = editor.nrows - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = start_row; i <= end_row; i++) {
|
||||||
|
row = &editor.row[i];
|
||||||
|
|
||||||
|
if (row->size == 0) {
|
||||||
|
if (in_paragraph) {
|
||||||
|
ab_append(&buf, "\n", 1);
|
||||||
|
in_paragraph = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ab_append(&buf, "\n", 1);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!in_paragraph) {
|
||||||
|
indent_len = 0;
|
||||||
|
while (indent_len < row->size &&
|
||||||
|
(row->line[indent_len] == ' ' ||
|
||||||
|
row->line[indent_len] == '\t')) {
|
||||||
|
indent[indent_len] = row->line[indent_len], indent_len++;
|
||||||
|
}
|
||||||
|
|
||||||
|
indent[indent_len] = '\0';
|
||||||
|
in_paragraph = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
ab_append(&buf, row->line + indent_len, row->size - indent_len);
|
||||||
|
ab_append(&buf, " ", 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (in_paragraph) {
|
||||||
|
ab_append(&buf, "\n", 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
p = buf.b;
|
||||||
|
col = 0;
|
||||||
|
|
||||||
|
while (p != NULL && *p != '\0') {
|
||||||
|
while (*p && isspace((unsigned char)*p) &&
|
||||||
|
*p != '\n') {
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*p == '\0') {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
wlen = 0;
|
||||||
|
while (*p && !isspace((unsigned char)*p)) {
|
||||||
|
if (wlen < REFLOW_MARGIN)
|
||||||
|
word[wlen++] = *p;
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
word[wlen] = '\0';
|
||||||
|
|
||||||
|
if (*p == '\n' && (p[1] == '\n' || p[1] == '\0')) {
|
||||||
|
ab_append(&out, "\n", 1); /* flush */
|
||||||
|
col = 0;
|
||||||
|
p++; /* consume the extra \n */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
this_len = wlen;
|
||||||
|
if (col > 0) {
|
||||||
|
this_len++; /* space before word */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (col == 0) {
|
||||||
|
ab_append(&out, indent, indent_len);
|
||||||
|
col = indent_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (col + this_len > REFLOW_MARGIN && col > 0) {
|
||||||
|
ab_append(&out, "\n", 1);
|
||||||
|
ab_append(&out, indent, indent_len);
|
||||||
|
col = indent_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (col > 0) {
|
||||||
|
ab_append(&out, " ", 1);
|
||||||
|
col++;
|
||||||
|
}
|
||||||
|
|
||||||
|
ab_append(&out, word, wlen);
|
||||||
|
col += wlen;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (col > 0) {
|
||||||
|
ab_append(&out, "\n", 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* the old switcharoo */
|
||||||
|
buf = out;
|
||||||
|
ab_free(&out);
|
||||||
|
|
||||||
|
|
||||||
|
for (i = end_row; i >= start_row; i--) {
|
||||||
|
delete_row(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
s = buf.b;
|
||||||
|
while ((e = strchr(s, '\n'))) {
|
||||||
|
erow_insert(start_row++, s, e - s);
|
||||||
|
s = e + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
ab_free(&buf);
|
||||||
|
|
||||||
|
editor.dirty++;
|
||||||
|
editor_set_status("Region reflowed to %d columns", REFLOW_MARGIN);
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user