ref: 8de0555a7ab8ac4d8d82854bdd54100bca13a8ec
dir: /sys/src/libcontrol/text.c/
#include <u.h>
#include <libc.h>
#include <draw.h>
#include <thread.h>
#include <mouse.h>
#include <keyboard.h>
#include <control.h>
static int debug = 0;
typedef struct Text Text;
struct Text
{
	Control;
	int		border;
	int		topline;
	int		scroll;
	int		nvis;
	int		lastbut;
	CFont	*font;
	CImage	*image;
	CImage	*textcolor;
	CImage	*bordercolor;
	CImage	*selectcolor;
	CImage	*selectingcolor;
	Rune		**line;
	int		selectmode;	// Selsingle, Selmulti
	int		selectstyle;	// Seldown, Selup (use Selup only with Selsingle)
	uchar	*selected;
	int		nline;
	int		warp;
	int		align;
	int		sel;		// line nr of selection made by last button down
	int		but;		// last button down (still being hold)
	int		offsel;	// we are on selection
};
enum
{
	Selsingle,
	Selmulti,
	Seldown,
	Selup,
};
enum{
	EAccumulate,
	EAdd,
	EAlign,
	EBorder,
	EBordercolor,
	EClear,
	EDelete,
	EFocus,
	EFont,
	EHide,
	EImage,
	ERect,
	EReplace,
	EReveal,
	EScroll,
	ESelect,
	ESelectcolor,
	ESelectingcolor,
	ESelectmode,
	ESelectstyle,
	EShow,
	ESize,
	ETextcolor,
	ETopline,
	EValue,
	EWarp,
};
static char *cmds[] = {
	[EAccumulate] =	"accumulate",
	[EAdd] =			"add",
	[EAlign] =			"align",
	[EBorder] =		"border",
	[EBordercolor] =	"bordercolor",
	[EClear] =			"clear",
	[EDelete] =		"delete",
	[EFocus] = 		"focus",
	[EFont] =			"font",
	[EHide] =			"hide",
	[EImage] =		"image",
	[ERect] =			"rect",
	[EReplace] =		"replace",
	[EReveal] =		"reveal",
	[EScroll] =			"scroll",
	[ESelect] =		"select",
	[ESelectcolor] =	"selectcolor",
	[ESelectingcolor] =	"selectingcolor",
	[ESelectmode] =	"selectmode",
	[ESelectstyle] =		"selectstyle",
	[EShow] =			"show",
	[ESize] =			"size",
	[ETextcolor] =		"textcolor",
	[ETopline] =		"topline",
	[EValue] =			"value",
	[EWarp] =			"warp",
	nil
};
static void	textshow(Text*);
static void	texttogglei(Text*, int);
static int	textline(Text*, Point);
static int	texttoggle(Text*, Point);
static void
textmouse(Control *c, Mouse *m)
{
	Text *t;
	int sel;
	t = (Text*)c;
	if (debug) fprint(2, "textmouse %s t->lastbut %d; m->buttons %d\n", t->name, t->lastbut, m->buttons);
	if (t->warp >= 0)
		return;
	if ((t->selectstyle == Selup) && (m->buttons&7)) {
		sel = textline(t, m->xy);
		if (t->sel >= 0) {
//			if (debug) fprint(2, "textmouse Selup %q sel=%d t->sel=%d t->but=%d\n",
//						t->name, sel, t->sel, t->but);
			t->offsel = (sel == t->sel) ? 0 : 1;
			if ((sel == t->sel &&
				    ((t->selected[t->sel] && !t->but) ||
				     ((!t->selected[t->sel]) && t->but))) ||
			    (sel != t->sel &&
				     ((t->selected[t->sel] && t->but) ||
                                         ((!t->selected[t->sel]) && (!t->but))))) {
				texttogglei(t, t->sel);
			}
		}
	}
	if(t->lastbut != (m->buttons&7)){
		if(m->buttons & 7){
			sel = texttoggle(t, m->xy);
			if(sel >= 0) {
				if (t->selectstyle == Seldown) {
					chanprint(t->event, "%q: select %d %d",
						t->name, sel, t->selected[sel] ? (m->buttons & 7) : 0);
					if (debug) fprint(2, "textmouse Seldown event %q: select %d %d\n",
						t->name, sel, t->selected[sel] ? (m->buttons & 7) : 0);
				} else {
					if (debug) fprint(2, "textmouse Selup no event yet %q: select %d %d\n",
						t->name, sel, t->selected[sel] ? (m->buttons & 7) : 0);
					t->sel = sel;
					t->but =  t->selected[sel] ? (m->buttons & 7) : 0;
				}
			}
		} else if (t->selectstyle == Selup) {
			sel = textline(t, m->xy);
			t->offsel = 0;
			if ((sel >= 0) && (sel == t->sel)) {
				chanprint(t->event, "%q: select %d %d",
					t->name, sel, t->but);
				if (debug) fprint(2, "textmouse Selup event %q: select %d %d\n",
					t->name, sel, t->but);
			} else if (sel != t->sel) {
				if  ((t->selected[t->sel] && t->but) ||
                                         ((!t->selected[t->sel]) && (!t->but))) {
					texttogglei(t, t->sel);
				} else {
					textshow(t);
				}
				if (debug) fprint(2, "textmouse Selup cancel %q: select %d %d\n",
					t->name, sel, t->but);
			}
			t->sel = -1;
			t->but = 0;
		}
		t->lastbut = m->buttons & 7;
	}
}
static void
textfree(Control *c)
{
	int i;
	Text *t;
	t = (Text*)c;
	_putctlfont(t->font);
	_putctlimage(t->image);
	_putctlimage(t->textcolor);
	_putctlimage(t->bordercolor);
	_putctlimage(t->selectcolor);
	_putctlimage(t->selectingcolor);
	for(i=0; i<t->nline; i++)
		free(t->line[i]);
	free(t->line);
	free(t->selected);
}
static void
textshow(Text *t)
{
	Rectangle r, tr;
	Point p;
	int i, ntext;
	Font *f;
	Rune *text;
	if (t->hidden)
		return;
	r = t->rect;
	f = t->font->font;
	draw(t->screen, r, t->image->image, nil, t->image->image->r.min);
	if(t->border > 0){
		border(t->screen, r, t->border, t->bordercolor->image, t->bordercolor->image->r.min);
		r = insetrect(r, t->border);
	}
	tr = r;
	t->nvis = Dy(r)/f->height;
	for(i=t->topline; i<t->nline && i<t->topline+t->nvis; i++){
		text = t->line[i];
		ntext = runestrlen(text);
		r.max.y = r.min.y+f->height;
		if(t->sel == i && t->offsel)
			draw(t->screen, r, t->selectingcolor->image, nil, ZP);
		else if(t->selected[i])
			draw(t->screen, r, t->selectcolor->image, nil, ZP);
		p = _ctlalignpoint(r,
			runestringnwidth(f, text, ntext),
			f->height, t->align);
		if(t->warp == i) {
			Point p2;
			 p2.x = p.x + 0.5*runestringnwidth(f, text, ntext);
			 p2.y = p.y + 0.5*f->height;
			moveto(t->controlset->mousectl, p2);
			t->warp = -1;
		}
		_string(t->screen, p, t->textcolor->image,
			ZP, f, nil, text, ntext, tr,
			nil, ZP, SoverD);
		r.min.y += f->height;
	}
	flushimage(display, 1);
}
static void
textctl(Control *c, CParse *cp)
{
	int cmd, i, n;
	Rectangle r;
	Text *t;
	Rune *rp;
	t = (Text*)c;
	cmd = _ctllookup(cp->args[0], cmds, nelem(cmds));
	switch(cmd){
	default:
		ctlerror("%q: unrecognized message '%s'", t->name, cp->str);
		break;
	case EAlign:
		_ctlargcount(t, cp, 2);
		t->align = _ctlalignment(cp->args[1]);
		break;
	case EBorder:
		_ctlargcount(t, cp, 2);
		if(cp->iargs[1] < 0)
			ctlerror("%q: bad border: %c", t->name, cp->str);
		t->border = cp->iargs[1];
		break;
	case EBordercolor:
		_ctlargcount(t, cp, 2);
		_setctlimage(t, &t->bordercolor, cp->args[1]);
		break;
	case EClear:
		_ctlargcount(t, cp, 1);
		for(i=0; i<t->nline; i++)
			free(t->line[i]);
		free(t->line);
		free(t->selected);
		t->line = ctlmalloc(sizeof(Rune*));
		t->selected = ctlmalloc(1);
		t->nline = 0;
		textshow(t);
		break;
	case EDelete:
		_ctlargcount(t, cp, 2);
		i = cp->iargs[1];
		if(i<0 || i>=t->nline)
			ctlerror("%q: line number out of range: %s", t->name, cp->str);
		free(t->line[i]);
		memmove(t->line+i, t->line+i+1, (t->nline-(i+1))*sizeof(Rune*));
		memmove(t->selected+i, t->selected+i+1, t->nline-(i+1));
		t->nline--;
		textshow(t);
		break;
	case EFocus:
		break;
	case EFont:
		_ctlargcount(t, cp, 2);
		_setctlfont(t, &t->font, cp->args[1]);
		break;
	case EHide:
		_ctlargcount(t, cp, 1);
		t->hidden = 1;
		break;
	case EImage:
		_ctlargcount(t, cp, 2);
		_setctlimage(t, &t->image, cp->args[1]);
		break;
	case ERect:
		_ctlargcount(t, cp, 5);
		r.min.x = cp->iargs[1];
		r.min.y = cp->iargs[2];
		r.max.x = cp->iargs[3];
		r.max.y = cp->iargs[4];
		if(Dx(r)<=0 || Dy(r)<=0)
			ctlerror("%q: bad rectangle: %s", t->name, cp->str);
		t->rect = r;
		t->nvis = (Dy(r)-2*t->border)/t->font->font->height;
		break;
	case EReplace:
		_ctlargcount(t, cp, 3);
		i = cp->iargs[1];
		if(i<0 || i>=t->nline)
			ctlerror("%q: line number out of range: %s", t->name, cp->str);
		free(t->line[i]);
		t->line[i] = _ctlrunestr(cp->args[2]);
		textshow(t);
		break;
	case EReveal:
		_ctlargcount(t, cp, 1);
		t->hidden = 0;
		textshow(t);
		break;
	case EScroll:
		_ctlargcount(t, cp, 2);
		t->scroll = cp->iargs[1];
		break;
	case ESelect:
		if(cp->nargs!=2 && cp->nargs!=3)
	badselect:
			ctlerror("%q: bad select message: %s", t->name, cp->str);
		if(cp->nargs == 2){
			if(strcmp(cp->args[1], "all") == 0){
				memset(t->selected, 1, t->nline);
				break;
			}
			if(strcmp(cp->args[1], "none") == 0){
				memset(t->selected, 0, t->nline);
				break;
			}
			if(cp->args[1][0]<'0' && '9'<cp->args[1][0])
				goto badselect;
			texttogglei(t, cp->iargs[1]);
			break;
		}
		if(cp->iargs[1]<0 || cp->iargs[1]>=t->nline)
			ctlerror("%q: selection index out of range (nline %d): %s", t->name, t->nline, cp->str);
		if(t->selected[cp->iargs[1]] != (cp->iargs[2]!=0))
			texttogglei(t, cp->iargs[1]);
		break;
	case ESelectcolor:
		_ctlargcount(t, cp, 2);
		_setctlimage(t, &t->selectcolor, cp->args[1]);
		break;
	case ESelectmode:
		_ctlargcount(t, cp, 2);
		if(strcmp(cp->args[1], "single") == 0)
			t->selectmode = Selsingle;
		else if(strncmp(cp->args[1], "multi", 5) == 0)
			t->selectmode = Selmulti;
		break;
	case ESelectstyle:
		_ctlargcount(t, cp, 2);
		 if(strcmp(cp->args[1], "down") == 0)
			t->selectstyle = Seldown;
		else if(strcmp(cp->args[1], "up") == 0)
			t->selectstyle = Selup;
		break;
	case EShow:
		_ctlargcount(t, cp, 1);
		textshow(t);
		break;
	case ESize:
		if (cp->nargs == 3)
			r.max = Pt(10000, 10000);
		else{
			_ctlargcount(t, cp, 5);
			r.max.x = cp->iargs[3];
			r.max.y = cp->iargs[4];
		}
		r.min.x = cp->iargs[1];
		r.min.y = cp->iargs[2];
		if(r.min.x<=0 || r.min.y<=0 || r.max.x<=0 || r.max.y<=0 || r.max.x < r.min.x || r.max.y < r.min.y)
			ctlerror("%q: bad sizes: %s", t->name, cp->str);
		t->size.min = r.min;
		t->size.max = r.max;
		break;
	case ETextcolor:
		_ctlargcount(t, cp, 2);
		_setctlimage(t, &t->textcolor, cp->args[1]);
		break;
	case ETopline:
		_ctlargcount(t, cp, 2);
		i = cp->iargs[1];
		if(i < 0)
			i = 0;
		if(i > t->nline)
			i = t->nline;
		if(t->topline != i){
			t->topline = i;
			textshow(t);
		}
		break;
	case EValue:
		/* set contents to single line */
		/* free existing text and fall through to add */
		for(i=0; i<t->nline; i++){
			free(t->line[i]);
			t->line[i] = nil;
		}
		t->nline = 0;
		t->topline = 0;
		/* fall through */
	case EAccumulate:
	case EAdd:
		switch (cp->nargs) {
		default:
			ctlerror("%q: wrong argument count in '%s'", t->name, cp->str);
		case 2:
			n = t->nline;
			break;
		case 3:
			n = cp->iargs[1];
			if(n<0 || n>t->nline)
				ctlerror("%q: line number out of range: %s", t->name, cp->str);
			break;
		}
		rp = _ctlrunestr(cp->args[cp->nargs-1]);
		t->line = ctlrealloc(t->line, (t->nline+1)*sizeof(Rune*));
		memmove(t->line+n+1, t->line+n, (t->nline-n)*sizeof(Rune*));
		t->line[n] = rp;
		t->selected = ctlrealloc(t->selected, t->nline+1);
		memmove(t->selected+n+1, t->selected+n, t->nline-n);
		t->selected[n] = (t->selectmode==Selmulti && cmd!=EAccumulate);
		t->nline++;
		if(t->scroll) {
			if(n > t->topline + (t->nvis - 1)){
				t->topline = n - (t->nvis - 1);
				if(t->topline < 0)
					t->topline = 0;
			}
			if(n < t->topline)
				t->topline = n;
		}
		if(cmd != EAccumulate)
			if(t->scroll || t->nline<=t->topline+t->nvis)
				textshow(t);
		break;
	case EWarp:
		_ctlargcount(t, cp, 2);
		i = cp->iargs[1];
		if(i <0 || i>=t->nline)
			ctlerror("%q: selection index out of range (nline %d): %s", t->name, t->nline, cp->str);
		if(i < t->topline || i >=  t->topline+t->nvis){
			t->topline = i;
		}
		t->warp = cp->iargs[1];
		textshow(t);
		t->warp = -1;
		break;
	}
}
static void
texttogglei(Text *t, int i)
{
	int prev;
	if(t->selectmode == Selsingle){
		/* clear the others */
		prev = t->selected[i];
		memset(t->selected, 0, t->nline);
		t->selected[i] = prev;
	}
	t->selected[i] ^= 1;
	textshow(t);
}
static int
textline(Text *t, Point p)
{
	Rectangle r;
	int i;
	r = t->rect;
	if(t->border > 0)
		r = insetrect(r, t->border);
	if(!ptinrect(p, r))
		return -1;
	i = (p.y-r.min.y)/t->font->font->height;
	i += t->topline;
	if(i >= t->nline)
		return -1;
	return i;
}
static int
texttoggle(Text *t, Point p)
{
	int i;
	i = textline(t, p);
	if (i >= 0)
		texttogglei(t, i);
	return i;
}
Control*
createtext(Controlset *cs, char *name)
{
	Text *t;
	t = (Text*)_createctl(cs, "text", sizeof(Text), name);
	t->line = ctlmalloc(sizeof(Rune*));
	t->selected = ctlmalloc(1);
	t->nline = 0;
	t->image = _getctlimage("white");
	t->textcolor = _getctlimage("black");
	t->bordercolor = _getctlimage("black");
	t->selectcolor = _getctlimage("yellow");
	t->selectingcolor = _getctlimage("paleyellow");
	t->font = _getctlfont("font");
	t->selectmode = Selsingle;
	t->selectstyle = Selup; // Seldown;
	t->lastbut = 0;
	t->mouse = textmouse;
	t->ctl = textctl;
	t->exit = textfree;
	t->warp = -1;
	t->sel = -1;
	t->offsel = 0;
	t->but = 0;
	return (Control *)t;
}