New bitmap method SetRGBConversionFunction which can be used to
[tangerine.git] / rom / intuition / inputhandler_actions.c
blob837ad158cfc4709d81e65b8c7510936261c887a4
1 /*
2 Copyright © 1995-2003, The AROS Development Team. All rights reserved.
3 Copyright © 2001-2003, The MorphOS Development Team. All Rights Reserved.
4 $Id$
6 Responsible for executing deferred Intuition actions like MoveWindow,
7 SizeWindow, ActivateWindow, etc.
8 */
10 #include <proto/exec.h>
11 #include <proto/intuition.h>
12 #include <proto/alib.h>
13 #include <proto/layers.h>
14 #include <proto/graphics.h>
15 #include <proto/keymap.h>
16 #include <proto/input.h>
17 #include <exec/memory.h>
18 #include <exec/alerts.h>
19 #include <exec/interrupts.h>
20 #include <exec/ports.h>
21 #include <intuition/intuition.h>
22 #include <intuition/intuitionbase.h>
23 #include <intuition/gadgetclass.h>
24 #include <intuition/cghooks.h>
25 #include <intuition/sghooks.h>
26 #include <devices/input.h>
27 #include <devices/inputevent.h>
28 #include "inputhandler.h"
30 #include "boopsigadgets.h"
31 #include "boolgadgets.h"
32 #include "propgadgets.h"
33 #include "strgadgets.h"
34 #include "gadgets.h"
35 #include "intuition_intern.h" /* EWFLG_xxx */
36 #include "inputhandler_support.h"
37 #include "inputhandler_actions.h"
38 #include "menus.h"
40 #undef DEBUG
41 #define DEBUG 0
42 #include <aros/debug.h>
44 #define LOCK_ACTIONS() ObtainSemaphore(&GetPrivIBase(IntuitionBase)->IntuiActionLock);
45 #define UNLOCK_ACTIONS() ReleaseSemaphore(&GetPrivIBase(IntuitionBase)->IntuiActionLock);
47 #ifndef __MORPHOS__
48 static void move_family(struct Window *, int , int);
49 #endif
51 /*******************************************************************************************************/
53 static void CheckLayerRefresh(struct Layer *lay, struct Screen *targetscreen,
54 struct IntuitionBase *IntuitionBase)
56 //if (lay->Flags & LAYERREFRESH)
58 struct Window *win = (struct Window *)lay->Window;
60 if (lay == targetscreen->BarLayer)
62 RenderScreenBar(targetscreen, TRUE, IntuitionBase);
64 else if (win)
66 /* Does it belong to a GZZ window and is it
67 the outer window of that GZZ window? */
68 if (IS_GZZWINDOW(win) && (lay == BLAYER(win)))
70 /* simply refresh that window's frame */
71 Gad_BeginUpdate(lay, IntuitionBase);
72 int_refreshwindowframe(win,REFRESHGAD_BORDER,0,IntuitionBase);
73 lay->Flags &= ~LAYERREFRESH;
74 Gad_EndUpdate(lay, TRUE, IntuitionBase);
76 /* Is it the window layer ? */
77 else if (lay == WLAYER(win))
79 WindowNeedsRefresh(win, IntuitionBase);
81 /* Otherwise, it's a requester. */
82 else
84 struct Requester *req;
86 for (req = win->FirstRequest; req && req->ReqLayer != lay; req = req->OlderRequest);
88 if (req)
90 /* FIXME: should send an IDCMP refresh message too. */
91 Gad_BeginUpdate(lay, IntuitionBase);
92 render_requester(req, IntuitionBase);
93 lay->Flags &= ~LAYERREFRESH;
94 Gad_EndUpdate(lay, TRUE, IntuitionBase);
99 } /* if (lay->Flags & LAYERREFRESH) */
102 /*******************************************************************************************************/
104 void CheckLayers(struct Screen *screen, struct IntuitionBase *IntuitionBase)
106 struct Layer *L;
108 LOCK_REFRESH(screen);
110 for (L = screen->LayerInfo.top_layer; L; L = L->back)
112 if (L->Flags & LAYERREFRESH)
114 CheckLayerRefresh(L, screen, IntuitionBase);
118 UNLOCK_REFRESH(screen);
121 void WindowSizeWillChange(struct Window *targetwindow, WORD dx, WORD dy,
122 struct IntuitionBase *IntuitionBase)
124 struct Rectangle *clipto = NULL;
125 struct Rectangle final_innerrect;
127 /* Erase the old frame on the right/lower side if
128 new size is bigger than old size
131 D(bug("********* WindowSizeWillChange ******** dx = %d dy = %d\n", dx, dy));
133 if (AVOID_WINBORDERERASE)
135 final_innerrect.MinX = targetwindow->BorderLeft;
136 final_innerrect.MinY = targetwindow->BorderTop;
137 final_innerrect.MaxX = targetwindow->Width + dx - 1 - targetwindow->BorderRight;
138 final_innerrect.MaxY = targetwindow->Height + dy - 1 - targetwindow->BorderBottom;
139 clipto = &final_innerrect;
142 if ( ((dx > 0) && (targetwindow->BorderRight > 0)) ||
143 ((dy > 0) && (targetwindow->BorderBottom > 0)) )
145 struct RastPort *rp = targetwindow->BorderRPort;
146 struct Layer *L = (BLAYER(targetwindow)) ? BLAYER(targetwindow) : WLAYER(targetwindow);
147 struct Rectangle rect;
148 struct Region *oldclipregion;
149 WORD ScrollX;
150 WORD ScrollY;
153 ** In case a clip region is installed then I have to
154 ** install the regular cliprects of the layer
155 ** first. Otherwise the frame might not get cleared correctly.
158 LockLayer(0, L);
160 oldclipregion = InstallClipRegion(L, NULL);
162 ScrollX = L->Scroll_X;
163 ScrollY = L->Scroll_Y;
165 L->Scroll_X = 0;
166 L->Scroll_Y = 0;
168 if ((dx > 0) && (targetwindow->BorderRight > 0))
170 rect.MinX = targetwindow->Width - targetwindow->BorderRight;
171 rect.MinY = 0;
172 rect.MaxX = targetwindow->Width - 1;
173 rect.MaxY = targetwindow->Height - 1;
175 OrRectRegion(L->DamageList, &rect);
176 L->Flags |= LAYERREFRESH;
178 if (!AVOID_WINBORDERERASE || AndRectRect(&rect, &final_innerrect, &rect))
180 EraseRect(rp, rect.MinX, rect.MinY, rect.MaxX, rect.MaxY);
185 if ((dy > 0) && (targetwindow->BorderBottom > 0))
188 rect.MinX = 0;
189 rect.MinY = targetwindow->Height - targetwindow->BorderBottom;
190 rect.MaxX = targetwindow->Width - 1;
191 rect.MaxY = targetwindow->Height - 1;
193 OrRectRegion(L->DamageList, &rect);
194 L->Flags |= LAYERREFRESH;
196 if (!AVOID_WINBORDERERASE || AndRectRect(&rect, &final_innerrect, &rect))
198 EraseRect(rp, rect.MinX, rect.MinY, rect.MaxX, rect.MaxY);
204 ** Reinstall the clipregions rectangles if there are any.
206 if (NULL != oldclipregion)
208 InstallClipRegion(L, oldclipregion);
211 L->Scroll_X = ScrollX;
212 L->Scroll_Y = ScrollY;
214 UnlockLayer(L);
216 } /* if ( ((dx > 0) && (targetwindow->BorderRight > 0)) || ((dy > 0) && (targetwindow->BorderBottom > 0)) ) */
218 /* Before resizing the layers eraserect the area of all
219 GFLG_REL??? gadgets and add the area to the damagelist */
221 EraseRelGadgetArea(targetwindow, clipto, FALSE, IntuitionBase);
225 /*******************************************************************************************************/
227 void WindowSizeHasChanged(struct Window *targetwindow, WORD dx, WORD dy,
228 BOOL is_sizewindow, struct IntuitionBase *IntuitionBase)
230 struct IIHData *iihdata = (struct IIHData *)GetPrivIBase(IntuitionBase)->InputHandler->is_Data;
231 struct Layer *lay;
233 D(bug("********* WindowSizeHasChanged ********\n"));
235 lay = (BLAYER(targetwindow)) ? BLAYER(targetwindow) : WLAYER(targetwindow);
237 /* Fix GadgetInfo domain if there is an active gadget in window
238 which was resized */
240 if ((iihdata->ActiveGadget) && (targetwindow == iihdata->GadgetInfo.gi_Window))
242 GetGadgetDomain(iihdata->ActiveGadget,
243 iihdata->GadgetInfo.gi_Screen,
244 iihdata->GadgetInfo.gi_Window,
245 NULL,
246 &iihdata->GadgetInfo.gi_Domain);
249 /* Relayout GFLG_REL??? gadgets */
250 DoGMLayout(targetwindow->FirstGadget, targetwindow, NULL, -1, FALSE, IntuitionBase);
252 /* Add the new area of all GFLG_REL??? gadgets to the damagelist, but
253 don't EraseRect() as the gadgets will be re-rendered at their new
254 position anyway */
257 struct Rectangle innerrect;
259 innerrect.MinX = targetwindow->BorderLeft;
260 innerrect.MinY = targetwindow->BorderTop;
261 innerrect.MaxX = targetwindow->Width - 1 - targetwindow->BorderRight;
262 innerrect.MaxY = targetwindow->Height - 1 - targetwindow->BorderBottom;
264 EraseRelGadgetArea(targetwindow, AVOID_WINBORDERERASE ? &innerrect : NULL, TRUE, IntuitionBase);
267 /* If new size is smaller than old size add right/bottom
268 frame to damagelist */
269 if ( ((dx < 0) && (targetwindow->BorderRight > 0)) ||
270 ((dx > 0) && (targetwindow->BorderTop > 0)) ||
271 ((dy < 0) && (targetwindow->BorderBottom > 0)) )
273 struct Rectangle rect;
275 LockLayer(0, lay);
277 if ((dx < 0) && (targetwindow->BorderRight > 0))
279 rect.MinX = targetwindow->Width - targetwindow->BorderRight;
280 rect.MinY = 0;
281 rect.MaxX = targetwindow->Width - 1;
282 rect.MaxY = targetwindow->Height - 1;
284 OrRectRegion(lay->DamageList, &rect);
285 lay->Flags |= LAYERREFRESH;
288 if ((dx > 0) && (targetwindow->BorderTop > 0))
290 rect.MinX = 0;
291 rect.MinY = 0;
292 rect.MaxX = targetwindow->Width - 1;
293 rect.MaxY = targetwindow->BorderTop - 1;
295 OrRectRegion(lay->DamageList, &rect);
296 lay->Flags |= LAYERREFRESH;
299 if ((dy < 0) && (targetwindow->BorderBottom > 0))
301 rect.MinX = 0;
302 rect.MinY = targetwindow->Height - targetwindow->BorderBottom;
303 rect.MaxX = targetwindow->Width - 1;
304 rect.MaxY = targetwindow->Height - 1;
306 OrRectRegion(lay->DamageList, &rect);
307 lay->Flags |= LAYERREFRESH;
310 UnlockLayer(lay);
312 } /* if ( ((dx < 0) && (targetwindow->BorderRight > 0)) || ((dy < 0) && (targetwindow->BorderBottom > 0)) ) */
314 ((struct IntWindow *)(targetwindow))->specialflags |= SPFLAG_LAYERRESIZED;
316 #if 0
317 if (IS_GZZWINDOW(targetwindow))
319 lay = targetwindow->BorderRPort->Layer;
321 if (lay->Flags & LAYERREFRESH)
323 Gad_BeginUpdate(lay, IntuitionBase);
324 RefreshWindowFrame(targetwindow);
325 lay->Flags &= ~LAYERREFRESH;
326 Gad_EndUpdate(lay, TRUE, IntuitionBase);
329 lay = targetwindow->WLayer;
331 if (lay->Flags & LAYERREFRESH)
333 Gad_BeginUpdate(lay, IntuitionBase);
334 int_refreshglist(targetwindow->FirstGadget, targetwindow, NULL, -1, 0, REFRESHGAD_BORDER, IntuitionBase);
335 Gad_EndUpdate(lay, IS_NOCAREREFRESH(targetwindow), IntuitionBase);
339 else
341 lay = targetwindow->WLayer;
343 if (lay->Flags & LAYERREFRESH)
345 Gad_BeginUpdate(lay, IntuitionBase);
346 RefreshWindowFrame(targetwindow);
347 int_refreshglist(targetwindow->FirstGadget, targetwindow, NULL, -1, 0, REFRESHGAD_BORDER, IntuitionBase);
348 Gad_EndUpdate(lay, IS_NOCAREREFRESH(targetwindow), IntuitionBase);
352 lay = targetwindow->WLayer;
354 if (IS_NOCAREREFRESH(targetwindow))
356 LockLayer(0, lay);
357 lay->Flags &= ~LAYERREFRESH;
358 UnlockLayer(lay);
360 #endif
362 #if 0
363 //if (is_sizewindow)
365 /* Send IDCMP_NEWSIZE to resized window */
367 ih_fire_intuimessage(targetwindow,
368 IDCMP_NEWSIZE,
370 targetwindow,
371 IntuitionBase);
374 if (ie = AllocInputEvent(iihdata))
376 ie->ie_Class = IECLASS_EVENT;
377 ie->ie_Code = IECODE_NEWSIZE;
378 ie->ie_EventAddress = targetwindow;
379 CurrentTime(&ie->ie_TimeStamp.tv_secs, &ie->ie_TimeStamp.tv_micro);
382 /* Send IDCMP_CHANGEWINDOW to resized window */
384 ih_fire_intuimessage(targetwindow,
385 IDCMP_CHANGEWINDOW,
386 CWCODE_MOVESIZE,
387 targetwindow,
388 IntuitionBase);
389 #endif
393 /*******************************************************************************************************/
395 void DoMoveSizeWindow(struct Window *targetwindow, LONG NewLeftEdge, LONG NewTopEdge,
396 LONG NewWidth, LONG NewHeight, BOOL send_newsize, struct IntuitionBase *IntuitionBase)
398 struct IIHData *iihdata = (struct IIHData *)GetPrivIBase(IntuitionBase)->InputHandler->is_Data;
399 //struct IntWindow *w = (struct IntWindow *)targetwindow;
400 struct Layer *targetlayer = WLAYER(targetwindow)/*, *L*/;
401 struct Requester *req;
402 struct InputEvent *ie;
403 LONG OldLeftEdge = targetwindow->LeftEdge;
404 LONG OldTopEdge = targetwindow->TopEdge;
405 LONG OldWidth = targetwindow->Width;
406 LONG OldHeight = targetwindow->Height;
407 LONG pos_dx, pos_dy, size_dx, size_dy;
409 /* correct new window coords if necessary */
411 FixWindowCoords(targetwindow, &NewLeftEdge, &NewTopEdge, &NewWidth, &NewHeight,IntuitionBase);
413 D(bug("DoMoveSizeWindow to %d,%d %d x %d\n", NewLeftEdge, NewTopEdge, NewWidth, NewHeight));
415 pos_dx = NewLeftEdge - OldLeftEdge;
416 pos_dy = NewTopEdge - OldTopEdge;
417 size_dx = NewWidth - OldWidth;
418 size_dy = NewHeight - OldHeight;
420 LOCK_REFRESH(targetwindow->WScreen);
422 /* jDc: intuition 68k doesn't care about that */
423 // if (pos_dx || pos_dy || size_dx || size_dy)
424 // {
426 if (size_dx || size_dy)
428 WindowSizeWillChange(targetwindow, size_dx, size_dy, IntuitionBase);
431 targetwindow->LeftEdge += pos_dx;
432 targetwindow->TopEdge += pos_dy;
433 #ifndef __MORPHOS__
434 targetwindow->RelLeftEdge += pos_dx;
435 targetwindow->RelTopEdge += pos_dy;
436 #endif
438 targetwindow->Width = NewWidth;
439 targetwindow->Height = NewHeight;
440 targetwindow->GZZWidth = targetwindow->Width - targetwindow->BorderLeft - targetwindow->BorderRight;
441 targetwindow->GZZHeight = targetwindow->Height - targetwindow->BorderTop - targetwindow->BorderBottom;
443 /* check for GZZ window */
444 if (BLAYER(targetwindow))
446 /* move outer window first */
447 MoveSizeLayer(BLAYER(targetwindow), pos_dx, pos_dy, size_dx, size_dy);
450 MoveSizeLayer(targetlayer, pos_dx, pos_dy, size_dx, size_dy);
452 for (req = targetwindow->FirstRequest; req; req = req->OlderRequest)
454 struct Layer *layer = req->ReqLayer;
456 if (layer)
458 int dx, dy, dw, dh;
459 int left, top, right, bottom;
461 left = NewLeftEdge + req->LeftEdge;
462 top = NewTopEdge + req->TopEdge;
463 right = left + req->Width - 1;
464 bottom = top + req->Height - 1;
466 if (left > NewLeftEdge + NewWidth - 1)
467 left = NewLeftEdge + NewWidth - 1;
469 if (top > NewTopEdge + NewHeight - 1)
470 top = NewTopEdge + NewHeight - 1;
472 if (right > NewLeftEdge + NewWidth - 1)
473 right = NewLeftEdge + NewWidth - 1;
475 if (bottom > NewTopEdge + NewHeight - 1)
476 bottom = NewTopEdge + NewHeight - 1;
478 dx = left - layer->bounds.MinX;
479 dy = top - layer->bounds.MinY;
480 dw = right - left - layer->bounds.MaxX + layer->bounds.MinX;
481 dh = bottom - top - layer->bounds.MaxY + layer->bounds.MinY;
483 MoveSizeLayer(layer, dx, dy, dw, dh);
487 #if 0
488 if (w->ZipLeftEdge != ~0) w->ZipLeftEdge = OldLeftEdge;
489 if (w->ZipTopEdge != ~0) w->ZipTopEdge = OldTopEdge;
490 if (w->ZipWidth != ~0) w->ZipWidth = OldWidth;
491 if (w->ZipHeight != ~0) w->ZipHeight = OldHeight;
492 #endif
494 if (pos_dx || pos_dy)
496 UpdateMouseCoords(targetwindow);
497 #ifndef __MORPHOS__
498 if (HAS_CHILDREN(targetwindow))
499 move_family(targetwindow, pos_dx, pos_dy);
500 #endif
503 // } /* if (pos_dx || pos_dy || size_dx || size_dy) */
505 if (size_dx || size_dy)
507 WindowSizeHasChanged(targetwindow, size_dx, size_dy, FALSE, IntuitionBase);
510 ih_fire_intuimessage(targetwindow,
511 IDCMP_CHANGEWINDOW,
512 CWCODE_MOVESIZE,
513 targetwindow,
514 IntuitionBase);
516 if (send_newsize)
518 /* Send IDCMP_NEWSIZE and IDCMP_CHANGEWINDOW to resized window, even
519 if there was no resizing/position change at all. BGUI for example
520 relies on this! */
522 ih_fire_intuimessage(targetwindow,
523 IDCMP_NEWSIZE,
525 targetwindow,
526 IntuitionBase);
528 if ((ie = AllocInputEvent(iihdata)))
530 ie->ie_Class = IECLASS_EVENT;
531 ie->ie_Code = IECODE_NEWSIZE;
532 ie->ie_EventAddress = targetwindow;
533 CurrentTime(&ie->ie_TimeStamp.tv_secs, &ie->ie_TimeStamp.tv_micro);
537 // jDc: CheckLayers calls LOCK_REFRESH, so there's no reason to UNLOCK here!
538 // UNLOCK_REFRESH(targetwindow->WScreen);
540 CheckLayers(targetwindow->WScreen, IntuitionBase);
542 UNLOCK_REFRESH(targetwindow->WScreen);
547 /*******************************************************************************************************/
549 void DoSyncAction(void (*func)(struct IntuiActionMsg *, struct IntuitionBase *),
550 struct IntuiActionMsg *msg,
551 struct IntuitionBase *IntuitionBase)
553 struct IIHData *iihd = (struct IIHData *)GetPrivIBase(IntuitionBase)->InputHandler->is_Data;
554 struct Task *me = FindTask(NULL);
556 if (me == iihd->InputDeviceTask)
558 func(msg, IntuitionBase);
560 else
562 #ifdef __MORPHOS__
563 struct IOStdReq req;
564 struct MsgPort port;
565 struct InputEvent ie;
566 #endif
568 msg->handler = func;
569 msg->task = me;
570 msg->done = FALSE;
572 ObtainSemaphore(&GetPrivIBase(IntuitionBase)->IntuiActionLock);
573 AddTail((struct List *)GetPrivIBase(IntuitionBase)->IntuiActionQueue, (struct Node *)msg);
574 ReleaseSemaphore(&GetPrivIBase(IntuitionBase)->IntuiActionLock);
576 #ifdef __MORPHOS__
577 port.mp_Flags = PA_SIGNAL;
578 port.mp_SigTask = me;
579 port.mp_SigBit = SIGB_INTUITION;
580 NEWLIST(&port.mp_MsgList);
582 req.io_Message.mn_ReplyPort = &port;
583 req.io_Device = GetPrivIBase(IntuitionBase)->InputIO->io_Device;
584 req.io_Unit = GetPrivIBase(IntuitionBase)->InputIO->io_Unit;
585 req.io_Command = IND_WRITEEVENT;
586 req.io_Length = sizeof(ie);
587 req.io_Data = &ie;
589 ie.ie_Class = IECLASS_NULL;
590 #endif
592 if (!msg->done)
594 #ifdef __MORPHOS__
595 DoIO((APTR)&req);
596 #else
597 AddNullEvent();
598 #endif
599 while (!msg->done)
601 Wait(SIGF_INTUITION);
607 /*******************************************************************************************************/
609 BOOL DoASyncAction(void (*func)(struct IntuiActionMsg *, struct IntuitionBase *),
610 struct IntuiActionMsg *msg, ULONG size,
611 struct IntuitionBase *IntuitionBase)
613 struct IIHData *iihd = (struct IIHData *)GetPrivIBase(IntuitionBase)->InputHandler->is_Data;
614 struct Task *me = FindTask(NULL);
615 struct IntuiActionMsg *new_msg;
617 if (me == iihd->InputDeviceTask)
619 func(msg, IntuitionBase);
620 return TRUE;
622 else if ((new_msg = AllocVecPooled(iihd->ActionsMemPool,size)))
624 new_msg->handler = func;
625 new_msg->task = NULL;
626 if (size > sizeof(*msg))
628 CopyMem(msg + 1, new_msg + 1, size - sizeof(*msg));
631 ObtainSemaphore(&GetPrivIBase(IntuitionBase)->IntuiActionLock);
632 AddTail((struct List *)GetPrivIBase(IntuitionBase)->IntuiActionQueue, (struct Node *)new_msg);
633 ReleaseSemaphore(&GetPrivIBase(IntuitionBase)->IntuiActionLock);
635 #ifndef __MORPHOS__
636 AddNullEvent();
637 #endif
638 return TRUE;
640 else
642 return FALSE;
646 /*******************************************************************************************************/
648 void HandleIntuiActions(struct IIHData *iihdata,
649 struct IntuitionBase *IntuitionBase)
651 struct IntuiActionMsg *am;
653 D(bug("Handle Intuition action messages\n"));
655 if (iihdata->ActiveSysGadget)
657 D(bug("Handle Intuition action messages. Doing nothing because of active drag or resize gadget!\n"));
658 return;
661 for (;;)
663 LOCK_ACTIONS();
664 am = (struct IntuiActionMsg *)RemHead((struct List *)&iihdata->IntuiActionQueue);
665 UNLOCK_ACTIONS();
667 if (!am)
668 break;
670 am->handler(am, IntuitionBase);
672 if (am->task)
674 Forbid();
675 am->done = TRUE;
676 Signal(am->task, SIGF_INTUITION);
677 Permit();
679 else
681 FreeVecPooled(iihdata->ActionsMemPool,am);
685 D(bug("Intuition action messages handled\n"));
688 #ifndef __MORPHOS__
689 static void move_family(struct Window * w, int dx, int dy)
691 struct Window * _w = w->firstchild;
693 while (_w)
695 _w->LeftEdge += dx,
696 _w->TopEdge += dy;
697 if (HAS_CHILDREN(_w))
698 move_family(_w,dx,dy);
699 _w=_w->nextchild;
702 #endif