/* * Copyright 2001-2005, Haiku. * Distributed under the terms of the MIT License. * * Authors: * DarkWyrm * Adrian Oanca * Stephan Aßmus * Stefano Ceccherini (burton666@libero.it) * Axel Dörfler, axeld@pinc-software.de */ #include #include #include #include #include #include #include #include #include #include #include #include #include "AppServer.h" #include "BGet++.h" #include "DebugInfoManager.h" #include "Desktop.h" #include "DisplayDriver.h" #include "HWInterface.h" #include "Layer.h" #include "MessagePrivate.h" #include "RAMLinkMsgReader.h" #include "RootLayer.h" #include "ServerApp.h" #include "ServerBitmap.h" #include "ServerPicture.h" #include "ServerProtocol.h" #include "TokenHandler.h" #include "Utils.h" #include "WinBorder.h" #include "Workspace.h" #include "WorkspacesLayer.h" #include "ServerWindow.h" //#define DEBUG_SERVERWINDOW //#define DEBUG_SERVERWINDOW_GRAPHICS #ifdef DEBUG_SERVERWINDOW # include # define STRACE(x) printf x #else # define STRACE(x) ; #endif #ifdef DEBUG_SERVERWINDOW_GRAPHICS # include # define DTRACE(x) printf x #else # define DTRACE(x) ; #endif static const uint32 kMsgWindowQuit = 'winQ'; /*! \brief Constructor Does a lot of stuff to set up for the window - new decorator, new winborder, spawn a monitor thread. */ ServerWindow::ServerWindow(const char *title, ServerApp *app, port_id clientPort, port_id looperPort, int32 handlerID) : BLocker(title && *title ? title : "Unnamed Window"), fTitle(title), fServerApp(app), fWinBorder(NULL), fClientTeam(app->ClientTeam()), fMessagePort(-1), fClientReplyPort(clientPort), fClientLooperPort(looperPort), fQuitting(false), fClientViewsWithInvalidCoords(B_VIEW_RESIZED), fHandlerToken(handlerID), fCurrentLayer(NULL) { STRACE(("ServerWindow(%s)::ServerWindow()\n", title)); fServerToken = BPrivate::gDefaultTokens.NewToken(B_SERVER_TOKEN, this); } //!Tears down all connections the main app_server objects, and deletes some internals. ServerWindow::~ServerWindow() { STRACE(("*ServerWindow (%s):~ServerWindow()\n", fTitle)); if (!fWinBorder->IsOffscreenWindow()) gDesktop->RemoveWinBorder(fWinBorder); delete fWinBorder; free(const_cast(fTitle)); BPrivate::gDefaultTokens.RemoveToken(fServerToken); STRACE(("#ServerWindow(%s) will exit NOW\n", fTitle)); } status_t ServerWindow::Init(BRect frame, uint32 look, uint32 feel, uint32 flags, uint32 workspace) { if (fTitle == NULL) fTitle = strdup("Unnamed Window"); if (fTitle == NULL) return B_NO_MEMORY; // fMessagePort is the port to which the app sends messages for the server fMessagePort = create_port(100, fTitle); if (fMessagePort < B_OK) return fMessagePort; fLink.SetSenderPort(fClientReplyPort); fLink.SetReceiverPort(fMessagePort); // We cannot call MakeWinBorder in the constructor, since it fWinBorder = MakeWinBorder(frame, fTitle, look, feel, flags, workspace); if (!fWinBorder) return B_NO_MEMORY; if (!fWinBorder->IsOffscreenWindow()) gDesktop->AddWinBorder(fWinBorder); return B_OK; } bool ServerWindow::Run() { BAutolock locker(this); char name[B_OS_NAME_LENGTH]; snprintf(name, sizeof(name), "w:%ld:%s", ClientTeam(), Title()); // Spawn our message-monitoring thread fThread = spawn_thread(_message_thread, name, B_NORMAL_PRIORITY, this); if (fThread < B_OK) return false; if (resume_thread(fThread) != B_OK) { kill_thread(fThread); fThread = -1; return false; } // Send a reply to our window - it is expecting fMessagePort // port and some other info fLink.StartMessage(SERVER_TRUE); fLink.Attach(fMessagePort); float minWidth, maxWidth, minHeight, maxHeight; fWinBorder->GetSizeLimits(&minWidth, &maxWidth, &minHeight, &maxHeight); fLink.Attach(fWinBorder->Frame()); fLink.Attach(minWidth); fLink.Attach(maxWidth); fLink.Attach(minHeight); fLink.Attach(maxHeight); fLink.Flush(); return true; } void ServerWindow::Quit() { fQuitting = true; if (fThread < B_OK) { delete this; return; } if (fThread == find_thread(NULL)) { App()->RemoveWindow(this); delete this; exit_thread(0); } else { PostMessage(AS_HIDE_WINDOW); PostMessage(kMsgWindowQuit); } } /*! \brief Send a message to the ServerWindow with no attachments \param code ID code of the message to post */ void ServerWindow::PostMessage(int32 code) { BPrivate::LinkSender link(fMessagePort); link.StartMessage(code); link.Flush(); } //! Forces the window border to update its decorator void ServerWindow::ReplaceDecorator() { if (!IsLocked()) debugger("you must lock a ServerWindow object before calling ::ReplaceDecorator()\n"); STRACE(("ServerWindow %s: Replace Decorator\n", fTitle)); fWinBorder->UpdateDecorator(); } //! Shows the window's WinBorder void ServerWindow::Show() { // NOTE: if you do something else, other than sending a port message, PLEASE lock STRACE(("ServerWindow %s: Show\n", Title())); if (fQuitting || !fWinBorder->IsHidden()) return; fWinBorder->GetRootLayer()->ShowWinBorder(fWinBorder); } //! Hides the window's WinBorder void ServerWindow::Hide() { // NOTE: if you do something else, other than sending a port message, PLEASE lock STRACE(("ServerWindow %s: Hide\n", Title())); if (fWinBorder->IsHidden()) return; fWinBorder->GetRootLayer()->HideWinBorder(fWinBorder); } void ServerWindow::SetTitle(const char* newTitle) { const char* oldTitle = fTitle; if (newTitle == NULL || !newTitle[0]) fTitle = strdup("Unnamed Window"); else fTitle = strdup(newTitle); if (fTitle == NULL) { fTitle = oldTitle; return; } free(const_cast(oldTitle)); if (Thread() >= B_OK) { char name[B_OS_NAME_LENGTH]; snprintf(name, sizeof(name), "w:%ld:%s", ClientTeam(), fTitle); rename_thread(Thread(), name); } if (fWinBorder != NULL) fWinBorder->SetName(newTitle); } //! Requests that the ServerWindow's BWindow quit void ServerWindow::NotifyQuitRequested() { // NOTE: if you do something else, other than sending a port message, PLEASE lock STRACE(("ServerWindow %s: Quit\n", fTitle)); BMessage msg(B_QUIT_REQUESTED); SendMessageToClient(&msg); } void ServerWindow::NotifyMinimize(bool minimize) { // NOTE: if you do something else, other than sending a port message, PLEASE lock // This function doesn't need much -- check to make sure that we should and // send the message to the client. According to the BeBook, the BWindow hook function // does all the heavy lifting for us. :) bool sendMessages = false; if (minimize) { if (!fWinBorder->IsHidden()) { Hide(); sendMessages = true; } } else { if (fWinBorder->IsHidden()) { Show(); sendMessages = true; } } if (sendMessages) { BMessage msg(B_MINIMIZE); msg.AddInt64("when", real_time_clock_usecs()); msg.AddBool("minimize", minimize); SendMessageToClient(&msg); } } //! Sends a message to the client to perform a Zoom void ServerWindow::NotifyZoom() { // NOTE: if you do something else, other than sending a port message, PLEASE lock BMessage msg(B_ZOOM); SendMessageToClient(&msg); } /*! \brief Notifies window of a change in screen resolution \param frame Size of the new resolution \param color_space Color space of the new screen mode */ void ServerWindow::NotifyScreenModeChanged(const BRect frame, const color_space colorSpace) { STRACE(("ServerWindow %s: ScreenModeChanged\n", fTitle)); BMessage msg(B_SCREEN_CHANGED); msg.AddRect("frame", frame); msg.AddInt32("mode", (int32)colorSpace); SendMessageToClient(&msg); } void ServerWindow::GetInfo(window_info& info) { info.team = ClientTeam(); info.server_token = ServerToken(); info.thread = Thread(); info.client_token = ClientToken(); info.client_port = fClientLooperPort; info.workspaces = fWinBorder->Workspaces(); info.layer = 0; // ToDo: what is this??? info.feel = fWinBorder->Feel(); info.flags = fWinBorder->WindowFlags(); info.window_left = (int)floor(fWinBorder->Frame().left); info.window_top = (int)floor(fWinBorder->Frame().top); info.window_right = (int)floor(fWinBorder->Frame().right); info.window_bottom = (int)floor(fWinBorder->Frame().bottom); info.show_hide_level = fWinBorder->IsHidden() ? 1 : -1; // ??? info.is_mini = fWinBorder->IsHidden(); } /*! \brief Sets the font state for a layer \param layer The layer to set the font */ inline void ServerWindow::SetLayerFontState(Layer *layer, BPrivate::LinkReceiver &link) { STRACE(("ServerWindow %s: SetLayerFontStateMessage for layer %s\n", fTitle, layer->Name())); // NOTE: no need to check for a lock. This is a private method. layer->fLayerData->ReadFontFromLink(link); } inline void ServerWindow::SetLayerState(Layer *layer, BPrivate::LinkReceiver &link) { STRACE(("ServerWindow %s: SetLayerState for layer %s\n", Title(), layer->Name())); // NOTE: no need to check for a lock. This is a private method. layer->fLayerData->ReadFromLink(link); // TODO: Rebuild clipping here? } inline Layer* ServerWindow::CreateLayerTree(BPrivate::LinkReceiver &link, Layer **_parent) { // NOTE: no need to check for a lock. This is a private method. int32 token; BRect frame; uint32 resizeMask; uint32 eventMask; uint32 eventOptions; uint32 flags; bool hidden; int32 parentToken; char *name = NULL; rgb_color viewColor; link.Read(&token); link.ReadString(&name); link.Read(&frame); link.Read(&resizeMask); link.Read(&eventMask); link.Read(&eventOptions); link.Read(&flags); link.Read(&hidden); link.Read(&viewColor); link.Read(&parentToken); STRACE(("ServerWindow(%s)::CreateLayerTree()-> layer %s, token %ld\n", fTitle, name, token)); Layer *newLayer; if (link.Code() == AS_LAYER_CREATE_ROOT && (fWinBorder->WindowFlags() & kWorkspacesWindowFlag) != 0) { // this is a workspaces window! newLayer = new WorkspacesLayer(frame, name, token, resizeMask, flags, fWinBorder->GetDisplayDriver()); } else { newLayer = new Layer(frame, name, token, resizeMask, flags, gDesktop->GetDisplayDriver()); } free(name); // there is no way of setting this, other than manually :-) newLayer->fViewColor = viewColor; newLayer->fHidden = hidden; newLayer->fEventMask = eventMask; newLayer->fEventOptions = eventOptions; newLayer->fOwner = fWinBorder; // TODO: rework the clipping stuff to remove RootLayer dependency and then // remove this hack: if (fWinBorder->IsOffscreenWindow()) { #ifndef NEW_CLIPPING newLayer->fVisible.Set(newLayer->fFrame); #else newLayer->fVisible2.Set(newLayer->fFrame); #endif } if (_parent) { Layer *parent = fWinBorder->FindLayer(parentToken); if (parent == NULL) CRITICAL("View token not found!\n"); *_parent = parent; } return newLayer; } void ServerWindow::_DispatchMessage(int32 code, BPrivate::LinkReceiver &link) { if (fCurrentLayer == NULL && code != AS_LAYER_CREATE_ROOT && code != AS_LAYER_CREATE) { printf("ServerWindow %s received unexpected code - message offset %ld before top_view attached.\n", Title(), code - SERVER_TRUE); return; } RootLayer *myRootLayer = fWinBorder->GetRootLayer(); // NOTE: is NULL when fWinBorder is offscreen! if (myRootLayer) myRootLayer->Lock(); switch (code) { //--------- BView Messages ----------------- case AS_LAYER_SCROLL: { DTRACE(("ServerWindow %s: Message AS_LAYER_SCROLL: Layer name: %s\n", fTitle, fCurrentLayer->Name())); float dh; float dv; link.Read(&dh); link.Read(&dv); #ifndef NEW_CLIPPING // scroll visually by using the CopyBits() implementation // this will also take care of invalidating previously invisible // areas (areas scrolled into view) BRect src = fCurrentLayer->Bounds(); BRect dst = src; // NOTE: if we scroll down, the contents are moved *up* dst.OffsetBy(-dh, -dv); // TODO: Are origin and scale handled in this conversion? src = fCurrentLayer->ConvertToTop(src); dst = fCurrentLayer->ConvertToTop(dst); int32 xOffset = (int32)(dst.left - src.left); int32 yOffset = (int32)(dst.top - src.top); // this little detail is where it differs from CopyBits() // -> it will invalidate areas previously out of screen dst = dst | src; fCurrentLayer->fLayerData->OffsetOrigin(BPoint(dh, dv)); fCurrentLayer->CopyBits(src, dst, xOffset, yOffset); #else fCurrentLayer->ScrollBy(dh, dv); #endif break; } case AS_LAYER_COPY_BITS: { BRect src; BRect dst; link.Read(&src); link.Read(&dst); // TODO: Are origin and scale handled in this conversion? src = fCurrentLayer->ConvertToTop(src); dst = fCurrentLayer->ConvertToTop(dst); int32 xOffset = (int32)(dst.left - src.left); int32 yOffset = (int32)(dst.top - src.top); fCurrentLayer->CopyBits(src, dst, xOffset, yOffset); break; } case AS_SET_CURRENT_LAYER: { int32 token; link.Read(&token); Layer *current = fWinBorder->FindLayer(token); if (current) { DTRACE(("ServerWindow %s: Message AS_SET_CURRENT_LAYER: %s, token %ld\n", fTitle, current->Name(), token)); } else { DTRACE(("ServerWindow %s: Message AS_SET_CURRENT_LAYER: layer not found, token %ld\n", fTitle, token)); } if (current) fCurrentLayer = current; else // hope this NEVER happens! :-) CRITICAL("Server PANIC: window cannot find Layer with ID\n"); // ToDo: if this happens, we probably want to kill the app and clean up break; } case AS_LAYER_CREATE_ROOT: { STRACE(("ServerWindow %s: Message AS_LAYER_CREATE_ROOT\n", fTitle)); // Start receiving top_view data -- pass NULL as the parent view. // This should be the *only* place where this happens. if (fCurrentLayer != NULL) break; fWinBorder->SetTopLayer(CreateLayerTree(link, NULL)); fCurrentLayer = fWinBorder->TopLayer(); break; } case AS_LAYER_CREATE: { STRACE(("ServerWindow %s: Message AS_LAYER_CREATE: Layer name: %s\n", fTitle, fCurrentLayer->Name())); Layer* parent = NULL; Layer* newLayer = CreateLayerTree(link, &parent); if (parent != NULL) parent->AddChild(newLayer, this); if (myRootLayer && !newLayer->IsHidden() && parent) #ifndef NEW_CLIPPING myRootLayer->GoInvalidate(newLayer, newLayer->fFull); #else { BRegion invalidRegion; newLayer->GetWantedRegion(invalidRegion); myRootLayer->GoInvalidate(newLayer, invalidRegion); } #endif break; } case AS_LAYER_DELETE: { // Received when a view is detached from a window. This is definitely // the less taxing operation - we call PruneTree() on the removed // layer, detach the layer itself, delete it, and invalidate the // area assuming that the view was visible when removed STRACE(("ServerWindow %s: AS_LAYER_DELETE(self)...\n", fTitle)); Layer *parent = fCurrentLayer->fParent; BRegion *invalidRegion = NULL; if (!fCurrentLayer->IsHidden() && parent) { #ifndef NEW_CLIPPING if (fCurrentLayer->fFullVisible.CountRects() > 0) invalidRegion = new BRegion(fCurrentLayer->fFullVisible); #else if (fCurrentLayer->FullVisible().Frame().IsValid()) invalidRegion = new BRegion(fCurrentLayer->FullVisible()); #endif } // here we remove current layer from list. fCurrentLayer->RemoveSelf(); fCurrentLayer->PruneTree(); if (invalidRegion && myRootLayer) { myRootLayer->GoInvalidate(parent, *invalidRegion); delete invalidRegion; } #ifdef DEBUG_SERVERWINDOW parent->PrintTree(); #endif STRACE(("DONE: ServerWindow %s: Message AS_DELETE_LAYER: Parent: %s Layer: %s\n", fTitle, parent->Name(), fCurrentLayer->Name())); delete fCurrentLayer; // TODO: Does the client know about this change? fCurrentLayer = parent; break; } case AS_LAYER_SET_STATE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_STATE: Layer name: %s\n", fTitle, fCurrentLayer->Name())); // SetLayerState(fCurrentLayer); SetLayerState(fCurrentLayer, link); // TODO: should this be moved into SetLayerState? // If it _always_ needs to be done afterwards, then yes! #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); #endif break; } case AS_LAYER_SET_FONT_STATE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_FONT_STATE: Layer name: %s\n", fTitle, fCurrentLayer->Name())); // SetLayerFontState(fCurrentLayer); SetLayerFontState(fCurrentLayer, link); #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); #endif break; } case AS_LAYER_GET_STATE: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_STATE: Layer name: %s\n", fTitle, fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); // attach state data fCurrentLayer->fLayerData->WriteToLink(fLink.Sender()); fLink.Flush(); break; } case AS_LAYER_SET_MOUSE_EVENT_MASK: { STRACE(("ServerWindow %s: Message AS_LAYER_SET_MOUSE_EVENT_MASK: Layer name: %s\n", fTitle, fCurrentLayer->Name())); uint32 mask; uint32 options; link.Read(&mask); link.Read(&options); if (myRootLayer) myRootLayer->SetEventMaskLayer(fCurrentLayer, mask, options); break; } case AS_LAYER_MOVE_TO: { STRACE(("ServerWindow %s: Message AS_LAYER_MOVE_TO: Layer name: %s\n", fTitle, fCurrentLayer->Name())); float x, y; link.Read(&x); link.Read(&y); float offsetX = x - fCurrentLayer->fFrame.left; float offsetY = y - fCurrentLayer->fFrame.top; fCurrentLayer->MoveBy(offsetX, offsetY); break; } case AS_LAYER_RESIZE_TO: { STRACE(("ServerWindow %s: Message AS_LAYER_RESIZE_TO: Layer name: %s\n", fTitle, fCurrentLayer->Name())); float newWidth, newHeight; link.Read(&newWidth); link.Read(&newHeight); // TODO: If fCurrentLayer is a window, check for minimum size allowed. // Need WinBorder::GetSizeLimits float deltaWidth = newWidth - fCurrentLayer->fFrame.Width(); float deltaHeight = newHeight - fCurrentLayer->fFrame.Height(); fCurrentLayer->ResizeBy(deltaWidth, deltaHeight); break; } case AS_LAYER_GET_COORD: { STRACE(("ServerWindow %s: Message AS_LAYER_GET_COORD: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); // our offset in the parent -> will be originX and originY in BView fLink.Attach(fCurrentLayer->fFrame.left); fLink.Attach(fCurrentLayer->fFrame.top); // convert frame to bounds fLink.Attach(fCurrentLayer->fFrame.OffsetToCopy(fCurrentLayer->BoundsOrigin())); fLink.Flush(); break; } case AS_LAYER_SET_ORIGIN: { STRACE(("ServerWindow %s: Message AS_LAYER_SET_ORIGIN: Layer: %s\n", Title(), fCurrentLayer->Name())); float x, y; link.Read(&x); link.Read(&y); fCurrentLayer->fLayerData->SetOrigin(BPoint(x, y)); break; } case AS_LAYER_GET_ORIGIN: { STRACE(("ServerWindow %s: Message AS_LAYER_GET_ORIGIN: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fCurrentLayer->fLayerData->Origin()); fLink.Flush(); break; } case AS_LAYER_RESIZE_MODE: { STRACE(("ServerWindow %s: Message AS_LAYER_RESIZE_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); link.Read(&(fCurrentLayer->fResizeMode)); break; } case AS_LAYER_CURSOR: { DTRACE(("ServerWindow %s: Message AS_LAYER_CURSOR: Layer: %s - NOT IMPLEMENTED\n", Title(), fCurrentLayer->Name())); int32 token; link.Read(&token); // TODO: implement; I think each Layer should have a member pointing // to this requested cursor. break; } case AS_LAYER_SET_FLAGS: { uint32 flags; link.Read(&flags); fCurrentLayer->SetFlags(flags); STRACE(("ServerWindow %s: Message AS_LAYER_SET_FLAGS: Layer: %s\n", Title(), fCurrentLayer->Name())); break; } case AS_LAYER_HIDE: { STRACE(("ServerWindow %s: Message AS_LAYER_HIDE: Layer: %s\n", Title(), fCurrentLayer->Name())); fCurrentLayer->Hide(); break; } case AS_LAYER_SHOW: { STRACE(("ServerWindow %s: Message AS_LAYER_SHOW: Layer: %s\n", Title(), fCurrentLayer->Name())); fCurrentLayer->Show(); break; } case AS_LAYER_SET_LINE_MODE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_LINE_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); int8 lineCap, lineJoin; float miterLimit; // TODO: Look into locking scheme relating to Layers and modifying redraw-related members link.Read(&lineCap); link.Read(&lineJoin); link.Read(&miterLimit); fCurrentLayer->fLayerData->SetLineCapMode((cap_mode)lineCap); fCurrentLayer->fLayerData->SetLineJoinMode((join_mode)lineJoin); fCurrentLayer->fLayerData->SetMiterLimit(miterLimit); break; } case AS_LAYER_GET_LINE_MODE: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_LINE_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach((int8)(fCurrentLayer->fLayerData->LineCapMode())); fLink.Attach((int8)(fCurrentLayer->fLayerData->LineJoinMode())); fLink.Attach(fCurrentLayer->fLayerData->MiterLimit()); fLink.Flush(); break; } case AS_LAYER_PUSH_STATE: { DTRACE(("ServerWindow %s: Message AS_LAYER_PUSH_STATE: Layer: %s\n", Title(), fCurrentLayer->Name())); fCurrentLayer->PushState(); #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); #endif break; } case AS_LAYER_POP_STATE: { DTRACE(("ServerWindow %s: Message AS_LAYER_POP_STATE: Layer: %s\n", Title(), fCurrentLayer->Name())); fCurrentLayer->PopState(); #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); #endif break; } case AS_LAYER_SET_SCALE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_SCALE: Layer: %s\n", Title(), fCurrentLayer->Name())); float scale; link.Read(&scale); // TODO: The BeBook says, if you call SetScale() it will be // multiplied with the scale from all previous states on the stack fCurrentLayer->fLayerData->SetScale(scale); break; } case AS_LAYER_GET_SCALE: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_SCALE: Layer: %s\n", Title(), fCurrentLayer->Name())); LayerData *ld = fCurrentLayer->fLayerData; // TODO: And here, we're taking that into account, but not above // -> refactor put scale into Layer, or better yet, when the // state stack is within Layer, PushState() should multiply // by the previous last states scale. Would fix the problem above too. float scale = ld->Scale(); while ((ld = ld->prevState)) scale *= ld->Scale(); fLink.StartMessage(SERVER_TRUE); fLink.Attach(scale); fLink.Flush(); break; } case AS_LAYER_SET_PEN_LOC: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_PEN_LOC: Layer: %s\n", Title(), fCurrentLayer->Name())); float x, y; link.Read(&x); link.Read(&y); fCurrentLayer->fLayerData->SetPenLocation(BPoint(x, y)); break; } case AS_LAYER_GET_PEN_LOC: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_PEN_LOC: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fCurrentLayer->fLayerData->PenLocation()); fLink.Flush(); break; } case AS_LAYER_SET_PEN_SIZE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_PEN_SIZE: Layer: %s\n", Title(), fCurrentLayer->Name())); float penSize; link.Read(&penSize); fCurrentLayer->fLayerData->SetPenSize(penSize); break; } case AS_LAYER_GET_PEN_SIZE: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_PEN_SIZE: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fCurrentLayer->fLayerData->PenSize()); fLink.Flush(); break; } case AS_LAYER_SET_VIEW_COLOR: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_VIEW_COLOR: Layer: %s\n", Title(), fCurrentLayer->Name())); rgb_color c; link.Read(&c, sizeof(rgb_color)); myRootLayer->Lock(); fCurrentLayer->SetViewColor(RGBColor(c)); #ifndef NEW_CLIPPING if (myRootLayer) myRootLayer->GoRedraw(fCurrentLayer, fCurrentLayer->fVisible); #else myRootLayer->GoRedraw(fCurrentLayer, fCurrentLayer->VisibleRegion()); #endif myRootLayer->Unlock(); break; } case AS_LAYER_GET_HIGH_COLOR: DTRACE(("ServerWindow %s: Message AS_LAYER_GET_HIGH_COLOR: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fCurrentLayer->fLayerData->HighColor().GetColor32()); fLink.Flush(); break; case AS_LAYER_GET_LOW_COLOR: DTRACE(("ServerWindow %s: Message AS_LAYER_GET_LOW_COLOR: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fCurrentLayer->fLayerData->LowColor().GetColor32()); fLink.Flush(); break; case AS_LAYER_GET_VIEW_COLOR: DTRACE(("ServerWindow %s: Message AS_LAYER_GET_VIEW_COLOR: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fCurrentLayer->ViewColor().GetColor32()); fLink.Flush(); break; case AS_LAYER_SET_BLEND_MODE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_BLEND_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); int8 srcAlpha, alphaFunc; link.Read(&srcAlpha); link.Read(&alphaFunc); fCurrentLayer->fLayerData->SetBlendingMode((source_alpha)srcAlpha, (alpha_function)alphaFunc); break; } case AS_LAYER_GET_BLEND_MODE: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_BLEND_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach((int8)(fCurrentLayer->fLayerData->AlphaSrcMode())); fLink.Attach((int8)(fCurrentLayer->fLayerData->AlphaFncMode())); fLink.Flush(); break; } case AS_LAYER_SET_DRAW_MODE: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_DRAW_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); int8 drawingMode; link.Read(&drawingMode); fCurrentLayer->fLayerData->SetDrawingMode((drawing_mode)drawingMode); break; } case AS_LAYER_GET_DRAW_MODE: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_DRAW_MODE: Layer: %s\n", Title(), fCurrentLayer->Name())); fLink.StartMessage(SERVER_TRUE); fLink.Attach((int8)(fCurrentLayer->fLayerData->GetDrawingMode())); fLink.Flush(); break; } case AS_LAYER_PRINT_ALIASING: { DTRACE(("ServerWindow %s: Message AS_LAYER_PRINT_ALIASING: Layer: %s\n", Title(), fCurrentLayer->Name())); bool fontAliasing; link.Read(&fontAliasing); fCurrentLayer->fLayerData->SetForceFontAliasing(fontAliasing); break; } case AS_LAYER_CLIP_TO_PICTURE: { DTRACE(("ServerWindow %s: Message AS_LAYER_CLIP_TO_PICTURE: Layer: %s\n", Title(), fCurrentLayer->Name())); // TODO: you are not allowed to use Layer regions here!!! // If there is no other way, then first lock RootLayer object first. int32 pictureToken; BPoint where; bool inverse = false; link.Read(&pictureToken); link.Read(&where); link.Read(&inverse); // search for a picture with the specified token. ServerPicture *picture = fServerApp->FindPicture(pictureToken); // TODO: Increase that picture's reference count.(~ allocate a picture) if (picture == NULL) break; BRegion region; // TODO: I think we also need the BView's token // I think PictureToRegion would fit better into the Layer class (?) if (PictureToRegion(picture, region, inverse, where) < B_OK) break; fCurrentLayer->fLayerData->SetClippingRegion(region); #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); #endif if (myRootLayer && !(fCurrentLayer->IsHidden()) && !fWinBorder->InUpdate()) #ifndef NEW_CLIPPING myRootLayer->GoInvalidate(fCurrentLayer, fCurrentLayer->fFull); #else { BRegion invalidRegion; fCurrentLayer->GetWantedRegion(invalidRegion); myRootLayer->GoInvalidate(fCurrentLayer, invalidRegion); } #endif break; } case AS_LAYER_GET_CLIP_REGION: { DTRACE(("ServerWindow %s: Message AS_LAYER_GET_CLIP_REGION: Layer: %s\n", Title(), fCurrentLayer->Name())); // if this Layer is hidden, it is clear that its visible region is void. if (fCurrentLayer->IsHidden()) { fLink.StartMessage(SERVER_TRUE); fLink.Attach(0L); fLink.Flush(); } else { // TODO: Watch out for the coordinate system in AS_LAYER_GET_CLIP_REGION BRegion region; // TODO: This could also be done more reliably in the Layer, // when the State stack is implemented there. There should be // DrawData::fCulmulatedClippingRegion... // TODO: the DrawData clipping region should be in local view coords. LayerData* layerData = fCurrentLayer->fLayerData; do { if (layerData->ClippingRegion()) region.IntersectWith(layerData->ClippingRegion()); } while ((layerData = layerData->prevState) != NULL); int32 rectCount = region.CountRects(); fLink.StartMessage(SERVER_TRUE); fLink.Attach(rectCount); for (int32 i = 0; i < rectCount; i++) fLink.Attach(fCurrentLayer->ConvertFromTop(region.RectAt(i))); fLink.Flush(); } break; } case AS_LAYER_SET_CLIP_REGION: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_CLIP_REGION: Layer: %s\n", Title(), fCurrentLayer->Name())); // TODO: Watch out for the coordinate system in AS_LAYER_SET_CLIP_REGION int32 noOfRects; BRect r; link.Read(&noOfRects); BRegion region; for (int i = 0; i < noOfRects; i++) { link.Read(&r); region.Include(fCurrentLayer->ConvertToTop(r)); } fCurrentLayer->fLayerData->SetClippingRegion(region); #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); #endif /* #ifndef NEW_CLIPPING fCurrentLayer->RebuildFullRegion(); if (myRootLayer && !(fCurrentLayer->IsHidden()) && !fWinBorder->InUpdate()) myRootLayer->GoInvalidate(fCurrentLayer, fCurrentLayer->fFull); #else if (myRootLayer && !(fCurrentLayer->IsHidden()) && !fWinBorder->InUpdate()) myRootLayer->GoInvalidate(fCurrentLayer, fCurrentLayer->Frame()); #endif */ break; } case AS_LAYER_INVAL_RECT: { DTRACE(("ServerWindow %s: Message AS_LAYER_INVAL_RECT: Layer: %s\n", Title(), fCurrentLayer->Name())); // TODO: handle transformation (origin and scale) prior to converting to top BRect invalRect; link.Read(&invalRect); if (myRootLayer) { BRect converted(fCurrentLayer->ConvertToTop(invalRect.LeftTop()), fCurrentLayer->ConvertToTop(invalRect.RightBottom())); BRegion invalidRegion(converted); #ifdef NEW_CLIPPING invalidRegion.IntersectWith(&fCurrentLayer->fVisible2); #endif myRootLayer->GoRedraw(fWinBorder, invalidRegion); // myRootLayer->RequestDraw(invalidRegion, fWinBorder); } break; } case AS_LAYER_INVAL_REGION: { DTRACE(("ServerWindow %s: Message AS_LAYER_INVAL_RECT: Layer: %s\n", Title(), fCurrentLayer->Name())); // TODO: handle transformation (origin and scale) prior to converting to top // TODO: Handle conversion to top BRegion invalReg; int32 noOfRects; BRect rect; link.Read(&noOfRects); for (int i = 0; i < noOfRects; i++) { link.Read(&rect); invalReg.Include(rect); } if (myRootLayer) myRootLayer->GoRedraw(fCurrentLayer, invalReg); break; } case AS_BEGIN_UPDATE: { DTRACE(("ServerWindowo %s: AS_BEGIN_UPDATE\n", Title())); fWinBorder->UpdateStart(); break; } case AS_END_UPDATE: { DTRACE(("ServerWindowo %s: AS_END_UPDATE\n", Title())); fWinBorder->UpdateEnd(); break; } // ********** END: BView Messages *********** // ********* BWindow Messages *********** case AS_LAYER_DELETE_ROOT: { // Received when a window deletes its internal top view // TODO: Implement AS_LAYER_DELETE_ROOT STRACE(("ServerWindow %s: Message Delete_Layer_Root unimplemented\n", Title())); break; } case AS_SHOW_WINDOW: { STRACE(("ServerWindow %s: Message AS_SHOW_WINDOW\n", Title())); Show(); break; } case AS_HIDE_WINDOW: { STRACE(("ServerWindow %s: Message AS_HIDE_WINDOW\n", Title())); Hide(); break; } #if 0 case AS_SEND_BEHIND: { // TODO: Implement AS_SEND_BEHIND STRACE(("ServerWindow %s: Message Send_Behind unimplemented\n", Title())); break; } case AS_ENABLE_UPDATES: { // TODO: Implement AS_ENABLE_UPDATES STRACE(("ServerWindow %s: Message Enable_Updates unimplemented\n", Title())); break; } case AS_DISABLE_UPDATES: { // TODO: Implement AS_DISABLE_UPDATES STRACE(("ServerWindow %s: Message Disable_Updates unimplemented\n", Title())); break; } case AS_NEEDS_UPDATE: { // TODO: Implement AS_NEEDS_UPDATE STRACE(("ServerWindow %s: Message Needs_Update unimplemented\n", Title())); break; } #endif case AS_SET_WINDOW_TITLE: { char* newTitle; if (link.ReadString(&newTitle) == B_OK) { SetTitle(newTitle); free(newTitle); } break; } case AS_ADD_TO_SUBSET: { STRACE(("ServerWindow %s: Message AS_ADD_TO_SUBSET\n", Title())); WinBorder *windowBorder; int32 mainToken; team_id teamID; link.Read(&mainToken); link.Read(&teamID, sizeof(team_id)); windowBorder = gDesktop->FindWinBorderByClientToken(mainToken, teamID); if (windowBorder) { fLink.StartMessage(SERVER_TRUE); fLink.Flush(); // ToDo: this is a pretty expensive and complicated way to send a message... BPrivate::PortLink msg(-1, -1); msg.StartMessage(AS_ROOTLAYER_ADD_TO_SUBSET); msg.Attach(fWinBorder); msg.Attach(windowBorder); fWinBorder->GetRootLayer()->EnqueueMessage(msg); } else { fLink.StartMessage(SERVER_FALSE); fLink.Flush(); } break; } case AS_REM_FROM_SUBSET: { STRACE(("ServerWindow %s: Message AS_REM_FROM_SUBSET\n", Title())); WinBorder *windowBorder; int32 mainToken; team_id teamID; link.Read(&mainToken); link.Read(&teamID, sizeof(team_id)); windowBorder = gDesktop->FindWinBorderByClientToken(mainToken, teamID); if (windowBorder) { fLink.StartMessage(SERVER_TRUE); fLink.Flush(); BPrivate::PortLink msg(-1, -1); msg.StartMessage(AS_ROOTLAYER_REMOVE_FROM_SUBSET); msg.Attach(fWinBorder); msg.Attach(windowBorder); fWinBorder->GetRootLayer()->EnqueueMessage(msg); } else { fLink.StartMessage(SERVER_FALSE); fLink.Flush(); } break; } #if 0 case AS_SET_LOOK: { // TODO: Implement AS_SET_LOOK STRACE(("ServerWindow %s: Message Set_Look unimplemented\n", Title())); break; } case AS_SET_FLAGS: { // TODO: Implement AS_SET_FLAGS STRACE(("ServerWindow %s: Message Set_Flags unimplemented\n", Title())); break; } case AS_SET_FEEL: { STRACE(("ServerWindow %s: Message AS_SET_FEEL\n", Title())); int32 newFeel; link.Read(&newFeel); if (myRootLayer) myRootLayer->GoChangeWinBorderFeel(fWinBorder, newFeel); break; } case AS_SET_ALIGNMENT: { // TODO: Implement AS_SET_ALIGNMENT STRACE(("ServerWindow %s: Message Set_Alignment unimplemented\n", Title())); break; } case AS_GET_ALIGNMENT: { // TODO: Implement AS_GET_ALIGNMENT STRACE(("ServerWindow %s: Message Get_Alignment unimplemented\n", Title())); break; } #endif case AS_GET_WORKSPACES: { STRACE(("ServerWindow %s: Message Get_Workspaces unimplemented\n", Title())); fLink.StartMessage(SERVER_TRUE); fLink.Attach(fWinBorder->Workspaces()); fLink.Flush(); break; } case AS_SET_WORKSPACES: { // TODO: Implement AS_SET_WORKSPACES STRACE(("ServerWindow %s: Message Set_Workspaces unimplemented\n", Title())); uint32 newWorkspaces; link.Read(&newWorkspaces); BPrivate::PortLink msg(-1, -1); msg.StartMessage(AS_ROOTLAYER_WINBORDER_SET_WORKSPACES); msg.Attach(fWinBorder); msg.Attach(fWinBorder->Workspaces()); msg.Attach(newWorkspaces); fWinBorder->GetRootLayer()->EnqueueMessage(msg); break; } case AS_WINDOW_RESIZE: { float xResizeBy; float yResizeBy; link.Read(&xResizeBy); link.Read(&yResizeBy); STRACE(("ServerWindow %s: Message AS_WINDOW_RESIZE %.1f, %.1f\n", Title(), xResizeBy, yResizeBy)); fWinBorder->ResizeBy(xResizeBy, yResizeBy); break; } case AS_WINDOW_MOVE: { float xMoveBy; float yMoveBy; link.Read(&xMoveBy); link.Read(&yMoveBy); STRACE(("ServerWindow %s: Message AS_WINDOW_MOVE: %.1f, %.1f\n", Title(), xMoveBy, yMoveBy)); fWinBorder->MoveBy(xMoveBy, yMoveBy); break; } case AS_SET_SIZE_LIMITS: { // Attached Data: // 1) float minimum width // 2) float maximum width // 3) float minimum height // 4) float maximum height float minWidth; float maxWidth; float minHeight; float maxHeight; link.Read(&minWidth); link.Read(&maxWidth); link.Read(&minHeight); link.Read(&maxHeight); fWinBorder->SetSizeLimits(minWidth, maxWidth, minHeight, maxHeight); // and now, sync the client to the limits that we were able to enforce fWinBorder->GetSizeLimits(&minWidth, &maxWidth, &minHeight, &maxHeight); fLink.StartMessage(SERVER_TRUE); fLink.Attach(minWidth); fLink.Attach(maxWidth); fLink.Attach(minHeight); fLink.Attach(maxHeight); fLink.Flush(); break; } case B_MINIMIZE: { // TODO: Implement B_MINIMIZE STRACE(("ServerWindow %s: Message Minimize unimplemented\n", Title())); break; } case B_WINDOW_ACTIVATED: { // TODO: Implement B_WINDOW_ACTIVATED STRACE(("ServerWindow %s: Message Window_Activated unimplemented\n", Title())); break; } case B_ZOOM: { // TODO: Implement B_ZOOM STRACE(("ServerWindow %s: Message Zoom unimplemented\n", Title())); break; } // Some BView drawing messages, but which don't need clipping case AS_LAYER_SET_HIGH_COLOR: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_HIGH_COLOR: Layer: %s\n", Title(), fCurrentLayer->Name())); rgb_color c; link.Read(&c, sizeof(rgb_color)); fCurrentLayer->fLayerData->SetHighColor(RGBColor(c)); break; } case AS_LAYER_SET_LOW_COLOR: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_LOW_COLOR: Layer: %s\n", Title(), fCurrentLayer->Name())); rgb_color c; link.Read(&c, sizeof(rgb_color)); fCurrentLayer->fLayerData->SetLowColor(RGBColor(c)); break; } case AS_LAYER_SET_PATTERN: { DTRACE(("ServerWindow %s: Message AS_LAYER_SET_PATTERN: Layer: %s\n", fTitle, fCurrentLayer->Name())); pattern pat; link.Read(&pat, sizeof(pattern)); fCurrentLayer->fLayerData->SetPattern(Pattern(pat)); break; } case AS_MOVEPENTO: { DTRACE(("ServerWindow %s: Message AS_MOVEPENTO\n", Title())); float x,y; link.Read(&x); link.Read(&y); if (fCurrentLayer && fCurrentLayer->fLayerData) fCurrentLayer->fLayerData->SetPenLocation(BPoint(x, y)); break; } case AS_SETPENSIZE: { DTRACE(("ServerWindow %s: Message AS_SETPENSIZE\n", Title())); float size; link.Read(&size); if (fCurrentLayer && fCurrentLayer->fLayerData) fCurrentLayer->fLayerData->SetPenSize(size); break; } case AS_SET_FONT: { DTRACE(("ServerWindow %s: Message AS_SET_FONT\n", Title())); // TODO: Implement AS_SET_FONT? // Confusing!! But it works already! break; } case AS_SET_FONT_SIZE: { DTRACE(("ServerWindow %s: Message AS_SET_FONT_SIZE\n", Title())); // TODO: Implement AS_SET_FONT_SIZE? break; } case AS_AREA_MESSAGE: { STRACE(("ServerWindow %s: Message AS_AREA_MESSAGE\n", Title())); // This occurs in only one kind of case: a message is too big to send over a port. This // is really an edge case, so this shouldn't happen *too* often // Attached Data: // 1) area_id id of an area already owned by the server containing the message // 2) size_t offset of the pointer in the area // 3) size_t size of the message area_id area; size_t offset; size_t msgsize; area_info ai; int8 *msgpointer; link.Read(&area); link.Read(&offset); link.Read(&msgsize); // Part sanity check, part get base pointer :) if(get_area_info(area,&ai)!=B_OK) break; msgpointer = (int8*)ai.address + offset; RAMLinkMsgReader mlink(msgpointer); _DispatchMessage(mlink.Code(), mlink); // This is a very special case in the sense that when ServerMemIO is used for this // purpose, it will be set to NOT automatically free the memory which it had // requested. This is the server's job once the message has been dispatched. fServerApp->AppAreaPool()->ReleaseBuffer(msgpointer); break; } case AS_SYNC: { // TODO: AS_SYNC is a no-op for now, just to get things working fLink.StartMessage(SERVER_TRUE); fLink.Flush(); break; } case AS_LAYER_DRAG_IMAGE: { // TODO: Implement AS_LAYER_DRAG_IMAGE STRACE(("ServerWindow %s: Message AS_DRAG_IMAGE unimplemented\n", Title())); DTRACE(("ServerWindow %s: Message AS_DRAG_IMAGE unimplemented\n", Title())); break; } case AS_LAYER_DRAG_RECT: { // TODO: Implement AS_LAYER_DRAG_RECT STRACE(("ServerWindow %s: Message AS_DRAG_RECT unimplemented\n", Title())); DTRACE(("ServerWindow %s: Message AS_DRAG_RECT unimplemented\n", Title())); break; } case AS_LAYER_GET_MOUSE_COORDS: { DTRACE(("ServerWindow %s: Message AS_GET_MOUSE_COORDS\n", fTitle)); fLink.StartMessage(SERVER_TRUE); // Returns // 1) BPoint mouse location // 2) int32 button state fLink.Attach(gDesktop->GetHWInterface()->GetCursorPosition()); fLink.Attach(gDesktop->ActiveRootLayer()->Buttons()); fLink.Flush(); break; } default: _DispatchGraphicsMessage(code, link); break; } if (myRootLayer) myRootLayer->Unlock(); } // -------------------- Graphics messages ---------------------------------- void ServerWindow::_DispatchGraphicsMessage(int32 code, BPrivate::LinkReceiver &link) { #ifndef NEW_CLIPPING BRegion rreg(fCurrentLayer->fVisible); #else BRegion rreg(fCurrentLayer->fVisible2); #endif if (fWinBorder->InUpdate()) rreg.IntersectWith(&fWinBorder->RegionToBeUpdated()); DisplayDriver* driver = fWinBorder->GetDisplayDriver(); driver->ConstrainClippingRegion(&rreg); // rgb_color rrr = fCurrentLayer->fLayerData->viewcolor.GetColor32(); // RGBColor c(rand()%255,rand()%255,rand()%255); // driver->FillRect(BRect(0,0,639,479), c); switch (code) { case AS_STROKE_LINE: { DTRACE(("ServerWindow %s: Message AS_STROKE_LINE\n", Title())); float x1, y1, x2, y2; link.Read(&x1); link.Read(&y1); link.Read(&x2); link.Read(&y2); if (fCurrentLayer && fCurrentLayer->fLayerData) { BPoint p1(x1,y1); BPoint p2(x2,y2); driver->StrokeLine(fCurrentLayer->ConvertToTop(p1), fCurrentLayer->ConvertToTop(p2), fCurrentLayer->fLayerData); // We update the pen here because many DisplayDriver calls which do not update the // pen position actually call StrokeLine // TODO: Decide where to put this, for example, it cannot be done // for DrawString(), also there needs to be a decision, if penlocation // is in View coordinates (I think it should be) or in screen coordinates. fCurrentLayer->fLayerData->SetPenLocation(p2); } break; } case AS_LAYER_INVERT_RECT: { DTRACE(("ServerWindow %s: Message AS_INVERT_RECT\n", Title())); BRect rect; link.Read(&rect); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->InvertRect(fCurrentLayer->ConvertToTop(rect)); break; } case AS_STROKE_RECT: { DTRACE(("ServerWindow %s: Message AS_STROKE_RECT\n", Title())); float left, top, right, bottom; link.Read(&left); link.Read(&top); link.Read(&right); link.Read(&bottom); BRect rect(left,top,right,bottom); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->StrokeRect(fCurrentLayer->ConvertToTop(rect), fCurrentLayer->fLayerData); break; } case AS_FILL_RECT: { DTRACE(("ServerWindow %s: Message AS_FILL_RECT\n", Title())); BRect rect; link.Read(&rect); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->FillRect(fCurrentLayer->ConvertToTop(rect), fCurrentLayer->fLayerData); break; } case AS_LAYER_DRAW_BITMAP_SYNC_AT_POINT: { DTRACE(("ServerWindow %s: Message AS_LAYER_DRAW_BITMAP_SYNC_AT_POINT: Layer name: %s\n", fTitle, fCurrentLayer->Name())); int32 bitmapToken; BPoint point; link.Read(&bitmapToken); link.Read(&point); ServerBitmap* sbmp = fServerApp->FindBitmap(bitmapToken); if (sbmp) { BRect src = sbmp->Bounds(); BRect dst = src.OffsetToCopy(point); dst = fCurrentLayer->ConvertToTop(dst); driver->DrawBitmap(sbmp, src, dst, fCurrentLayer->fLayerData); } // TODO: Adi -- shouldn't AS_LAYER_DRAW_BITMAP_SYNC_AT_POINT sync with the client? break; } case AS_LAYER_DRAW_BITMAP_ASYNC_AT_POINT: { DTRACE(("ServerWindow %s: Message AS_LAYER_DRAW_BITMAP_ASYNC_AT_POINT: Layer name: %s\n", fTitle, fCurrentLayer->Name())); int32 bitmapToken; BPoint point; link.Read(&bitmapToken); link.Read(&point); ServerBitmap* sbmp = fServerApp->FindBitmap(bitmapToken); if (sbmp) { BRect src = sbmp->Bounds(); BRect dst = src.OffsetToCopy(point); dst = fCurrentLayer->ConvertToTop(dst); driver->DrawBitmap(sbmp, src, dst, fCurrentLayer->fLayerData); } break; } case AS_LAYER_DRAW_BITMAP_SYNC_IN_RECT: { DTRACE(("ServerWindow %s: Message AS_LAYER_DRAW_BITMAP_SYNC_IN_RECT: Layer name: %s\n", fTitle, fCurrentLayer->Name())); int32 bitmapToken; BRect srcRect, dstRect; link.Read(&bitmapToken); link.Read(&dstRect); link.Read(&srcRect); ServerBitmap* sbmp = fServerApp->FindBitmap(bitmapToken); if (sbmp) { dstRect = fCurrentLayer->ConvertToTop(dstRect); driver->DrawBitmap(sbmp, srcRect, dstRect, fCurrentLayer->fLayerData); } // TODO: Adi -- shouldn't AS_LAYER_DRAW_BITMAP_SYNC_IN_RECT sync with the client? break; } case AS_LAYER_DRAW_BITMAP_ASYNC_IN_RECT: { DTRACE(("ServerWindow %s: Message AS_LAYER_DRAW_BITMAP_ASYNC_IN_RECT: Layer name: %s\n", fTitle, fCurrentLayer->Name())); int32 bitmapToken; BRect srcRect, dstRect; link.Read(&bitmapToken); link.Read(&dstRect); link.Read(&srcRect); ServerBitmap* sbmp = fServerApp->FindBitmap(bitmapToken); if (sbmp) { dstRect = fCurrentLayer->ConvertToTop(dstRect); driver->DrawBitmap(sbmp, srcRect, dstRect, fCurrentLayer->fLayerData); } break; } case AS_STROKE_ARC: { DTRACE(("ServerWindow %s: Message AS_STROKE_ARC\n", Title())); float angle, span; BRect r; link.Read(&r); link.Read(&angle); link.Read(&span); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->StrokeArc(fCurrentLayer->ConvertToTop(r),angle,span, fCurrentLayer->fLayerData); break; } case AS_FILL_ARC: { DTRACE(("ServerWindow %s: Message AS_FILL_ARC\n", Title())); float angle, span; BRect r; link.Read(&r); link.Read(&angle); link.Read(&span); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->FillArc(fCurrentLayer->ConvertToTop(r),angle,span, fCurrentLayer->fLayerData); break; } case AS_STROKE_BEZIER: { DTRACE(("ServerWindow %s: Message AS_STROKE_BEZIER\n", Title())); BPoint *pts; int i; pts = new BPoint[4]; for (i=0; i<4; i++) link.Read(&(pts[i])); if (fCurrentLayer && fCurrentLayer->fLayerData) { for (i=0; i<4; i++) pts[i]=fCurrentLayer->ConvertToTop(pts[i]); driver->StrokeBezier(pts, fCurrentLayer->fLayerData); } delete [] pts; break; } case AS_FILL_BEZIER: { DTRACE(("ServerWindow %s: Message AS_FILL_BEZIER\n", Title())); BPoint *pts; int i; pts = new BPoint[4]; for (i=0; i<4; i++) link.Read(&(pts[i])); if (fCurrentLayer && fCurrentLayer->fLayerData) { for (i=0; i<4; i++) pts[i]=fCurrentLayer->ConvertToTop(pts[i]); driver->FillBezier(pts, fCurrentLayer->fLayerData); } delete [] pts; break; } case AS_STROKE_ELLIPSE: { DTRACE(("ServerWindow %s: Message AS_STROKE_ELLIPSE\n", Title())); BRect rect; link.Read(&rect); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->StrokeEllipse(fCurrentLayer->ConvertToTop(rect), fCurrentLayer->fLayerData); break; } case AS_FILL_ELLIPSE: { DTRACE(("ServerWindow %s: Message AS_FILL_ELLIPSE\n", Title())); BRect rect; link.Read(&rect); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->FillEllipse(fCurrentLayer->ConvertToTop(rect), fCurrentLayer->fLayerData); break; } case AS_STROKE_ROUNDRECT: { DTRACE(("ServerWindow %s: Message AS_STROKE_ROUNDRECT\n", Title())); BRect rect; float xrad,yrad; link.Read(&rect); link.Read(&xrad); link.Read(&yrad); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->StrokeRoundRect(fCurrentLayer->ConvertToTop(rect),xrad,yrad, fCurrentLayer->fLayerData); break; } case AS_FILL_ROUNDRECT: { DTRACE(("ServerWindow %s: Message AS_FILL_ROUNDRECT\n", Title())); BRect rect; float xrad,yrad; link.Read(&rect); link.Read(&xrad); link.Read(&yrad); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->FillRoundRect(fCurrentLayer->ConvertToTop(rect),xrad,yrad, fCurrentLayer->fLayerData); break; } case AS_STROKE_TRIANGLE: { DTRACE(("ServerWindow %s: Message AS_STROKE_TRIANGLE\n", Title())); BPoint pts[3]; BRect rect; for (int i = 0; i < 3; i++) link.Read(&(pts[i])); link.Read(&rect); if (fCurrentLayer && fCurrentLayer->fLayerData) { for (int i = 0;i < 3; i++) pts[i] = fCurrentLayer->ConvertToTop(pts[i]); driver->StrokeTriangle(pts, fCurrentLayer->ConvertToTop(rect), fCurrentLayer->fLayerData); } break; } case AS_FILL_TRIANGLE: { DTRACE(("ServerWindow %s: Message AS_FILL_TRIANGLE\n", Title())); BPoint pts[3]; BRect rect; for (int i = 0; i < 3; i++) link.Read(&(pts[i])); link.Read(&rect); if (fCurrentLayer && fCurrentLayer->fLayerData) { for (int i = 0; i < 3; i++) pts[i] = fCurrentLayer->ConvertToTop(pts[i]); driver->FillTriangle(pts, fCurrentLayer->ConvertToTop(rect), fCurrentLayer->fLayerData); } break; } // TODO: get rid of all this code duplication!! case AS_STROKE_POLYGON: { DTRACE(("ServerWindow %s: Message AS_STROKE_POLYGON\n", Title())); BRect polyframe; bool isclosed; int32 pointcount; BPoint *pointlist; link.Read(&polyframe); link.Read(&isclosed); link.Read(&pointcount); pointlist = new BPoint[pointcount]; link.Read(pointlist, sizeof(BPoint)*pointcount); for (int32 i = 0; i < pointcount; i++) pointlist[i] = fCurrentLayer->ConvertToTop(pointlist[i]); driver->StrokePolygon(pointlist,pointcount,polyframe, fCurrentLayer->fLayerData,isclosed); delete [] pointlist; break; } case AS_FILL_POLYGON: { DTRACE(("ServerWindow %s: Message AS_FILL_POLYGON\n", Title())); BRect polyframe; int32 pointcount; BPoint *pointlist; link.Read(&polyframe); link.Read(&pointcount); pointlist = new BPoint[pointcount]; link.Read(pointlist, sizeof(BPoint) * pointcount); for (int32 i = 0; i < pointcount; i++) pointlist[i] = fCurrentLayer->ConvertToTop(pointlist[i]); driver->FillPolygon(pointlist,pointcount,polyframe, fCurrentLayer->fLayerData); delete [] pointlist; break; } case AS_STROKE_SHAPE: { DTRACE(("ServerWindow %s: Message AS_STROKE_SHAPE\n", Title())); BRect shaperect; int32 opcount; int32 ptcount; int32 *oplist; BPoint *ptlist; link.Read(&shaperect); link.Read(&opcount); link.Read(&ptcount); oplist = new int32[opcount]; ptlist = new BPoint[ptcount]; link.Read(oplist, sizeof(int32) * opcount); link.Read(ptlist, sizeof(BPoint) * ptcount); for (int32 i = 0; i < ptcount; i++) ptlist[i] = fCurrentLayer->ConvertToTop(ptlist[i]); driver->StrokeShape(shaperect, opcount, oplist, ptcount, ptlist, fCurrentLayer->fLayerData); delete[] oplist; delete[] ptlist; break; } case AS_FILL_SHAPE: { DTRACE(("ServerWindow %s: Message AS_FILL_SHAPE\n", Title())); BRect shaperect; int32 opcount; int32 ptcount; int32 *oplist; BPoint *ptlist; link.Read(&shaperect); link.Read(&opcount); link.Read(&ptcount); oplist = new int32[opcount]; ptlist = new BPoint[ptcount]; link.Read(oplist, sizeof(int32) * opcount); link.Read(ptlist, sizeof(BPoint) * ptcount); for (int32 i = 0; i < ptcount; i++) ptlist[i] = fCurrentLayer->ConvertToTop(ptlist[i]); driver->FillShape(shaperect, opcount, oplist, ptcount, ptlist, fCurrentLayer->fLayerData); delete[] oplist; delete[] ptlist; break; } case AS_FILL_REGION: { DTRACE(("ServerWindow %s: Message AS_FILL_REGION\n", Title())); int32 count; link.Read(&count); BRect *rects = new BRect[count]; if (link.Read(rects, sizeof(BRect) * count) != B_OK) { delete[] rects; break; } // Between the client-side conversion to BRects from clipping_rects to the overhead // in repeatedly calling FillRect(), this is definitely in need of optimization. At // least it works for now. :) BRegion region; for (int32 i = 0; i < count; i++) { region.Include(fCurrentLayer->ConvertToTop(rects[i])); } driver->FillRegion(region, fCurrentLayer->fLayerData); delete[] rects; // TODO: create support for clipping_rect usage for faster BRegion display. // Tweaks to DisplayDriver are necessary along with conversion routines in Layer break; } case AS_STROKE_LINEARRAY: { DTRACE(("ServerWindow %s: Message AS_STROKE_LINEARRAY\n", Title())); // Attached Data: // 1) int32 Number of lines in the array // 2) array of struct _array_data_ objects, as defined in ViewAux.h int32 linecount; link.Read(&linecount); if (linecount > 0) { LineArrayData linedata[linecount], *index; for (int32 i = 0; i < linecount; i++) { index = &linedata[i]; link.Read(&(index->pt1.x)); link.Read(&(index->pt1.y)); link.Read(&(index->pt2.x)); link.Read(&(index->pt2.y)); link.Read(&(index->color)); index->pt1 = fCurrentLayer->ConvertToTop(index->pt1); index->pt2 = fCurrentLayer->ConvertToTop(index->pt2); } driver->StrokeLineArray(linecount,linedata,fCurrentLayer->fLayerData); } break; } case AS_DRAW_STRING: { DTRACE(("ServerWindow %s: Message AS_DRAW_STRING\n", Title())); char *string; int32 length; BPoint location; escapement_delta delta; link.Read(&length); link.Read(&location); link.Read(&delta); link.ReadString(&string); if (fCurrentLayer && fCurrentLayer->fLayerData) driver->DrawString(string, length, fCurrentLayer->ConvertToTop(location), fCurrentLayer->fLayerData); free(string); break; } case AS_LAYER_BEGIN_PICTURE: CRITICAL("AS_LAYER_BEGIN_PICTURE not implemented\n"); break; case AS_LAYER_APPEND_TO_PICTURE: CRITICAL("AS_LAYER_APPEND_TO_PICTURE not implemented\n"); break; case AS_LAYER_END_PICTURE: CRITICAL("AS_LAYER_END_PICTURE not implemented\n"); break; default: printf("ServerWindow %s received unexpected code - message offset %ld\n", Title(), code - SERVER_TRUE); if (link.NeedsReply()) { // the client is now blocking and waiting for a reply! fLink.StartMessage(SERVER_FALSE); fLink.Flush(); } break; } driver->ConstrainClippingRegion(NULL); } /*! \brief Message-dispatching loop starter \param data The thread's ServerWindow */ int32 ServerWindow::_message_thread(void *_window) { ServerWindow *window = (ServerWindow *)_window; window->_MessageLooper(); return 0; } /*! \brief Message-dispatching loop for the ServerWindow Watches the ServerWindow's message port and dispatches as necessary */ void ServerWindow::_MessageLooper() { BPrivate::LinkReceiver& receiver = fLink.Receiver(); bool quitting = false; int32 code; status_t err = B_OK; while (!quitting) { STRACE(("info: ServerWindow::MonitorWin listening on port %ld.\n", fMessagePort)); err = receiver.GetNextMessage(code); if (err < B_OK) { // that shouldn't happen, it's our port // ToDo: do something about it, anyway! return; } Lock(); switch (code) { case AS_DELETE_WINDOW: case kMsgWindowQuit: { // this means the client has been killed STRACE(("ServerWindow %s received 'AS_DELETE_WINDOW' message code\n", Title())); // ToDo: what's this? //RootLayer *rootLayer = fWinBorder->GetRootLayer(); // we are preparing to delete a ServerWindow, RootLayer should be aware // of that and stop for a moment. // also we must wait a bit for the associated WinBorder to become hidden //while(1) { // myRootLayer->Lock(); // if (IsHidden()) // break; // else // rootLayer->Unlock(); //} // ServerWindow's destructor takes care of pulling this object off the desktop. if (!fWinBorder->IsHidden()) CRITICAL("ServerWindow: a window must be hidden before it's deleted\n"); Quit(); // does not return break; } case B_QUIT_REQUESTED: STRACE(("ServerWindow %s received quit request\n", Title())); NotifyQuitRequested(); break; default: _DispatchMessage(code, receiver); break; } Unlock(); } } /* void ServerWindow::_CopyBits(RootLayer* rootLayer, Layer* layer, BRect& src, BRect& dst, int32 xOffset, int32 yOffset) const { // NOTE: The correct behaviour is this: // * The region that is copied is the // src rectangle, no matter if it fits // into the dst rectangle. It is copied // by the offset dst.LeftTop() - src.LeftTop() // * The dst rectangle is used for invalidation: // Any area in the dst rectangle that could // not be copied from src (because either the // src rectangle was not big enough, or because there // were parts cut off by the current layer clipping), // are triggering BView::Draw() to be called // and for these parts only. #ifndef NEW_CLIPPING // the region that is going to be copied BRegion copyRegion(src); // apply the current clipping of the layer copyRegion.IntersectWith(&layer->fVisible); // offset the region to the destination // and apply the current clipping there as well copyRegion.OffsetBy(xOffset, yOffset); copyRegion.IntersectWith(&layer->fVisible); // the region at the destination that needs invalidation BRegion invalidRegion(dst); // exclude the region drawn by the copy operation invalidRegion.Exclude(©Region); // apply the current clipping as well invalidRegion.IntersectWith(&layer->fVisible); // move the region back for the actual operation copyRegion.OffsetBy(-xOffset, -yOffset); layer->GetDisplayDriver()->CopyRegion(©Region, xOffset, yOffset); // trigger the redraw if (rootLayer) { // the following code solves a "concurrency" problem: // since the scrolling might happen more often // than redrawing, we need to keep track of the region // pending for redraw that might fall into the area // that is scrolled. BRegion scrolledInvalid(fWinBorder->CulmulatedUpdateRegion()); scrolledInvalid.IntersectWith(&layer->fVisible); if (scrolledInvalid.Frame().IsValid()) { //printf("the layer has pending updates that will be scrolled\n"); scrolledInvalid.OffsetBy(xOffset, yOffset); invalidRegion.Include(&scrolledInvalid); } rootLayer->GoRedraw(fWinBorder, invalidRegion); } #endif }*/ void ServerWindow::SendMessageToClient(const BMessage* msg, int32 target, bool usePreferred) const { ssize_t size = msg->FlattenedSize(); char* buffer = new char[size]; if (msg->Flatten(buffer, size) == B_OK) { status_t ret = BMessage::Private::SendFlattenedMessage(buffer, size, fClientLooperPort, target, usePreferred, 100000); if (ret < B_OK) fprintf(stderr, "ServerWindow::SendMessageToClient(): %s\n", strerror(ret)); } else printf("PANIC: ServerWindow %s: can't flatten message in 'SendMessageToClient()'\n", fTitle); delete[] buffer; } // MakeWinBorder WinBorder* ServerWindow::MakeWinBorder(BRect frame, const char* name, uint32 look, uint32 feel, uint32 flags, uint32 workspace) { // The non-offscreen ServerWindow uses the DisplayDriver instance from the desktop. return new(nothrow) WinBorder(frame, name, look, feel, flags, workspace, this, gDesktop->GetDisplayDriver()); } status_t ServerWindow::PictureToRegion(ServerPicture *picture, BRegion ®ion, bool inverse, BPoint where) { fprintf(stderr, "ServerWindow::PictureToRegion() not implemented\n"); region.MakeEmpty(); return B_ERROR; }