mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2024-12-13 06:00:42 +00:00
2572011a4b
The editor now uses the vertical scrollbar for handling mouse wheel scrolling, thus keeping the scrollbar in sync. Scrollbar index can now cover the full range (not sure why I had that -1), and the potential divide by zero is avoided properly. Thumb-tab now positioned properly when the range is 0.
640 lines
13 KiB
R
640 lines
13 KiB
R
#include <QF/keys.h>
|
|
#include <string.h>
|
|
#include "ruamoko/qwaq/qwaq-app.h"
|
|
#include "ruamoko/qwaq/editor/editor.h"
|
|
#include "ruamoko/qwaq/ui/listener.h"
|
|
#include "ruamoko/qwaq/ui/scrollbar.h"
|
|
|
|
@implementation Editor
|
|
|
|
static int
|
|
center (unsigned v, int len)
|
|
{
|
|
return v > len / 2 ? v / 2 : 0;
|
|
}
|
|
|
|
static void
|
|
trackCursor (Editor *self)
|
|
{
|
|
unsigned cx = [self.buffer charPos:self.line_index at:self.char_index];
|
|
if (self.cursor.x != cx) {
|
|
if (self.char_index < [self.buffer getEOT]) {
|
|
int c = [self.buffer getChar:self.char_index];
|
|
}
|
|
}
|
|
}
|
|
|
|
-initWithRect:(Rect) rect file:(string) filename path:(string) filepath
|
|
{
|
|
if (!(self = [super initWithRect: rect])) {
|
|
return nil;
|
|
}
|
|
self.filename = str_hold (filename);
|
|
if (filepath != filename) {
|
|
self.filepath = str_hold (filepath);
|
|
} else {
|
|
self.filepath = filename;
|
|
}
|
|
buffer = [[EditBuffer withFile:filepath] retain];
|
|
line_count = [buffer countLines: {0, [buffer textSize]}];
|
|
linebuffer = [[DrawBuffer buffer: { xlen, 1 }] retain];
|
|
growMode = gfGrowHi;
|
|
options = ofCanFocus | ofRelativeEvents;
|
|
[onViewScrolled addListener:self :@selector(onScroll:)];
|
|
[self setCursorVisible:1];
|
|
|
|
return self;
|
|
}
|
|
|
|
+(Editor *)withRect:(Rect)rect file:(string)filename
|
|
{
|
|
return [[[self alloc] initWithRect:rect
|
|
file:filename
|
|
path:filename] autorelease];
|
|
}
|
|
|
|
+(Editor *)withRect:(Rect)rect file:(string)filename path:(string)filepath
|
|
{
|
|
return [[[self alloc] initWithRect:rect
|
|
file:filename
|
|
path:filepath] autorelease];
|
|
}
|
|
|
|
-(void)dealloc
|
|
{
|
|
if (filepath != filename) {
|
|
str_free (filepath);
|
|
}
|
|
str_free (filename);
|
|
[vScrollBar release];
|
|
[buffer release];
|
|
[linebuffer release];
|
|
[super dealloc];
|
|
}
|
|
|
|
-(string)filename
|
|
{
|
|
return filename;
|
|
}
|
|
|
|
-(string)filepath
|
|
{
|
|
return filepath;
|
|
}
|
|
|
|
-(Point)cursor
|
|
{
|
|
return cursor;
|
|
}
|
|
|
|
-setStatusView:(EditStatus *)status
|
|
{
|
|
self.status = status;
|
|
[status setModified:modified];
|
|
[status setCursorMode:cursorMode];
|
|
[status redraw];
|
|
return self;
|
|
}
|
|
|
|
-trackCursor:(int)fwd
|
|
{
|
|
unsigned tx = [buffer charPos:line_index at:char_index];
|
|
|
|
cursorMode &= ~cmVInsert;
|
|
if (tx != cursor.x) {
|
|
if (char_index < [buffer getEOT]) {
|
|
int c = [buffer getChar:char_index];
|
|
|
|
if (virtualInsert) {
|
|
if (c != '\t' || cursorThroughTabs) {
|
|
cursorMode |= cmVInsert;
|
|
goto done;
|
|
}
|
|
}
|
|
if (c == '\t' && fwd) {
|
|
tx = [buffer charPos:line_index at:++char_index];
|
|
}
|
|
} else if (virtualInsert) {
|
|
cursorMode |= cmVInsert;
|
|
goto done;
|
|
}
|
|
cursor.x = tx;
|
|
}
|
|
done:
|
|
[status setCursorMode:cursorMode];
|
|
[status redraw];
|
|
return self;
|
|
}
|
|
|
|
-draw
|
|
{
|
|
[super draw];
|
|
unsigned lind = base_index;
|
|
int *lbuf = [linebuffer buffer];
|
|
for (int y = 0; y < ylen; y++) {
|
|
lind = [buffer formatLine:lind from:base.x into:lbuf width:xlen
|
|
highlight:selection colors: {color_palette[047],
|
|
color_palette[007]}];
|
|
[textContext blitFromBuffer: linebuffer to: {xpos, ypos + y}
|
|
from: [linebuffer rect]];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
-resize: (Extent) delta
|
|
{
|
|
[super resize: delta];
|
|
[linebuffer resizeTo: {xlen, 1}];
|
|
return self;
|
|
}
|
|
|
|
static int
|
|
handleEvent (Editor *self, qwaq_event_t *event)
|
|
{
|
|
if (event.what & qe_mouse) {
|
|
if (event.what == qe_mouseclick) {
|
|
if (event.mouse.buttons & (1 << 3)) {
|
|
[self.vScrollBar page:1 dir:0];
|
|
return 1;
|
|
}
|
|
if (event.mouse.buttons & (1 << 4)) {
|
|
[self.vScrollBar page:1 dir:1];
|
|
return 1;
|
|
}
|
|
if (event.mouse.buttons & (1 << 5)) {
|
|
[self scrollLeft: 1];
|
|
return 1;
|
|
}
|
|
if (event.mouse.buttons & (1 << 6)) {
|
|
[self scrollRight: 1];
|
|
return 1;
|
|
}
|
|
}
|
|
} else if (event.what == qe_keydown) {
|
|
switch (event.key.code) {
|
|
case QFK_PAGEUP:
|
|
if (event.key.shift & qe_control) {
|
|
[self moveBOT];
|
|
} else {
|
|
[self pageUp];
|
|
}
|
|
return 1;
|
|
case QFK_PAGEDOWN:
|
|
if (event.key.shift & qe_control) {
|
|
[self moveEOT];
|
|
} else {
|
|
[self pageDown];
|
|
}
|
|
return 1;
|
|
case QFK_UP:
|
|
if (event.key.shift & qe_control) {
|
|
[self linesUp];
|
|
} else {
|
|
[self charUp];
|
|
}
|
|
return 1;
|
|
case QFK_DOWN:
|
|
if (event.key.shift & qe_control) {
|
|
[self linesDown];
|
|
} else {
|
|
[self charDown];
|
|
}
|
|
return 1;
|
|
case QFK_LEFT:
|
|
if (event.key.shift & qe_control) {
|
|
[self wordLeft];
|
|
} else {
|
|
[self charLeft];
|
|
}
|
|
return 1;
|
|
case QFK_RIGHT:
|
|
if (event.key.shift & qe_control) {
|
|
[self wordRight];
|
|
} else {
|
|
[self charRight];
|
|
}
|
|
return 1;
|
|
case QFK_HOME:
|
|
if (event.key.shift & qe_control) {
|
|
[self moveBOS];
|
|
} else {
|
|
[self moveBOL];
|
|
}
|
|
return 1;
|
|
case QFK_END:
|
|
if (event.key.shift & qe_control) {
|
|
[self moveEOS];
|
|
} else {
|
|
[self moveEOL];
|
|
}
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
-handleEvent:(qwaq_event_t *) event
|
|
{
|
|
[super handleEvent: event];
|
|
|
|
if (handleEvent (self, event)) {
|
|
event.what = qe_none;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
-scrollLeft:(unsigned) count
|
|
{
|
|
if (base.x > count) {
|
|
base.x -= count;
|
|
} else {
|
|
base.x = 0;
|
|
}
|
|
[self redraw];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-scrollRight:(unsigned) count
|
|
{
|
|
if (1024 - base.x > count) {
|
|
base.x += count;
|
|
} else {
|
|
base.x = 1024;
|
|
}
|
|
[self redraw];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-scrollTo:(unsigned)target
|
|
{
|
|
if (target > base.y) {
|
|
base_index = [buffer nextLine:base_index :target - base.y];
|
|
} else if (target < base.y) {
|
|
base_index = [buffer prevLine:base_index :base.y - target];
|
|
}
|
|
base.y = target;
|
|
[vScrollBar setIndex:base.y];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-pageUp
|
|
{
|
|
[self recenter:0];
|
|
unsigned count = cursor.y;
|
|
|
|
if (count > ylen) {
|
|
count = ylen;
|
|
}
|
|
if (count) {
|
|
cursor.y -= count;
|
|
[vScrollBar setIndex:[vScrollBar index] - count];
|
|
line_index = [buffer prevLine:line_index :count];
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
-pageDown
|
|
{
|
|
[self recenter:0];
|
|
unsigned count = line_count - cursor.y;
|
|
|
|
if (count > ylen) {
|
|
count = ylen;
|
|
}
|
|
if (count) {
|
|
cursor.y += count;
|
|
[vScrollBar setIndex:[vScrollBar index] + count];
|
|
line_index = [buffer nextLine:line_index :count];
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
-linesUp
|
|
{
|
|
while (line_index > 0) {
|
|
line_index = [buffer prevLine: line_index];
|
|
cursor.y--;
|
|
unsigned p = [buffer nextNonSpace: line_index];
|
|
if ([buffer getChar:p] == '\n') {
|
|
continue;
|
|
}
|
|
int x = [buffer charPos:line_index at:p];
|
|
if (x <= cursor.x) {
|
|
break;
|
|
}
|
|
}
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
|
|
[self recenter:0];
|
|
[self trackCursor:1];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
|
|
return self;
|
|
}
|
|
|
|
-linesDown
|
|
{
|
|
unsigned end = [buffer getEOT];
|
|
unsigned last = [buffer getBOL:end];
|
|
while (line_index < last) {
|
|
line_index = [buffer nextLine: line_index];
|
|
cursor.y++;
|
|
unsigned p = [buffer nextNonSpace: line_index];
|
|
if (p >= end || [buffer getChar:p] == '\n') {
|
|
continue;
|
|
}
|
|
int x = [buffer charPos:line_index at:p];
|
|
if (x <= cursor.x) {
|
|
break;
|
|
}
|
|
}
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
|
|
[self recenter:0];
|
|
[self trackCursor:1];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
|
|
return self;
|
|
}
|
|
|
|
-charUp
|
|
{
|
|
[self recenter:0];
|
|
if (cursor.y < 1) {
|
|
return self;
|
|
}
|
|
cursor.y--;
|
|
line_index = [buffer prevLine:line_index :1];
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
|
|
if (base.y > cursor.y) {
|
|
[vScrollBar setIndex:cursor.y];
|
|
}
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-charDown
|
|
{
|
|
[self recenter:0];
|
|
if (cursor.y >= line_count) {
|
|
return self;
|
|
}
|
|
cursor.y++;
|
|
line_index = [buffer nextLine:line_index :1];
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
|
|
if (base.y + ylen - 1 < cursor.y) {
|
|
[vScrollBar setIndex:cursor.y + 1 - ylen];
|
|
}
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-charLeft
|
|
{
|
|
[self recenter:0];
|
|
if (cursor.x < 1) {
|
|
return self;
|
|
}
|
|
cursor.x--;
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:0];
|
|
|
|
if (base.x > cursor.x) {
|
|
[hScrollBar setIndex:cursor.x];
|
|
}
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-charRight
|
|
{
|
|
[self recenter:0];
|
|
// FIXME handle horizontal scrolling and how to deal with max scroll
|
|
cursor.x++;
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
|
|
if (base.x + xlen - 1 < cursor.x) {
|
|
[hScrollBar setIndex:cursor.x + 1 - xlen];
|
|
}
|
|
if (base.x + xlen - 1 < cursor.x) {
|
|
cursor.x = base.x + xlen - 1;
|
|
}
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-wordLeft
|
|
{
|
|
[self recenter:0];
|
|
//unsigned oc = char_index;
|
|
Point b = base;
|
|
|
|
do {
|
|
if (char_index && char_index == line_index) {
|
|
line_index = [buffer prevLine: line_index];
|
|
char_index = [buffer getEOL: line_index];
|
|
if (--cursor.y < b.y) {
|
|
b.y = cursor.y;
|
|
}
|
|
}
|
|
char_index = [buffer prevWord:char_index];
|
|
} while (char_index && char_index < [buffer getEOT]
|
|
&& ![buffer isWord:char_index]);
|
|
|
|
cursor.x = [buffer charPos:line_index at:char_index];
|
|
if (cursor.x < b.x) {
|
|
b.x = cursor.x;
|
|
} else if (cursor.x >= b.x) {
|
|
b.x = center (b.x, xlen);
|
|
}
|
|
base.x = b.x;
|
|
[vScrollBar setIndex:b.y];
|
|
trackCursor (self);
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
|
|
return self;
|
|
}
|
|
|
|
-wordRight
|
|
{
|
|
[self recenter:0];
|
|
//unsigned oc = char_index;
|
|
Point b = base;
|
|
|
|
if (char_index >= [buffer getEOT]) {
|
|
return self;
|
|
}
|
|
if ([buffer getChar:char_index] == '\n') {
|
|
while ([buffer getChar:char_index] == '\n') {
|
|
char_index = line_index = [buffer nextLine:line_index];
|
|
if (++cursor.y >= b.y + ylen) {
|
|
b.y = cursor.y + 1 - ylen;
|
|
}
|
|
if (char_index >= [buffer getEOT]) {
|
|
break;
|
|
}
|
|
if (![buffer isWord:char_index]) {
|
|
char_index = [buffer nextWord: char_index];
|
|
}
|
|
}
|
|
} else {
|
|
char_index = [buffer nextWord: char_index];
|
|
}
|
|
|
|
cursor.x = [buffer charPos:line_index at:char_index];
|
|
if (cursor.x < b.x) {
|
|
b.x = cursor.x;
|
|
} else if (cursor.x >= b.x) {
|
|
b.x = center (b.x, xlen);
|
|
}
|
|
base.x = b.x;
|
|
[vScrollBar setIndex:b.y];
|
|
trackCursor (self);
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
|
|
return self;
|
|
}
|
|
|
|
-moveBOT
|
|
{
|
|
line_index = base_index = char_index = 0;;
|
|
cursor = nil;
|
|
base.x = 0;
|
|
[vScrollBar setIndex:0];
|
|
[self trackCursor:1];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-moveEOT
|
|
{
|
|
char_index = [buffer getEOT];
|
|
line_index = [buffer getBOL:char_index];
|
|
cursor.x = [buffer charPos:line_index at:char_index];
|
|
cursor.y = line_count;
|
|
[self recenter:0];
|
|
[self trackCursor:1];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-moveBOS
|
|
{
|
|
line_index = base_index;
|
|
cursor.y = base.y;
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-moveEOS
|
|
{
|
|
unsigned count = line_count - base.y;
|
|
if (count > ylen - 1) {
|
|
count = ylen - 1;
|
|
}
|
|
line_index = [buffer nextLine:base_index :count];
|
|
cursor.y = base.y + count;
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self trackCursor:1];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-moveBOL
|
|
{
|
|
char_index = line_index;
|
|
cursor.x = 0;
|
|
base.x = 0;
|
|
[self recenter:0];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-moveEOL
|
|
{
|
|
char_index = [buffer getEOL:line_index];
|
|
cursor.x = [buffer charPos:line_index at:char_index];
|
|
[self recenter:0];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-(void)onScroll:(id)sender
|
|
{
|
|
base.x = scroll.x;
|
|
if (base.y != scroll.y) {
|
|
[self scrollTo:scroll.y];
|
|
}
|
|
}
|
|
|
|
-setVerticalScrollBar:(ScrollBar *)scrollbar
|
|
{
|
|
[super setVerticalScrollBar:scrollbar];
|
|
[vScrollBar setRange:line_count];
|
|
[vScrollBar setPageStep: ylen];
|
|
return self;
|
|
}
|
|
|
|
-recenter:(int) force
|
|
{
|
|
if (force || cursor.y < base.y || cursor.y - base.y >= ylen) {
|
|
[self scrollTo: center (cursor.y, ylen)];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
-gotoLine:(unsigned) line
|
|
{
|
|
if (line > cursor.y) {
|
|
line_index = [buffer nextLine:line_index :line - cursor.y];
|
|
} else if (line < cursor.y) {
|
|
line_index = [buffer prevLine:line_index :cursor.y - line];
|
|
}
|
|
cursor.y = line;
|
|
char_index = [buffer charPtr:line_index at:cursor.x];
|
|
[self recenter:0];
|
|
[self moveCursor: {cursor.x - base.x, cursor.y - base.y}];
|
|
return self;
|
|
}
|
|
|
|
-highlightLine
|
|
{
|
|
selection.start = line_index;
|
|
selection.length = [buffer nextLine: line_index] - line_index;
|
|
if (!selection.length) {
|
|
selection.length = [buffer getEOL: line_index] - line_index;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
-(string)getWordAt:(Point) pos
|
|
{
|
|
if (pos.x < 0 || pos.y < 0 || pos.x >= xlen || pos.y >= ylen) {
|
|
return nil;
|
|
}
|
|
pos.x += base.x;
|
|
unsigned lind = [buffer nextLine:base_index :pos.y];
|
|
unsigned cind = [buffer charPtr:lind at:pos.x];
|
|
eb_sel_t word = [buffer getWord: cind];
|
|
return [buffer readString:word];
|
|
}
|
|
|
|
@end
|