revert between 56095 -> 55830 in arch
[AROS.git] / rom / devs / console / charmapconclass.c
blob1757b0bb5ba6afcb4be568881433fbfa0d9fcd5f
1 /*
2 Copyright © 1995-2017, The AROS Development Team. All rights reserved.
3 $Id$
5 Desc: Code for CONU_CHARMAP console units.
6 Lang: english
7 */
9 #include <string.h>
11 #include <exec/ports.h>
12 #include <proto/graphics.h>
13 #include <proto/intuition.h>
14 #include <proto/alib.h>
15 #include <intuition/intuition.h>
17 #include <intuition/imageclass.h>
18 #include <intuition/gadgetclass.h>
19 #include <intuition/sghooks.h>
20 #include <libraries/gadtools.h>
22 #include <graphics/rastport.h>
23 #include <aros/asmcall.h>
25 #include <stdlib.h>
27 #define SDEBUG 0
28 //#define DEBUG 1
29 #define DEBUG 0
30 #include <aros/debug.h>
32 #include "console_gcc.h"
33 #include "consoleif.h"
35 #include "charmap.h"
37 #define PROP_FLAGS \
38 AUTOKNOB | FREEVERT | PROPNEWLOOK | PROPBORDERLESS
40 #define CODE_COPY 'C'
41 #define CODE_PASTE 'V'
43 struct MyEditHookMsg
45 struct Message msg;
46 struct SGWork *sgw;
47 WORD code;
50 static const STRPTR CONCLIP_PORTNAME = "ConClip.rendezvous";
51 struct Scroll
53 struct Gadget scroller; /* proportionnal gadget */
54 struct Gadget down; /* down gadget */
55 struct Gadget up; /* up gadget */
56 struct PropInfo pinfo; /* PropInfo for scroller */
57 struct Image simage; /* image for scroller */
58 struct Image *upimage; /* Boopsi image for up arrow */
59 struct Image *downimage; /* ditto for down arrow */
62 // FIXME: Abstract out the non-GUI aspects
63 struct charmapcondata
65 /* Start of the scrollback buffer */
66 struct charmap_line *top_of_scrollback;
67 /* The line currently displayed at the top of the screen */
68 struct charmap_line *top_of_window;
69 /* Saved position for the top of the screen at the end of
70 the buffer; where the buffer is reset to if there is
71 output while scrolling */
72 struct charmap_line *saved_top_of_window;
73 ULONG saved_scrollback_pos;
75 ULONG scrollback_size; /* Total size of the scrollback buffer */
76 ULONG scrollback_pos; /* Position of the top of the window */
78 ULONG scrollback_max; /* Maximum number of lines in scrollback
79 buffer on top of CHAR_YMAX(o) */
81 BOOL unrendered; /* Unrendered cursor while scrolled back? */
83 /* FIXME: Belongs in snipmap class */
84 /* Current selection */
85 LONG select_x_min;
86 LONG select_y_min;
87 struct charmap_line *select_line_min;
88 LONG select_x_max;
89 LONG select_y_max;
90 struct charmap_line *select_line_max;
91 BOOL active_selection; /* If true, mouse move will affect the selection */
92 BOOL ignore_drag;
95 UBYTE boopsigad; /* Type of right prop gadget of window */
96 struct Scroll *prop;
98 /* Active gadget */
99 APTR activeGad;
101 /* Libraries */
102 struct Library *ccd_GfxBase;
106 /* Template used to quickly fill constant fields */
107 CONST struct Scroll ScrollBar = {
109 /* PropGadget */
110 NULL, 0, 0, 0, 0,
111 GFLG_RELRIGHT | GFLG_RELHEIGHT,
112 GACT_RIGHTBORDER | GACT_FOLLOWMOUSE | GACT_IMMEDIATE |
113 GACT_RELVERIFY,
114 GTYP_PROPGADGET,
115 NULL, NULL, NULL, 0, NULL,
116 0, NULL
119 /* Up-gadget */
120 NULL, 0, 0, 0, 0,
121 GFLG_RELRIGHT | GFLG_RELBOTTOM | GFLG_GADGHIMAGE |
122 GFLG_GADGIMAGE,
123 GACT_RIGHTBORDER | GACT_RELVERIFY | GACT_IMMEDIATE,
124 GTYP_BOOLGADGET,
125 NULL, NULL, NULL, 0, NULL,
126 1, NULL
129 /* Down-Gadget */
130 NULL, 0, 0, 0, 0,
131 GFLG_RELRIGHT | GFLG_RELBOTTOM | GFLG_GADGHIMAGE |
132 GFLG_GADGIMAGE,
133 GACT_RIGHTBORDER | GACT_RELVERIFY | GACT_IMMEDIATE,
134 GTYP_BOOLGADGET,
135 NULL, NULL, NULL, 0, NULL,
136 2, NULL
139 /* PropInfo */
140 PROP_FLAGS,
141 MAXPOT, MAXPOT,
142 MAXBODY, MAXBODY
144 /* Other values may be NULL */
148 #ifdef ConsoleDevice
149 #undef ConsoleDevice
150 #endif
152 #ifdef GfxBase
153 #undef GfxBase
154 #endif
155 //#define GfxBase (((struct charmapcondata *)INST_DATA(cl, o))->ccd_GfxBase)
157 static VOID charmapcon_refresh(Class *cl, Object *o, LONG off);
159 /*** Allocate and attach a prop gadget to the window ***/
160 static VOID charmapcon_add_prop(Class *cl, Object *o)
162 struct charmapcondata *data = INST_DATA(cl, o);
163 struct ConsoleBase *ConsoleDevice = (struct ConsoleBase *)cl->cl_UserData;
164 struct Scroll *pg;
165 struct Image *dummy;
166 struct Window *win = CU(o)->cu_Window;
168 UWORD height, size_width, size_height;
170 /* If the window is a backdrop'ed one, use a simplified BOOPSI propgadget
171 * because the next propgadget aspect depends on window activated state */
172 if (win->Flags & WFLG_BACKDROP)
174 /* Yes this is actually a (struct Gadget *)... */
175 if ((data->prop = pg =
176 (struct Scroll *)NewObject(NULL, "propgclass", GA_Top, 0,
177 GA_Left, win->Width - 10, GA_Width, 10, GA_Height,
178 win->Height, GA_RelVerify, TRUE, GA_FollowMouse, TRUE,
179 GA_Immediate, TRUE, PGA_VertPot, MAXPOT, PGA_VertBody,
180 MAXBODY, PGA_Freedom, FREEVERT, PGA_NewLook, TRUE,
181 TAG_END)))
183 /* And finally, add it to the window */
184 AddGList(win, (struct Gadget *)pg, 0, 1, NULL);
185 RefreshGList((struct Gadget *)pg, win, NULL, 1);
188 data->boopsigad = TRUE;
189 return;
191 data->boopsigad = FALSE;
193 /* Get memory */
194 if ((data->prop = pg = (void *)AllocMem(sizeof(*pg), MEMF_PUBLIC)))
196 /* Copy default flags/modes/etc. */
197 CopyMem(&ScrollBar, pg, sizeof(*pg));
198 pg->pinfo.Flags = PROP_FLAGS;
199 struct DrawInfo *di;
201 di = (void *)GetScreenDrawInfo(win->WScreen);
203 /* We need to get size-gadget height, to adjust properly arrows */
204 if ((dummy = (struct Image *)NewObject(NULL, "sysiclass",
205 SYSIA_Which, SIZEIMAGE,
206 SYSIA_DrawInfo, (IPTR) di, TAG_END)))
208 size_width = dummy->Width; /* width of up/down-gadgets */
209 size_height = dummy->Height; /* bottom offset */
211 /* We don't need the image anymore */
212 DisposeObject(dummy);
214 /* Get the boopsi image of the up and down arrow */
215 if ((pg->upimage = (struct Image *)NewObject(NULL, "sysiclass",
216 SYSIA_Which, UPIMAGE,
217 SYSIA_DrawInfo, (IPTR) di, TAG_END)))
219 pg->up.GadgetRender = pg->up.SelectRender =
220 (APTR) pg->upimage;
221 height = pg->upimage->Height;
223 if ((pg->downimage =
224 (struct Image *)NewObject(NULL, "sysiclass",
225 SYSIA_Which, DOWNIMAGE, SYSIA_DrawInfo,
226 (IPTR) di, TAG_END)))
228 struct Gadget *G = (void *)pg;
229 WORD hoffset = size_width / 4;
231 pg->down.GadgetRender = pg->down.SelectRender =
232 (APTR) pg->downimage;
234 /* Release drawinfo */
235 FreeScreenDrawInfo(win->WScreen, di);
237 /* Now init all sizes/positions relative to window's
238 * borders */
239 G->Height =
240 -(win->BorderTop + size_height + 2 * height + 2);
241 G->TopEdge = win->BorderTop + 1;
242 G->Width = size_width - hoffset * 2 + 2;
243 G->LeftEdge = -(size_width - hoffset);
244 G++;
245 pg->up.LeftEdge = G->LeftEdge = -(size_width - 1);
246 G->Width = pg->up.Width = size_width;
247 G->Height = pg->up.Height = height;
248 G->TopEdge = -(size_height + height - 1);
249 pg->up.TopEdge = G->TopEdge - height;
251 /* Other fields */
252 pg->scroller.GadgetRender = (APTR) &pg->simage;
253 pg->scroller.SpecialInfo = (APTR) &pg->pinfo;
255 /* Link gadgets */
256 pg->scroller.NextGadget = &pg->up;
257 pg->up.NextGadget = &pg->down;
259 /* And finally, add them to the window */
260 AddGList(win, &pg->scroller, 0, 3, NULL);
261 RefreshGList(&pg->scroller, win, NULL, 3);
263 return;
265 DisposeObject(pg->upimage);
268 FreeMem(pg, sizeof(*pg));
269 FreeScreenDrawInfo(win->WScreen, di);
271 return;
274 static VOID charmapcon_adj_prop(Class *cl, Object *o)
276 struct charmapcondata *data = INST_DATA(cl, o);
277 struct ConsoleBase *ConsoleDevice = (struct ConsoleBase *)cl->cl_UserData;
278 struct Window *w = CU(o)->cu_Window;
279 ULONG VertBody, VertPot;
281 ULONG hidden =
282 data->scrollback_size >
283 CHAR_YMAX(o) ? data->scrollback_size - CHAR_YMAX(o) - 1 : 0;
285 if (hidden > 0)
287 VertPot = (data->scrollback_pos) * MAXPOT / hidden;
288 VertBody = CHAR_YMAX(o) * MAXBODY / data->scrollback_size;
290 else
292 VertPot = 0;
293 VertBody = MAXBODY;
296 if (VertPot > MAXPOT)
298 VertPot = MAXPOT;
299 D(bug("VERTPOT SET TOO HIGH. Adjusted\n"));
302 NewModifyProp((struct Gadget *)&(data->prop->scroller), w, NULL,
303 ((struct PropInfo *)data->prop->scroller.SpecialInfo)->Flags,
304 MAXPOT, VertPot, MAXBODY, VertBody, 1);
307 /*** Free resources allocated for scroller ***/
308 void charmapcon_free_prop(Class *cl, Object *o)
310 struct charmapcondata *data = INST_DATA(cl, o);
311 struct ConsoleBase *ConsoleDevice = (struct ConsoleBase *)cl->cl_UserData;
312 struct Window *win = CU(o)->cu_Window;
314 if (data->prop)
316 if (win)
318 if (data->boopsigad)
319 RemoveGadget(win, (struct Gadget *)data->prop);
320 else
321 RemoveGList(win, &data->prop->scroller, 3);
323 if (data->boopsigad)
324 DisposeObject(data->prop);
325 else
327 /* Free elements */
328 DisposeObject(data->prop->upimage);
329 DisposeObject(data->prop->downimage);
331 /* Free struct */
332 FreeMem(data->prop, sizeof(*data->prop));
339 /*********** CharMapCon::New() **********************/
341 static Object *charmapcon_new(Class *cl, Object *o, struct opSet *msg)
343 EnterFunc(bug("CharMapCon::New()\n"));
344 APTR newGfxBase = TaggedOpenLibrary(TAGGEDOPEN_GRAPHICS);
345 if (newGfxBase == NULL)
346 return NULL;
348 o = (Object *) DoSuperMethodA(cl, o, (Msg) msg);
349 if (o)
351 struct charmapcondata *data = INST_DATA(cl, o);
353 /* Clear for checking inside dispose() whether stuff was allocated.
354 Basically this is bug-prevention.
356 memset(data, 0, sizeof(struct charmapcondata));
358 data->scrollback_max = 500; /* FIXME: Don't hardcode it */
359 data->ccd_GfxBase = newGfxBase;
360 charmapcon_add_prop(cl, o);
362 ReturnPtr("CharMapCon::New", Object *, o);
365 CloseLibrary(newGfxBase);
366 ReturnPtr("CharMapCon::New", Object *, NULL);
370 /*********** CharMapCon::Dispose() **************************/
372 static VOID charmapcon_dispose(Class *cl, Object *o, Msg msg)
374 struct charmapcondata *data = INST_DATA(cl, o);
376 charmap_dispose_lines(data->top_of_scrollback);
377 charmapcon_free_prop(cl, o);
379 CloseLibrary(data->ccd_GfxBase);
381 DoSuperMethodA(cl, o, msg);
384 /********* CharMapCon::DoCommand() ****************************/
386 static struct charmap_line *charmapcon_find_line(Class *cl, Object *o,
387 ULONG ycp)
389 struct charmapcondata *data = INST_DATA(cl, o);
391 // Find the line. This is inefficient but the number of lines on screen
392 // should never be very high.
393 // FIXME: Optimizing the case of appending to the end (e.g. know what line
394 // is on the last line of the buffer).
396 struct charmap_line *line = data->top_of_window;
397 if (!line)
399 D(bug("Initializing charmap\n"));
400 data->top_of_window = data->top_of_scrollback = line =
401 charmap_newline(0, 0);
402 data->scrollback_size = 1;
405 D(bug("Finding line %ld\n", ycp));
406 while (ycp > 0)
408 if (!line->next)
410 charmap_newline(0, line);
411 data->scrollback_size += 1;
413 line = line->next;
414 ycp -= 1;
417 while (data->scrollback_size > data->scrollback_max + CHAR_YMAX(o) &&
418 data->top_of_window != data->top_of_scrollback)
420 data->scrollback_size -= 1;
421 data->scrollback_pos -= 1;
423 /* FIXME: Needs testing... */
424 if (data->select_line_max == data->select_line_min &&
425 data->select_line_min == data->top_of_scrollback)
427 /* The entire selection has scrolled out, but we keep it in case
428 it is still active
430 data->select_line_min = data->select_line_min->next;
431 data->select_line_max = data->select_line_min;
432 data->select_x_min = 0;
433 data->select_x_max = 0;
434 data->select_y_min += 1;
435 data->select_y_max += 1;
437 else if (data->top_of_scrollback == data->select_line_min)
439 data->select_line_min = data->select_line_min->next;
440 data->select_y_min += 1;
441 data->select_x_min = 0;
443 else if (data->top_of_scrollback == data->select_line_max)
445 /* "Reversed" selection */
446 data->select_line_max = data->select_line_max->next;
447 data->select_y_max += 1;
448 data->select_x_max = 0;
451 data->top_of_scrollback =
452 charmap_dispose_line(data->top_of_scrollback);
455 return line;
458 static VOID charmap_ascii(Class *cl, Object *o, ULONG xcp, ULONG ycp,
459 char *str, ULONG len)
461 struct charmap_line *line = charmapcon_find_line(cl, o, ycp);
462 ULONG oldsize = line->size;
464 // Ensure the line has sufficient capacity.
465 if (line->size < xcp + len)
466 charmap_resize(line, xcp + len);
468 // .. copy the required data
469 memset(line->fgpen + xcp, CU(o)->cu_FgPen, len);
470 memset(line->bgpen + xcp, CU(o)->cu_BgPen, len);
471 memset(line->flags + xcp, CU(o)->cu_TxFlags, len);
472 memcpy(line->text + xcp, str, len);
474 // If cursor output is moved further right on the screen than
475 // the last output, we need to fill the line
476 if (oldsize < xcp)
478 memset(line->fgpen + oldsize, CU(o)->cu_FgPen, xcp - oldsize);
479 memset(line->bgpen + oldsize, CU(o)->cu_BgPen, xcp - oldsize);
480 memset(line->flags + oldsize, CU(o)->cu_TxFlags, xcp - oldsize);
481 memset(line->text + oldsize, ' ', xcp - oldsize);
485 static VOID charmap_scroll_up(Class *cl, Object *o, ULONG y)
487 struct charmapcondata *data = INST_DATA(cl, o);
489 if (!data->top_of_window)
490 return;
492 while (y--)
494 if (!data->top_of_window->next)
496 charmap_newline(0, data->top_of_window);
497 data->scrollback_size += 1;
499 data->top_of_window = data->top_of_window->next;
500 data->scrollback_pos += 1;
501 data->select_y_max -= 1;
502 data->select_y_min -= 1;
505 if (data->scrollback_size - CHAR_YMAX(o) - 1 <= data->scrollback_pos &&
506 data->unrendered)
508 Console_RenderCursor(o);
509 data->unrendered = 0;
512 while (data->scrollback_size > data->scrollback_max + CHAR_YMAX(o) &&
513 data->top_of_window != data->top_of_scrollback)
515 data->scrollback_size -= 1;
516 data->scrollback_pos -= 1;
517 data->top_of_scrollback =
518 charmap_dispose_line(data->top_of_scrollback);
522 static VOID charmap_scroll_down(Class *cl, Object *o, ULONG y)
524 // FIXME: Need to adjust cursor position or reset to bottom when editing.
525 struct charmapcondata *data = INST_DATA(cl, o);
526 if (!data->unrendered)
528 Console_UnRenderCursor(o);
529 data->unrendered = 1;
530 data->saved_scrollback_pos = data->scrollback_pos;
531 data->saved_top_of_window = data->top_of_window;
533 // FIXME: Select position.
535 if (data->top_of_window)
537 while (y-- && data->top_of_window->prev)
539 data->top_of_window = data->top_of_window->prev;
540 data->scrollback_pos -= 1;
541 data->select_y_max += 1;
542 data->select_y_min += 1;
547 static VOID charmapcon_scroll_to(Class *cl, Object *o, ULONG y)
549 struct charmapcondata *data = INST_DATA(cl, o);
550 struct Library *GfxBase = data->ccd_GfxBase;
551 struct Window *w = CU(o)->cu_Window;
552 struct RastPort *rp = w->RPort;
553 LONG off = data->scrollback_pos - y;
554 LONG old_pos = data->scrollback_pos;
556 if (off == 0)
557 return;
559 Console_UnRenderCursor(o);
561 if (off > 0)
562 charmap_scroll_down(cl, o, off);
563 else
564 charmap_scroll_up(cl, o, -off);
566 /* Correct offset to account for the fact we might reach the
567 * top or bottom of the buffer:
569 off = old_pos - data->scrollback_pos;
571 /* A whole screenful? If so we have no choice but a full refresh
572 * (though we could double buffer... Not sure that's worth the
573 * memory cost)
575 if (abs(off) > CHAR_YMAX(o))
577 charmapcon_refresh(cl, o, 0);
578 Console_RenderCursor(o);
579 return;
582 /* Avoid a full refresh by scrolling the rastport.
583 * Use "standard" background to reduce flicker
585 SetBPen(rp, CU(o)->cu_BgPen);
586 if (off > 0)
588 ScrollRaster(rp,
590 -YRSIZE * off,
591 GFX_XMIN(o), GFX_YMIN(o), GFX_XMAX(o), GFX_YMAX(o));
593 else
595 ScrollRaster(rp,
597 -YRSIZE * off,
598 GFX_XMIN(o), GFX_YMIN(o), GFX_XMAX(o), GFX_YMAX(o));
602 /* Partial refresh */
603 charmapcon_refresh(cl, o, off);
605 Console_RenderCursor(o);
609 static VOID charmap_delete_char(Class *cl, Object *o, ULONG x, ULONG y)
611 struct charmap_line *line = charmapcon_find_line(cl, o, y);
613 if (!line || x >= line->size)
614 return;
616 // FIXME: Shrink the buffer, or keep track of capacity separately.
617 if (x + 1 >= line->size)
619 line->text[x] = 0;
620 return;
623 memmove(line->fgpen + x, line->fgpen + x + 1, 1);
624 memmove(line->bgpen + x, line->bgpen + x + 1, 1);
625 memmove(line->flags + x, line->flags + x + 1, 1);
626 memmove(line->text + x, line->text + x + 1, 1);
629 static VOID charmap_insert_char(Class *cl, Object *o, ULONG x, ULONG y)
631 struct charmap_line *line = charmapcon_find_line(cl, o, y);
633 if (x >= line->size)
634 return;
636 /* FIXME: This is wasteful, since it copies the buffers straight over,
637 * so we have to do memmove's further down. */
638 charmap_resize(line, line->size + 1);
640 memmove(line->fgpen + x + 1, line->fgpen + x, line->size - x - 1);
641 memmove(line->bgpen + x + 1, line->bgpen + x, line->size - x - 1);
642 memmove(line->flags + x + 1, line->flags + x, line->size - x - 1);
643 memmove(line->text + x + 1, line->text + x, line->size - x - 1);
645 line->fgpen[x] = CU(o)->cu_FgPen;
646 line->bgpen[x] = CU(o)->cu_BgPen;
647 line->flags[x] = CU(o)->cu_TxFlags;
648 line->text[x] = ' ';
651 static VOID charmap_formfeed(Class *cl, Object *o)
653 struct charmapcondata *data = INST_DATA(cl, o);
654 struct charmap_line *line = data->top_of_window;
656 while (line)
658 charmap_resize(line, 0);
659 line = line->next;
663 static VOID charmapcon_docommand(Class *cl, Object *o,
664 struct P_Console_DoCommand *msg)
666 IPTR *params = msg->Params;
667 struct charmapcondata *data = INST_DATA(cl, o);
669 EnterFunc(bug
670 ("CharMapCon::DoCommand(o=%p, cmd=%d, params=%p) x=%d, y=%d, ymax=%d\n",
671 o, msg->Command, params, XCP, YCP, CHAR_YMAX(o)));
673 // This is a bit of a hack: Set position to bottom in order to prevent
674 // output while scrolled.
676 ULONG old_scrollback_size = data->scrollback_size;
677 ULONG old_scrollback_pos = data->scrollback_pos;
679 if (data->unrendered)
681 data->unrendered = 0;
682 data->scrollback_pos = data->saved_scrollback_pos;
683 data->select_y_min += old_scrollback_pos - data->scrollback_pos;
684 data->select_y_max += old_scrollback_pos - data->scrollback_pos;
685 data->top_of_window = data->saved_top_of_window;
686 charmapcon_refresh(cl, o, 0);
687 Console_RenderCursor(o);
690 switch (msg->Command)
692 case C_ASCII:
693 charmap_ascii(cl, o, XCP, YCP, (char *)&params[0], 1);
694 DoSuperMethodA(cl, o, (Msg) msg);
695 break;
697 case C_ASCII_STRING:
698 charmap_ascii(cl, o, XCP, YCP, (char *)params[0], (int)params[1]);
699 DoSuperMethodA(cl, o, (Msg) msg);
700 break;
702 case C_FORMFEED:
703 charmap_formfeed(cl, o);
704 DoSuperMethodA(cl, o, (Msg) msg);
705 break;
707 case C_DELETE_CHAR: /* FIXME: can it have params!? */
708 charmap_delete_char(cl, o, XCP, YCP);
709 DoSuperMethodA(cl, o, (Msg) msg);
710 break;
712 case C_INSERT_CHAR:
713 charmap_insert_char(cl, o, XCP, YCP);
714 DoSuperMethodA(cl, o, (Msg) msg);
715 break;
717 case C_SCROLL_UP:
719 D(bug("C_SCROLL_UP area (%d, %d) to (%d, %d), %d\n",
720 GFX_XMIN(o), GFX_YMIN(o), GFX_XMAX(o), GFX_YMAX(o),
721 YRSIZE * params[0]));
722 charmap_scroll_up(cl, o, params[0]);
723 DoSuperMethodA(cl, o, (Msg) msg);
724 break;
727 case C_SCROLL_DOWN:
729 D(bug("C_SCROLL_DOWN area (%d, %d) to (%d, %d), %d\n",
730 GFX_XMIN(o), GFX_YMIN(o), GFX_XMAX(o), GFX_YMAX(o),
731 YRSIZE * params[0]));
732 charmap_scroll_down(cl, o, params[0]);
733 DoSuperMethodA(cl, o, (Msg) msg);
734 break;
737 default:
738 DoSuperMethodA(cl, o, (Msg) msg);
739 break;
742 if (old_scrollback_size != data->scrollback_size ||
743 old_scrollback_pos != data->scrollback_pos)
744 charmapcon_adj_prop(cl, o);
746 ReturnVoid("CharMapCon::DoCommand");
749 /**************************
750 ** CharMapCon::ClearCell() **
751 **************************/
752 static VOID charmapcon_clearcell(Class *cl, Object *o,
753 struct P_Console_ClearCell *msg)
755 // FIXME, insert space.
756 DoSuperMethodA(cl, o, (Msg) msg);
759 static VOID charmapcon_refresh_lines(Class *cl, Object *o, LONG fromLine,
760 LONG toLine)
762 struct Window *w = CU(o)->cu_Window;
763 struct RastPort *rp = w->RPort;
764 struct charmapcondata *data = INST_DATA(cl, o);
765 struct Library *GfxBase = data->ccd_GfxBase;
767 if (fromLine < CHAR_YMIN(o))
768 fromLine = CHAR_YMIN(o);
769 if (toLine > CHAR_YMAX(o))
770 toLine = CHAR_YMAX(o);
772 D(bug("fromLine: %ld, toLine: %ld, char_ymax: %ld\n", fromLine, toLine,
773 CHAR_YMAX(o)));
775 if (toLine < fromLine)
776 return;
778 Console_UnRenderCursor(o);
780 D(bug("Rendering charmap\n"));
782 struct charmap_line *line = charmapcon_find_line(cl, o, fromLine);
783 ULONG y = GFX_YMIN(o) + fromLine * YRSIZE + rp->Font->tf_Baseline;
784 ULONG yc = fromLine;
785 UBYTE flags = 255;
787 LONG selectstart_x, selectstart_y, selectend_x, selectend_y;
788 if (data->select_y_min == data->select_y_max)
790 selectstart_y = selectend_y = data->select_y_min;
791 selectstart_x = MIN(data->select_x_min, data->select_x_max);
792 selectend_x = MAX(data->select_x_min, data->select_x_max);
794 else if (data->select_y_min < data->select_y_max)
796 selectstart_y = data->select_y_min;
797 selectstart_x = data->select_x_min;
798 selectend_y = data->select_y_max;
799 selectend_x = data->select_x_max;
801 else
803 selectstart_y = data->select_y_max;
804 selectstart_x = data->select_x_max;
805 selectend_y = data->select_y_min;
806 selectend_x = data->select_x_min;
809 while (line && yc <= toLine)
811 const char *str = line->text;
812 ULONG start = 0;
813 ULONG remaining_space = CHAR_XMAX(o) + 1;
814 Move(rp, GFX_XMIN(o), y);
815 while (line->size > start && remaining_space > 0 && str[start])
817 /* Identify a batch of characters with the same fgpen/bgpen
818 to avoid having to move/set pens and do Text() on single
819 characters */
821 UBYTE fgpen = line->fgpen[start];
822 UBYTE bgpen = line->bgpen[start];
824 /* Is any part of this line part of a selection?
825 * If so, we bake in a state transition on "stop".
826 * This code is messy - there must be a nicer way.
828 ULONG stop = 9999999;
829 BOOL in_selection = 0;
831 if (yc > selectstart_y && yc < selectend_y)
833 in_selection = 1;
835 else if (yc == selectstart_y)
837 if (yc == selectend_y)
839 if (start >= selectstart_x && start < selectend_x)
841 in_selection = 1;
842 stop = selectend_x;
844 else if (start < selectstart_x)
846 stop = selectstart_x;
849 else
851 if (start >= selectstart_x)
852 in_selection = 1;
853 else
854 stop = selectstart_x;
857 else if (yc == selectend_y)
859 /* In this case, the selection *ends* on selectend_x */
860 if (start < selectend_x)
862 in_selection = 1;
863 stop = selectend_x;
867 if (stop == start)
868 stop += 1;
870 if (in_selection)
872 fgpen = line->bgpen[start];
873 bgpen = line->fgpen[start];
876 ULONG len = 0;
877 while (line->size > start + len && str[start + len] &&
878 len < remaining_space &&
879 line->fgpen[start] == line->fgpen[start + len] &&
880 line->bgpen[start] == line->bgpen[start + len] &&
881 line->flags[start] == line->flags[start + len] &&
882 start + len < stop)
883 len += 1;
885 setabpen(GfxBase, rp, line->flags[start], fgpen, bgpen);
886 if ((line->flags[start] & CON_TXTFLAGS_MASK) !=
887 (flags & CON_TXTFLAGS_MASK))
889 SetSoftStyle(rp, line->flags[start], CON_TXTFLAGS_MASK);
891 flags = line->flags[start];
893 Text(rp, &str[start], len);
895 start += len;
896 remaining_space -= len;
899 /* Clear to EOL, without overwriting scroll bar (ClearEOL does) */
900 SetAPen(rp, CU(o)->cu_BgPen);
901 RectFill(rp,
902 GFX_X(o, start), GFX_Y(o, yc),
903 GFX_XMAX(o), GFX_Y(o, yc + 1) - 1);
904 y += YRSIZE;
905 yc++;
907 /* We want to make sure we have lines covering the window once
908 there's something to scroll back, as that simplies resize handling
909 etc.
911 if (!line->next && yc <= toLine)
913 line->next = charmap_newline(0, line);
914 data->scrollback_size += 1;
916 line = line->next;
919 if (yc < toLine)
921 SetAPen(rp, CU(o)->cu_BgPen);
922 RectFill(rp,
923 GFX_XMIN(o), GFX_Y(o, yc), GFX_XMAX(o), GFX_Y(o, toLine));
926 Console_RenderCursor(o);
931 * Refresh the full console unless "off" is provided.
932 * If off is set to a positive value, it indicates the
933 * number of rows from the top we start rendering.
934 * If off is set to a negative value, it indicates the
935 * number of rows from the top we stop rendering.
936 * This is used for partial refreshes when the screen is
937 * scrolled up/down.
939 static VOID charmapcon_refresh(Class *cl, Object *o, LONG off)
941 LONG fromLine = 0;
942 LONG toLine = CHAR_YMAX(o) - CHAR_YMIN(o);
944 if (off > 0)
945 toLine = off - 1;
946 if (off < 0)
947 fromLine = CHAR_YMAX(o) + off + 1;
949 charmapcon_refresh_lines(cl, o, fromLine, toLine);
953 static ULONG charmapcon_calc_selection_size(struct charmap_line *first,
954 struct charmap_line *last, ULONG minx, ULONG maxx)
956 struct charmap_line *cur = first;
957 ULONG size = 0;
958 while (cur)
960 if (cur->text)
962 if (cur == first)
964 if (cur == last)
965 size += abs(maxx - minx);
966 else
967 size += cur->size - minx;
969 else if (cur == last)
971 size += maxx;
973 else
975 size += cur->size;
978 if (cur != last || (cur == last && maxx == cur->size))
979 size += 1; /* line feed */
981 if (cur == last)
982 break;
983 cur = cur->next;
985 return size;
988 char *charmapcon_get_selection(ULONG size,
989 struct charmap_line *first,
990 struct charmap_line *last, ULONG minx, ULONG maxx)
992 char *buf = AllocMem(size, MEMF_ANY);
993 char *bufpos = buf;
994 struct charmap_line *cur = first;
996 if (!bufpos)
997 return 0;
999 while (cur)
1001 if (cur->text)
1003 /* empty lines may not have text ptrs */
1004 if (cur == first)
1006 if (cur == last)
1008 CopyMem(cur->text + MIN(minx, maxx), bufpos,
1009 abs(maxx - minx));
1010 bufpos += abs(maxx - minx);
1012 else
1014 CopyMem(cur->text + minx, bufpos, cur->size - minx);
1015 bufpos += cur->size - minx;
1018 else if (cur == last)
1020 CopyMem(cur->text, bufpos, maxx);
1021 bufpos += maxx;
1023 else
1025 CopyMem(cur->text, bufpos, cur->size);
1026 bufpos += cur->size;
1029 if (cur != last || (cur == last && maxx == cur->size)) /* line feed */
1031 *bufpos = '\r';
1032 bufpos += 1;
1034 if (cur == last)
1035 break;
1036 cur = cur->next;
1038 return buf;
1041 struct ConClipData
1043 ULONG flags; /* always zero? */
1044 ULONG size; /* does not include NUL termination */
1045 APTR buffer; /* NUL-terminated string! */
1048 /* FIXME: Belongs in snipmapcon - here temporarily until refactored out
1049 selection code */
1050 static VOID charmapcon_copy(Class *cl, Object *o, Msg copymsg)
1052 struct charmapcondata *data = INST_DATA(cl, o);
1053 struct ConsoleBase *ConsoleDevice = (struct ConsoleBase *)cl->cl_UserData;
1054 struct MsgPort replyport, *port;
1055 struct SGWork sgw;
1056 struct MyEditHookMsg msg;
1057 char *buf;
1059 struct charmap_line *first, *last;
1060 ULONG minx, maxx, size;
1062 /* Create a string from the contents of the scrollback buffer */
1063 if (data->select_y_min < data->select_y_max)
1065 first = data->select_line_min;
1066 last = data->select_line_max;
1067 minx = data->select_x_min;
1068 maxx = data->select_x_max;
1070 else
1072 first = data->select_line_max;
1073 last = data->select_line_min;
1074 minx = data->select_x_max;
1075 maxx = data->select_x_min;
1078 size = charmapcon_calc_selection_size(first, last, minx, maxx);
1079 buf = charmapcon_get_selection(size, first, last, minx, maxx);
1080 D(bug("%d bytes copied\n"));
1082 /* If Conclip is running, we prefer using that */
1083 if (IsListEmpty(&ConsoleDevice->sniphooks)
1084 && (port = FindPort(CONCLIP_PORTNAME)))
1086 /* AROS conclip format */
1087 D(bug("AROS conclip\n"));
1089 memset( &replyport, 0, sizeof( replyport ) );
1091 replyport.mp_Node.ln_Type = NT_MSGPORT;
1092 replyport.mp_Flags = PA_SIGNAL;
1093 replyport.mp_SigBit = SIGB_SINGLE;
1094 replyport.mp_SigTask = FindTask(NULL);
1095 NewList(&replyport.mp_MsgList);
1097 msg.msg.mn_Node.ln_Type = NT_MESSAGE;
1098 msg.msg.mn_ReplyPort = &replyport;
1099 msg.msg.mn_Length = sizeof(msg);
1101 msg.code = CODE_COPY;
1102 msg.sgw = &sgw;
1104 sgw.Gadget = 0;
1105 sgw.WorkBuffer = buf;
1106 sgw.PrevBuffer = 0;
1107 sgw.IEvent = 0;
1108 sgw.Code = CODE_COPY;
1109 sgw.Actions = 0;
1110 sgw.LongInt = 0;
1111 sgw.GadgetInfo = 0;
1112 sgw.EditOp = EO_BIGCHANGE;
1113 sgw.BufferPos = 0;
1114 sgw.NumChars = size;
1116 SetSignal(0, SIGF_SINGLE);
1117 PutMsg(port, &msg.msg);
1118 WaitPort(&replyport);
1120 FreeMem(buf, size);
1121 return;
1124 ObtainSemaphore(&ConsoleDevice->copyBufferLock);
1125 FreeMem((APTR) ConsoleDevice->copyBuffer,
1126 ConsoleDevice->copyBufferSize);
1128 ConsoleDevice->copyBuffer = buf;
1129 if (ConsoleDevice->copyBuffer)
1130 ConsoleDevice->copyBufferSize = size;
1131 else
1132 ConsoleDevice->copyBufferSize = 0;
1134 if (!IsListEmpty(&ConsoleDevice->sniphooks))
1136 /* OS2-3.x compatible conclip format */
1137 struct Hook *conhook;
1138 struct ConClipData ccd;
1140 D(bug("AOS conclip\n"));
1141 if (ConsoleDevice->copyBufferSize)
1143 ccd.flags = 0;
1144 ccd.size = ConsoleDevice->copyBufferSize;
1145 /* must be NUL-terminated */
1146 ccd.buffer = AllocVec(ccd.size + 1, MEMF_CLEAR);
1147 if (ccd.buffer)
1149 CopyMem(ConsoleDevice->copyBuffer, ccd.buffer, ccd.size);
1150 ForeachNode(&ConsoleDevice->sniphooks, conhook)
1152 D(bug("Calling AOS conclip hook %p\n", conhook));
1153 CALLHOOKPKT(conhook, NULL, &ccd);
1155 FreeVec(ccd.buffer);
1159 ReleaseSemaphore(&ConsoleDevice->copyBufferLock);
1163 /**********************************
1164 ** CharMapCon::NewWindowSize() **
1165 **********************************/
1166 static VOID charmapcon_newwindowsize(Class *cl, Object *o,
1167 struct P_Console_NewWindowSize *msg)
1169 struct charmapcondata *data = INST_DATA(cl, o);
1171 WORD old_ycp = YCP;
1173 DoSuperMethodA(cl, o, (Msg) msg);
1174 D(bug("CharMapCon::NewWindowSize(o=%p) x=%d, y=%d, ymax=%d\n",
1175 o, XCP, YCP, CHAR_YMAX(o)));
1177 // Is console empty? Unlikely, but anyway.
1178 if (!data->top_of_window)
1179 return;
1181 // Scroll up if new window size has forced the cursor up
1182 if (old_ycp > CHAR_YMAX(o))
1184 charmap_scroll_up(cl, o, old_ycp - CHAR_YMAX(o));
1187 charmapcon_refresh(cl, o, 0);
1190 static VOID charmapcon_handlemouse(Class *cl, Object *o,
1191 struct P_Console_HandleGadgets *msg)
1193 struct charmapcondata *data = INST_DATA(cl, o);
1194 struct Window *w = CU(o)->cu_Window;
1195 struct InputEvent *e = msg->Event;
1197 /* We have the following states:
1198 * - No active selection and no mouse button => ignore
1199 * - No active selection and left mouse button => Start selection
1200 * - Active selection and no mouse button => End selection
1201 * - Active selection and left mouse button => Update selection
1204 LONG x, y;
1207 * Take window-relative mouse position.
1208 * The original code here failed when the screen was moved away from (0, 0)
1209 * position. x and y were unaware of shifted display (ie_X and ie_Y are
1210 * raw physical coordinates, and w->LeftEdge and w->TopEdge are screen-
1211 * relative), and this caused lockups here.
1212 * TODO: Verify this intuition's behavior with AmigaOS3.1 using screentest
1213 * program. (sonic)
1215 x = e->ie_X - w->LeftEdge;
1216 y = e->ie_Y - w->TopEdge;
1218 x = w->MouseX;
1219 y = w->MouseY;
1221 /* Active selection */
1223 if (!(e->ie_Qualifier & IEQUALIFIER_LEFTBUTTON))
1225 /* End selection */
1226 data->active_selection = 0;
1227 data->ignore_drag = 0;
1228 return;
1231 if (!data->active_selection)
1233 if (e->ie_Qualifier & IEQUALIFIER_LEFTBUTTON && !data->ignore_drag)
1235 /* Inside the console area? */
1236 if (x >= GFX_XMIN(o) && x <= GFX_XMAX(o) &&
1237 y >= GFX_YMIN(o) && y <= GFX_YMAX(o))
1239 D(bug("activated selection with x: %ld, y: %ld,"
1240 " xmin: %ld, ymin: %Ld, xmax: %ld, ymax: %ld\n",
1241 x, y, GFX_XMIN(o), GFX_YMIN(o), GFX_XMAX(o),
1242 GFX_YMAX(o)));
1244 /* We need to clear these lines
1246 LONG old_min_y =
1247 MIN(data->select_y_min, data->select_y_max);
1248 LONG old_max_y =
1249 MAX(data->select_y_min, data->select_y_max);
1251 /* Yes, so start selection */
1252 data->active_selection = 1;
1253 data->ignore_drag = 0;
1254 data->select_x_min = (x - GFX_XMIN(o)) / XRSIZE;
1255 data->select_y_min = (y - GFX_YMIN(o)) / YRSIZE;
1256 data->select_line_min =
1257 charmapcon_find_line(cl, o, data->select_y_min);
1259 data->select_x_max = data->select_x_min;
1260 data->select_y_max = data->select_y_min;
1262 /* Clear */
1264 /* FIXME: Determine exactly which lines are affected, as
1265 follows:
1267 - If max_y has increased, render from old_max_y to new
1268 - If min_y has decreased, refresh from new min y to old
1269 - If both have changed, refresh in two batches.
1270 - If there's been a reversal of the relative positions of
1271 y_min/y_max, refresh the entire range.
1273 charmapcon_refresh_lines(cl, o, old_min_y, old_max_y);
1276 else
1277 data->ignore_drag = 1;
1279 return;
1282 /* Update selection. */
1283 if (x >= GFX_XMIN(o) && x <= GFX_XMAX(o) &&
1284 y >= GFX_YMIN(o) && y <= GFX_YMAX(o))
1287 LONG xmax, ymax;
1288 xmax = data->select_x_max;
1289 ymax = data->select_y_max;
1290 data->select_x_max = (x - GFX_XMIN(o)) / XRSIZE;
1291 data->select_y_max = (y - GFX_YMIN(o)) / YRSIZE;
1292 data->select_line_max =
1293 charmapcon_find_line(cl, o, data->select_y_max);
1294 /* FIXME: More intelligent refresh */
1295 if (xmax != data->select_x_max || ymax != data->select_y_max)
1297 charmapcon_refresh_lines(cl, o, MIN(ymax,
1298 MIN(data->select_y_min, data->select_y_max)), MAX(ymax,
1299 MAX(data->select_y_min, data->select_y_max)));
1302 else
1304 /* FIXME: Outside the console area, we need to scroll the window
1305 and just update the selection based on that */
1310 static VOID charmapcon_handlegadgets(Class *cl, Object *o,
1311 struct P_Console_HandleGadgets *msg)
1313 struct InputEvent *e = msg->Event;
1315 if (e->ie_Class == IECLASS_RAWMOUSE)
1317 charmapcon_handlemouse(cl, o, msg);
1318 return;
1321 struct charmapcondata *data = INST_DATA(cl, o);
1323 if (e->ie_Class == IECLASS_GADGETUP)
1325 data->activeGad = 0;
1326 return;
1329 if (e->ie_Class == IECLASS_GADGETDOWN)
1331 /* We pass 0 from consoletask if the mouse wheel is being used */
1332 if ((IPTR) e->ie_EventAddress == 1)
1333 data->activeGad = (APTR) &(data->prop->up);
1334 else if ((IPTR) e->ie_EventAddress == 2)
1335 data->activeGad = (APTR) &(data->prop->down);
1336 else
1337 data->activeGad = e->ie_EventAddress;
1340 if (data->activeGad == (APTR) &(data->prop->scroller))
1342 ULONG hidden =
1343 data->scrollback_size >
1344 CHAR_YMAX(o) ? data->scrollback_size - CHAR_YMAX(o) - 1 : 0;
1345 ULONG pos =
1346 (((struct PropInfo *)((struct Gadget *)&(data->prop->
1347 scroller))->SpecialInfo)->VertPot * hidden +
1348 (MAXPOT / 2)) / MAXPOT;
1350 if (pos != data->scrollback_pos)
1351 charmapcon_scroll_to(cl, o, pos);
1353 else if (data->activeGad == (APTR) &(data->prop->down))
1355 if (data->scrollback_pos + CHAR_YMAX(o) < data->scrollback_size - 1)
1357 charmap_scroll_up(cl, o, 1);
1358 charmapcon_refresh(cl, o, 0);
1359 charmapcon_adj_prop(cl, o);
1362 else if (data->activeGad == (APTR) &(data->prop->up))
1364 if (data->top_of_window != data->top_of_scrollback)
1366 charmap_scroll_down(cl, o, 1);
1367 charmapcon_refresh(cl, o, 0);
1368 charmapcon_adj_prop(cl, o);
1375 AROS_UFH3S(IPTR, dispatch_charmapconclass,
1376 AROS_UFHA(Class *, cl, A0),
1377 AROS_UFHA(Object *, o, A2), AROS_UFHA(Msg, msg, A1))
1379 AROS_USERFUNC_INIT
1381 IPTR retval = 0UL;
1383 switch (msg->MethodID)
1385 case OM_NEW:
1386 retval = (IPTR) charmapcon_new(cl, o, (struct opSet *)msg);
1387 break;
1389 case OM_DISPOSE:
1390 charmapcon_dispose(cl, o, msg);
1391 break;
1393 case M_Console_DoCommand:
1394 charmapcon_docommand(cl, o, (struct P_Console_DoCommand *)msg);
1395 break;
1397 case M_Console_ClearCell:
1398 // FIXME: scroll down to end here if it's not there already.
1399 charmapcon_clearcell(cl, o, (struct P_Console_ClearCell *)msg);
1400 break;
1402 case M_Console_NewWindowSize:
1403 charmapcon_newwindowsize(cl, o,
1404 (struct P_Console_NewWindowSize *)msg);
1405 break;
1407 case M_Console_HandleGadgets:
1408 //D(bug("CharMapCon::HandleGadgets\n"));
1409 charmapcon_handlegadgets(cl, o,
1410 (struct P_Console_HandleGadgets *)msg);
1411 break;
1413 /* FIXME: Belongs in snimapcon - here temporarily until refactored out
1414 selection code */
1415 case M_Console_Copy:
1416 charmapcon_copy(cl, o, msg);
1417 break;
1419 default:
1420 retval = DoSuperMethodA(cl, o, msg);
1421 break;
1424 return retval;
1426 AROS_USERFUNC_EXIT
1429 #undef ConsoleDevice
1431 Class *makeCharMapConClass(struct ConsoleBase *ConsoleDevice)
1434 Class *cl;
1436 cl = MakeClass(NULL, NULL, STDCONCLASSPTR,
1437 sizeof(struct charmapcondata), 0UL);
1438 if (cl)
1440 cl->cl_Dispatcher.h_Entry = (APTR) dispatch_charmapconclass;
1441 cl->cl_Dispatcher.h_SubEntry = NULL;
1443 cl->cl_UserData = (IPTR) ConsoleDevice;
1445 return (cl);
1447 return (NULL);