xref: /plan9/sys/src/libframe/frdelete.c (revision 7dd7cddf99dd7472612f1413b4da293630e6b1bc)
13e12c5d1SDavid du Colombier #include <u.h>
2219b2ee8SDavid du Colombier #include <libc.h>
3*7dd7cddfSDavid du Colombier #include <draw.h>
4*7dd7cddfSDavid du Colombier #include <thread.h>
5*7dd7cddfSDavid du Colombier #include <mouse.h>
63e12c5d1SDavid du Colombier #include <frame.h>
73e12c5d1SDavid du Colombier 
83e12c5d1SDavid du Colombier int
frdelete(Frame * f,ulong p0,ulong p1)93e12c5d1SDavid du Colombier frdelete(Frame *f, ulong p0, ulong p1)
103e12c5d1SDavid du Colombier {
113e12c5d1SDavid du Colombier 	Point pt0, pt1, ppt0;
123e12c5d1SDavid du Colombier 	Frbox *b;
133e12c5d1SDavid du Colombier 	int n0, n1, n;
14*7dd7cddfSDavid du Colombier 	ulong cn1;
153e12c5d1SDavid du Colombier 	Rectangle r;
163e12c5d1SDavid du Colombier 	int nn0;
17*7dd7cddfSDavid du Colombier 	Image *col;
183e12c5d1SDavid du Colombier 
19*7dd7cddfSDavid du Colombier 	if(p0>=f->nchars || p0==p1 || f->b==nil)
203e12c5d1SDavid du Colombier 		return 0;
213e12c5d1SDavid du Colombier 	if(p1 > f->nchars)
223e12c5d1SDavid du Colombier 		p1 = f->nchars;
23*7dd7cddfSDavid du Colombier 	n0 = _frfindbox(f, 0, 0, p0);
24*7dd7cddfSDavid du Colombier 	if(n0 == f->nbox)
25*7dd7cddfSDavid du Colombier 		drawerror(f->display, "off end in frdelete");
263e12c5d1SDavid du Colombier 	n1 = _frfindbox(f, n0, p0, p1);
273e12c5d1SDavid du Colombier 	pt0 = _frptofcharnb(f, p0, n0);
283e12c5d1SDavid du Colombier 	pt1 = frptofchar(f, p1);
29*7dd7cddfSDavid du Colombier 	if(f->p0 == f->p1)
30*7dd7cddfSDavid du Colombier 		frtick(f, frptofchar(f, f->p0), 0);
313e12c5d1SDavid du Colombier 	nn0 = n0;
323e12c5d1SDavid du Colombier 	ppt0 = pt0;
333e12c5d1SDavid du Colombier 	_frfreebox(f, n0, n1-1);
343e12c5d1SDavid du Colombier 	f->modified = 1;
353e12c5d1SDavid du Colombier 
363e12c5d1SDavid du Colombier 	/*
373e12c5d1SDavid du Colombier 	 * Invariants:
38*7dd7cddfSDavid du Colombier 	 *  - pt0 points to beginning, pt1 points to end
39*7dd7cddfSDavid du Colombier 	 *  - n0 is box containing beginning of stuff being deleted
40*7dd7cddfSDavid du Colombier 	 *  - n1, b are box containing beginning of stuff to be kept after deletion
41*7dd7cddfSDavid du Colombier 	 *  - cn1 is char position of n1
42*7dd7cddfSDavid du Colombier 	 *  - f->p0 and f->p1 are not adjusted until after all deletion is done
433e12c5d1SDavid du Colombier 	 */
443e12c5d1SDavid du Colombier 	b = &f->box[n1];
45*7dd7cddfSDavid du Colombier 	cn1 = p1;
463e12c5d1SDavid du Colombier 	while(pt1.x!=pt0.x && n1<f->nbox){
473e12c5d1SDavid du Colombier 		_frcklinewrap0(f, &pt0, b);
483e12c5d1SDavid du Colombier 		_frcklinewrap(f, &pt1, b);
493e12c5d1SDavid du Colombier 		n = _frcanfit(f, pt0, b);
503e12c5d1SDavid du Colombier 		if(n==0)
51*7dd7cddfSDavid du Colombier 			drawerror(f->display, "_frcanfit==0");
52*7dd7cddfSDavid du Colombier 		r.min = pt0;
53*7dd7cddfSDavid du Colombier 		r.max = pt0;
54*7dd7cddfSDavid du Colombier 		r.max.y += f->font->height;
55*7dd7cddfSDavid du Colombier 		if(b->nrune > 0){
563e12c5d1SDavid du Colombier 			if(n != b->nrune){
573e12c5d1SDavid du Colombier 				_frsplitbox(f, n1, n);
583e12c5d1SDavid du Colombier 				b = &f->box[n1];
593e12c5d1SDavid du Colombier 			}
603e12c5d1SDavid du Colombier 			r.max.x += b->wid;
61*7dd7cddfSDavid du Colombier 			draw(f->b, r, f->b, nil, pt1);
62*7dd7cddfSDavid du Colombier 			cn1 += b->nrune;
63*7dd7cddfSDavid du Colombier 		}else{
64*7dd7cddfSDavid du Colombier 			r.max.x += _frnewwid0(f, pt0, b);
65*7dd7cddfSDavid du Colombier 			if(r.max.x > f->r.max.x)
66*7dd7cddfSDavid du Colombier 				r.max.x = f->r.max.x;
67*7dd7cddfSDavid du Colombier 			col = f->cols[BACK];
68*7dd7cddfSDavid du Colombier 			if(f->p0<=cn1 && cn1<f->p1)
69*7dd7cddfSDavid du Colombier 				col = f->cols[HIGH];
70*7dd7cddfSDavid du Colombier 			draw(f->b, r, col, nil, pt0);
71*7dd7cddfSDavid du Colombier 			cn1++;
723e12c5d1SDavid du Colombier 		}
733e12c5d1SDavid du Colombier 		_fradvance(f, &pt1, b);
743e12c5d1SDavid du Colombier 		pt0.x += _frnewwid(f, pt0, b);
753e12c5d1SDavid du Colombier 		f->box[n0++] = f->box[n1++];
763e12c5d1SDavid du Colombier 		b++;
773e12c5d1SDavid du Colombier 	}
78*7dd7cddfSDavid du Colombier 	if(n1==f->nbox && pt0.x!=pt1.x)	/* deleting last thing in window; must clean up */
79*7dd7cddfSDavid du Colombier 		frselectpaint(f, pt0, pt1, f->cols[BACK]);
803e12c5d1SDavid du Colombier 	if(pt1.y != pt0.y){
813e12c5d1SDavid du Colombier 		Point pt2;
823e12c5d1SDavid du Colombier 
833e12c5d1SDavid du Colombier 		pt2 = _frptofcharptb(f, 32767, pt1, n1);
843e12c5d1SDavid du Colombier 		if(pt2.y > f->r.max.y)
85*7dd7cddfSDavid du Colombier 			drawerror(f->display, "frptofchar in frdelete");
863e12c5d1SDavid du Colombier 		if(n1 < f->nbox){
873e12c5d1SDavid du Colombier 			int q0, q1, q2;
883e12c5d1SDavid du Colombier 
893e12c5d1SDavid du Colombier 			q0 = pt0.y+f->font->height;
903e12c5d1SDavid du Colombier 			q1 = pt1.y+f->font->height;
913e12c5d1SDavid du Colombier 			q2 = pt2.y+f->font->height;
92*7dd7cddfSDavid du Colombier 			if(q2 > f->r.max.y)
93*7dd7cddfSDavid du Colombier 				q2 = f->r.max.y;
94*7dd7cddfSDavid du Colombier 			draw(f->b, Rect(pt0.x, pt0.y, pt0.x+(f->r.max.x-pt1.x), q0),
95*7dd7cddfSDavid du Colombier 				f->b, nil, pt1);
96*7dd7cddfSDavid du Colombier 			draw(f->b, Rect(f->r.min.x, q0, f->r.max.x, q0+(q2-q1)),
97*7dd7cddfSDavid du Colombier 				f->b, nil, Pt(f->r.min.x, q1));
98*7dd7cddfSDavid du Colombier 			frselectpaint(f, Pt(pt2.x, pt2.y-(pt1.y-pt0.y)), pt2, f->cols[BACK]);
993e12c5d1SDavid du Colombier 		}else
100*7dd7cddfSDavid du Colombier 			frselectpaint(f, pt0, pt2, f->cols[BACK]);
1013e12c5d1SDavid du Colombier 	}
1023e12c5d1SDavid du Colombier 	_frclosebox(f, n0, n1-1);
103*7dd7cddfSDavid du Colombier 	if(nn0>0 && f->box[nn0-1].nrune>=0 && ppt0.x-f->box[nn0-1].wid>=(int)f->r.min.x){
1043e12c5d1SDavid du Colombier 		--nn0;
1053e12c5d1SDavid du Colombier 		ppt0.x -= f->box[nn0].wid;
1063e12c5d1SDavid du Colombier 	}
1073e12c5d1SDavid du Colombier 	_frclean(f, ppt0, nn0, n0<f->nbox-1? n0+1 : n0);
1083e12c5d1SDavid du Colombier 	if(f->p1 > p1)
1093e12c5d1SDavid du Colombier 		f->p1 -= p1-p0;
1103e12c5d1SDavid du Colombier 	else if(f->p1 > p0)
1113e12c5d1SDavid du Colombier 		f->p1 = p0;
1123e12c5d1SDavid du Colombier 	if(f->p0 > p1)
1133e12c5d1SDavid du Colombier 		f->p0 -= p1-p0;
1143e12c5d1SDavid du Colombier 	else if(f->p0 > p0)
1153e12c5d1SDavid du Colombier 		f->p0 = p0;
1163e12c5d1SDavid du Colombier 	f->nchars -= p1-p0;
117*7dd7cddfSDavid du Colombier 	if(f->p0 == f->p1)
118*7dd7cddfSDavid du Colombier 		frtick(f, frptofchar(f, f->p0), 1);
1193e12c5d1SDavid du Colombier 	pt0 = frptofchar(f, f->nchars);
1203e12c5d1SDavid du Colombier 	n = f->nlines;
121*7dd7cddfSDavid du Colombier 	f->nlines = (pt0.y-f->r.min.y)/f->font->height+(pt0.x>f->r.min.x);
1223e12c5d1SDavid du Colombier 	return n - f->nlines;
1233e12c5d1SDavid du Colombier }
124