/* Haiku window system support. Hey, Emacs, this is -*- C++ -*- Copyright (C) 2021-2024 Free Software Foundation, Inc. This file is part of GNU Emacs. GNU Emacs is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. GNU Emacs is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNU Emacs. If not, see . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef USE_BE_CAIRO #include #endif #include "haiku_support.h" /* Some messages that Emacs sends to itself. */ enum { SCROLL_BAR_UPDATE = 3000, WAIT_FOR_RELEASE = 3001, RELEASE_NOW = 3002, CANCEL_DROP = 3003, SHOW_MENU_BAR = 3004, BE_MENU_BAR_OPEN = 3005, QUIT_APPLICATION = 3006, REPLAY_MENU_BAR = 3007, FONT_FAMILY_SELECTED = 3008, FONT_STYLE_SELECTED = 3009, FILE_PANEL_SELECTION = 3010, QUIT_PREVIEW_DIALOG = 3011, SET_FONT_INDICES = 3012, SET_PREVIEW_DIALOG = 3013, UPDATE_PREVIEW_DIALOG = 3014, SEND_MOVE_FRAME_EVENT = 3015, SET_DISABLE_ANTIALIASING = 3016, }; /* X11 keysyms that we use. */ enum { KEY_BACKSPACE = 0xff08, KEY_TAB = 0xff09, KEY_RETURN = 0xff0d, KEY_PAUSE = 0xff13, KEY_ESCAPE = 0xff1b, KEY_DELETE = 0xffff, KEY_HOME = 0xff50, KEY_LEFT_ARROW = 0xff51, KEY_UP_ARROW = 0xff52, KEY_RIGHT_ARROW = 0xff53, KEY_DOWN_ARROW = 0xff54, KEY_PAGE_UP = 0xff55, KEY_PAGE_DOWN = 0xff56, KEY_END = 0xff57, KEY_PRINT = 0xff61, KEY_INSERT = 0xff63, /* This is used to indicate the first function key. */ KEY_F1 = 0xffbe, /* These are found on some multilingual keyboards. */ KEY_HANGUL = 0xff31, KEY_HANGUL_HANJA = 0xff34, KEY_HIRIGANA_KATAGANA = 0xff27, KEY_ZENKAKU_HANKAKU = 0xff2a, }; struct font_selection_dialog_message { /* Whether or not font selection was canceled. */ bool_bf cancel : 1; /* Whether or not a size was explicitly specified. */ bool_bf size_specified : 1; /* Whether or not antialiasing should be disabled. */ bool_bf disable_antialias : 1; /* The index of the selected font family. */ int family_idx; /* The index of the selected font style. */ int style_idx; /* The selected font size. */ int size; }; /* The color space of the main screen. B_NO_COLOR_SPACE means it has not yet been computed. */ static color_space dpy_color_space = B_NO_COLOR_SPACE; /* The keymap, or NULL if it has not been initialized. */ static key_map *key_map; /* Indices of characters into the keymap. */ static char *key_chars; /* Lock around keymap data, since it's touched from different threads. */ static BLocker key_map_lock; /* The locking semantics of BWindows running in multiple threads are so complex that child frame state (which is the only state that is shared between different BWindows at runtime) does best with a single global lock. */ static BLocker child_frame_lock; /* Variable where the popup menu thread returns the chosen menu item. */ static BMessage volatile *popup_track_message; /* Variable in which alert dialog threads return the selected button number. */ static int32 volatile alert_popup_value; /* The view that has the passive grab. */ static void *grab_view; /* The locker for that variable. */ static BLocker grab_view_locker; /* Whether or not a drag-and-drop operation is in progress. */ static bool drag_and_drop_in_progress; /* Many places require us to lock the child frame data, and then lock the locker of some random window. Unfortunately, locking such a window might be delayed due to an arriving message, which then calls a callback inside that window that tries to lock the child frame data but doesn't finish since the child frame lock is already held, not letting the code that held the child frame lock proceed, thereby causing a deadlock. Rectifying that problem is simple: all code in a looper callback must lock the child frame data with this macro instead. IOW, if some other code is already running with the child frame lock held, don't interfere: wait until it's finished before continuing. */ #define CHILD_FRAME_LOCK_INSIDE_LOOPER_CALLBACK \ if (child_frame_lock.LockWithTimeout (200) != B_OK) \ { \ /* The Haiku equivalent of XPutBackEvent. */ \ if (CurrentMessage ()) \ PostMessage (CurrentMessage ()); \ } \ else /* This could be a private API, but it's used by (at least) the Qt port, so it's probably here to stay. */ extern status_t get_subpixel_antialiasing (bool *); /* The ID of the thread the BApplication is running in. */ static thread_id app_thread; _Noreturn void gui_abort (const char *msg) { fprintf (stderr, "Abort in GUI code: %s\n", msg); fprintf (stderr, "Under Haiku, Emacs cannot recover from errors in GUI code\n"); fprintf (stderr, "App Server disconnects usually manifest as bitmap " "initialization failures or lock failures."); abort (); } struct be_popup_menu_data { int x, y; BPopUpMenu *menu; }; static int32 be_popup_menu_thread_entry (void *thread_data) { struct be_popup_menu_data *data; struct haiku_dummy_event dummy; BMenuItem *it; data = (struct be_popup_menu_data *) thread_data; it = data->menu->Go (BPoint (data->x, data->y)); if (it) popup_track_message = it->Message (); else popup_track_message = NULL; haiku_write (DUMMY_EVENT, &dummy); return 0; } /* Convert a raw character RAW produced by the keycode KEY into a key symbol and place it in KEYSYM. If RAW cannot be converted into a keysym, value is 0. If RAW can be converted into a keysym, but it should be ignored, value is -1. Any other value means success, and that the keysym should be used instead of mapping the keycode into a character. */ static int keysym_from_raw_char (int32 raw, int32 key, unsigned *code) { switch (raw) { case B_BACKSPACE: *code = KEY_BACKSPACE; break; case B_RETURN: *code = KEY_RETURN; break; case B_TAB: *code = KEY_TAB; break; case B_ESCAPE: *code = KEY_ESCAPE; break; case B_LEFT_ARROW: *code = KEY_LEFT_ARROW; break; case B_RIGHT_ARROW: *code = KEY_RIGHT_ARROW; break; case B_UP_ARROW: *code = KEY_UP_ARROW; break; case B_DOWN_ARROW: *code = KEY_DOWN_ARROW; break; case B_INSERT: *code = KEY_INSERT; break; case B_DELETE: *code = KEY_DELETE; break; case B_HOME: *code = KEY_HOME; break; case B_END: *code = KEY_END; break; case B_PAGE_UP: *code = KEY_PAGE_UP; break; case B_PAGE_DOWN: *code = KEY_PAGE_DOWN; break; case B_FUNCTION_KEY: *code = KEY_F1 + key - 2; if (*code - KEY_F1 == 12) *code = KEY_PRINT; else if (*code - KEY_F1 == 13) /* Okay, Scroll Lock is a bit too much: keyboard.c doesn't know about it yet, and it shouldn't, since that's a modifier key. *code = KEY_SCROLL_LOCK; */ return -1; else if (*code - KEY_F1 == 14) *code = KEY_PAUSE; break; case B_HANGUL: *code = KEY_HANGUL; break; case B_HANGUL_HANJA: *code = KEY_HANGUL_HANJA; break; case B_KATAKANA_HIRAGANA: *code = KEY_HIRIGANA_KATAGANA; break; case B_HANKAKU_ZENKAKU: *code = KEY_ZENKAKU_HANKAKU; break; default: return 0; } return 1; } static void map_key (char *chars, int32 offset, uint32_t *c) { int size = chars[offset++]; switch (size) { case 0: break; case 1: *c = chars[offset]; break; default: { char str[5]; int i = (size <= 4) ? size : 4; strncpy (str, &(chars[offset]), i); str[i] = '0'; *c = BUnicodeChar::FromUTF8 ((char *) &str); break; } } } static void map_shift (uint32_t kc, uint32_t *ch) { if (!key_map_lock.Lock ()) gui_abort ("Failed to lock keymap"); if (!key_map) get_key_map (&key_map, &key_chars); if (!key_map) return; if (kc >= 128) return; int32_t m = key_map->shift_map[kc]; map_key (key_chars, m, ch); key_map_lock.Unlock (); } static void map_caps (uint32_t kc, uint32_t *ch) { if (!key_map_lock.Lock ()) gui_abort ("Failed to lock keymap"); if (!key_map) get_key_map (&key_map, &key_chars); if (!key_map) return; if (kc >= 128) return; int32_t m = key_map->caps_map[kc]; map_key (key_chars, m, ch); key_map_lock.Unlock (); } static void map_caps_shift (uint32_t kc, uint32_t *ch) { if (!key_map_lock.Lock ()) gui_abort ("Failed to lock keymap"); if (!key_map) get_key_map (&key_map, &key_chars); if (!key_map) return; if (kc >= 128) return; int32_t m = key_map->caps_shift_map[kc]; map_key (key_chars, m, ch); key_map_lock.Unlock (); } static void map_normal (uint32_t kc, uint32_t *ch) { if (!key_map_lock.Lock ()) gui_abort ("Failed to lock keymap"); if (!key_map) get_key_map (&key_map, &key_chars); if (!key_map) return; if (kc >= 128) return; int32_t m = key_map->normal_map[kc]; map_key (key_chars, m, ch); key_map_lock.Unlock (); } static BRect get_zoom_rect (BWindow *window) { BScreen screen; BDeskbar deskbar; BRect screen_frame; BRect frame; BRect deskbar_frame; BRect window_frame; BRect decorator_frame; if (!screen.IsValid ()) gui_abort ("Failed to calculate screen rect"); screen_frame = frame = screen.Frame (); deskbar_frame = deskbar.Frame (); if (!(modifiers () & B_SHIFT_KEY) && !deskbar.IsAutoHide ()) { switch (deskbar.Location ()) { case B_DESKBAR_TOP: frame.top = deskbar_frame.bottom + 2; break; case B_DESKBAR_BOTTOM: case B_DESKBAR_LEFT_BOTTOM: case B_DESKBAR_RIGHT_BOTTOM: frame.bottom = deskbar_frame.top - 2; break; case B_DESKBAR_LEFT_TOP: if (!deskbar.IsExpanded ()) frame.top = deskbar_frame.bottom + 2; else if (!deskbar.IsAlwaysOnTop () && !deskbar.IsAutoRaise ()) frame.left = deskbar_frame.right + 2; break; default: if (deskbar.IsExpanded () && !deskbar.IsAlwaysOnTop () && !deskbar.IsAutoRaise ()) frame.right = deskbar_frame.left - 2; } } if (window) { window_frame = window->Frame (); decorator_frame = window->DecoratorFrame (); frame.top += (window_frame.top - decorator_frame.top); frame.bottom -= (decorator_frame.bottom - window_frame.bottom); frame.left += (window_frame.left - decorator_frame.left); frame.right -= (decorator_frame.right - window_frame.right); if (frame.top > deskbar_frame.bottom || frame.bottom < deskbar_frame.top) { frame.left = screen_frame.left + (window_frame.left - decorator_frame.left); frame.right = screen_frame.right - (decorator_frame.right - window_frame.right); } } return frame; } /* Invisible window used to get B_SCREEN_CHANGED events. */ class EmacsScreenChangeMonitor : public BWindow { BRect previous_screen_frame; public: EmacsScreenChangeMonitor (void) : BWindow (BRect (-100, -100, 0, 0), "", B_NO_BORDER_WINDOW_LOOK, B_FLOATING_ALL_WINDOW_FEEL, B_AVOID_FRONT | B_AVOID_FOCUS) { BScreen screen (this); if (!screen.IsValid ()) return; previous_screen_frame = screen.Frame (); /* Immediately show this window upon creation. It will not steal the focus or become visible. */ Show (); if (!LockLooper ()) return; Hide (); UnlockLooper (); } void DispatchMessage (BMessage *msg, BHandler *handler) { struct haiku_screen_changed_event rq; BRect frame; if (msg->what == B_SCREEN_CHANGED) { if (msg->FindInt64 ("when", &rq.when) != B_OK) rq.when = 0; if (msg->FindRect ("frame", &frame) != B_OK || frame != previous_screen_frame) { haiku_write (SCREEN_CHANGED_EVENT, &rq); if (frame.IsValid ()) previous_screen_frame = frame; } } BWindow::DispatchMessage (msg, handler); } }; #if 0 /* Return the ID of this team. */ static team_id my_team_id (void) { thread_id id; thread_info info; id = find_thread (NULL); get_thread_info (id, &info); return info.team; } #endif /* 0 */ class Emacs : public BApplication { public: BMessage settings; bool settings_valid_p; EmacsScreenChangeMonitor *monitor; Emacs (void) : BApplication ("application/x-vnd.GNU-emacs"), settings_valid_p (false) { BPath settings_path; if (find_directory (B_USER_SETTINGS_DIRECTORY, &settings_path) != B_OK) return; settings_path.Append (PACKAGE_NAME); BEntry entry (settings_path.Path ()); BFile settings_file (&entry, B_READ_ONLY | B_CREATE_FILE); if (settings.Unflatten (&settings_file) != B_OK) return; settings_valid_p = true; monitor = new EmacsScreenChangeMonitor; } ~Emacs (void) { if (monitor->LockLooper ()) monitor->Quit (); else delete monitor; } void AboutRequested (void) { BAlert *about = new BAlert (PACKAGE_NAME, PACKAGE_STRING "\nThe extensible, self-documenting, " "real-time display editor.", "Close"); about->Go (); } bool QuitRequested (void) { struct haiku_app_quit_requested_event rq; struct haiku_session_manager_reply reply; int32 reply_type; haiku_write (APP_QUIT_REQUESTED_EVENT, &rq); if (read_port (port_emacs_to_session_manager, &reply_type, &reply, sizeof reply) < B_OK) /* Return true so the system kills us, since there's no real alternative if this read fails. */ return true; return reply.quit_reply; } void MessageReceived (BMessage *msg) { struct haiku_clipboard_changed_event rq; if (msg->what == QUIT_APPLICATION) Quit (); else if (msg->what == B_CLIPBOARD_CHANGED) haiku_write (CLIPBOARD_CHANGED_EVENT, &rq); else if (msg->what == B_KEY_MAP_LOADED) { /* Install the new keymap. Or rather, clear key_map -- Emacs will fetch it again from the main thread the next time it is needed. */ if (key_map_lock.Lock ()) { if (key_map) free (key_map); if (key_chars) free (key_chars); key_map = NULL; key_chars = NULL; key_map_lock.Unlock (); } } else BApplication::MessageReceived (msg); } /* The code below doesn't function; see `be_display_notification' for further specifics. */ #if 0 void ArgvReceived (int32 argc, char **argv) { struct haiku_notification_click_event rq; intmax_t id; team_id team; /* ArgvReceived is called after Emacs is first started, with each command line argument passed to Emacs. It is, moreover, called with ARGC set to 1 and ARGV[0] a string starting with -Notification, after a notification is clicked. This string both incorporates the team ID and the notification ID. */ if (argc == 1 && sscanf (argv[0], "-Notification,%d.%jd", &team, &id) == 2) { /* Since this is a valid notification message, generate an event if the team ID matches. */ if (team == my_team_id ()) { rq.id = id; haiku_write (NOTIFICATION_CLICK_EVENT, &rq); } } BApplication::ArgvReceived (argc, argv); } #endif /* 0 */ }; class EmacsWindow : public BWindow { public: struct child_frame { struct child_frame *next; int xoff, yoff; EmacsWindow *window; } *subset_windows; EmacsWindow *parent; BRect pre_fullscreen_rect; BRect pre_zoom_rect; bool shown_flag; volatile bool was_shown_p; bool menu_bar_active_p; bool override_redirect_p; window_look pre_override_redirect_look; window_feel pre_override_redirect_feel; uint32 pre_override_redirect_workspaces; int window_id; bool *menus_begun; enum haiku_z_group z_group; bool tooltip_p; enum haiku_fullscreen_mode fullscreen_mode; EmacsWindow () : BWindow (BRect (0, 0, 0, 0), "", B_TITLED_WINDOW_LOOK, B_NORMAL_WINDOW_FEEL, B_NO_SERVER_SIDE_WINDOW_MODIFIERS), subset_windows (NULL), parent (NULL), shown_flag (false), was_shown_p (false), menu_bar_active_p (false), override_redirect_p (false), menus_begun (NULL), z_group (Z_GROUP_NONE), tooltip_p (false), fullscreen_mode (FULLSCREEN_MODE_NONE) { /* This pulse rate is used by scroll bars for repeating a button action while a button is held down. */ SetPulseRate (30000); } ~EmacsWindow () { if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); struct child_frame *next; for (struct child_frame *f = subset_windows; f; f = next) { if (f->window->LockLooper ()) gui_abort ("Failed to lock looper for unparent"); f->window->Unparent (); f->window->UnlockLooper (); next = f->next; delete f; } if (this->parent) UnparentAndUnlink (); child_frame_lock.Unlock (); } void RecomputeFeel (void) { if (override_redirect_p || tooltip_p) SetFeel (kMenuWindowFeel); else if (parent) SetFeel (B_FLOATING_SUBSET_WINDOW_FEEL); else if (z_group == Z_GROUP_ABOVE) SetFeel (B_FLOATING_ALL_WINDOW_FEEL); else SetFeel (B_NORMAL_WINDOW_FEEL); } void UpwardsSubset (EmacsWindow *w) { for (; w; w = w->parent) AddToSubset (w); } void UpwardsSubsetChildren (EmacsWindow *w) { if (!LockLooper ()) gui_abort ("Failed to lock looper for subset"); if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); UpwardsSubset (w); for (struct child_frame *f = subset_windows; f; f = f->next) f->window->UpwardsSubsetChildren (w); child_frame_lock.Unlock (); UnlockLooper (); } void UpwardsUnSubset (EmacsWindow *w) { for (; w; w = w->parent) RemoveFromSubset (w); } void UpwardsUnSubsetChildren (EmacsWindow *w) { if (!LockLooper ()) gui_abort ("Failed to lock looper for unsubset"); if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); UpwardsUnSubset (w); for (struct child_frame *f = subset_windows; f; f = f->next) f->window->UpwardsUnSubsetChildren (w); child_frame_lock.Unlock (); UnlockLooper (); } void Unparent (void) { EmacsWindow *parent; if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); parent = this->parent; this->parent = NULL; RecomputeFeel (); UpwardsUnSubsetChildren (parent); this->RemoveFromSubset (this); child_frame_lock.Unlock (); } void UnparentAndUnlink (void) { if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); this->parent->UnlinkChild (this); this->Unparent (); child_frame_lock.Unlock (); } void UnlinkChild (EmacsWindow *window) { struct child_frame *last = NULL; struct child_frame *tem = subset_windows; for (; tem; last = tem, tem = tem->next) { if (tem->window == window) { if (last) last->next = tem->next; else subset_windows = tem->next; delete tem; return; } } gui_abort ("Failed to unlink child frame"); } void ParentTo (EmacsWindow *window) { if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); if (this->parent) UnparentAndUnlink (); this->parent = window; RecomputeFeel (); this->AddToSubset (this); if (!IsHidden () && this->parent) UpwardsSubsetChildren (parent); window->LinkChild (this); child_frame_lock.Unlock (); } void LinkChild (EmacsWindow *window) { struct child_frame *f = new struct child_frame; for (struct child_frame *f = subset_windows; f; f = f->next) { if (window == f->window) gui_abort ("Trying to link a child frame that is already present"); } f->window = window; f->next = subset_windows; f->xoff = -1; f->yoff = -1; subset_windows = f; } void MoveToIncludingFrame (int x, int y) { BRect decorator, frame; decorator = DecoratorFrame (); frame = Frame (); MoveTo (x + frame.left - decorator.left, y + frame.top - decorator.top); } void DoMove (struct child_frame *f) { BRect frame = this->Frame (); f->window->MoveToIncludingFrame (frame.left + f->xoff, frame.top + f->yoff); } void DoUpdateWorkspace (struct child_frame *f) { f->window->SetWorkspaces (this->Workspaces ()); } void MoveChild (EmacsWindow *window, int xoff, int yoff, int weak_p) { if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); for (struct child_frame *f = subset_windows; f; f = f->next) { if (window == f->window) { f->xoff = xoff; f->yoff = yoff; if (!weak_p) DoMove (f); child_frame_lock.Unlock (); return; } } child_frame_lock.Unlock (); gui_abort ("Trying to move a child frame that doesn't exist"); } void WindowActivated (bool activated) { struct haiku_activation_event rq; rq.window = this; rq.activated_p = activated; haiku_write (ACTIVATION, &rq); } void MessageReceived (BMessage *msg) { if (msg->WasDropped ()) { BPoint whereto; int64 threadid; struct haiku_drag_and_drop_event rq; if (msg->FindInt64 ("emacs:thread_id", &threadid) == B_OK && threadid == find_thread (NULL)) return; whereto = msg->DropPoint (); this->ConvertFromScreen (&whereto); rq.window = this; rq.message = DetachCurrentMessage (); rq.x = whereto.x; rq.y = whereto.y; haiku_write (DRAG_AND_DROP_EVENT, &rq); } else if (msg->GetPointer ("menuptr")) { struct haiku_menu_bar_select_event rq; rq.window = this; rq.ptr = (void *) msg->GetPointer ("menuptr"); haiku_write (MENU_BAR_SELECT_EVENT, &rq); } else BWindow::MessageReceived (msg); } void DispatchMessage (BMessage *msg, BHandler *handler) { if (msg->what == B_KEY_DOWN || msg->what == B_KEY_UP) { struct haiku_key_event rq; /* Pass through key events to the regular dispatch mechanism if the menu bar active, so that key navigation can work. */ if (menu_bar_active_p) { BWindow::DispatchMessage (msg, handler); return; } rq.window = this; int32 raw, key; int ret; msg->FindInt32 ("raw_char", &raw); msg->FindInt32 ("key", &key); msg->FindInt64 ("when", &rq.time); rq.modifiers = 0; rq.keysym = 0; uint32_t mods = modifiers (); if (haiku_should_pass_control_tab_to_system() && (mods & B_CONTROL_KEY) && key == 38) { BWindow::DispatchMessage (msg, handler); return; } if (mods & B_SHIFT_KEY) rq.modifiers |= HAIKU_MODIFIER_SHIFT; if (mods & B_CONTROL_KEY) rq.modifiers |= HAIKU_MODIFIER_CTRL; if (mods & B_COMMAND_KEY) rq.modifiers |= HAIKU_MODIFIER_ALT; if (mods & B_OPTION_KEY) rq.modifiers |= HAIKU_MODIFIER_SUPER; /* mods & B_SHIFT_KEY should be inverted if keycode is situated in the numeric keypad and Num Lock is set, for this transformation is not effected on key events themselves. */ if (mods & B_NUM_LOCK) { switch (key) { case 0x37: case 0x38: case 0x39: case 0x48: case 0x49: case 0x4a: case 0x58: case 0x59: case 0x5a: case 0x64: case 0x65: mods ^= B_SHIFT_KEY; /* If shift is set at this juncture, map these keys to the digits they represent. Because raw is not affected by Num Lock, keysym_from_raw_char will map this to the keysym yielded by this key in the absence of any modifiers. */ if (mods & B_SHIFT_KEY) goto map_keysym; } } ret = keysym_from_raw_char (raw, key, &rq.keysym); if (ret < 0) return; rq.multibyte_char = 0; if (!rq.keysym) { if (mods & B_SHIFT_KEY) { map_keysym: if (mods & B_CAPS_LOCK) map_caps_shift (key, &rq.multibyte_char); else map_shift (key, &rq.multibyte_char); } else { if (mods & B_CAPS_LOCK) map_caps (key, &rq.multibyte_char); else map_normal (key, &rq.multibyte_char); } } haiku_write (msg->what == B_KEY_DOWN ? KEY_DOWN : KEY_UP, &rq); } else if (msg->what == B_MOUSE_WHEEL_CHANGED) { struct haiku_wheel_move_event rq; rq.window = this; rq.modifiers = 0; uint32_t mods = modifiers (); if (mods & B_SHIFT_KEY) rq.modifiers |= HAIKU_MODIFIER_SHIFT; if (mods & B_CONTROL_KEY) rq.modifiers |= HAIKU_MODIFIER_CTRL; if (mods & B_COMMAND_KEY) rq.modifiers |= HAIKU_MODIFIER_ALT; if (mods & B_OPTION_KEY) rq.modifiers |= HAIKU_MODIFIER_SUPER; float dx, dy; if (msg->FindFloat ("be:wheel_delta_x", &dx) == B_OK && msg->FindFloat ("be:wheel_delta_y", &dy) == B_OK) { rq.delta_x = dx; rq.delta_y = dy; haiku_write (WHEEL_MOVE_EVENT, &rq); }; } else if (msg->what == SEND_MOVE_FRAME_EVENT) FrameMoved (Frame ().LeftTop ()); else if (msg->what == B_SCREEN_CHANGED) { if (fullscreen_mode != FULLSCREEN_MODE_NONE) SetFullscreen (fullscreen_mode); BWindow::DispatchMessage (msg, handler); } else BWindow::DispatchMessage (msg, handler); } void MenusBeginning (void) { struct haiku_menu_bar_state_event rq; rq.window = this; if (!menus_begun) haiku_write (MENU_BAR_OPEN, &rq); else *menus_begun = true; menu_bar_active_p = true; } void MenusEnded () { struct haiku_menu_bar_state_event rq; rq.window = this; haiku_write (MENU_BAR_CLOSE, &rq); menu_bar_active_p = false; } void FrameResized (float newWidth, float newHeight) { struct haiku_resize_event rq; rq.window = this; rq.width = newWidth + 1.0f; rq.height = newHeight + 1.0f; haiku_write (FRAME_RESIZED, &rq); BWindow::FrameResized (newWidth, newHeight); } void FrameMoved (BPoint new_position) { struct haiku_move_event rq; BRect frame, decorator_frame; struct child_frame *f; if (fullscreen_mode == FULLSCREEN_MODE_WIDTH && new_position.x != 0) { MoveTo (0, new_position.y); return; } if (fullscreen_mode == FULLSCREEN_MODE_HEIGHT && new_position.y != 0) { MoveTo (new_position.x, 0); return; } rq.window = this; rq.x = std::lrint (new_position.x); rq.y = std::lrint (new_position.y); frame = Frame (); decorator_frame = DecoratorFrame (); rq.decorator_width = std::lrint (frame.left - decorator_frame.left); rq.decorator_height = std::lrint (frame.top - decorator_frame.top); haiku_write (MOVE_EVENT, &rq); CHILD_FRAME_LOCK_INSIDE_LOOPER_CALLBACK { for (f = subset_windows; f; f = f->next) DoMove (f); child_frame_lock.Unlock (); BWindow::FrameMoved (new_position); } } void WorkspacesChanged (uint32_t old, uint32_t n) { struct child_frame *f; CHILD_FRAME_LOCK_INSIDE_LOOPER_CALLBACK { for (f = subset_windows; f; f = f->next) DoUpdateWorkspace (f); child_frame_lock.Unlock (); } } void EmacsMoveTo (int x, int y) { if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); if (!this->parent) this->MoveToIncludingFrame (x, y); else this->parent->MoveChild (this, x, y, 0); child_frame_lock.Unlock (); } bool QuitRequested () { struct haiku_quit_requested_event rq; rq.window = this; haiku_write (QUIT_REQUESTED, &rq); return false; } void Minimize (bool minimized_p) { struct haiku_iconification_event rq; rq.window = this; rq.iconified_p = !parent && minimized_p; haiku_write (ICONIFICATION, &rq); BWindow::Minimize (minimized_p); } void EmacsHide (void) { if (this->IsHidden ()) return; if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); Hide (); if (this->parent) UpwardsUnSubsetChildren (this->parent); child_frame_lock.Unlock (); } void EmacsShow (void) { if (!this->IsHidden ()) return; if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); if (!was_shown_p) { /* This window is being shown for the first time, which means Show will unlock the looper. In this case, it should be locked again, since the looper is unlocked when the window is first created. */ if (!LockLooper ()) gui_abort ("Failed to lock looper during first window show"); was_shown_p = true; } if (this->parent) shown_flag = 1; Show (); if (this->parent) UpwardsSubsetChildren (this->parent); child_frame_lock.Unlock (); } BRect ClearFullscreen (enum haiku_fullscreen_mode target_mode) { BRect original_frame; switch (fullscreen_mode) { case FULLSCREEN_MODE_MAXIMIZED: original_frame = pre_zoom_rect; if (target_mode == FULLSCREEN_MODE_NONE) BWindow::Zoom (pre_zoom_rect.LeftTop (), BE_RECT_WIDTH (pre_zoom_rect) - 1, BE_RECT_HEIGHT (pre_zoom_rect) - 1); break; case FULLSCREEN_MODE_BOTH: case FULLSCREEN_MODE_HEIGHT: case FULLSCREEN_MODE_WIDTH: original_frame = pre_fullscreen_rect; SetFlags (Flags () & ~(B_NOT_MOVABLE | B_NOT_ZOOMABLE | B_NOT_RESIZABLE)); if (target_mode != FULLSCREEN_MODE_NONE) goto out; MoveTo (pre_fullscreen_rect.LeftTop ()); ResizeTo (BE_RECT_WIDTH (pre_fullscreen_rect) - 1, BE_RECT_HEIGHT (pre_fullscreen_rect) - 1); break; case FULLSCREEN_MODE_NONE: original_frame = Frame (); break; } out: fullscreen_mode = FULLSCREEN_MODE_NONE; return original_frame; } BRect FullscreenRectForMode (enum haiku_fullscreen_mode mode) { BScreen screen (this); BRect frame; if (!screen.IsValid ()) return BRect (0, 0, 0, 0); frame = screen.Frame (); if (mode == FULLSCREEN_MODE_HEIGHT) frame.right -= BE_RECT_WIDTH (frame) / 2; else if (mode == FULLSCREEN_MODE_WIDTH) frame.bottom -= BE_RECT_HEIGHT (frame) / 2; return frame; } void SetFullscreen (enum haiku_fullscreen_mode mode) { BRect zoom_rect, frame; frame = ClearFullscreen (mode); switch (mode) { case FULLSCREEN_MODE_MAXIMIZED: pre_zoom_rect = frame; zoom_rect = get_zoom_rect (this); BWindow::Zoom (zoom_rect.LeftTop (), BE_RECT_WIDTH (zoom_rect) - 1, BE_RECT_HEIGHT (zoom_rect) - 1); break; case FULLSCREEN_MODE_BOTH: SetFlags (Flags () | B_NOT_MOVABLE); FALLTHROUGH; case FULLSCREEN_MODE_HEIGHT: case FULLSCREEN_MODE_WIDTH: SetFlags (Flags () | B_NOT_ZOOMABLE | B_NOT_RESIZABLE); pre_fullscreen_rect = frame; zoom_rect = FullscreenRectForMode (mode); ResizeTo (BE_RECT_WIDTH (zoom_rect) - 1, BE_RECT_HEIGHT (zoom_rect) - 1); MoveTo (zoom_rect.left, zoom_rect.top); break; case FULLSCREEN_MODE_NONE: break; } fullscreen_mode = mode; } void Zoom (BPoint origin, float width, float height) { struct haiku_zoom_event rq; rq.window = this; rq.fullscreen_mode = fullscreen_mode; haiku_write (ZOOM_EVENT, &rq); } void OffsetChildRect (BRect *r, EmacsWindow *c) { if (!child_frame_lock.Lock ()) gui_abort ("Failed to lock child frame state lock"); for (struct child_frame *f; f; f = f->next) if (f->window == c) { r->top -= f->yoff; r->bottom -= f->yoff; r->left -= f->xoff; r->right -= f->xoff; child_frame_lock.Unlock (); return; } child_frame_lock.Lock (); gui_abort ("Trying to calculate offsets for a child frame that doesn't exist"); } }; class EmacsMenuBar : public BMenuBar { bool tracking_p; public: EmacsMenuBar () : BMenuBar (BRect (0, 0, 0, 0), NULL) { } void AttachedToWindow (void) { BWindow *window = Window (); window->SetKeyMenuBar (this); } void FrameResized (float newWidth, float newHeight) { struct haiku_menu_bar_resize_event rq; rq.window = this->Window (); rq.height = std::lrint (newHeight + 1); rq.width = std::lrint (newWidth + 1); haiku_write (MENU_BAR_RESIZE, &rq); BMenuBar::FrameResized (newWidth, newHeight); } void MouseDown (BPoint point) { struct haiku_menu_bar_click_event rq; EmacsWindow *ew = (EmacsWindow *) Window (); rq.window = ew; rq.x = std::lrint (point.x); rq.y = std::lrint (point.y); if (!ew->menu_bar_active_p) haiku_write (MENU_BAR_CLICK, &rq); else BMenuBar::MouseDown (point); } void MouseMoved (BPoint point, uint32 transit, const BMessage *msg) { struct haiku_menu_bar_left_event rq; if (transit == B_EXITED_VIEW) { rq.x = std::lrint (point.x); rq.y = std::lrint (point.y); rq.window = this->Window (); haiku_write (MENU_BAR_LEFT, &rq); } BMenuBar::MouseMoved (point, transit, msg); } void MessageReceived (BMessage *msg) { BRect frame; BPoint pt, l; EmacsWindow *window; bool menus_begun; if (msg->what == SHOW_MENU_BAR) { window = (EmacsWindow *) Window (); frame = Frame (); pt = frame.LeftTop (); l = pt; menus_begun = false; Parent ()->ConvertToScreen (&pt); window->menus_begun = &menus_begun; set_mouse_position (pt.x, pt.y); BMenuBar::MouseDown (l); window->menus_begun = NULL; if (!menus_begun) msg->SendReply (msg); else msg->SendReply (BE_MENU_BAR_OPEN); } else if (msg->what == REPLAY_MENU_BAR) { window = (EmacsWindow *) Window (); menus_begun = false; window->menus_begun = &menus_begun; if (msg->FindPoint ("emacs:point", &pt) == B_OK) BMenuBar::MouseDown (pt); window->menus_begun = NULL; if (!menus_begun) msg->SendReply (msg); else msg->SendReply (BE_MENU_BAR_OPEN); } else BMenuBar::MessageReceived (msg); } }; class EmacsView : public BView { public: int looper_locked_count; BRegion sb_region; BRegion invalid_region; BView *offscreen_draw_view; BBitmap *offscreen_draw_bitmap_1; BBitmap *copy_bitmap; #ifdef USE_BE_CAIRO cairo_surface_t *cr_surface; cairo_t *cr_context; BLocker cr_surface_lock; #endif BMessage *wait_for_release_message; int64 grabbed_buttons; BScreen screen; bool use_frame_synchronization; EmacsView () : BView (BRect (0, 0, 0, 0), "Emacs", B_FOLLOW_NONE, B_WILL_DRAW), looper_locked_count (0), offscreen_draw_view (NULL), offscreen_draw_bitmap_1 (NULL), copy_bitmap (NULL), #ifdef USE_BE_CAIRO cr_surface (NULL), cr_context (NULL), #endif wait_for_release_message (NULL), grabbed_buttons (0), use_frame_synchronization (false) { } ~EmacsView () { if (wait_for_release_message) { wait_for_release_message->SendReply (wait_for_release_message); delete wait_for_release_message; } TearDownDoubleBuffering (); if (!grab_view_locker.Lock ()) gui_abort ("Couldn't lock grab view locker"); if (grab_view == this) grab_view = NULL; grab_view_locker.Unlock (); } void SetFrameSynchronization (bool sync) { if (LockLooper ()) { use_frame_synchronization = sync; UnlockLooper (); } } void MessageReceived (BMessage *msg) { uint32 buttons; BLooper *looper = Looper (); if (msg->what == WAIT_FOR_RELEASE) { if (wait_for_release_message) gui_abort ("Wait for release message already exists"); GetMouse (NULL, &buttons, false); if (!buttons) msg->SendReply (msg); else wait_for_release_message = looper->DetachCurrentMessage (); } else if (msg->what == RELEASE_NOW) { if (wait_for_release_message) wait_for_release_message->SendReply (msg); delete wait_for_release_message; wait_for_release_message = NULL; } else BView::MessageReceived (msg); } #ifdef USE_BE_CAIRO void DetachCairoSurface (void) { if (!cr_surface_lock.Lock ()) gui_abort ("Could not lock cr surface during detachment"); if (!cr_surface) gui_abort ("Trying to detach window cr surface when none exists"); cairo_destroy (cr_context); cairo_surface_destroy (cr_surface); cr_surface = NULL; cr_context = NULL; cr_surface_lock.Unlock (); } void AttachCairoSurface (void) { if (!cr_surface_lock.Lock ()) gui_abort ("Could not lock cr surface during attachment"); if (cr_surface) gui_abort ("Trying to attach cr surface when one already exists"); BRect bounds = offscreen_draw_bitmap_1->Bounds (); cr_surface = cairo_image_surface_create_for_data ((unsigned char *) offscreen_draw_bitmap_1->Bits (), CAIRO_FORMAT_ARGB32, BE_RECT_WIDTH (bounds), BE_RECT_HEIGHT (bounds), offscreen_draw_bitmap_1->BytesPerRow ()); if (!cr_surface) gui_abort ("Cr surface allocation failed for double-buffered view"); cr_context = cairo_create (cr_surface); if (!cr_context) gui_abort ("cairo_t allocation failed for double-buffered view"); cr_surface_lock.Unlock (); } #endif void TearDownDoubleBuffering (void) { if (offscreen_draw_view) { if (Window ()) ClearViewBitmap (); if (copy_bitmap) { delete copy_bitmap; copy_bitmap = NULL; } if (!looper_locked_count) if (!offscreen_draw_view->LockLooper ()) gui_abort ("Failed to lock offscreen draw view"); #ifdef USE_BE_CAIRO if (cr_surface) DetachCairoSurface (); #endif offscreen_draw_view->RemoveSelf (); delete offscreen_draw_view; offscreen_draw_view = NULL; delete offscreen_draw_bitmap_1; offscreen_draw_bitmap_1 = NULL; } } void AfterResize (void) { if (offscreen_draw_view) { if (!LockLooper ()) gui_abort ("Failed to lock looper after resize"); if (!offscreen_draw_view->LockLooper ()) gui_abort ("Failed to lock offscreen draw view after resize"); #ifdef USE_BE_CAIRO DetachCairoSurface (); #endif offscreen_draw_view->RemoveSelf (); delete offscreen_draw_bitmap_1; offscreen_draw_bitmap_1 = new BBitmap (Frame (), B_RGBA32, 1); if (offscreen_draw_bitmap_1->InitCheck () != B_OK) gui_abort ("Offscreen draw bitmap initialization failed"); BRect frame = Frame (); offscreen_draw_view->MoveTo (frame.left, frame.top); offscreen_draw_view->ResizeTo (BE_RECT_WIDTH (frame), BE_RECT_HEIGHT (frame)); offscreen_draw_bitmap_1->AddChild (offscreen_draw_view); #ifdef USE_BE_CAIRO AttachCairoSurface (); #endif if (looper_locked_count) offscreen_draw_bitmap_1->Lock (); UnlockLooper (); } } void Draw (BRect expose_bounds) { struct haiku_expose_event rq; EmacsWindow *w = (EmacsWindow *) Window (); if (w->shown_flag && offscreen_draw_view) { PushState (); SetDrawingMode (B_OP_ERASE); FillRect (Frame ()); PopState (); return; } if (!offscreen_draw_view) { if (sb_region.Contains (std::lrint (expose_bounds.left), std::lrint (expose_bounds.top)) && sb_region.Contains (std::lrint (expose_bounds.right), std::lrint (expose_bounds.top)) && sb_region.Contains (std::lrint (expose_bounds.left), std::lrint (expose_bounds.bottom)) && sb_region.Contains (std::lrint (expose_bounds.right), std::lrint (expose_bounds.bottom))) return; rq.x = std::floor (expose_bounds.left); rq.y = std::floor (expose_bounds.top); rq.width = std::ceil (expose_bounds.right - expose_bounds.left + 1); rq.height = std::ceil (expose_bounds.bottom - expose_bounds.top + 1); if (!rq.width) rq.width = 1; if (!rq.height) rq.height = 1; rq.window = this->Window (); haiku_write (FRAME_EXPOSED, &rq); } } void FlipBuffers (void) { EmacsWindow *w; if (!LockLooper ()) gui_abort ("Failed to lock looper during buffer flip"); if (!offscreen_draw_view) gui_abort ("Failed to lock offscreen view during buffer flip"); offscreen_draw_view->Sync (); w = (EmacsWindow *) Window (); w->shown_flag = 0; if (copy_bitmap && copy_bitmap->Bounds () != offscreen_draw_bitmap_1->Bounds ()) { delete copy_bitmap; copy_bitmap = NULL; } if (!copy_bitmap) { copy_bitmap = new BBitmap (offscreen_draw_bitmap_1); SetViewBitmap (copy_bitmap, Frame (), Frame (), B_FOLLOW_NONE, 0); } else copy_bitmap->ImportBits (offscreen_draw_bitmap_1); if (copy_bitmap->InitCheck () != B_OK) gui_abort ("Failed to init copy bitmap during buffer flip"); /* Wait for VBLANK. If responding to the invalidation or buffer flipping takes longer than the blanking period, we lose. */ if (use_frame_synchronization) screen.WaitForRetrace (); Invalidate (&invalid_region); invalid_region.MakeEmpty (); UnlockLooper (); return; } void SetUpDoubleBuffering (void) { if (!LockLooper ()) gui_abort ("Failed to lock self setting up double buffering"); if (offscreen_draw_view) gui_abort ("Failed to lock offscreen view setting up double buffering"); offscreen_draw_bitmap_1 = new BBitmap (Frame (), B_RGBA32, 1); if (offscreen_draw_bitmap_1->InitCheck () != B_OK) gui_abort ("Failed to init offscreen bitmap"); #ifdef USE_BE_CAIRO AttachCairoSurface (); #endif offscreen_draw_view = new BView (Frame (), NULL, B_FOLLOW_NONE, B_WILL_DRAW); offscreen_draw_bitmap_1->AddChild (offscreen_draw_view); if (looper_locked_count) { if (!offscreen_draw_bitmap_1->Lock ()) gui_abort ("Failed to lock bitmap after double buffering was set up"); } invalid_region.MakeEmpty (); UnlockLooper (); Invalidate (); } void MouseMoved (BPoint point, uint32 transit, const BMessage *drag_msg) { struct haiku_mouse_motion_event rq; int64 threadid; EmacsWindow *window; window = (EmacsWindow *) Window (); if (transit == B_EXITED_VIEW) rq.just_exited_p = true; else rq.just_exited_p = false; rq.x = point.x; rq.y = point.y; rq.window = window; rq.time = system_time (); if (drag_msg && (drag_msg->IsSourceRemote () || drag_msg->FindInt64 ("emacs:thread_id", &threadid) != B_OK || threadid != find_thread (NULL))) rq.dnd_message = true; else rq.dnd_message = false; if (!grab_view_locker.Lock ()) gui_abort ("Couldn't lock grab view locker"); if (grab_view && this != grab_view) { grab_view_locker.Unlock (); return; } grab_view_locker.Unlock (); haiku_write (MOUSE_MOTION, &rq); } void BasicMouseDown (BPoint point, BView *scroll_bar, BMessage *message) { struct haiku_button_event rq; int64 when; int32 mods, buttons, button; if (message->FindInt64 ("when", &when) != B_OK || message->FindInt32 ("modifiers", &mods) != B_OK || message->FindInt32 ("buttons", &buttons) != B_OK) return; /* Find which button was pressed by comparing the previous button mask to the current one. This assumes that B_MOUSE_DOWN will be sent for each button press. */ button = buttons & ~grabbed_buttons; grabbed_buttons = buttons; if (!scroll_bar) { if (!grab_view_locker.Lock ()) gui_abort ("Couldn't lock grab view locker"); grab_view = this; grab_view_locker.Unlock (); } rq.window = this->Window (); rq.scroll_bar = scroll_bar; if (button == B_PRIMARY_MOUSE_BUTTON) rq.btn_no = 0; else if (button == B_SECONDARY_MOUSE_BUTTON) rq.btn_no = 2; else if (button == B_TERTIARY_MOUSE_BUTTON) rq.btn_no = 1; else /* We don't know which button was pressed. This usually happens when a B_MOUSE_UP is sent to a view that didn't receive a corresponding B_MOUSE_DOWN event, so simply ignore the message. */ return; rq.x = point.x; rq.y = point.y; rq.modifiers = 0; if (mods & B_SHIFT_KEY) rq.modifiers |= HAIKU_MODIFIER_SHIFT; if (mods & B_CONTROL_KEY) rq.modifiers |= HAIKU_MODIFIER_CTRL; if (mods & B_COMMAND_KEY) rq.modifiers |= HAIKU_MODIFIER_ALT; if (mods & B_OPTION_KEY) rq.modifiers |= HAIKU_MODIFIER_SUPER; if (!scroll_bar) SetMouseEventMask (B_POINTER_EVENTS, (B_LOCK_WINDOW_FOCUS | B_NO_POINTER_HISTORY)); rq.time = when; haiku_write (BUTTON_DOWN, &rq); } void MouseDown (BPoint point) { BMessage *msg; BLooper *looper; looper = Looper (); msg = (looper ? looper->CurrentMessage () : NULL); if (msg) BasicMouseDown (point, NULL, msg); } void BasicMouseUp (BPoint point, BView *scroll_bar, BMessage *message) { struct haiku_button_event rq; int64 when; int32 mods, button, buttons; if (message->FindInt64 ("when", &when) != B_OK || message->FindInt32 ("modifiers", &mods) != B_OK || message->FindInt32 ("buttons", &buttons) != B_OK) return; if (!scroll_bar) { if (!grab_view_locker.Lock ()) gui_abort ("Couldn't lock grab view locker"); if (!buttons) grab_view = NULL; grab_view_locker.Unlock (); } button = (grabbed_buttons & ~buttons); grabbed_buttons = buttons; if (wait_for_release_message) { if (!grabbed_buttons) { wait_for_release_message->SendReply (wait_for_release_message); delete wait_for_release_message; wait_for_release_message = NULL; } return; } rq.window = this->Window (); rq.scroll_bar = scroll_bar; if (button == B_PRIMARY_MOUSE_BUTTON) rq.btn_no = 0; else if (button == B_SECONDARY_MOUSE_BUTTON) rq.btn_no = 2; else if (button == B_TERTIARY_MOUSE_BUTTON) rq.btn_no = 1; else return; rq.x = point.x; rq.y = point.y; rq.modifiers = 0; if (mods & B_SHIFT_KEY) rq.modifiers |= HAIKU_MODIFIER_SHIFT; if (mods & B_CONTROL_KEY) rq.modifiers |= HAIKU_MODIFIER_CTRL; if (mods & B_COMMAND_KEY) rq.modifiers |= HAIKU_MODIFIER_ALT; if (mods & B_OPTION_KEY) rq.modifiers |= HAIKU_MODIFIER_SUPER; rq.time = when; haiku_write (BUTTON_UP, &rq); } void MouseUp (BPoint point) { BMessage *msg; BLooper *looper; looper = Looper (); msg = (looper ? looper->CurrentMessage () : NULL); if (msg) BasicMouseUp (point, NULL, msg); } }; class EmacsScrollBar : public BScrollBar { public: int dragging; bool horizontal; enum haiku_scroll_bar_part current_part; float old_value; scroll_bar_info info; /* How many button events were passed to the parent without release. */ int handle_button_count; bool in_overscroll; bool can_overscroll; bool maybe_overscroll; BPoint last_overscroll; int last_reported_overscroll_value; int max_value, real_max_value; int overscroll_start_value; bigtime_t repeater_start; EmacsView *parent; EmacsScrollBar (int x, int y, int x1, int y1, bool horizontal_p, EmacsView *parent) : BScrollBar (BRect (x, y, x1, y1), NULL, NULL, 0, 0, horizontal_p ? B_HORIZONTAL : B_VERTICAL), dragging (0), handle_button_count (0), in_overscroll (false), can_overscroll (false), maybe_overscroll (false), parent (parent) { BView *vw = (BView *) this; vw->SetResizingMode (B_FOLLOW_NONE); horizontal = horizontal_p; get_scroll_bar_info (&info); SetSteps (5000, 10000); } void MessageReceived (BMessage *msg) { int32 portion, range, dragging, value; float proportion; if (msg->what == SCROLL_BAR_UPDATE) { portion = msg->GetInt32 ("emacs:portion", 0); range = msg->GetInt32 ("emacs:range", 0); dragging = msg->GetInt32 ("emacs:dragging", 0); proportion = ((range <= 0 || portion <= 0) ? 1.0f : (float) portion / range); value = msg->GetInt32 ("emacs:units", 0); can_overscroll = msg->GetBool ("emacs:overscroll", false); if (value < 0) value = 0; if (dragging != 1) { if (in_overscroll || dragging != -1) { /* Set the value to the smallest possible one. Otherwise, the call to SetRange could lead to spurious updates. */ old_value = 0; SetValue (0); /* Unlike on Motif, PORTION isn't included in the total range of the scroll bar. */ SetRange (0, range - portion); SetProportion (proportion); max_value = range - portion; real_max_value = range; if (in_overscroll || value > max_value) value = max_value; old_value = roundf (value); SetValue (old_value); } else { value = Value (); old_value = 0; SetValue (0); SetRange (0, range - portion); SetProportion (proportion); old_value = value; SetValue (value); max_value = range - portion; real_max_value = range; } } } BScrollBar::MessageReceived (msg); } void Pulse (void) { struct haiku_scroll_bar_part_event rq; BPoint point; uint32 buttons; if (!dragging) { SetFlags (Flags () & ~B_PULSE_NEEDED); return; } if (repeater_start < system_time ()) { GetMouse (&point, &buttons, false); if (ButtonRegionFor (current_part).Contains (point)) { rq.scroll_bar = this; rq.window = Window (); rq.part = current_part; haiku_write (SCROLL_BAR_PART_EVENT, &rq); } } BScrollBar::Pulse (); } void ValueChanged (float new_value) { struct haiku_scroll_bar_value_event rq; new_value = Value (); if (dragging) { if (new_value != old_value) { if (dragging > 1) { SetValue (old_value); SetFlags (Flags () | B_PULSE_NEEDED); } else dragging++; } return; } if (new_value != old_value) { rq.scroll_bar = this; rq.window = Window (); rq.position = new_value; old_value = new_value; haiku_write (SCROLL_BAR_VALUE_EVENT, &rq); } } BRegion ButtonRegionFor (enum haiku_scroll_bar_part button) { BRegion region; BRect bounds; BRect rect; float button_size; bounds = Bounds (); bounds.InsetBy (0.0, 0.0); if (horizontal) button_size = bounds.Height () + 1.0f; else button_size = bounds.Width () + 1.0f; rect = BRect (bounds.left, bounds.top, bounds.left + button_size - 1.0f, bounds.top + button_size - 1.0f); if (button == HAIKU_SCROLL_BAR_UP_BUTTON) { if (!horizontal) { region.Include (rect); if (info.double_arrows) region.Include (rect.OffsetToCopy (bounds.left, bounds.bottom - 2 * button_size + 1)); } else { region.Include (rect); if (info.double_arrows) region.Include (rect.OffsetToCopy (bounds.right - 2 * button_size, bounds.top)); } } else { if (!horizontal) { region.Include (rect.OffsetToCopy (bounds.left, bounds.bottom - button_size)); if (info.double_arrows) region.Include (rect.OffsetByCopy (0.0, button_size)); } else { region.Include (rect.OffsetToCopy (bounds.right - button_size, bounds.top)); if (info.double_arrows) region.Include (rect.OffsetByCopy (button_size, 0.0)); } } return region; } void MouseDown (BPoint pt) { struct haiku_scroll_bar_drag_event rq; struct haiku_scroll_bar_part_event part; BRegion r; BLooper *looper; BMessage *message; int32 buttons, mods; looper = Looper (); message = NULL; if (!looper) GetMouse (&pt, (uint32 *) &buttons, false); else { message = looper->CurrentMessage (); if (!message || message->FindInt32 ("buttons", &buttons) != B_OK) GetMouse (&pt, (uint32 *) &buttons, false); } if (message && (message->FindInt32 ("modifiers", &mods) == B_OK) && mods & B_CONTROL_KEY) { /* Allow C-mouse-3 to split the window on a scroll bar. */ handle_button_count += 1; SetMouseEventMask (B_POINTER_EVENTS, (B_SUSPEND_VIEW_FOCUS | B_LOCK_WINDOW_FOCUS)); parent->BasicMouseDown (ConvertToParent (pt), this, message); return; } repeater_start = system_time () + 300000; if (buttons == B_PRIMARY_MOUSE_BUTTON) { r = ButtonRegionFor (HAIKU_SCROLL_BAR_UP_BUTTON); if (r.Contains (pt)) { part.scroll_bar = this; part.window = Window (); part.part = HAIKU_SCROLL_BAR_UP_BUTTON; dragging = 1; current_part = HAIKU_SCROLL_BAR_UP_BUTTON; haiku_write (SCROLL_BAR_PART_EVENT, &part); goto out; } r = ButtonRegionFor (HAIKU_SCROLL_BAR_DOWN_BUTTON); if (r.Contains (pt)) { part.scroll_bar = this; part.window = Window (); part.part = HAIKU_SCROLL_BAR_DOWN_BUTTON; dragging = 1; current_part = HAIKU_SCROLL_BAR_DOWN_BUTTON; if (Value () == max_value) { SetFlags (Flags () | B_PULSE_NEEDED); dragging = 2; } haiku_write (SCROLL_BAR_PART_EVENT, &part); goto out; } maybe_overscroll = true; } rq.dragging_p = 1; rq.window = Window (); rq.scroll_bar = this; SetMouseEventMask (B_POINTER_EVENTS, (B_SUSPEND_VIEW_FOCUS | B_LOCK_WINDOW_FOCUS)); haiku_write (SCROLL_BAR_DRAG_EVENT, &rq); out: BScrollBar::MouseDown (pt); } void MouseUp (BPoint pt) { struct haiku_scroll_bar_drag_event rq; BMessage *msg; BLooper *looper; in_overscroll = false; maybe_overscroll = false; if (handle_button_count) { handle_button_count--; looper = Looper (); msg = (looper ? looper->CurrentMessage () : NULL); if (msg) parent->BasicMouseUp (ConvertToParent (pt), this, msg); return; } rq.dragging_p = 0; rq.scroll_bar = this; rq.window = Window (); haiku_write (SCROLL_BAR_DRAG_EVENT, &rq); dragging = 0; BScrollBar::MouseUp (pt); } void MouseMoved (BPoint point, uint32 transit, const BMessage *msg) { struct haiku_menu_bar_left_event rq; struct haiku_scroll_bar_value_event value_event; int range, diff, value, trough_size; BRect bounds; BPoint conv; uint32 buttons; GetMouse (NULL, &buttons, false); if (transit == B_EXITED_VIEW) { conv = ConvertToParent (point); rq.x = std::lrint (conv.x); rq.y = std::lrint (conv.y); rq.window = this->Window (); haiku_write (MENU_BAR_LEFT, &rq); } if (in_overscroll) { if (horizontal) diff = point.x - last_overscroll.x; else diff = point.y - last_overscroll.y; if (diff < 0) { in_overscroll = false; goto allow; } range = real_max_value; bounds = Bounds (); bounds.InsetBy (1.0, 1.0); value = overscroll_start_value; trough_size = (horizontal ? BE_RECT_WIDTH (bounds) : BE_RECT_HEIGHT (bounds)); trough_size -= (horizontal ? BE_RECT_HEIGHT (bounds) : BE_RECT_WIDTH (bounds)) / 2; if (info.double_arrows) trough_size -= (horizontal ? BE_RECT_HEIGHT (bounds) : BE_RECT_WIDTH (bounds)) / 2; value += ((double) range / trough_size) * diff; if (value != last_reported_overscroll_value) { last_reported_overscroll_value = value; value_event.scroll_bar = this; value_event.window = Window (); value_event.position = value; haiku_write (SCROLL_BAR_VALUE_EVENT, &value_event); return; } } else if (can_overscroll && (buttons == B_PRIMARY_MOUSE_BUTTON) && maybe_overscroll) { value = Value (); if (value >= max_value) { BScrollBar::MouseMoved (point, transit, msg); if (value == Value ()) { overscroll_start_value = value; in_overscroll = true; last_overscroll = point; last_reported_overscroll_value = value; MouseMoved (point, transit, msg); return; } } } allow: BScrollBar::MouseMoved (point, transit, msg); } }; class EmacsTitleMenuItem : public BMenuItem { public: EmacsTitleMenuItem (const char *str) : BMenuItem (str, NULL) { SetEnabled (0); } void DrawContent (void) { BMenu *menu = Menu (); menu->PushState (); menu->SetFont (be_bold_font); menu->SetHighColor (ui_color (B_MENU_ITEM_TEXT_COLOR)); BMenuItem::DrawContent (); menu->PopState (); } }; class EmacsMenuItem : public BMenuItem { public: int menu_bar_id; void *menu_ptr; void *wind_ptr; char *key; char *help; EmacsMenuItem (const char *key_label, const char *label, const char *help, BMessage *message = NULL) : BMenuItem (label, message), menu_bar_id (-1), menu_ptr (NULL), wind_ptr (NULL), key (NULL), help (NULL) { if (key_label) key = strdup (key_label); if (help) this->help = strdup (help); } ~EmacsMenuItem () { if (key) free (key); if (help) free (help); } void DrawContent (void) { BMenu *menu = Menu (); BMenuItem::DrawContent (); if (key) { BRect r = Frame (); int w; menu->PushState (); menu->ClipToRect (r); menu->SetFont (be_plain_font); w = menu->StringWidth (key); menu->MovePenTo (BPoint (BE_RECT_WIDTH (r) - w - 4, menu->PenLocation ().y)); menu->DrawString (key); menu->PopState (); } } void GetContentSize (float *w, float *h) { BMenuItem::GetContentSize (w, h); if (Menu () && key) *w += 4 + Menu ()->StringWidth (key); } void Highlight (bool highlight_p) { struct haiku_menu_bar_help_event rq; struct haiku_dummy_event dummy; BMenu *menu = Menu (); BRect r; BPoint pt; uint32 buttons; if (help) menu->SetToolTip (highlight_p ? help : NULL); else { rq.window = wind_ptr; rq.mb_idx = highlight_p ? menu_bar_id : -1; rq.highlight_p = highlight_p; rq.data = menu_ptr; r = Frame (); menu->GetMouse (&pt, &buttons); if (!highlight_p || r.Contains (pt)) { if (menu_bar_id > 0) haiku_write (MENU_BAR_HELP_EVENT, &rq); else { haiku_write_without_signal (MENU_BAR_HELP_EVENT, &rq, true); haiku_write (DUMMY_EVENT, &dummy); } } } BMenuItem::Highlight (highlight_p); } }; class EmacsFontPreviewDialog : public BWindow { BStringView text_view; BMessenger preview_source; BFont *current_font; bool is_visible; void DoLayout (void) { float width, height; text_view.GetPreferredSize (&width, &height); text_view.ResizeTo (width - 1, height - 1); SetSizeLimits (width, width, height, height); ResizeTo (width - 1, height - 1); } bool QuitRequested (void) { preview_source.SendMessage (QUIT_PREVIEW_DIALOG); return false; } void MessageReceived (BMessage *message) { int32 family, style; uint32 flags; font_family name; font_style sname; status_t rc; const char *size_name; int size; if (message->what == SET_FONT_INDICES) { size_name = message->FindString ("emacs:size"); if (message->FindInt32 ("emacs:family", &family) != B_OK || message->FindInt32 ("emacs:style", &style) != B_OK) return; rc = get_font_family (family, &name, &flags); if (rc != B_OK) return; rc = get_font_style (name, style, &sname, &flags); if (rc != B_OK) return; if (current_font) delete current_font; current_font = new BFont; current_font->SetFamilyAndStyle (name, sname); if (message->GetBool ("emacs:disable_antialiasing", false)) current_font->SetFlags (B_DISABLE_ANTIALIASING); if (size_name && strlen (size_name)) { size = atoi (size_name); current_font->SetSize (size); } text_view.SetFont (current_font); DoLayout (); return; } BWindow::MessageReceived (message); } public: EmacsFontPreviewDialog (BWindow *target) : BWindow (BRect (45, 45, 500, 300), "Preview font", B_FLOATING_WINDOW_LOOK, B_MODAL_APP_WINDOW_FEEL, B_NOT_ZOOMABLE | B_NOT_RESIZABLE), text_view (BRect (0, 0, 0, 0), NULL, "The quick brown fox " "jumped over the lazy dog"), preview_source (target), current_font (NULL) { AddChild (&text_view); DoLayout (); } ~EmacsFontPreviewDialog (void) { text_view.RemoveSelf (); if (current_font) delete current_font; } }; class TripleLayoutView : public BView { BScrollView *view_1; BView *view_2, *view_3; void FrameResized (float new_width, float new_height) { BRect frame; float width, height, height_1, width_1; float basic_height; frame = Frame (); view_2->GetPreferredSize (&width, &height); view_3->GetPreferredSize (&width_1, &height_1); basic_height = height + height_1; view_1->MoveTo (0, 0); view_1->ResizeTo (BE_RECT_WIDTH (frame), BE_RECT_HEIGHT (frame) - basic_height); view_2->MoveTo (2, BE_RECT_HEIGHT (frame) - basic_height); view_2->ResizeTo (BE_RECT_WIDTH (frame) - 4, height); view_3->MoveTo (2, BE_RECT_HEIGHT (frame) - height_1); view_3->ResizeTo (BE_RECT_WIDTH (frame) - 4, height_1); BView::FrameResized (new_width, new_height); } /* This is called by the BSplitView. */ BSize MinSize (void) { float width, height; float width_1, height_1; BSize size_1; size_1 = view_1->MinSize (); view_2->GetPreferredSize (&width, &height); view_3->GetPreferredSize (&width_1, &height_1); return BSize (std::max (size_1.width, std::max (width_1, width)), std::max (size_1.height, height + height_1)); } public: TripleLayoutView (BScrollView *first, BView *second, BView *third) : BView (NULL, B_FRAME_EVENTS), view_1 (first), view_2 (second), view_3 (third) { FrameResized (801, 801); } }; class EmacsFontSelectionDialog : public BWindow { BView basic_view; BCheckBox antialias_checkbox; BCheckBox preview_checkbox; BSplitView split_view; BListView font_family_pane; BListView font_style_pane; BScrollView font_family_scroller; BScrollView font_style_scroller; TripleLayoutView style_view; BObjectList all_families; BObjectList all_styles; BButton cancel_button, ok_button; BTextControl size_entry; port_id comm_port; bool allow_monospace_only; int pending_selection_idx; EmacsFontPreviewDialog *preview; void ShowPreview (void) { if (!preview) { preview = new EmacsFontPreviewDialog (this); preview->Show (); UpdatePreview (); } } void UpdatePreview (void) { int family, style; BMessage message; BMessenger messenger (preview); family = font_family_pane.CurrentSelection (); style = font_style_pane.CurrentSelection (); message.what = SET_FONT_INDICES; message.AddInt32 ("emacs:family", family); message.AddInt32 ("emacs:style", style); if (antialias_checkbox.Value () == B_CONTROL_ON) message.AddBool ("emacs:disable_antialiasing", true); message.AddString ("emacs:size", size_entry.Text ()); messenger.SendMessage (&message); } void HidePreview (void) { if (preview) { if (preview->LockLooper ()) preview->Quit (); /* I hope this works. */ else delete preview; preview = NULL; } } void UpdateStylesForIndex (int idx) { int n, i, previous_selection; uint32 flags; font_family family; font_style style; BStringItem *item; char *current_style; n = all_styles.CountItems (); current_style = NULL; previous_selection = font_style_pane.CurrentSelection (); if (previous_selection >= 0) { item = all_styles.ItemAt (previous_selection); current_style = strdup (item->Text ()); } font_style_pane.MakeEmpty (); all_styles.MakeEmpty (); if (get_font_family (idx, &family, &flags) == B_OK) { n = count_font_styles (family); for (i = 0; i < n; ++i) { if (get_font_style (family, i, &style, &flags) == B_OK) item = new BStringItem (style); else item = new BStringItem (""); if (current_style && pending_selection_idx < 0 && !strcmp (current_style, style)) pending_selection_idx = i; font_style_pane.AddItem (item); all_styles.AddItem (item); } } if (pending_selection_idx >= 0) { font_style_pane.Select (pending_selection_idx); font_style_pane.ScrollToSelection (); } pending_selection_idx = -1; UpdateForSelectedStyle (); if (current_style) free (current_style); } bool QuitRequested (void) { struct font_selection_dialog_message rq; rq.cancel = true; write_port (comm_port, 0, &rq, sizeof rq); return false; } void UpdateForSelectedStyle (void) { int style = font_style_pane.CurrentSelection (); if (style < 0) ok_button.SetEnabled (false); else ok_button.SetEnabled (true); if (style >= 0 && preview) UpdatePreview (); } void MessageReceived (BMessage *msg) { const char *text; int idx; struct font_selection_dialog_message rq; if (msg->what == FONT_FAMILY_SELECTED) { idx = font_family_pane.CurrentSelection (); UpdateStylesForIndex (idx); } else if (msg->what == FONT_STYLE_SELECTED) UpdateForSelectedStyle (); else if (msg->what == B_OK && font_style_pane.CurrentSelection () >= 0) { text = size_entry.Text (); rq.cancel = false; rq.family_idx = font_family_pane.CurrentSelection (); rq.style_idx = font_style_pane.CurrentSelection (); rq.size = atoi (text); rq.size_specified = rq.size > 0 || strlen (text); if (antialias_checkbox.Value () == B_CONTROL_ON) rq.disable_antialias = true; else rq.disable_antialias = false; write_port (comm_port, 0, &rq, sizeof rq); } else if (msg->what == B_CANCEL) { rq.cancel = true; write_port (comm_port, 0, &rq, sizeof rq); } else if (msg->what == SET_PREVIEW_DIALOG) { if (preview_checkbox.Value () == B_CONTROL_OFF) HidePreview (); else ShowPreview (); } else if (msg->what == QUIT_PREVIEW_DIALOG) { preview_checkbox.SetValue (B_CONTROL_OFF); HidePreview (); } else if (msg->what == UPDATE_PREVIEW_DIALOG) { if (preview) UpdatePreview (); } else if (msg->what == SET_DISABLE_ANTIALIASING) { if (preview) UpdatePreview (); } BWindow::MessageReceived (msg); } public: ~EmacsFontSelectionDialog (void) { if (preview) { if (preview->LockLooper ()) preview->Quit (); /* I hope this works. */ else delete preview; } font_family_pane.MakeEmpty (); font_style_pane.MakeEmpty (); font_family_pane.RemoveSelf (); font_style_pane.RemoveSelf (); antialias_checkbox.RemoveSelf (); preview_checkbox.RemoveSelf (); style_view.RemoveSelf (); font_family_scroller.RemoveSelf (); font_style_scroller.RemoveSelf (); cancel_button.RemoveSelf (); ok_button.RemoveSelf (); size_entry.RemoveSelf (); basic_view.RemoveSelf (); if (comm_port >= B_OK) delete_port (comm_port); } EmacsFontSelectionDialog (bool monospace_only, int initial_family_idx, int initial_style_idx, int initial_size, bool initial_antialias) : BWindow (BRect (0, 0, 500, 500), "Select font from list", B_TITLED_WINDOW_LOOK, B_MODAL_APP_WINDOW_FEEL, 0), basic_view (NULL, 0), antialias_checkbox ("Disable antialiasing", "Disable antialiasing", new BMessage (SET_DISABLE_ANTIALIASING)), preview_checkbox ("Show preview", "Show preview", new BMessage (SET_PREVIEW_DIALOG)), font_family_pane (BRect (0, 0, 0, 0), NULL, B_SINGLE_SELECTION_LIST, B_FOLLOW_ALL_SIDES), font_style_pane (BRect (0, 0, 0, 0), NULL, B_SINGLE_SELECTION_LIST, B_FOLLOW_ALL_SIDES), font_family_scroller (NULL, &font_family_pane, B_FOLLOW_LEFT | B_FOLLOW_TOP, 0, false, true), font_style_scroller (NULL, &font_style_pane, B_FOLLOW_ALL_SIDES, B_SUPPORTS_LAYOUT, false, true), style_view (&font_style_scroller, &antialias_checkbox, &preview_checkbox), all_families (20, true), all_styles (20, true), cancel_button ("Cancel", "Cancel", new BMessage (B_CANCEL)), ok_button ("OK", "OK", new BMessage (B_OK)), size_entry (NULL, "Size:", NULL, new BMessage (UPDATE_PREVIEW_DIALOG)), allow_monospace_only (monospace_only), pending_selection_idx (initial_style_idx), preview (NULL) { BStringItem *family_item; int i, n_families; font_family name; uint32 flags, c; BMessage *selection; BTextView *size_text; char format_buffer[4]; AddChild (&basic_view); basic_view.AddChild (&split_view); basic_view.AddChild (&cancel_button); basic_view.AddChild (&ok_button); basic_view.AddChild (&size_entry); split_view.AddChild (&font_family_scroller, 0.7); split_view.AddChild (&style_view, 0.3); style_view.AddChild (&font_style_scroller); style_view.AddChild (&antialias_checkbox); style_view.AddChild (&preview_checkbox); basic_view.SetViewUIColor (B_PANEL_BACKGROUND_COLOR); style_view.SetViewUIColor (B_PANEL_BACKGROUND_COLOR); FrameResized (801, 801); UpdateForSelectedStyle (); selection = new BMessage (FONT_FAMILY_SELECTED); font_family_pane.SetSelectionMessage (selection); selection = new BMessage (FONT_STYLE_SELECTED); font_style_pane.SetSelectionMessage (selection); selection = new BMessage (B_OK); font_style_pane.SetInvocationMessage (selection); selection = new BMessage (UPDATE_PREVIEW_DIALOG); size_entry.SetModificationMessage (selection); comm_port = create_port (1, "font dialog port"); n_families = count_font_families (); for (i = 0; i < n_families; ++i) { if (get_font_family (i, &name, &flags) == B_OK) { family_item = new BStringItem (name); all_families.AddItem (family_item); font_family_pane.AddItem (family_item); family_item->SetEnabled (!allow_monospace_only || flags & B_IS_FIXED); } else { family_item = new BStringItem (""); all_families.AddItem (family_item); font_family_pane.AddItem (family_item); } } if (initial_family_idx >= 0) { font_family_pane.Select (initial_family_idx); font_family_pane.ScrollToSelection (); } size_text = size_entry.TextView (); for (c = 0; c <= 47; ++c) size_text->DisallowChar (c); for (c = 58; c <= 127; ++c) size_text->DisallowChar (c); if (initial_size > 0 && initial_size < 1000) { sprintf (format_buffer, "%d", initial_size); size_entry.SetText (format_buffer); } if (!initial_antialias) antialias_checkbox.SetValue (B_CONTROL_ON); } void FrameResized (float new_width, float new_height) { BRect frame; float ok_height, ok_width; float cancel_height, cancel_width; float size_width, size_height; float bone; int max_height; ok_button.GetPreferredSize (&ok_width, &ok_height); cancel_button.GetPreferredSize (&cancel_width, &cancel_height); size_entry.GetPreferredSize (&size_width, &size_height); max_height = std::max (std::max (ok_height, cancel_height), size_height); SetSizeLimits (cancel_width + ok_width + size_width + 6, 65535, max_height + 64, 65535); frame = Frame (); basic_view.ResizeTo (BE_RECT_WIDTH (frame), BE_RECT_HEIGHT (frame)); split_view.ResizeTo (BE_RECT_WIDTH (frame) - 1, BE_RECT_HEIGHT (frame) - 4 - max_height); bone = BE_RECT_HEIGHT (frame) - 2 - max_height / 2; ok_button.MoveTo ((BE_RECT_WIDTH (frame) - 4 - cancel_width - ok_width), bone - ok_height / 2); cancel_button.MoveTo (BE_RECT_WIDTH (frame) - 2 - cancel_width, bone - cancel_height / 2); size_entry.MoveTo (2, bone - size_height / 2); ok_button.ResizeTo (ok_width, ok_height); cancel_button.ResizeTo (cancel_width, cancel_height); size_entry.ResizeTo (std::max (size_width, BE_RECT_WIDTH (frame) / 4), size_height); } void WaitForChoice (struct font_selection_dialog_message *msg, void (*process_pending_signals_function) (void), bool (*should_quit_function) (void)) { int32 reply_type; struct object_wait_info infos[2]; ssize_t status; infos[0].object = port_application_to_emacs; infos[0].type = B_OBJECT_TYPE_PORT; infos[0].events = B_EVENT_READ; infos[1].object = comm_port; infos[1].type = B_OBJECT_TYPE_PORT; infos[1].events = B_EVENT_READ; while (true) { status = wait_for_objects (infos, 2); if (status < B_OK) continue; if (infos[1].events & B_EVENT_READ) { if (read_port (comm_port, &reply_type, msg, sizeof *msg) >= B_OK) return; goto cancel; } if (infos[0].events & B_EVENT_READ) process_pending_signals_function (); if (should_quit_function ()) goto cancel; infos[0].events = B_EVENT_READ; infos[1].events = B_EVENT_READ; } cancel: msg->cancel = true; return; } status_t InitCheck (void) { return comm_port >= B_OK ? B_OK : comm_port; } }; class EmacsFilePanelCallbackLooper : public BLooper { port_id comm_port; void MessageReceived (BMessage *msg) { const char *str_path, *name; char *file_name, *str_buf; BEntry entry; BPath path; entry_ref ref; int32 old_what; if (msg->what == FILE_PANEL_SELECTION || ((msg->FindInt32 ("old_what", &old_what) == B_OK && old_what == FILE_PANEL_SELECTION))) { file_name = NULL; if (msg->FindRef ("refs", &ref) == B_OK && entry.SetTo (&ref, 0) == B_OK && entry.GetPath (&path) == B_OK) { str_path = path.Path (); if (str_path) file_name = strdup (str_path); } else if (msg->FindRef ("directory", &ref) == B_OK && entry.SetTo (&ref, 0) == B_OK && entry.GetPath (&path) == B_OK) { name = msg->GetString ("name"); str_path = path.Path (); if (name) { str_buf = (char *) alloca (std::strlen (str_path) + std::strlen (name) + 2); sprintf (str_buf, "%s/%s", str_path, name); file_name = strdup (str_buf); } } write_port (comm_port, 0, &file_name, sizeof file_name); } BLooper::MessageReceived (msg); } public: EmacsFilePanelCallbackLooper (void) : BLooper () { comm_port = create_port (1, "file panel port"); } ~EmacsFilePanelCallbackLooper (void) { delete_port (comm_port); } char * ReadFileName (void (*process_pending_signals_function) (void)) { object_wait_info infos[2]; ssize_t status; int32 reply_type; char *file_name; file_name = NULL; infos[0].object = port_application_to_emacs; infos[0].type = B_OBJECT_TYPE_PORT; infos[0].events = B_EVENT_READ; infos[1].object = comm_port; infos[1].type = B_OBJECT_TYPE_PORT; infos[1].events = B_EVENT_READ; while (true) { status = wait_for_objects (infos, 2); if (status == B_INTERRUPTED || status == B_WOULD_BLOCK) continue; if (infos[0].events & B_EVENT_READ) process_pending_signals_function (); if (infos[1].events & B_EVENT_READ) { status = read_port (comm_port, &reply_type, &file_name, sizeof file_name); if (status < B_OK) file_name = NULL; goto out; } infos[0].events = B_EVENT_READ; infos[1].events = B_EVENT_READ; } out: return file_name; } status_t InitCheck (void) { return comm_port >= B_OK ? B_OK : comm_port; } }; /* A view that is added as a child of a tooltip's text view, and prevents motion events from reaching it (thereby moving the tooltip). */ class EmacsMotionSuppressionView : public BView { void AttachedToWindow (void) { BView *text_view, *tooltip_view; /* We know that this view is a child of the text view, whose parent is the tooltip view, and that the tooltip view has already set its mouse event mask. */ text_view = Parent (); if (!text_view) return; tooltip_view = text_view->Parent (); if (!tooltip_view) return; tooltip_view->SetEventMask (B_KEYBOARD_EVENTS, 0); } public: EmacsMotionSuppressionView (void) : BView (BRect (-1, -1, 1, 1), NULL, 0, 0) { return; } }; static int32 start_running_application (void *data) { Emacs *app = (Emacs *) data; haiku_io_init_in_app_thread (); if (!app->Lock ()) gui_abort ("Failed to lock application"); app->Run (); app->Unlock (); return 0; } /* Take BITMAP, a reference to a BBitmap, and return a pointer to its data. */ void * BBitmap_data (void *bitmap) { return ((BBitmap *) bitmap)->Bits (); } /* Convert bitmap if required, placing the new bitmap in NEW_BITMAP, and return non-null if bitmap was successfully converted. Bitmaps should be freed with `BBitmap_free'. */ int BBitmap_convert (void *_bitmap, void **new_bitmap) { BBitmap *bitmap = (BBitmap *) _bitmap; if (bitmap->ColorSpace () == B_RGBA32) return -1; BRect bounds = bitmap->Bounds (); BBitmap *bmp = new (std::nothrow) BBitmap (bounds, B_RGBA32); if (!bmp || bmp->InitCheck () != B_OK) { if (bmp) delete bmp; return 0; } if (bmp->ImportBits (bitmap) != B_OK) { delete bmp; return 0; } *(BBitmap **) new_bitmap = bmp; return 1; } void BBitmap_free (void *bitmap) { delete (BBitmap *) bitmap; } /* Create new bitmap in RGB32 format, or in GRAY1 if MONO_P is non-zero. */ void * BBitmap_new (int width, int height, int mono_p) { BBitmap *bn = new (std::nothrow) BBitmap (BRect (0, 0, width - 1, height - 1), mono_p ? B_GRAY1 : B_RGB32); return bn->InitCheck () == B_OK ? (void *) bn : (void *) (delete bn, NULL); } void BBitmap_dimensions (void *bitmap, int *left, int *top, int *right, int *bottom, int32_t *bytes_per_row, int *mono_p) { BRect rect = ((BBitmap *) bitmap)->Bounds (); *left = rect.left; *top = rect.top; *right = rect.right; *bottom = rect.bottom; *bytes_per_row = ((BBitmap *) bitmap)->BytesPerRow (); *mono_p = (((BBitmap *) bitmap)->ColorSpace () == B_GRAY1); } static void wait_for_exit_of_app_thread (void) { status_t ret; be_app->PostMessage (QUIT_APPLICATION); wait_for_thread (app_thread, &ret); } /* Set up an application and return it. If starting the application thread fails, abort Emacs. */ void * BApplication_setup (void) { thread_id id; Emacs *app; if (be_app) return be_app; app = new Emacs; app->Unlock (); if ((id = spawn_thread (start_running_application, "Emacs app thread", B_DEFAULT_MEDIA_PRIORITY, app)) < 0) gui_abort ("spawn_thread failed"); resume_thread (id); app_thread = id; atexit (wait_for_exit_of_app_thread); return app; } /* Set up and return a window with its view put in VIEW. */ void * BWindow_new (void **view) { BWindow *window; BView *vw; window = new (std::nothrow) EmacsWindow; if (!window) { *view = NULL; return window; } vw = new (std::nothrow) EmacsView; if (!vw) { *view = NULL; window->LockLooper (); window->Quit (); return NULL; } /* Windows are created locked by the current thread, but calling Show for the first time causes them to be unlocked. To avoid a deadlock when a frame is created invisible in one thread, and another thread later tries to lock it, the window is unlocked here, and EmacsShow will lock it manually if it's being shown for the first time. */ window->UnlockLooper (); window->AddChild (vw); *view = vw; return window; } void BWindow_quit (void *window) { BWindow *w = (BWindow *) window; w->LockLooper (); w->Quit (); } /* Set WINDOW's offset to X, Y. */ void BWindow_set_offset (void *window, int x, int y) { BWindow *wn = (BWindow *) window; EmacsWindow *w = dynamic_cast (wn); if (w) { if (!w->LockLooper ()) gui_abort ("Failed to lock window looper setting offset"); w->EmacsMoveTo (x, y); w->UnlockLooper (); } else wn->MoveTo (x, y); } void BWindow_dimensions (void *window, int *width, int *height) { BWindow *w = (BWindow *) window; BRect frame = w->Frame (); *width = BE_RECT_WIDTH (frame); *height = BE_RECT_HEIGHT (frame); } /* Iconify WINDOW. */ void BWindow_iconify (void *window) { if (((BWindow *) window)->IsHidden ()) BWindow_set_visible (window, true); ((BWindow *) window)->Minimize (true); } /* Show or hide WINDOW. */ void BWindow_set_visible (void *window, int visible_p) { EmacsWindow *win = (EmacsWindow *) window; if (visible_p) { if (win->IsMinimized ()) win->Minimize (false); win->EmacsShow (); } else if (!win->IsHidden ()) { if (win->IsMinimized ()) win->Minimize (false); win->EmacsHide (); } } /* Change the title of WINDOW to the multibyte string TITLE. */ void BWindow_retitle (void *window, const char *title) { ((BWindow *) window)->SetTitle (title); } /* Resize WINDOW to WIDTH by HEIGHT. */ void BWindow_resize (void *window, int width, int height) { ((BWindow *) window)->ResizeTo (width - 1, height - 1); } /* Activate WINDOW, making it the subject of keyboard focus and bringing it to the front of the screen. */ void BWindow_activate (void *window) { ((BWindow *) window)->Activate (); } /* Return the pixel dimensions of the main screen in WIDTH and HEIGHT. */ void be_get_screen_dimensions (int *width, int *height) { BScreen screen; BRect frame; if (!screen.IsValid ()) gui_abort ("Invalid screen"); frame = screen.Frame (); *width = BE_RECT_WIDTH (frame); *height = BE_RECT_HEIGHT (frame); } /* Resize VIEW to WIDTH, HEIGHT. */ void BView_resize_to (void *view, int width, int height) { EmacsView *vw = (EmacsView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view for resize"); vw->ResizeTo (width, height); vw->AfterResize (); vw->UnlockLooper (); } void be_delete_cursor (void *cursor) { if (cursor) delete (BCursor *) cursor; } void * be_create_cursor_from_id (int id) { return new BCursor ((enum BCursorID) id); } void BView_set_view_cursor (void *view, void *cursor) { BView *v = (BView *) view; if (!v->LockLooper ()) gui_abort ("Failed to lock view setting cursor"); v->SetViewCursor ((BCursor *) cursor); v->UnlockLooper (); } void BWindow_Flush (void *window) { ((BWindow *) window)->Flush (); } /* Make a scrollbar, attach it to VIEW's window, and return it. */ void * be_make_scroll_bar_for_view (void *view, int horizontal_p, int x, int y, int x1, int y1) { EmacsScrollBar *scroll_bar; BView *vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock scrollbar owner"); scroll_bar = new EmacsScrollBar (x, y, x1, y1, horizontal_p, (EmacsView *) vw); vw->AddChild (scroll_bar); vw->UnlockLooper (); return scroll_bar; } void BScrollBar_delete (void *sb) { BView *view = (BView *) sb; BView *pr = view->Parent (); if (!pr->LockLooper ()) gui_abort ("Failed to lock scrollbar parent"); pr->RemoveChild (view); pr->UnlockLooper (); delete (EmacsScrollBar *) sb; } void BView_move_frame (void *view, int x, int y, int x1, int y1) { BView *vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view moving frame"); vw->MoveTo (x, y); vw->ResizeTo (x1 - x, y1 - y); vw->UnlockLooper (); } /* DRAGGING can either be 0 (which means to update everything), 1 (which means to update nothing), or -1 (which means to update only the thumb size and range). */ void BView_scroll_bar_update (void *sb, int portion, int whole, int position, int dragging, bool can_overscroll) { BScrollBar *bar = (BScrollBar *) sb; BMessage msg = BMessage (SCROLL_BAR_UPDATE); BMessenger mr = BMessenger (bar); msg.AddInt32 ("emacs:range", whole); msg.AddInt32 ("emacs:units", position); msg.AddInt32 ("emacs:portion", portion); msg.AddInt32 ("emacs:dragging", dragging); msg.AddBool ("emacs:overscroll", can_overscroll); mr.SendMessage (&msg); } /* Return the default scrollbar size. */ int BScrollBar_default_size (int horizontal_p) { return be_control_look->GetScrollBarWidth (horizontal_p ? B_HORIZONTAL : B_VERTICAL); } /* Invalidate VIEW, causing it to be drawn again. */ void BView_invalidate (void *view) { BView *vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Couldn't lock view while invalidating it"); vw->Invalidate (); vw->UnlockLooper (); } /* Lock VIEW in preparation for drawing operations. This should be called before any attempt to draw onto VIEW or to lock it for Cairo drawing. `BView_draw_unlock' should be called afterwards. If any drawing is going to take place, INVALID_REGION should be true, and X, Y, WIDTH, HEIGHT should specify a rectangle in which the drawing will take place. */ void BView_draw_lock (void *view, bool invalidate_region, int x, int y, int width, int height) { EmacsView *vw = (EmacsView *) view; if (vw->looper_locked_count) { vw->looper_locked_count++; if (invalidate_region && vw->offscreen_draw_view) vw->invalid_region.Include (BRect (x, y, x + width - 1, y + height - 1)); return; } BView *v = (BView *) find_appropriate_view_for_draw (vw); if (v != vw) { if (!vw->offscreen_draw_bitmap_1->Lock ()) gui_abort ("Failed to lock offscreen bitmap while acquiring draw lock"); } else if (!v->LockLooper ()) gui_abort ("Failed to lock draw view while acquiring draw lock"); if (v != vw && !vw->LockLooper ()) gui_abort ("Failed to lock view while acquiring draw lock"); if (invalidate_region && vw->offscreen_draw_view) vw->invalid_region.Include (BRect (x, y, x + width - 1, y + height - 1)); vw->looper_locked_count++; } void BView_invalidate_region (void *view, int x, int y, int width, int height) { EmacsView *vw = (EmacsView *) view; if (vw->offscreen_draw_view) vw->invalid_region.Include (BRect (x, y, x + width - 1, y + height - 1)); } void BView_draw_unlock (void *view) { EmacsView *vw = (EmacsView *) view; if (--vw->looper_locked_count) return; BView *v = (BView *) find_appropriate_view_for_draw (view); if (v == vw) vw->UnlockLooper (); else { vw->offscreen_draw_bitmap_1->Unlock (); vw->UnlockLooper (); } } void BWindow_center_on_screen (void *window) { BWindow *w = (BWindow *) window; w->CenterOnScreen (); } /* Import fringe bitmap (short array, low bit rightmost) BITS into BITMAP using the B_GRAY1 colorspace. */ void BBitmap_import_fringe_bitmap (void *bitmap, unsigned short *bits, int wd, int h) { BBitmap *bmp = (BBitmap *) bitmap; unsigned char *data = (unsigned char *) bmp->Bits (); int i; for (i = 0; i < h; i++) { if (wd <= 8) data[0] = bits[i] & 0xff; else { data[1] = bits[i] & 0xff; data[0] = bits[i] >> 8; } data += bmp->BytesPerRow (); } } /* Make a scrollbar at X, Y known to the view VIEW. */ void BView_publish_scroll_bar (void *view, int x, int y, int width, int height) { EmacsView *vw = (EmacsView *) view; if (vw->LockLooper ()) { vw->sb_region.Include (BRect (x, y, x - 1 + width, y - 1 + height)); vw->UnlockLooper (); } } void BView_forget_scroll_bar (void *view, int x, int y, int width, int height) { EmacsView *vw = (EmacsView *) view; if (vw->LockLooper ()) { vw->sb_region.Exclude (BRect (x, y, x - 1 + width, y - 1 + height)); vw->UnlockLooper (); } } bool BView_inside_scroll_bar (void *view, int x, int y) { EmacsView *vw = (EmacsView *) view; bool val; if (vw->LockLooper ()) { val = vw->sb_region.Contains (BPoint (x, y)); vw->UnlockLooper (); } else val = false; return val; } void BView_get_mouse (void *view, int *x, int *y) { BPoint l; BView *vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view in BView_get_mouse"); vw->GetMouse (&l, NULL, 1); vw->UnlockLooper (); *x = std::lrint (l.x); *y = std::lrint (l.y); } /* Perform an in-place conversion of X and Y from VIEW's coordinate system to its screen's coordinate system. */ void BView_convert_to_screen (void *view, int *x, int *y) { BPoint l = BPoint (*x, *y); BView *vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view in convert_to_screen"); vw->ConvertToScreen (&l); vw->UnlockLooper (); *x = std::lrint (l.x); *y = std::lrint (l.y); } void BView_convert_from_screen (void *view, int *x, int *y) { BPoint l = BPoint (*x, *y); BView *vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view in convert_from_screen"); vw->ConvertFromScreen (&l); vw->UnlockLooper (); *x = std::lrint (l.x); *y = std::lrint (l.y); } /* Decorate or undecorate WINDOW depending on DECORATE_P. */ void BWindow_change_decoration (void *window, int decorate_p) { EmacsWindow *w = (EmacsWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window while changing its decorations"); if (!w->override_redirect_p) { if (decorate_p) w->SetLook (B_TITLED_WINDOW_LOOK); else w->SetLook (B_NO_BORDER_WINDOW_LOOK); } else { if (decorate_p) w->pre_override_redirect_look = B_TITLED_WINDOW_LOOK; else w->pre_override_redirect_look = B_NO_BORDER_WINDOW_LOOK; } w->UnlockLooper (); } /* Decorate WINDOW appropriately for use as a tooltip. */ void BWindow_set_tooltip_decoration (void *window) { EmacsWindow *w = (EmacsWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window while setting ttip decoration"); w->tooltip_p = true; w->RecomputeFeel (); w->SetLook (B_BORDERED_WINDOW_LOOK); w->SetFlags (B_NOT_ZOOMABLE | B_NOT_MINIMIZABLE | B_AVOID_FRONT | B_AVOID_FOCUS); w->UnlockLooper (); } /* Set B_AVOID_FOCUS on WINDOW if AVOID_FOCUS_P is non-nil, or clear it otherwise. */ void BWindow_set_avoid_focus (void *window, int avoid_focus_p) { BWindow *w = (BWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window while setting avoid focus"); if (!avoid_focus_p) w->SetFlags (w->Flags () & ~B_AVOID_FOCUS); else w->SetFlags (w->Flags () | B_AVOID_FOCUS); w->UnlockLooper (); } void BView_emacs_delete (void *view) { EmacsView *vw = (EmacsView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view while deleting it"); vw->RemoveSelf (); delete vw; } /* Create a popup menu. */ void * BPopUpMenu_new (const char *name) { BPopUpMenu *menu = new BPopUpMenu (name); menu->SetRadioMode (0); return menu; } /* Add a title item to MENU. These items cannot be highlighted or triggered, and their labels will display as bold text. */ void BMenu_add_title (void *menu, const char *text) { BMenu *be_menu = (BMenu *) menu; EmacsTitleMenuItem *it; it = new EmacsTitleMenuItem (text); be_menu->AddItem (it); } /* Add an item to the menu MENU. */ void BMenu_add_item (void *menu, const char *label, void *ptr, bool enabled_p, bool marked_p, bool mbar_p, void *mbw_ptr, const char *key, const char *help) { BMenu *m = (BMenu *) menu; BMessage *msg; if (ptr) msg = new BMessage (); EmacsMenuItem *it = new EmacsMenuItem (key, label, help, ptr ? msg : NULL); it->SetTarget (m->Window ()); it->SetEnabled (enabled_p); it->SetMarked (marked_p); if (mbar_p) { it->menu_bar_id = (intptr_t) ptr; it->wind_ptr = mbw_ptr; } it->menu_ptr = ptr; if (ptr) msg->AddPointer ("menuptr", ptr); m->AddItem (it); } /* Add a separator to the menu MENU. */ void BMenu_add_separator (void *menu) { BMenu *m = (BMenu *) menu; m->AddSeparatorItem (); } /* Create a submenu and attach it to MENU. */ void * BMenu_new_submenu (void *menu, const char *label, bool enabled_p) { BMenu *m = (BMenu *) menu; BMenu *mn = new BMenu (label, B_ITEMS_IN_COLUMN); mn->SetRadioMode (0); BMenuItem *i = new BMenuItem (mn); i->SetEnabled (enabled_p); m->AddItem (i); return mn; } /* Create a submenu that notifies Emacs upon opening. */ void * BMenu_new_menu_bar_submenu (void *menu, const char *label) { BMenu *m = (BMenu *) menu; BMenu *mn = new BMenu (label, B_ITEMS_IN_COLUMN); mn->SetRadioMode (0); BMenuItem *i = new BMenuItem (mn); i->SetEnabled (1); m->AddItem (i); return mn; } /* Run MENU, waiting for it to close, and return a pointer to the data of the selected item (if one exists), or NULL. X, Y should be in the screen coordinate system. */ void * BMenu_run (void *menu, int x, int y, void (*run_help_callback) (void *, void *), void (*block_input_function) (void), void (*unblock_input_function) (void), struct timespec (*process_pending_signals_function) (void), void *run_help_callback_data) { BPopUpMenu *mn = (BPopUpMenu *) menu; enum haiku_event_type type; void *buf; void *ptr = NULL; struct be_popup_menu_data data; struct object_wait_info infos[3]; struct haiku_menu_bar_help_event *event; BMessage *msg; ssize_t stat; struct timespec next_time; bigtime_t timeout; block_input_function (); port_popup_menu_to_emacs = create_port (1800, "popup menu port"); data.x = x; data.y = y; data.menu = mn; unblock_input_function (); if (port_popup_menu_to_emacs < B_OK) return NULL; block_input_function (); mn->SetRadioMode (0); buf = alloca (200); infos[0].object = port_popup_menu_to_emacs; infos[0].type = B_OBJECT_TYPE_PORT; infos[0].events = B_EVENT_READ; infos[1].object = spawn_thread (be_popup_menu_thread_entry, "Menu tracker", B_DEFAULT_MEDIA_PRIORITY, (void *) &data); infos[1].type = B_OBJECT_TYPE_THREAD; infos[1].events = B_EVENT_INVALID; infos[2].object = port_application_to_emacs; infos[2].type = B_OBJECT_TYPE_PORT; infos[2].events = B_EVENT_READ; unblock_input_function (); if (infos[1].object < B_OK) { block_input_function (); delete_port (port_popup_menu_to_emacs); unblock_input_function (); return NULL; } block_input_function (); resume_thread (infos[1].object); unblock_input_function (); while (true) { next_time = process_pending_signals_function (); if (next_time.tv_nsec < 0) timeout = 10000000000; else timeout = (next_time.tv_sec * 1000000 + next_time.tv_nsec / 1000); if ((stat = wait_for_objects_etc ((object_wait_info *) &infos, 3, B_RELATIVE_TIMEOUT, timeout)) < B_OK) { if (stat == B_INTERRUPTED || stat == B_TIMED_OUT || stat == B_WOULD_BLOCK) continue; else gui_abort ("Failed to wait for popup"); } if (infos[0].events & B_EVENT_READ) { while (!haiku_read_with_timeout (&type, buf, 200, 0, true)) { switch (type) { case MENU_BAR_HELP_EVENT: event = (struct haiku_menu_bar_help_event *) buf; run_help_callback (event->highlight_p ? event->data : NULL, run_help_callback_data); break; default: gui_abort ("Unknown popup menu event"); } } } if (infos[1].events & B_EVENT_INVALID) { block_input_function (); msg = (BMessage *) popup_track_message; if (popup_track_message) ptr = (void *) msg->GetPointer ("menuptr"); delete_port (port_popup_menu_to_emacs); unblock_input_function (); return ptr; } infos[0].events = B_EVENT_READ; infos[1].events = B_EVENT_INVALID; infos[2].events = B_EVENT_READ; } } /* Delete the entire menu hierarchy of MENU, and then delete MENU itself. */ void BPopUpMenu_delete (void *menu) { delete (BPopUpMenu *) menu; } /* Create a menubar, attach it to VIEW, and return it. */ void * BMenuBar_new (void *view) { BView *vw = (BView *) view; EmacsMenuBar *bar = new EmacsMenuBar (); if (!vw->LockLooper ()) gui_abort ("Failed to lock menu bar parent"); vw->AddChild ((BView *) bar); vw->UnlockLooper (); return bar; } /* Delete MENUBAR along with all subitems. */ void BMenuBar_delete (void *menubar) { BView *vw = (BView *) menubar; BView *p = vw->Parent (); EmacsWindow *window = (EmacsWindow *) p->Window (); if (!p->LockLooper ()) gui_abort ("Failed to lock menu bar parent while removing menubar"); window->SetKeyMenuBar (NULL); /* MenusEnded isn't called if the menu bar is destroyed before it closes. */ window->menu_bar_active_p = false; vw->RemoveSelf (); p->UnlockLooper (); delete vw; } /* Delete all items from MENU. */ void BMenu_delete_all (void *menu) { BMenu *mn = (BMenu *) menu; mn->RemoveItems (0, mn->CountItems (), true); } /* Delete COUNT items from MENU starting from START. */ void BMenu_delete_from (void *menu, int start, int count) { BMenu *mn = (BMenu *) menu; mn->RemoveItems (start, count, true); } /* Count items in menu MENU. */ int BMenu_count_items (void *menu) { return ((BMenu *) menu)->CountItems (); } /* Find the item in MENU at IDX. */ void * BMenu_item_at (void *menu, int idx) { return ((BMenu *) menu)->ItemAt (idx); } /* Set ITEM's label to LABEL. */ void BMenu_item_set_label (void *item, const char *label) { ((BMenuItem *) item)->SetLabel (label); } /* Get ITEM's menu. */ void * BMenu_item_get_menu (void *item) { return ((BMenuItem *) item)->Submenu (); } /* Emit a beep noise. */ void haiku_ring_bell (void) { beep (); } /* Create a BAlert with TEXT. */ void * BAlert_new (const char *text, enum haiku_alert_type type) { return new BAlert (NULL, text, NULL, NULL, NULL, B_WIDTH_AS_USUAL, (enum alert_type) type); } /* Add a button to ALERT and return the button. */ void * BAlert_add_button (void *alert, const char *text) { BAlert *al = (BAlert *) alert; al->AddButton (text); return al->ButtonAt (al->CountButtons () - 1); } /* Make sure the leftmost button is grouped to the left hand side of the alert. */ void BAlert_set_offset_spacing (void *alert) { BAlert *al = (BAlert *) alert; al->SetButtonSpacing (B_OFFSET_SPACING); } static int32 be_alert_thread_entry (void *thread_data) { BAlert *alert = (BAlert *) thread_data; int32 value; if (alert->LockLooper ()) value = alert->Go (); else value = -1; alert_popup_value = value; return 0; } /* Run ALERT, returning the number of the button that was selected, or -1 if no button was selected before the alert was closed. */ int32 BAlert_go (void *alert, void (*block_input_function) (void), void (*unblock_input_function) (void), void (*process_pending_signals_function) (void)) { struct object_wait_info infos[2]; ssize_t stat; BAlert *alert_object = (BAlert *) alert; infos[0].object = port_application_to_emacs; infos[0].type = B_OBJECT_TYPE_PORT; infos[0].events = B_EVENT_READ; block_input_function (); /* Alerts are created locked, just like other windows. */ alert_object->UnlockLooper (); infos[1].object = spawn_thread (be_alert_thread_entry, "Popup tracker", B_DEFAULT_MEDIA_PRIORITY, alert); infos[1].type = B_OBJECT_TYPE_THREAD; infos[1].events = B_EVENT_INVALID; unblock_input_function (); if (infos[1].object < B_OK) return -1; block_input_function (); resume_thread (infos[1].object); unblock_input_function (); while (true) { stat = wait_for_objects ((object_wait_info *) &infos, 2); if (stat == B_INTERRUPTED) continue; else if (stat < B_OK) gui_abort ("Failed to wait for popup dialog"); if (infos[1].events & B_EVENT_INVALID) return alert_popup_value; if (infos[0].events & B_EVENT_READ) process_pending_signals_function (); infos[0].events = B_EVENT_READ; infos[1].events = B_EVENT_INVALID; } } /* Enable or disable BUTTON depending on ENABLED_P. */ void BButton_set_enabled (void *button, int enabled_p) { ((BButton *) button)->SetEnabled (enabled_p); } /* Set VIEW's tooltip to TOOLTIP. */ void BView_set_tooltip (void *view, const char *tooltip) { ((BView *) view)->SetToolTip (tooltip); } /* Set VIEW's tooltip to a sticky tooltip at X by Y. */ void be_show_sticky_tooltip (void *view, const char *tooltip_text, int x, int y) { BToolTip *tooltip; BView *vw, *tooltip_view; BPoint point; vw = (BView *) view; if (!vw->LockLooper ()) gui_abort ("Failed to lock view while showing sticky tooltip"); vw->SetToolTip ((const char *) NULL); /* If the tooltip text is empty, then a tooltip object won't be created by SetToolTip. */ if (tooltip_text[0] == '\0') tooltip_text = " "; vw->SetToolTip (tooltip_text); tooltip = vw->ToolTip (); vw->GetMouse (&point, NULL, 1); point.x -= x; point.y -= y; point.x = -point.x; point.y = -point.y; /* We don't have to make the tooltip sticky since not receiving mouse movement is enough to prevent it from being hidden. */ tooltip->SetMouseRelativeLocation (point); /* Prevent the tooltip from moving in response to mouse movement. */ tooltip_view = tooltip->View (); if (tooltip_view) tooltip_view->AddChild (new EmacsMotionSuppressionView); vw->ShowToolTip (tooltip); vw->UnlockLooper (); } /* Delete ALERT. */ void BAlert_delete (void *alert) { delete (BAlert *) alert; } /* Place the resolution of the monitor in DPI in X_OUT and Y_OUT. */ void be_get_display_resolution (double *x_out, double *y_out) { BScreen s (B_MAIN_SCREEN_ID); monitor_info i; double x_inches, y_inches; BRect frame; if (!s.IsValid ()) gui_abort ("Invalid screen for resolution checks"); if (s.GetMonitorInfo (&i) == B_OK) { frame = s.Frame (); x_inches = (double) i.width * 25.4; y_inches = (double) i.height * 25.4; *x_out = (double) BE_RECT_WIDTH (frame) / x_inches; *y_out = (double) BE_RECT_HEIGHT (frame) / y_inches; return; } *x_out = 72.0; *y_out = 72.0; } /* Add WINDOW to OTHER_WINDOW's subset and parent it to OTHER_WINDOW. */ void EmacsWindow_parent_to (void *window, void *other_window) { EmacsWindow *w = (EmacsWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window while parenting"); w->ParentTo ((EmacsWindow *) other_window); w->UnlockLooper (); } void EmacsWindow_unparent (void *window) { EmacsWindow *w = (EmacsWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window while unparenting"); w->UnparentAndUnlink (); w->UnlockLooper (); } /* Place text describing the current version of Haiku in VERSION, which should be a buffer LEN bytes wide. */ void be_get_version_string (char *version, int len) { std::strncpy (version, "Unknown Haiku release", len - 1); version[len - 1] = '\0'; BPath path; if (find_directory (B_BEOS_LIB_DIRECTORY, &path) == B_OK) { path.Append ("libbe.so"); BAppFileInfo appFileInfo; version_info versionInfo; BFile file; if (file.SetTo (path.Path (), B_READ_ONLY) == B_OK && appFileInfo.SetTo (&file) == B_OK && appFileInfo.GetVersionInfo (&versionInfo, B_APP_VERSION_KIND) == B_OK && versionInfo.short_info[0] != '\0') { std::strncpy (version, versionInfo.short_info, len - 1); version[len - 1] = '\0'; } } } /* Return the amount of color planes in the current display. */ int be_get_display_planes (void) { color_space space = dpy_color_space; BScreen screen; if (space == B_NO_COLOR_SPACE) { if (!screen.IsValid ()) gui_abort ("Invalid screen"); space = dpy_color_space = screen.ColorSpace (); } switch (space) { case B_RGB32: case B_RGB24: return 24; case B_RGB16: return 16; case B_RGB15: return 15; case B_CMAP8: case B_GRAY8: return 8; case B_GRAY1: return 1; default: gui_abort ("Bad colorspace for screen"); } /* https://www.haiku-os.org/docs/api/classBScreen.html says a valid screen can't be anything else. */ return -1; } /* Return the amount of colors the display can handle. */ int be_get_display_color_cells (void) { BScreen screen; color_space space = dpy_color_space; if (space == B_NO_COLOR_SPACE) { if (!screen.IsValid ()) gui_abort ("Invalid screen"); space = dpy_color_space = screen.ColorSpace (); } switch (space) { case B_RGB32: case B_RGB24: return 16777216; case B_RGB16: return 65536; case B_RGB15: return 32768; case B_CMAP8: case B_GRAY8: return 256; case B_GRAY1: return 2; default: gui_abort ("Bad colorspace for screen"); } return -1; } /* Return whether or not the current display is only capable of producing grayscale colors. */ bool be_is_display_grayscale (void) { BScreen screen; color_space space = dpy_color_space; if (space == B_NO_COLOR_SPACE) { if (!screen.IsValid ()) gui_abort ("Invalid screen"); space = dpy_color_space = screen.ColorSpace (); } return space == B_GRAY8 || space == B_GRAY1; } /* Warp the pointer to X by Y. */ void be_warp_pointer (int x, int y) { /* We're not supposed to use the following function without a BWindowScreen object, but in Haiku nothing actually prevents us from doing so. */ set_mouse_position (x, y); } /* Update the position of CHILD in WINDOW without actually moving it. */ void EmacsWindow_move_weak_child (void *window, void *child, int xoff, int yoff) { EmacsWindow *w = (EmacsWindow *) window; EmacsWindow *c = (EmacsWindow *) child; if (!w->LockLooper ()) gui_abort ("Couldn't lock window for weak move"); w->MoveChild (c, xoff, yoff, 1); w->UnlockLooper (); } /* Find an appropriate view to draw onto. If VW is double-buffered, this will be the view used for double buffering instead of VW itself. */ void * find_appropriate_view_for_draw (void *vw) { BView *v = (BView *) vw; EmacsView *ev = dynamic_cast(v); if (!ev) return v; return ev->offscreen_draw_view ? ev->offscreen_draw_view : vw; } /* Set up double buffering for VW. */ void EmacsView_set_up_double_buffering (void *vw) { EmacsView *view = (EmacsView *) vw; if (!view->LockLooper ()) gui_abort ("Couldn't lock view while setting up double buffering"); if (view->offscreen_draw_view) { view->UnlockLooper (); return; } view->SetUpDoubleBuffering (); view->UnlockLooper (); } /* Flip and invalidate the view VW. */ void EmacsView_flip_and_blit (void *vw) { EmacsView *view = (EmacsView *) vw; if (!view->offscreen_draw_view) return; if (!view->LockLooper ()) gui_abort ("Couldn't lock view in flip_and_blit"); view->FlipBuffers (); view->UnlockLooper (); } /* Disable double buffering for VW. */ void EmacsView_disable_double_buffering (void *vw) { EmacsView *view = (EmacsView *) vw; if (!view->LockLooper ()) gui_abort ("Couldn't lock view tearing down double buffering"); view->TearDownDoubleBuffering (); view->UnlockLooper (); } /* Return non-0 if VW is double-buffered. */ int EmacsView_double_buffered_p (void *vw) { EmacsView *view = (EmacsView *) vw; if (!view->LockLooper ()) gui_abort ("Couldn't lock view testing double buffering status"); int db_p = !!view->offscreen_draw_view; view->UnlockLooper (); return db_p; } /* Popup a file dialog. */ char * be_popup_file_dialog (int open_p, const char *default_dir, int must_match_p, int dir_only_p, void *window, const char *save_text, const char *prompt, void (*process_pending_signals_function) (void)) { BWindow *panel_window; BEntry path; BMessage msg (FILE_PANEL_SELECTION); BFilePanel panel (open_p ? B_OPEN_PANEL : B_SAVE_PANEL, NULL, NULL, (dir_only_p ? B_DIRECTORY_NODE : B_FILE_NODE | B_DIRECTORY_NODE)); char *file_name; EmacsFilePanelCallbackLooper *looper; looper = new EmacsFilePanelCallbackLooper; if (looper->InitCheck () < B_OK) { delete looper; return NULL; } if (default_dir) { if (path.SetTo (default_dir, 0) != B_OK) default_dir = NULL; } panel_window = panel.Window (); if (default_dir) panel.SetPanelDirectory (&path); if (save_text) panel.SetSaveText (save_text); panel_window->SetTitle (prompt); panel_window->SetFeel (B_MODAL_APP_WINDOW_FEEL); panel.SetHideWhenDone (false); panel.SetTarget (BMessenger (looper)); panel.SetMessage (&msg); panel.Show (); looper->Run (); file_name = looper->ReadFileName (process_pending_signals_function); if (looper->Lock ()) looper->Quit (); return file_name; } /* Move the pointer into MBAR and start tracking. Return whether the menu bar was opened correctly. */ bool BMenuBar_start_tracking (void *mbar) { EmacsMenuBar *mb = (EmacsMenuBar *) mbar; BMessenger messenger (mb); BMessage reply; messenger.SendMessage (SHOW_MENU_BAR, &reply); return reply.what == BE_MENU_BAR_OPEN; } #ifdef HAVE_NATIVE_IMAGE_API int be_can_translate_type_to_bitmap_p (const char *mime) { BTranslatorRoster *r = BTranslatorRoster::Default (); translator_id *ids; int32 id_len; if (r->GetAllTranslators (&ids, &id_len) != B_OK) return 0; int found_in = 0; int found_out = 0; for (int i = 0; i < id_len; ++i) { found_in = 0; found_out = 0; const translation_format *i_fmts; const translation_format *o_fmts; int32 i_count, o_count; if (r->GetInputFormats (ids[i], &i_fmts, &i_count) != B_OK) continue; if (r->GetOutputFormats (ids[i], &o_fmts, &o_count) != B_OK) continue; for (int x = 0; x < i_count; ++x) { if (!strcmp (i_fmts[x].MIME, mime)) { found_in = 1; break; } } for (int x = 0; x < i_count; ++x) { if (!strcmp (o_fmts[x].MIME, "image/x-be-bitmap") || !strcmp (o_fmts[x].MIME, "image/x-vnd.Be-bitmap")) { found_out = 1; break; } } if (found_in && found_out) break; } delete [] ids; return found_in && found_out; } void * be_translate_bitmap_from_file_name (const char *filename) { BBitmap *bm = BTranslationUtils::GetBitmap (filename); return bm; } void * be_translate_bitmap_from_memory (const void *buf, size_t bytes) { BMemoryIO io (buf, bytes); BBitmap *bm = BTranslationUtils::GetBitmap (&io); return bm; } #endif /* Return the size of BITMAP's data, in bytes. */ size_t BBitmap_bytes_length (void *bitmap) { BBitmap *bm = (BBitmap *) bitmap; return bm->BitsLength (); } /* Show VIEW's tooltip. */ void BView_show_tooltip (void *view) { BView *vw = (BView *) view; if (vw->LockLooper ()) { vw->ShowToolTip (vw->ToolTip ()); vw->UnlockLooper (); } } #ifdef USE_BE_CAIRO /* Return VIEW's cairo context. */ cairo_t * EmacsView_cairo_context (void *view) { EmacsView *vw = (EmacsView *) view; return vw->cr_context; } /* Transfer each clip rectangle in VIEW to the cairo context CTX. */ void BView_cr_dump_clipping (void *view, cairo_t *ctx) { BView *vw = (BView *) find_appropriate_view_for_draw (view); BRegion cr; vw->GetClippingRegion (&cr); for (int i = 0; i < cr.CountRects (); ++i) { BRect r = cr.RectAt (i); cairo_rectangle (ctx, r.left, r.top, BE_RECT_WIDTH (r), BE_RECT_HEIGHT (r)); } cairo_clip (ctx); } /* Lock WINDOW in preparation for drawing using Cairo. */ void EmacsWindow_begin_cr_critical_section (void *window) { BWindow *w = (BWindow *) window; BView *vw = (BView *) w->FindView ("Emacs"); EmacsView *ev = dynamic_cast (vw); if (ev && !ev->cr_surface_lock.Lock ()) gui_abort ("Couldn't lock view cairo surface"); } /* Unlock WINDOW in preparation for drawing using Cairo. */ void EmacsWindow_end_cr_critical_section (void *window) { BWindow *w = (BWindow *) window; BView *vw = (BView *) w->FindView ("Emacs"); EmacsView *ev = dynamic_cast (vw); if (ev) ev->cr_surface_lock.Unlock (); } #endif /* Get the width of STR in the plain font. */ int be_string_width_with_plain_font (const char *str) { return be_plain_font->StringWidth (str); } /* Get the ascent + descent of the plain font. */ int be_plain_font_height (void) { struct font_height fheight; be_plain_font->GetHeight (&fheight); return fheight.ascent + fheight.descent; } /* Return the number of physical displays connected. */ int be_get_display_screens (void) { int count = 1; BScreen scr; if (!scr.IsValid ()) gui_abort ("Main screen vanished!"); while (scr.SetToNext () == B_OK && scr.IsValid ()) ++count; return count; } /* Set the minimum width the user can resize WINDOW to. */ /* Synchronize WINDOW's connection to the App Server. */ void BWindow_sync (void *window) { BWindow *w = (BWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window looper for sync"); w->Sync (); w->UnlockLooper (); } /* Set the alignment of WINDOW's dimensions. */ void BWindow_set_size_alignment (void *window, int align_width, int align_height) { BWindow *w = (BWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window looper setting alignment"); #if 0 /* Haiku does not currently implement SetWindowAlignment. */ if (w->SetWindowAlignment (B_PIXEL_ALIGNMENT, -1, -1, align_width, align_width, -1, -1, align_height, align_height) != B_NO_ERROR) gui_abort ("Invalid pixel alignment"); #endif w->UnlockLooper (); } void BWindow_send_behind (void *window, void *other_window) { BWindow *w = (BWindow *) window; BWindow *other = (BWindow *) other_window; if (!w->LockLooper ()) gui_abort ("Failed to lock window in order to send it behind another"); w->SendBehind (other); w->UnlockLooper (); } bool BWindow_is_active (void *window) { BWindow *w = (BWindow *) window; return w->IsActive (); } bool be_use_subpixel_antialiasing (void) { bool current_subpixel_antialiasing; if (get_subpixel_antialiasing (¤t_subpixel_antialiasing) != B_OK) return false; return current_subpixel_antialiasing; } void BWindow_set_override_redirect (void *window, bool override_redirect_p) { EmacsWindow *w = (EmacsWindow *) window; if (w->LockLooper ()) { if (override_redirect_p && !w->override_redirect_p) { w->override_redirect_p = true; w->pre_override_redirect_look = w->Look (); w->RecomputeFeel (); w->SetLook (B_NO_BORDER_WINDOW_LOOK); w->pre_override_redirect_workspaces = w->Workspaces (); w->SetWorkspaces (B_ALL_WORKSPACES); } else if (w->override_redirect_p) { w->override_redirect_p = false; w->SetLook (w->pre_override_redirect_look); w->RecomputeFeel (); w->SetWorkspaces (w->pre_override_redirect_workspaces); } w->UnlockLooper (); } } /* Find a resource by the name NAME inside the settings file. The string returned is in UTF-8 encoding, and will stay allocated as long as the BApplication (a.k.a display) is alive. */ const char * be_find_setting (const char *name) { Emacs *app = (Emacs *) be_app; const char *value; /* Note that this is thread-safe since the constructor of `Emacs' runs in the main thread. */ if (!app->settings_valid_p) return NULL; if (app->settings.FindString (name, 0, &value) != B_OK) return NULL; return value; } void BMessage_delete (void *message) { delete (BMessage *) message; } static int32 be_drag_message_thread_entry (void *thread_data) { BMessenger *messenger; BMessage reply; messenger = (BMessenger *) thread_data; messenger->SendMessage (WAIT_FOR_RELEASE, &reply); return 0; } bool be_drag_message (void *view, void *message, bool allow_same_view, void (*block_input_function) (void), void (*unblock_input_function) (void), void (*process_pending_signals_function) (void), bool (*should_quit_function) (void)) { EmacsView *vw = (EmacsView *) view; EmacsWindow *window = (EmacsWindow *) vw->Window (); BMessage *msg = (BMessage *) message; BMessage wait_for_release; BMessenger messenger (vw); BMessage cancel_message (CANCEL_DROP); struct object_wait_info infos[2]; ssize_t stat; thread_id window_thread; block_input_function (); if (!allow_same_view) window_thread = window->Looper ()->Thread (); if (!allow_same_view && (msg->ReplaceInt64 ("emacs:thread_id", window_thread) == B_NAME_NOT_FOUND)) msg->AddInt64 ("emacs:thread_id", window_thread); if (!vw->LockLooper ()) gui_abort ("Failed to lock view looper for drag"); vw->DragMessage (msg, BRect (0, 0, 0, 0)); vw->UnlockLooper (); infos[0].object = port_application_to_emacs; infos[0].type = B_OBJECT_TYPE_PORT; infos[0].events = B_EVENT_READ; infos[1].object = spawn_thread (be_drag_message_thread_entry, "Drag waiter thread", B_DEFAULT_MEDIA_PRIORITY, (void *) &messenger); infos[1].type = B_OBJECT_TYPE_THREAD; infos[1].events = B_EVENT_INVALID; unblock_input_function (); if (infos[1].object < B_OK) return false; block_input_function (); resume_thread (infos[1].object); unblock_input_function (); drag_and_drop_in_progress = true; while (true) { block_input_function (); stat = wait_for_objects ((struct object_wait_info *) &infos, 2); unblock_input_function (); if (stat == B_INTERRUPTED || stat == B_TIMED_OUT || stat == B_WOULD_BLOCK) continue; if (stat < B_OK) gui_abort ("Failed to wait for drag"); if (infos[0].events & B_EVENT_READ) process_pending_signals_function (); if (should_quit_function ()) { /* Do the best we can to prevent something from being dropped, since Haiku doesn't provide a way to actually cancel drag-and-drop. */ if (vw->LockLooper ()) { vw->DragMessage (&cancel_message, BRect (0, 0, 0, 0)); vw->UnlockLooper (); } messenger.SendMessage (CANCEL_DROP); drag_and_drop_in_progress = false; return true; } if (infos[1].events & B_EVENT_INVALID) { drag_and_drop_in_progress = false; return false; } infos[0].events = B_EVENT_READ; infos[1].events = B_EVENT_INVALID; } } bool be_drag_and_drop_in_progress (void) { return drag_and_drop_in_progress; } /* Replay the menu bar click event EVENT. Return whether or not the menu bar actually opened. */ bool be_replay_menu_bar_event (void *menu_bar, struct haiku_menu_bar_click_event *event) { BMenuBar *m = (BMenuBar *) menu_bar; BMessenger messenger (m); BMessage reply, msg (REPLAY_MENU_BAR); msg.AddPoint ("emacs:point", BPoint (event->x, event->y)); messenger.SendMessage (&msg, &reply); return reply.what == BE_MENU_BAR_OPEN; } void BWindow_set_z_group (void *window, enum haiku_z_group z_group) { EmacsWindow *w = (EmacsWindow *) window; if (w->LockLooper ()) { if (w->z_group != z_group) { w->z_group = z_group; w->RecomputeFeel (); if (w->z_group == Z_GROUP_BELOW) w->SetFlags (w->Flags () | B_AVOID_FRONT); else w->SetFlags (w->Flags () & ~B_AVOID_FRONT); } w->UnlockLooper (); } } int be_get_ui_color (const char *name, uint32_t *color) { color_which which; rgb_color rgb; which = which_ui_color (name); if (which == B_NO_COLOR) return 1; rgb = ui_color (which); *color = (rgb.blue | rgb.green << 8 | rgb.red << 16 | 255 << 24); return 0; } bool be_select_font (void (*process_pending_signals_function) (void), bool (*should_quit_function) (void), haiku_font_family_or_style *family, haiku_font_family_or_style *style, int *size, bool allow_monospace_only, int initial_family, int initial_style, int initial_size, bool initial_antialias, bool *disable_antialias) { EmacsFontSelectionDialog *dialog; struct font_selection_dialog_message msg; uint32 flags; font_family family_buffer; font_style style_buffer; dialog = new EmacsFontSelectionDialog (allow_monospace_only, initial_family, initial_style, initial_size, initial_antialias); dialog->CenterOnScreen (); if (dialog->InitCheck () < B_OK) { dialog->Quit (); return false; } dialog->Show (); dialog->WaitForChoice (&msg, process_pending_signals_function, should_quit_function); if (!dialog->LockLooper ()) gui_abort ("Failed to lock font selection dialog looper"); dialog->Quit (); if (msg.cancel) return false; if (get_font_family (msg.family_idx, &family_buffer, &flags) != B_OK || get_font_style (family_buffer, msg.style_idx, &style_buffer, &flags) != B_OK) return false; memcpy (family, family_buffer, sizeof *family); memcpy (style, style_buffer, sizeof *style); *size = msg.size_specified ? msg.size : -1; *disable_antialias = msg.disable_antialias; return true; } void BWindow_set_sticky (void *window, bool sticky) { BWindow *w = (BWindow *) window; if (w->LockLooper ()) { w->SetFlags (sticky ? (w->Flags () | B_SAME_POSITION_IN_ALL_WORKSPACES) : w->Flags () & ~B_SAME_POSITION_IN_ALL_WORKSPACES); w->UnlockLooper (); } } status_t be_roster_launch (const char *type, const char *file, char **cargs, ptrdiff_t nargs, void *message, team_id *team_id) { BEntry entry; entry_ref ref; if (type) { if (message) return be_roster->Launch (type, (BMessage *) message, team_id); return be_roster->Launch (type, (nargs > INT_MAX ? INT_MAX : nargs), cargs, team_id); } if (entry.SetTo (file) != B_OK) return B_ERROR; if (entry.GetRef (&ref) != B_OK) return B_ERROR; if (message) return be_roster->Launch (&ref, (BMessage *) message, team_id); return be_roster->Launch (&ref, (nargs > INT_MAX ? INT_MAX : nargs), cargs, team_id); } void * be_create_pixmap_cursor (void *bitmap, int x, int y) { BBitmap *bm; BCursor *cursor; bm = (BBitmap *) bitmap; cursor = new BCursor (bm, BPoint (x, y)); if (cursor->InitCheck () != B_OK) { delete cursor; return NULL; } return cursor; } void be_get_window_decorator_dimensions (void *window, int *left, int *top, int *right, int *bottom) { BWindow *wnd; BRect frame, window_frame; wnd = (BWindow *) window; if (!wnd->LockLooper ()) gui_abort ("Failed to lock window looper frame"); frame = wnd->DecoratorFrame (); window_frame = wnd->Frame (); if (left) *left = window_frame.left - frame.left; if (top) *top = window_frame.top - frame.top; if (right) *right = frame.right - window_frame.right; if (bottom) *bottom = frame.bottom - window_frame.bottom; wnd->UnlockLooper (); } void be_get_window_decorator_frame (void *window, int *left, int *top, int *width, int *height) { BWindow *wnd; BRect frame; wnd = (BWindow *) window; if (!wnd->LockLooper ()) gui_abort ("Failed to lock window looper frame"); frame = wnd->DecoratorFrame (); *left = frame.left; *top = frame.top; *width = BE_RECT_WIDTH (frame); *height = BE_RECT_HEIGHT (frame); wnd->UnlockLooper (); } /* Request that a MOVE_EVENT be sent for WINDOW. This is so that frame offsets can be updated after a frame parameter affecting decorators changes. Sending an event instead of updating the offsets directly avoids race conditions where events with older information are received after the update happens. */ void be_send_move_frame_event (void *window) { BWindow *wnd = (BWindow *) window; BMessenger msg (wnd); msg.SendMessage (SEND_MOVE_FRAME_EVENT); } void be_lock_window (void *window) { BWindow *wnd = (BWindow *) window; if (!wnd->LockLooper ()) gui_abort ("Failed to lock window looper"); } void be_unlock_window (void *window) { BWindow *wnd = (BWindow *) window; wnd->UnlockLooper (); } void be_set_window_fullscreen_mode (void *window, enum haiku_fullscreen_mode mode) { EmacsWindow *w = (EmacsWindow *) window; if (!w->LockLooper ()) gui_abort ("Failed to lock window to set fullscreen mode"); w->SetFullscreen (mode); w->UnlockLooper (); } bool be_get_explicit_workarea (int *x, int *y, int *width, int *height) { BDeskbar deskbar; BRect zoom; deskbar_location location; location = deskbar.Location (); if (location != B_DESKBAR_TOP && location != B_DESKBAR_BOTTOM) return false; zoom = get_zoom_rect (NULL); *x = zoom.left; *y = zoom.top; *width = BE_RECT_WIDTH (zoom); *height = BE_RECT_HEIGHT (zoom); return true; } /* Clear the grab view. This has to be called manually from some places, since we don't get B_MOUSE_UP messages after a popup menu is run. */ void be_clear_grab_view (void) { if (grab_view_locker.Lock ()) { grab_view = NULL; grab_view_locker.Unlock (); } } void be_set_use_frame_synchronization (void *view, bool sync) { EmacsView *vw; vw = (EmacsView *) view; vw->SetFrameSynchronization (sync); } status_t be_write_node_message (const char *path, const char *name, void *message) { BNode node (path); status_t rc; ssize_t flat, result; char *buffer; BMessage *msg; rc = node.InitCheck (); msg = (BMessage *) message; if (rc < B_OK) return rc; flat = msg->FlattenedSize (); if (flat < B_OK) return flat; buffer = new (std::nothrow) char[flat]; if (!buffer) return B_NO_MEMORY; rc = msg->Flatten (buffer, flat); if (rc < B_OK) { delete[] buffer; return rc; } result = node.WriteAttr (name, B_MIME_TYPE, 0, buffer, flat); delete[] buffer; if (result < B_OK) return result; if (result != flat) return B_ERROR; return B_OK; } void be_send_message (const char *app_id, void *message) { BMessenger messenger (app_id); messenger.SendMessage ((BMessage *) message); }