/* Functions for the NeXT/Open/GNUstep and macOS window system. Copyright (C) 1989, 1992-1994, 2005-2006, 2008-2021 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 . */ /* Originally by Carl Edman Updated by Christian Limpach (chris@nice.ch) OpenStep/Rhapsody port by Scott Bender (sbender@harmony-ds.com) macOS/Aqua port by Christophe de Dinechin (descubes@earthlink.net) GNUstep port and post-20 update by Adrian Robert (arobert@cogsci.ucsd.edu) */ /* This should be the first include, as it may set up #defines affecting interpretation of even the system includes. */ #include #include #include #include "lisp.h" #include "blockinput.h" #include "nsterm.h" #include "window.h" #include "character.h" #include "buffer.h" #include "keyboard.h" #include "termhooks.h" #include "fontset.h" #include "font.h" #ifdef NS_IMPL_COCOA #include #include "macfont.h" #endif #ifdef HAVE_NS static EmacsTooltip *ns_tooltip = nil; /* Static variables to handle AppleScript execution. */ static Lisp_Object as_script, *as_result; static int as_status; static ptrdiff_t image_cache_refcount; static struct ns_display_info *ns_display_info_for_name (Lisp_Object); /* ========================================================================== Internal utility functions ========================================================================== */ /* Let the user specify a Nextstep display with a Lisp object. OBJECT may be nil, a frame or a terminal object. nil stands for the selected frame--or, if that is not a Nextstep frame, the first Nextstep display on the list. */ static struct ns_display_info * check_ns_display_info (Lisp_Object object) { struct ns_display_info *dpyinfo = NULL; if (NILP (object)) { struct frame *sf = XFRAME (selected_frame); if (FRAME_NS_P (sf) && FRAME_LIVE_P (sf)) dpyinfo = FRAME_DISPLAY_INFO (sf); else if (x_display_list != 0) dpyinfo = x_display_list; else error ("Nextstep windows are not in use or not initialized"); } else if (TERMINALP (object)) { struct terminal *t = decode_live_terminal (object); if (t->type != output_ns) error ("Terminal %d is not a Nextstep display", t->id); dpyinfo = t->display_info.ns; } else if (STRINGP (object)) dpyinfo = ns_display_info_for_name (object); else { struct frame *f = decode_window_system_frame (object); dpyinfo = FRAME_DISPLAY_INFO (f); } return dpyinfo; } static id ns_get_window (Lisp_Object maybeFrame) { id view =nil, window =nil; if (!FRAMEP (maybeFrame) || !FRAME_NS_P (XFRAME (maybeFrame))) maybeFrame = selected_frame; /* wrong_type_argument (Qframep, maybeFrame); */ if (!NILP (maybeFrame)) view = FRAME_NS_VIEW (XFRAME (maybeFrame)); if (view) window =[view window]; return window; } /* Return the X display structure for the display named NAME. Open a new connection if necessary. */ static struct ns_display_info * ns_display_info_for_name (Lisp_Object name) { struct ns_display_info *dpyinfo; CHECK_STRING (name); for (dpyinfo = x_display_list; dpyinfo; dpyinfo = dpyinfo->next) if (!NILP (Fstring_equal (XCAR (dpyinfo->name_list_element), name))) return dpyinfo; error ("Emacs for Nextstep does not yet support multi-display"); Fx_open_connection (name, Qnil, Qnil); dpyinfo = x_display_list; if (dpyinfo == 0) error ("Display on %s not responding.\n", SDATA (name)); return dpyinfo; } static NSString * ns_filename_from_panel (NSSavePanel *panel) { #ifdef NS_IMPL_COCOA NSURL *url = [panel URL]; NSString *str = [url path]; return str; #else return [panel filename]; #endif } static NSString * ns_directory_from_panel (NSSavePanel *panel) { #ifdef NS_IMPL_COCOA NSURL *url = [panel directoryURL]; NSString *str = [url path]; return str; #else return [panel directory]; #endif } #ifndef NS_IMPL_COCOA static Lisp_Object interpret_services_menu (NSMenu *menu, Lisp_Object prefix, Lisp_Object old) /* -------------------------------------------------------------------------- Turn the input menu (an NSMenu) into a lisp list for tracking on lisp side. -------------------------------------------------------------------------- */ { int i, count; NSMenuItem *item; const char *name; Lisp_Object nameStr; unsigned short key; NSString *keys; Lisp_Object res; count = [menu numberOfItems]; for (i = 0; ioutput_data.ns->foreground_color release]; f->output_data.ns->foreground_color = col; [col getRed: &r green: &g blue: &b alpha: &alpha]; FRAME_FOREGROUND_PIXEL (f) = ARGB_TO_ULONG ((unsigned long) (alpha * 0xff), (unsigned long) (r * 0xff), (unsigned long) (g * 0xff), (unsigned long) (b * 0xff)); if (FRAME_NS_VIEW (f)) { update_face_from_frame_parameter (f, Qforeground_color, arg); /* recompute_basic_faces (f); */ if (FRAME_VISIBLE_P (f)) SET_FRAME_GARBAGED (f); } unblock_input (); } static void ns_set_background_color (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { struct face *face; NSColor *col; NSView *view = FRAME_NS_VIEW (f); EmacsCGFloat r, g, b, alpha; block_input (); if (ns_lisp_to_color (arg, &col)) { store_frame_param (f, Qbackground_color, oldval); unblock_input (); error ("Unknown color"); } [col retain]; [f->output_data.ns->background_color release]; f->output_data.ns->background_color = col; [col getRed: &r green: &g blue: &b alpha: &alpha]; FRAME_BACKGROUND_PIXEL (f) = ARGB_TO_ULONG ((unsigned long) (alpha * 0xff), (unsigned long) (r * 0xff), (unsigned long) (g * 0xff), (unsigned long) (b * 0xff)); if (view != nil) { [[view window] setBackgroundColor: col]; if (alpha != (EmacsCGFloat) 1.0) [[view window] setOpaque: NO]; else [[view window] setOpaque: YES]; face = FRAME_DEFAULT_FACE (f); if (face) { col = ns_lookup_indexed_color (NS_FACE_BACKGROUND (face), f); face->background = ns_index_color ([col colorWithAlphaComponent: alpha], f); update_face_from_frame_parameter (f, Qbackground_color, arg); } if (FRAME_VISIBLE_P (f)) { SET_FRAME_GARBAGED (f); ns_clear_frame (f); } } unblock_input (); } static void ns_set_cursor_color (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { NSColor *col; block_input (); if (ns_lisp_to_color (arg, &col)) { store_frame_param (f, Qcursor_color, oldval); unblock_input (); error ("Unknown color"); } [FRAME_CURSOR_COLOR (f) release]; FRAME_CURSOR_COLOR (f) = [col retain]; if (FRAME_VISIBLE_P (f)) { gui_update_cursor (f, 0); gui_update_cursor (f, 1); } update_face_from_frame_parameter (f, Qcursor_color, arg); unblock_input (); } static void ns_set_icon_name (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { NSView *view = FRAME_NS_VIEW (f); NSTRACE ("ns_set_icon_name"); /* See if it's changed. */ if (STRINGP (arg)) { if (STRINGP (oldval) && EQ (Fstring_equal (oldval, arg), Qt)) return; } else if (!STRINGP (oldval) && NILP (oldval) == NILP (arg)) return; fset_icon_name (f, arg); if (NILP (arg)) { if (!NILP (f->title)) arg = f->title; else /* Explicit name and no icon-name -> explicit_name. */ if (f->explicit_name) arg = f->name; else { /* No explicit name and no icon-name -> name has to be rebuild from icon_title_format. */ windows_or_buffers_changed = 62; return; } } /* Don't change the name if it's already NAME. */ if ([[view window] miniwindowTitle] && ([[[view window] miniwindowTitle] isEqualToString: [NSString stringWithLispString:arg]])) return; [[view window] setMiniwindowTitle: [NSString stringWithLispString:arg]]; } static void ns_set_name_internal (struct frame *f, Lisp_Object name) { NSView *view = FRAME_NS_VIEW (f); NSString *str = [NSString stringWithLispString: name]; /* Don't change the name if it's already NAME. */ if (! [[[view window] title] isEqualToString: str]) [[view window] setTitle: str]; if (STRINGP (f->icon_name)) str = [NSString stringWithLispString: f->icon_name]; if ([[view window] miniwindowTitle] && ! [[[view window] miniwindowTitle] isEqualToString: str]) [[view window] setMiniwindowTitle: str]; } static void ns_set_name (struct frame *f, Lisp_Object name, int explicit) { NSTRACE ("ns_set_name"); /* Make sure that requests from lisp code override requests from Emacs redisplay code. */ if (explicit) { /* If we're switching from explicit to implicit, we had better update the mode lines and thereby update the title. */ if (f->explicit_name && NILP (name)) update_mode_lines = 21; f->explicit_name = ! NILP (name); } else if (f->explicit_name) return; if (NILP (name)) name = [ns_app_name lispString]; else CHECK_STRING (name); /* Don't change the name if it's already NAME. */ if (! NILP (Fstring_equal (name, f->name))) return; fset_name (f, name); /* Title overrides explicit name. */ if (! NILP (f->title)) name = f->title; ns_set_name_internal (f, name); } static void ns_set_represented_filename (struct frame *f) { Lisp_Object filename; Lisp_Object buf = XWINDOW (f->selected_window)->contents; NSAutoreleasePool *pool; NSString *fstr; NSView *view = FRAME_NS_VIEW (f); NSTRACE ("ns_set_represented_filename"); if (f->explicit_name || ! NILP (f->title)) return; block_input (); pool = [[NSAutoreleasePool alloc] init]; filename = BVAR (XBUFFER (buf), filename); if (! NILP (filename)) { fstr = [NSString stringWithLispString:filename]; if (fstr == nil) fstr = @""; } else fstr = @""; #if defined (NS_IMPL_COCOA) && defined (MAC_OS_X_VERSION_10_7) /* Work around for Mach port leaks on macOS 10.15 (bug#38618). */ NSURL *fileURL = [NSURL fileURLWithPath:fstr isDirectory:NO]; NSNumber *isUbiquitousItem = [NSNumber numberWithBool:YES]; [fileURL getResourceValue:(id *)&isUbiquitousItem forKey:NSURLIsUbiquitousItemKey error:nil]; if ([isUbiquitousItem boolValue]) fstr = @""; #endif #ifdef NS_IMPL_COCOA /* Work around a bug observed on 10.3 and later where setTitleWithRepresentedFilename does not clear out previous state if given filename does not exist. */ if (! [[NSFileManager defaultManager] fileExistsAtPath: fstr]) [[view window] setRepresentedFilename: @""]; #endif [[view window] setRepresentedFilename: fstr]; [pool release]; unblock_input (); } /* This function should be called when the user's lisp code has specified a name for the frame; the name will override any set by the redisplay code. */ static void ns_explicitly_set_name (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { NSTRACE ("ns_explicitly_set_name"); ns_set_name (f, arg, 1); } /* This function should be called by Emacs redisplay code to set the name; names set this way will never override names set by the user's lisp code. */ void ns_implicitly_set_name (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { NSTRACE ("ns_implicitly_set_name"); if (ns_use_proxy_icon) ns_set_represented_filename (f); ns_set_name (f, arg, 0); } /* Change the title of frame F to NAME. If NAME is nil, use the frame name as the title. */ static void ns_set_title (struct frame *f, Lisp_Object name, Lisp_Object old_name) { NSTRACE ("ns_set_title"); /* Don't change the title if it's already NAME. */ if (EQ (name, f->title)) return; update_mode_lines = 22; fset_title (f, name); if (NILP (name)) name = f->name; else CHECK_STRING (name); ns_set_name_internal (f, name); } void ns_set_doc_edited (void) { NSAutoreleasePool *pool; Lisp_Object tail, frame; block_input (); pool = [[NSAutoreleasePool alloc] init]; FOR_EACH_FRAME (tail, frame) { BOOL edited = NO; struct frame *f = XFRAME (frame); struct window *w; NSView *view; if (! FRAME_NS_P (f)) continue; w = XWINDOW (FRAME_SELECTED_WINDOW (f)); view = FRAME_NS_VIEW (f); if (!MINI_WINDOW_P (w)) edited = ! NILP (Fbuffer_modified_p (w->contents)) && ! NILP (Fbuffer_file_name (w->contents)); [[view window] setDocumentEdited: edited]; } [pool release]; unblock_input (); } static void ns_set_menu_bar_lines (struct frame *f, Lisp_Object value, Lisp_Object oldval) { int nlines; if (FRAME_MINIBUF_ONLY_P (f)) return; if (TYPE_RANGED_FIXNUMP (int, value)) nlines = XFIXNUM (value); else nlines = 0; FRAME_MENU_BAR_LINES (f) = 0; if (nlines) { FRAME_EXTERNAL_MENU_BAR (f) = 1; /* Does for all frames, whereas we just want for one frame [NSMenu setMenuBarVisible: YES]; */ } else { if (FRAME_EXTERNAL_MENU_BAR (f) == 1) free_frame_menubar (f); /* [NSMenu setMenuBarVisible: NO]; */ FRAME_EXTERNAL_MENU_BAR (f) = 0; } } /* tabbar support */ static void ns_set_tab_bar_lines (struct frame *f, Lisp_Object value, Lisp_Object oldval) { /* Currently unimplemented. */ NSTRACE ("ns_set_tab_bar_lines"); } /* toolbar support */ static void ns_set_tool_bar_lines (struct frame *f, Lisp_Object value, Lisp_Object oldval) { /* Currently, when the tool bar changes state, the frame is resized. TODO: It would be better if this didn't occur when 1) the frame is full height or maximized or 2) when specified by `frame-inhibit-implied-resize'. */ int nlines; NSTRACE ("ns_set_tool_bar_lines"); if (FRAME_MINIBUF_ONLY_P (f)) return; if (RANGED_FIXNUMP (0, value, INT_MAX)) nlines = XFIXNAT (value); else nlines = 0; if (nlines) { FRAME_EXTERNAL_TOOL_BAR (f) = 1; update_frame_tool_bar (f); } else { if (FRAME_EXTERNAL_TOOL_BAR (f)) { free_frame_tool_bar (f); FRAME_EXTERNAL_TOOL_BAR (f) = 0; { EmacsView *view = FRAME_NS_VIEW (f); int fs_state = [view fullscreenState]; if (fs_state == FULLSCREEN_MAXIMIZED) { [view setFSValue:FULLSCREEN_WIDTH]; } else if (fs_state == FULLSCREEN_HEIGHT) { [view setFSValue:FULLSCREEN_NONE]; } } } } adjust_frame_size (f, -1, -1, 2, false, Qtool_bar_lines); } static void ns_set_child_frame_border_width (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { int border; if (NILP (arg)) border = -1; else if (RANGED_FIXNUMP (0, arg, INT_MAX)) border = XFIXNAT (arg); else signal_error ("Invalid child frame border width", arg); if (border != FRAME_CHILD_FRAME_BORDER_WIDTH (f)) { f->child_frame_border_width = border; if (FRAME_NATIVE_WINDOW (f) != 0) adjust_frame_size (f, -1, -1, 3, 0, Qchild_frame_border_width); SET_FRAME_GARBAGED (f); } } static void ns_set_internal_border_width (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { int old_width = FRAME_INTERNAL_BORDER_WIDTH (f); int new_width = check_int_nonnegative (arg); if (new_width == old_width) return; f->internal_border_width = new_width; if (FRAME_NATIVE_WINDOW (f) != 0) adjust_frame_size (f, -1, -1, 3, 0, Qinternal_border_width); SET_FRAME_GARBAGED (f); } static void ns_implicitly_set_icon_type (struct frame *f) { Lisp_Object tem; EmacsView *view = FRAME_NS_VIEW (f); id image = nil; Lisp_Object chain, elt; NSAutoreleasePool *pool; BOOL setMini = YES; NSTRACE ("ns_implicitly_set_icon_type"); block_input (); pool = [[NSAutoreleasePool alloc] init]; if (f->output_data.ns->miniimage && [[NSString stringWithLispString:f->name] isEqualToString: [(NSImage *)f->output_data.ns->miniimage name]]) { [pool release]; unblock_input (); return; } tem = assq_no_quit (Qicon_type, f->param_alist); if (CONSP (tem) && ! NILP (XCDR (tem))) { [pool release]; unblock_input (); return; } for (chain = Vns_icon_type_alist; image == nil && CONSP (chain); chain = XCDR (chain)) { elt = XCAR (chain); /* Special case: t means go by file type. */ if (SYMBOLP (elt) && EQ (elt, Qt) && SSDATA (f->name)[0] == '/') { NSString *str = [NSString stringWithLispString:f->name]; if ([[NSFileManager defaultManager] fileExistsAtPath: str]) image = [[[NSWorkspace sharedWorkspace] iconForFile: str] retain]; } else if (CONSP (elt) && STRINGP (XCAR (elt)) && STRINGP (XCDR (elt)) && fast_string_match (XCAR (elt), f->name) >= 0) { image = [EmacsImage allocInitFromFile: XCDR (elt)]; if (image == nil) image = [[NSImage imageNamed: [NSString stringWithLispString:XCDR (elt)]] retain]; } } if (image == nil) { image = [[[NSWorkspace sharedWorkspace] iconForFileType: @"text"] retain]; setMini = NO; } [f->output_data.ns->miniimage release]; f->output_data.ns->miniimage = image; [view setMiniwindowImage: setMini]; [pool release]; unblock_input (); } static void ns_set_icon_type (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { EmacsView *view = FRAME_NS_VIEW (f); id image = nil; BOOL setMini = YES; NSTRACE ("ns_set_icon_type"); if (!NILP (arg) && SYMBOLP (arg)) { arg =build_string (SSDATA (SYMBOL_NAME (arg))); store_frame_param (f, Qicon_type, arg); } /* Do it the implicit way. */ if (NILP (arg)) { ns_implicitly_set_icon_type (f); return; } CHECK_STRING (arg); image = [EmacsImage allocInitFromFile: arg]; if (image == nil) image =[NSImage imageNamed: [NSString stringWithLispString:arg]]; if (image == nil) { image = [NSImage imageNamed: @"text"]; setMini = NO; } f->output_data.ns->miniimage = image; [view setMiniwindowImage: setMini]; } /* This is the same as the xfns.c definition. */ static void ns_set_cursor_type (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { set_frame_cursor_types (f, arg); } /* called to set mouse pointer color, but all other terms use it to initialize pointer types (and don't set the color ;) */ static void ns_set_mouse_color (struct frame *f, Lisp_Object arg, Lisp_Object oldval) { /* Don't think we can do this on Nextstep. */ } #define Str(x) #x #define Xstr(x) Str(x) static Lisp_Object ns_appkit_version_str (void) { NSString *tmp; #ifdef NS_IMPL_GNUSTEP tmp = [NSString stringWithFormat:@"gnustep-gui-%s", Xstr(GNUSTEP_GUI_VERSION)]; #elif defined (NS_IMPL_COCOA) tmp = [NSString stringWithFormat:@"appkit-%.2f %@", NSAppKitVersionNumber, [[NSProcessInfo processInfo] operatingSystemVersionString]]; #else tmp = [NSString initWithUTF8String:@"ns-unknown"]; #endif return [tmp lispString]; } /* This is for use by x-server-version and collapses all version info we have into a single int. For a better picture of the implementation running, use ns_appkit_version_str. */ static int ns_appkit_version_int (void) { #ifdef NS_IMPL_GNUSTEP return GNUSTEP_GUI_MAJOR_VERSION * 100 + GNUSTEP_GUI_MINOR_VERSION; #elif defined (NS_IMPL_COCOA) return (int)NSAppKitVersionNumber; #endif return 0; } static void ns_icon (struct frame *f, Lisp_Object parms) /* -------------------------------------------------------------------------- Strangely-named function to set icon position parameters in frame. This is irrelevant under macOS, but might be needed under GNUstep, depending on the window manager used. Note, this is not a standard frame parameter-setter; it is called directly from x-create-frame. -------------------------------------------------------------------------- */ { Lisp_Object icon_x, icon_y; struct ns_display_info *dpyinfo = check_ns_display_info (Qnil); f->output_data.ns->icon_top = -1; f->output_data.ns->icon_left = -1; /* Set the position of the icon. */ icon_x = gui_display_get_arg (dpyinfo, parms, Qicon_left, 0, 0, RES_TYPE_NUMBER); icon_y = gui_display_get_arg (dpyinfo, parms, Qicon_top, 0, 0, RES_TYPE_NUMBER); if (!EQ (icon_x, Qunbound) && !EQ (icon_y, Qunbound)) { CHECK_FIXNUM (icon_x); CHECK_FIXNUM (icon_y); f->output_data.ns->icon_top = XFIXNUM (icon_y); f->output_data.ns->icon_left = XFIXNUM (icon_x); } else if (!EQ (icon_x, Qunbound) || !EQ (icon_y, Qunbound)) error ("Both left and top icon corners of icon must be specified"); } /* Note: see frame.c for template, also where generic functions are implemented. */ frame_parm_handler ns_frame_parm_handlers[] = { gui_set_autoraise, /* generic OK */ gui_set_autolower, /* generic OK */ ns_set_background_color, 0, /* x_set_border_color, may be impossible under Nextstep */ 0, /* x_set_border_width, may be impossible under Nextstep */ ns_set_cursor_color, ns_set_cursor_type, gui_set_font, /* generic OK */ ns_set_foreground_color, ns_set_icon_name, ns_set_icon_type, ns_set_child_frame_border_width, ns_set_internal_border_width, gui_set_right_divider_width, /* generic OK */ gui_set_bottom_divider_width, /* generic OK */ ns_set_menu_bar_lines, ns_set_mouse_color, ns_explicitly_set_name, gui_set_scroll_bar_width, /* generic OK */ gui_set_scroll_bar_height, /* generic OK */ ns_set_title, gui_set_unsplittable, /* generic OK */ gui_set_vertical_scroll_bars, /* generic OK */ gui_set_horizontal_scroll_bars, /* generic OK */ gui_set_visibility, /* generic OK */ ns_set_tab_bar_lines, ns_set_tool_bar_lines, 0, /* x_set_scroll_bar_foreground, will ignore (not possible on NS) */ 0, /* x_set_scroll_bar_background, will ignore (not possible on NS) */ gui_set_screen_gamma, /* generic OK */ gui_set_line_spacing, /* generic OK, sets f->extra_line_spacing to int */ gui_set_left_fringe, /* generic OK */ gui_set_right_fringe, /* generic OK */ 0, /* x_set_wait_for_wm, will ignore */ gui_set_fullscreen, /* generic OK */ gui_set_font_backend, /* generic OK */ gui_set_alpha, 0, /* x_set_sticky */ 0, /* x_set_tool_bar_position */ 0, /* x_set_inhibit_double_buffering */ #ifdef NS_IMPL_COCOA ns_set_undecorated, #else 0, /* ns_set_undecorated */ #endif ns_set_parent_frame, 0, /* x_set_skip_taskbar */ ns_set_no_focus_on_map, ns_set_no_accept_focus, ns_set_z_group, 0, /* x_set_override_redirect */ gui_set_no_special_glyphs, #ifdef NS_IMPL_COCOA ns_set_appearance, ns_set_transparent_titlebar, #endif }; /* Handler for signals raised during x_create_frame. FRAME is the frame which is partially constructed. */ static void unwind_create_frame (Lisp_Object frame) { struct frame *f = XFRAME (frame); /* If frame is already dead, nothing to do. This can happen if the display is disconnected after the frame has become official, but before x_create_frame removes the unwind protect. */ if (!FRAME_LIVE_P (f)) return; /* If frame is ``official'', nothing to do. */ if (NILP (Fmemq (frame, Vframe_list))) { #if defined GLYPH_DEBUG && defined ENABLE_CHECKING struct ns_display_info *dpyinfo = FRAME_DISPLAY_INFO (f); #endif /* If the frame's image cache refcount is still the same as our private shadow variable, it means we are unwinding a frame for which we didn't yet call init_frame_faces, where the refcount is incremented. Therefore, we increment it here, so that free_frame_faces, called in ns_free_frame_resources below, will not mistakenly decrement the counter that was not incremented yet to account for this new frame. */ if (FRAME_IMAGE_CACHE (f) != NULL && FRAME_IMAGE_CACHE (f)->refcount == image_cache_refcount) FRAME_IMAGE_CACHE (f)->refcount++; ns_free_frame_resources (f); free_glyphs (f); #if defined GLYPH_DEBUG && defined ENABLE_CHECKING /* Check that reference counts are indeed correct. */ eassert (dpyinfo->terminal->image_cache->refcount == image_cache_refcount); #endif } } /* * Read geometry related parameters from preferences if not in PARMS. * Returns the union of parms and any preferences read. */ static Lisp_Object get_geometry_from_preferences (struct ns_display_info *dpyinfo, Lisp_Object parms) { struct { const char *val; const char *cls; Lisp_Object tem; } r[] = { { "width", "Width", Qwidth }, { "height", "Height", Qheight }, { "left", "Left", Qleft }, { "top", "Top", Qtop }, }; int i; for (i = 0; i < ARRAYELTS (r); ++i) { if (NILP (Fassq (r[i].tem, parms))) { Lisp_Object value = gui_display_get_arg (dpyinfo, parms, r[i].tem, r[i].val, r[i].cls, RES_TYPE_NUMBER); if (! EQ (value, Qunbound)) parms = Fcons (Fcons (r[i].tem, value), parms); } } return parms; } /* ========================================================================== Lisp definitions ========================================================================== */ DEFUN ("x-create-frame", Fx_create_frame, Sx_create_frame, 1, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object parms) { struct frame *f; Lisp_Object frame, tem; Lisp_Object name; int minibuffer_only = 0; long window_prompting = 0; ptrdiff_t count = specpdl_ptr - specpdl; Lisp_Object display; struct ns_display_info *dpyinfo = NULL; Lisp_Object parent, parent_frame; struct kboard *kb; static int desc_ctr = 1; /* gui_display_get_arg modifies parms. */ parms = Fcopy_alist (parms); /* Use this general default value to start with until we know if this frame has a specified name. */ Vx_resource_name = Vinvocation_name; display = gui_display_get_arg (dpyinfo, parms, Qterminal, 0, 0, RES_TYPE_STRING); if (EQ (display, Qunbound)) display = Qnil; dpyinfo = check_ns_display_info (display); kb = dpyinfo->terminal->kboard; if (!dpyinfo->terminal->name) error ("Terminal is not live, can't create new frames on it"); name = gui_display_get_arg (dpyinfo, parms, Qname, 0, 0, RES_TYPE_STRING); if (!STRINGP (name) && ! EQ (name, Qunbound) && ! NILP (name)) error ("Invalid frame name--not a string or nil"); if (STRINGP (name)) Vx_resource_name = name; parent = gui_display_get_arg (dpyinfo, parms, Qparent_id, 0, 0, RES_TYPE_NUMBER); if (EQ (parent, Qunbound)) parent = Qnil; if (! NILP (parent)) CHECK_FIXNUM (parent); /* make_frame_without_minibuffer can run Lisp code and garbage collect. */ /* No need to protect DISPLAY because that's not used after passing it to make_frame_without_minibuffer. */ frame = Qnil; tem = gui_display_get_arg (dpyinfo, parms, Qminibuffer, "minibuffer", "Minibuffer", RES_TYPE_SYMBOL); if (EQ (tem, Qnone) || NILP (tem)) f = make_frame_without_minibuffer (Qnil, kb, display); else if (EQ (tem, Qonly)) { f = make_minibuffer_frame (); minibuffer_only = 1; } else if (WINDOWP (tem)) f = make_frame_without_minibuffer (tem, kb, display); else f = make_frame (1); XSETFRAME (frame, f); f->terminal = dpyinfo->terminal; f->output_method = output_ns; f->output_data.ns = xzalloc (sizeof *f->output_data.ns); FRAME_FONTSET (f) = -1; fset_icon_name (f, gui_display_get_arg (dpyinfo, parms, Qicon_name, "iconName", "Title", RES_TYPE_STRING)); if (! STRINGP (f->icon_name)) fset_icon_name (f, Qnil); FRAME_DISPLAY_INFO (f) = dpyinfo; /* With FRAME_DISPLAY_INFO set up, this unwind-protect is safe. */ record_unwind_protect (unwind_create_frame, frame); f->output_data.ns->window_desc = desc_ctr++; if (TYPE_RANGED_FIXNUMP (Window, parent)) { f->output_data.ns->parent_desc = XFIXNAT (parent); f->output_data.ns->explicit_parent = 1; } else { f->output_data.ns->parent_desc = FRAME_DISPLAY_INFO (f)->root_window; f->output_data.ns->explicit_parent = 0; } /* Set the name; the functions to which we pass f expect the name to be set. */ if (EQ (name, Qunbound) || NILP (name) || ! STRINGP (name)) { fset_name (f, [ns_app_name lispString]); f->explicit_name = 0; } else { fset_name (f, name); f->explicit_name = 1; specbind (Qx_resource_name, name); } block_input (); #ifdef NS_IMPL_COCOA mac_register_font_driver (f); #else register_font_driver (&nsfont_driver, f); #endif image_cache_refcount = FRAME_IMAGE_CACHE (f) ? FRAME_IMAGE_CACHE (f)->refcount : 0; gui_default_parameter (f, parms, Qfont_backend, Qnil, "fontBackend", "FontBackend", RES_TYPE_STRING); { /* use for default font name */ id font = [NSFont userFixedPitchFontOfSize: -1.0]; /* default */ gui_default_parameter (f, parms, Qfontsize, make_fixnum (0 /* (int)[font pointSize] */), "fontSize", "FontSize", RES_TYPE_NUMBER); // Remove ' Regular', not handled by backends. char *fontname = xstrdup ([[font displayName] UTF8String]); int len = strlen (fontname); if (len > 8 && strcmp (fontname + len - 8, " Regular") == 0) fontname[len-8] = '\0'; gui_default_parameter (f, parms, Qfont, build_string (fontname), "font", "Font", RES_TYPE_STRING); xfree (fontname); } unblock_input (); gui_default_parameter (f, parms, Qborder_width, make_fixnum (0), "borderwidth", "BorderWidth", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qinternal_border_width, make_fixnum (2), "internalBorderWidth", "InternalBorderWidth", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qchild_frame_border_width, Qnil, "childFrameBorderWidth", "childFrameBorderWidth", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qright_divider_width, make_fixnum (0), NULL, NULL, RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qbottom_divider_width, make_fixnum (0), NULL, NULL, RES_TYPE_NUMBER); /* default vertical scrollbars on right on Mac */ { Lisp_Object spos #ifdef NS_IMPL_GNUSTEP = Qt; #else = Qright; #endif gui_default_parameter (f, parms, Qvertical_scroll_bars, spos, "verticalScrollBars", "VerticalScrollBars", RES_TYPE_SYMBOL); } gui_default_parameter (f, parms, Qhorizontal_scroll_bars, Qnil, "horizontalScrollBars", "HorizontalScrollBars", RES_TYPE_SYMBOL); gui_default_parameter (f, parms, Qforeground_color, build_string ("Black"), "foreground", "Foreground", RES_TYPE_STRING); gui_default_parameter (f, parms, Qbackground_color, build_string ("White"), "background", "Background", RES_TYPE_STRING); gui_default_parameter (f, parms, Qline_spacing, Qnil, "lineSpacing", "LineSpacing", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qleft_fringe, Qnil, "leftFringe", "LeftFringe", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qright_fringe, Qnil, "rightFringe", "RightFringe", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qno_special_glyphs, Qnil, NULL, NULL, RES_TYPE_BOOLEAN); init_frame_faces (f); /* Read comment about this code in corresponding place in xfns.c. */ tem = gui_display_get_arg (dpyinfo, parms, Qmin_width, NULL, NULL, RES_TYPE_NUMBER); if (FIXNUMP (tem)) store_frame_param (f, Qmin_width, tem); tem = gui_display_get_arg (dpyinfo, parms, Qmin_height, NULL, NULL, RES_TYPE_NUMBER); if (FIXNUMP (tem)) store_frame_param (f, Qmin_height, tem); adjust_frame_size (f, FRAME_COLS (f) * FRAME_COLUMN_WIDTH (f), FRAME_LINES (f) * FRAME_LINE_HEIGHT (f), 5, 1, Qx_create_frame_1); tem = gui_display_get_arg (dpyinfo, parms, Qundecorated, NULL, NULL, RES_TYPE_BOOLEAN); FRAME_UNDECORATED (f) = !NILP (tem) && !EQ (tem, Qunbound); store_frame_param (f, Qundecorated, FRAME_UNDECORATED (f) ? Qt : Qnil); #ifdef NS_IMPL_COCOA tem = gui_display_get_arg (dpyinfo, parms, Qns_appearance, NULL, NULL, RES_TYPE_SYMBOL); if (EQ (tem, Qdark)) FRAME_NS_APPEARANCE (f) = ns_appearance_vibrant_dark; else if (EQ (tem, Qlight)) FRAME_NS_APPEARANCE (f) = ns_appearance_aqua; else FRAME_NS_APPEARANCE (f) = ns_appearance_system_default; store_frame_param (f, Qns_appearance, (!NILP (tem) && !EQ (tem, Qunbound)) ? tem : Qnil); tem = gui_display_get_arg (dpyinfo, parms, Qns_transparent_titlebar, NULL, NULL, RES_TYPE_BOOLEAN); FRAME_NS_TRANSPARENT_TITLEBAR (f) = !NILP (tem) && !EQ (tem, Qunbound); store_frame_param (f, Qns_transparent_titlebar, FRAME_NS_TRANSPARENT_TITLEBAR (f) ? Qt : Qnil); #endif parent_frame = gui_display_get_arg (dpyinfo, parms, Qparent_frame, NULL, NULL, RES_TYPE_SYMBOL); /* Accept parent-frame iff parent-id was not specified. */ if (!NILP (parent) || EQ (parent_frame, Qunbound) || NILP (parent_frame) || !FRAMEP (parent_frame) || !FRAME_LIVE_P (XFRAME (parent_frame))) parent_frame = Qnil; fset_parent_frame (f, parent_frame); store_frame_param (f, Qparent_frame, parent_frame); gui_default_parameter (f, parms, Qz_group, Qnil, NULL, NULL, RES_TYPE_SYMBOL); gui_default_parameter (f, parms, Qno_focus_on_map, Qnil, NULL, NULL, RES_TYPE_BOOLEAN); gui_default_parameter (f, parms, Qno_accept_focus, Qnil, NULL, NULL, RES_TYPE_BOOLEAN); /* The resources controlling the menu-bar and tool-bar are processed specially at startup, and reflected in the mode variables; ignore them here. */ gui_default_parameter (f, parms, Qmenu_bar_lines, NILP (Vmenu_bar_mode) ? make_fixnum (0) : make_fixnum (1), NULL, NULL, RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qtool_bar_lines, NILP (Vtool_bar_mode) ? make_fixnum (0) : make_fixnum (1), NULL, NULL, RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qbuffer_predicate, Qnil, "bufferPredicate", "BufferPredicate", RES_TYPE_SYMBOL); gui_default_parameter (f, parms, Qtitle, Qnil, "title", "Title", RES_TYPE_STRING); parms = get_geometry_from_preferences (dpyinfo, parms); window_prompting = gui_figure_window_size (f, parms, false, true); tem = gui_display_get_arg (dpyinfo, parms, Qunsplittable, 0, 0, RES_TYPE_BOOLEAN); f->no_split = minibuffer_only || (!EQ (tem, Qunbound) && !NILP (tem)); /* NOTE: on other terms, this is done in set_mouse_color, however this was not getting called under Nextstep. */ f->output_data.ns->text_cursor = [NSCursor IBeamCursor]; f->output_data.ns->nontext_cursor = [NSCursor arrowCursor]; f->output_data.ns->modeline_cursor = [NSCursor pointingHandCursor]; f->output_data.ns->hand_cursor = [NSCursor pointingHandCursor]; f->output_data.ns->hourglass_cursor = [NSCursor disappearingItemCursor]; f->output_data.ns->horizontal_drag_cursor = [NSCursor resizeLeftRightCursor]; f->output_data.ns->vertical_drag_cursor = [NSCursor resizeUpDownCursor]; f->output_data.ns->left_edge_cursor = [NSCursor resizeLeftRightCursor]; f->output_data.ns->top_left_corner_cursor = [NSCursor arrowCursor]; f->output_data.ns->top_edge_cursor = [NSCursor resizeUpDownCursor]; f->output_data.ns->top_right_corner_cursor = [NSCursor arrowCursor]; f->output_data.ns->right_edge_cursor = [NSCursor resizeLeftRightCursor]; f->output_data.ns->bottom_right_corner_cursor = [NSCursor arrowCursor]; f->output_data.ns->bottom_edge_cursor = [NSCursor resizeUpDownCursor]; f->output_data.ns->bottom_left_corner_cursor = [NSCursor arrowCursor]; FRAME_DISPLAY_INFO (f)->vertical_scroll_bar_cursor = [NSCursor arrowCursor]; FRAME_DISPLAY_INFO (f)->horizontal_scroll_bar_cursor = [NSCursor arrowCursor]; f->output_data.ns->current_pointer = f->output_data.ns->text_cursor; f->output_data.ns->in_animation = NO; [[EmacsView alloc] initFrameFromEmacs: f]; ns_icon (f, parms); /* ns_display_info does not have a reference_count. */ f->terminal->reference_count++; /* It is now ok to make the frame official even if we get an error below. The frame needs to be on Vframe_list or making it visible won't work. */ Vframe_list = Fcons (frame, Vframe_list); gui_default_parameter (f, parms, Qicon_type, Qnil, "bitmapIcon", "BitmapIcon", RES_TYPE_SYMBOL); gui_default_parameter (f, parms, Qauto_raise, Qnil, "autoRaise", "AutoRaiseLower", RES_TYPE_BOOLEAN); gui_default_parameter (f, parms, Qauto_lower, Qnil, "autoLower", "AutoLower", RES_TYPE_BOOLEAN); gui_default_parameter (f, parms, Qcursor_type, Qbox, "cursorType", "CursorType", RES_TYPE_SYMBOL); gui_default_parameter (f, parms, Qscroll_bar_width, Qnil, "scrollBarWidth", "ScrollBarWidth", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qscroll_bar_height, Qnil, "scrollBarHeight", "ScrollBarHeight", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qalpha, Qnil, "alpha", "Alpha", RES_TYPE_NUMBER); gui_default_parameter (f, parms, Qfullscreen, Qnil, "fullscreen", "Fullscreen", RES_TYPE_SYMBOL); /* Allow set_window_size_hook, now. */ f->can_set_window_size = true; adjust_frame_size (f, FRAME_TEXT_WIDTH (f), FRAME_TEXT_HEIGHT (f), 0, true, Qx_create_frame_2); if (! f->output_data.ns->explicit_parent) { Lisp_Object visibility; visibility = gui_display_get_arg (dpyinfo, parms, Qvisibility, 0, 0, RES_TYPE_SYMBOL); if (EQ (visibility, Qunbound)) visibility = Qt; if (EQ (visibility, Qicon)) ns_iconify_frame (f); else if (! NILP (visibility)) { ns_make_frame_visible (f); [[FRAME_NS_VIEW (f) window] makeKeyWindow]; } else { /* Must have been Qnil. */ f->was_invisible = true; } } if (FRAME_HAS_MINIBUF_P (f) && (!FRAMEP (KVAR (kb, Vdefault_minibuffer_frame)) || !FRAME_LIVE_P (XFRAME (KVAR (kb, Vdefault_minibuffer_frame))))) kset_default_minibuffer_frame (kb, frame); /* All remaining specified parameters, which have not been "used" by gui_display_get_arg and friends, now go in the misc. alist of the frame. */ for (tem = parms; CONSP (tem); tem = XCDR (tem)) if (CONSP (XCAR (tem)) && !NILP (XCAR (XCAR (tem)))) fset_param_alist (f, Fcons (XCAR (tem), f->param_alist)); if (window_prompting & USPosition) ns_set_offset (f, f->left_pos, f->top_pos, 1); /* Make sure windows on this frame appear in calls to next-window and similar functions. */ Vwindow_list = Qnil; return unbind_to (count, frame); } static BOOL ns_window_is_ancestor (NSWindow *win, NSWindow *candidate) /* Test whether CANDIDATE is an ancestor window of WIN. */ { if (candidate == NULL) return NO; else if (win == candidate) return YES; else return ns_window_is_ancestor(win, [candidate parentWindow]); } DEFUN ("ns-frame-list-z-order", Fns_frame_list_z_order, Sns_frame_list_z_order, 0, 1, 0, doc: /* Return list of Emacs' frames, in Z (stacking) order. If TERMINAL is non-nil and specifies a live frame, return the child frames of that frame in Z (stacking) order. Frames are listed from topmost (first) to bottommost (last). */) (Lisp_Object terminal) { Lisp_Object frames = Qnil; NSWindow *parent = nil; if (FRAMEP (terminal) && FRAME_LIVE_P (XFRAME (terminal))) parent = [FRAME_NS_VIEW (XFRAME (terminal)) window]; for (NSWindow *win in [[NSApp orderedWindows] reverseObjectEnumerator]) { Lisp_Object frame; /* Check against [win parentWindow] so that it doesn't match itself. */ if ([[win delegate] isKindOfClass:[EmacsView class]] && (parent == nil || ns_window_is_ancestor (parent, [win parentWindow]))) { XSETFRAME (frame, ((EmacsView *)[win delegate])->emacsframe); frames = Fcons(frame, frames); } } return frames; } DEFUN ("ns-frame-restack", Fns_frame_restack, Sns_frame_restack, 2, 3, 0, doc: /* Restack FRAME1 below FRAME2. This means that if both frames are visible and the display areas of these frames overlap, FRAME2 (partially) obscures FRAME1. If optional third argument ABOVE is non-nil, restack FRAME1 above FRAME2. This means that if both frames are visible and the display areas of these frames overlap, FRAME1 (partially) obscures FRAME2. Some window managers may refuse to restack windows. */) (Lisp_Object frame1, Lisp_Object frame2, Lisp_Object above) { struct frame *f1 = decode_live_frame (frame1); struct frame *f2 = decode_live_frame (frame2); if (FRAME_NS_VIEW (f1) && FRAME_NS_VIEW (f2)) { EmacsWindow *window = (EmacsWindow *)[FRAME_NS_VIEW (f1) window]; NSWindow *window2 = [FRAME_NS_VIEW (f2) window]; if ([window restackWindow:window2 above:!NILP (above)]) return Qt; else return Qnil; } else { error ("Cannot restack frames"); return Qnil; } } DEFUN ("ns-popup-font-panel", Fns_popup_font_panel, Sns_popup_font_panel, 0, 1, "", doc: /* Pop up the font panel. */) (Lisp_Object frame) { struct frame *f = decode_window_system_frame (frame); id fm = [NSFontManager sharedFontManager]; struct font *font = f->output_data.ns->font; NSFont *nsfont; #ifdef NS_IMPL_GNUSTEP nsfont = ((struct nsfont_info *)font)->nsfont; #endif #ifdef NS_IMPL_COCOA nsfont = (NSFont *) macfont_get_nsctfont (font); #endif [fm setSelectedFont: nsfont isMultiple: NO]; [fm orderFrontFontPanel: NSApp]; return Qnil; } DEFUN ("ns-popup-color-panel", Fns_popup_color_panel, Sns_popup_color_panel, 0, 1, "", doc: /* Pop up the color panel. */) (Lisp_Object frame) { check_window_system (NULL); [NSApp orderFrontColorPanel: NSApp]; return Qnil; } static struct { id panel; BOOL ret; #ifdef NS_IMPL_GNUSTEP NSString *dirS, *initS; BOOL no_types; #endif } ns_fd_data; void ns_run_file_dialog (void) { if (ns_fd_data.panel == nil) return; #ifdef NS_IMPL_COCOA ns_fd_data.ret = [ns_fd_data.panel runModal]; #else if (ns_fd_data.no_types) { ns_fd_data.ret = [ns_fd_data.panel runModalForDirectory: ns_fd_data.dirS file: ns_fd_data.initS]; } else { ns_fd_data.ret = [ns_fd_data.panel runModalForDirectory: ns_fd_data.dirS file: ns_fd_data.initS types: nil]; } #endif ns_fd_data.panel = nil; } #ifdef NS_IMPL_COCOA #if MAC_OS_X_VERSION_MAX_ALLOWED > 1090 #define MODAL_OK_RESPONSE NSModalResponseOK #endif #endif #ifndef MODAL_OK_RESPONSE #define MODAL_OK_RESPONSE NSOKButton #endif DEFUN ("ns-read-file-name", Fns_read_file_name, Sns_read_file_name, 1, 5, 0, doc: /* Use a graphical panel to read a file name, using prompt PROMPT. Optional arg DIR, if non-nil, supplies a default directory. Optional arg MUSTMATCH, if non-nil, means the returned file or directory must exist. Optional arg INIT, if non-nil, provides a default file name to use. Optional arg DIR_ONLY_P, if non-nil, means choose only directories. */) (Lisp_Object prompt, Lisp_Object dir, Lisp_Object mustmatch, Lisp_Object init, Lisp_Object dir_only_p) { static id fileDelegate = nil; BOOL isSave = NILP (mustmatch) && NILP (dir_only_p); id panel; Lisp_Object fname = Qnil; NSString *promptS = NILP (prompt) || !STRINGP (prompt) ? nil : [NSString stringWithLispString:prompt]; NSString *dirS = NILP (dir) || !STRINGP (dir) ? [NSString stringWithLispString:BVAR (current_buffer, directory)] : [NSString stringWithLispString:dir]; NSString *initS = NILP (init) || !STRINGP (init) ? nil : [NSString stringWithLispString:init]; NSEvent *nxev; check_window_system (NULL); if (fileDelegate == nil) fileDelegate = [EmacsFileDelegate new]; [NSCursor setHiddenUntilMouseMoves: NO]; if ([dirS characterAtIndex: 0] == '~') dirS = [dirS stringByExpandingTildeInPath]; panel = isSave ? (id)[NSSavePanel savePanel] : (id)[NSOpenPanel openPanel]; [panel setTitle: promptS]; [panel setAllowsOtherFileTypes: YES]; [panel setTreatsFilePackagesAsDirectories: YES]; [panel setDelegate: fileDelegate]; if (! NILP (dir_only_p)) { [panel setCanChooseDirectories: YES]; [panel setCanChooseFiles: NO]; } else if (! isSave) { /* This is not quite what the documentation says, but it is compatible with the Gtk+ code. Also, the menu entry says "Open File...". */ [panel setCanChooseDirectories: NO]; [panel setCanChooseFiles: YES]; } block_input (); ns_fd_data.panel = panel; ns_fd_data.ret = NO; #ifdef NS_IMPL_COCOA if (! NILP (mustmatch) || ! NILP (dir_only_p)) [panel setAllowedFileTypes: nil]; if (dirS) [panel setDirectoryURL: [NSURL fileURLWithPath: dirS]]; if (initS && NILP (Ffile_directory_p (init))) [panel setNameFieldStringValue: [initS lastPathComponent]]; else [panel setNameFieldStringValue: @""]; #else ns_fd_data.no_types = NILP (mustmatch) && NILP (dir_only_p); ns_fd_data.dirS = dirS; ns_fd_data.initS = initS; #endif /* runModalForDirectory/runModal restarts the main event loop when done, so we must start an event loop and then pop up the file dialog. The file dialog may pop up a confirm dialog after Ok has been pressed, so we can not simply pop down on the Ok/Cancel press. */ nxev = [NSEvent otherEventWithType: NSEventTypeApplicationDefined location: NSMakePoint (0, 0) modifierFlags: 0 timestamp: 0 windowNumber: [[NSApp mainWindow] windowNumber] context: [NSApp context] subtype: 0 data1: 0 data2: NSAPP_DATA2_RUNFILEDIALOG]; [NSApp postEvent: nxev atStart: NO]; while (ns_fd_data.panel != nil) [NSApp run]; if (ns_fd_data.ret == MODAL_OK_RESPONSE) { NSString *str = ns_filename_from_panel (panel); if (! str) str = ns_directory_from_panel (panel); if (str) fname = [str lispString]; } [[FRAME_NS_VIEW (SELECTED_FRAME ()) window] makeKeyWindow]; unblock_input (); return fname; } const char * ns_get_defaults_value (const char *key) { NSObject *obj = [[NSUserDefaults standardUserDefaults] objectForKey: [NSString stringWithUTF8String: key]]; if (!obj) return NULL; return [[NSString stringWithFormat: @"%@", obj] UTF8String]; } DEFUN ("ns-get-resource", Fns_get_resource, Sns_get_resource, 2, 2, 0, doc: /* Return the value of the property NAME of OWNER from the defaults database. If OWNER is nil, Emacs is assumed. */) (Lisp_Object owner, Lisp_Object name) { const char *value; check_window_system (NULL); if (NILP (owner)) owner = [ns_app_name lispString]; CHECK_STRING (name); value = ns_get_defaults_value (SSDATA (name)); if (value) return build_string (value); return Qnil; } DEFUN ("ns-set-resource", Fns_set_resource, Sns_set_resource, 3, 3, 0, doc: /* Set property NAME of OWNER to VALUE, from the defaults database. If OWNER is nil, Emacs is assumed. If VALUE is nil, the default is removed. */) (Lisp_Object owner, Lisp_Object name, Lisp_Object value) { check_window_system (NULL); if (NILP (owner)) owner = [ns_app_name lispString]; CHECK_STRING (name); if (NILP (value)) { [[NSUserDefaults standardUserDefaults] removeObjectForKey: [NSString stringWithLispString:name]]; } else { CHECK_STRING (value); [[NSUserDefaults standardUserDefaults] setObject: [NSString stringWithLispString:value] forKey: [NSString stringWithLispString:name]]; } return Qnil; } DEFUN ("x-server-max-request-size", Fx_server_max_request_size, Sx_server_max_request_size, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); /* This function has no real equivalent under Nextstep. Return nil to indicate this. */ return Qnil; } DEFUN ("x-server-vendor", Fx_server_vendor, Sx_server_vendor, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); #ifdef NS_IMPL_GNUSTEP return build_string ("GNU"); #else return build_string ("Apple"); #endif } DEFUN ("x-server-version", Fx_server_version, Sx_server_version, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); /* NOTE: it is unclear what would best correspond with "protocol"; we return 10.3, meaning Panther, since this is roughly the level that GNUstep's APIs correspond to. The last number is where we distinguish between the Apple and GNUstep implementations ("distributor-specific release number") and give int'ized versions of major.minor. */ return list3i (10, 3, ns_appkit_version_int ()); } DEFUN ("x-display-screens", Fx_display_screens, Sx_display_screens, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); return make_fixnum (1); } DEFUN ("x-display-mm-height", Fx_display_mm_height, Sx_display_mm_height, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { struct ns_display_info *dpyinfo = check_ns_display_info (terminal); return make_fixnum (ns_display_pixel_height (dpyinfo) / (92.0/25.4)); } DEFUN ("x-display-mm-width", Fx_display_mm_width, Sx_display_mm_width, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { struct ns_display_info *dpyinfo = check_ns_display_info (terminal); return make_fixnum (ns_display_pixel_width (dpyinfo) / (92.0/25.4)); } DEFUN ("x-display-backing-store", Fx_display_backing_store, Sx_display_backing_store, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); /* Note that the xfns.c version has different return values. */ switch ([ns_get_window (terminal) backingType]) { case NSBackingStoreBuffered: return intern ("buffered"); #if defined (NS_IMPL_GNUSTEP) || MAC_OS_X_VERSION_MIN_REQUIRED < 101300 case NSBackingStoreRetained: return intern ("retained"); case NSBackingStoreNonretained: return intern ("non-retained"); #endif default: error ("Strange value for backingType parameter of frame"); } return Qnil; /* not reached, shut compiler up */ } DEFUN ("x-display-visual-class", Fx_display_visual_class, Sx_display_visual_class, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { NSWindowDepth depth; check_ns_display_info (terminal); depth = [[[NSScreen screens] objectAtIndex:0] depth]; if ( depth == NSBestDepth (NSCalibratedWhiteColorSpace, 2, 2, YES, NULL)) return intern ("static-gray"); else if (depth == NSBestDepth (NSCalibratedWhiteColorSpace, 8, 8, YES, NULL)) return intern ("gray-scale"); else if ( depth == NSBestDepth (NSCalibratedRGBColorSpace, 8, 8, YES, NULL)) return intern ("pseudo-color"); else if ( depth == NSBestDepth (NSCalibratedRGBColorSpace, 4, 12, NO, NULL)) return intern ("true-color"); else if ( depth == NSBestDepth (NSCalibratedRGBColorSpace, 8, 24, NO, NULL)) return intern ("direct-color"); else /* Color management as far as we do it is really handled by Nextstep itself anyway. */ return intern ("direct-color"); } DEFUN ("x-display-save-under", Fx_display_save_under, Sx_display_save_under, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); switch ([ns_get_window (terminal) backingType]) { case NSBackingStoreBuffered: return Qt; #if defined (NS_IMPL_GNUSTEP) || MAC_OS_X_VERSION_MIN_REQUIRED < 101300 case NSBackingStoreRetained: case NSBackingStoreNonretained: return Qnil; #endif default: error ("Strange value for backingType parameter of frame"); } return Qnil; /* not reached, shut compiler up */ } DEFUN ("x-open-connection", Fx_open_connection, Sx_open_connection, 1, 3, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object display, Lisp_Object resource_string, Lisp_Object must_succeed) { struct ns_display_info *dpyinfo; CHECK_STRING (display); nxatoms_of_nsselect (); dpyinfo = ns_term_init (display); if (dpyinfo == 0) { if (!NILP (must_succeed)) fatal ("Display on %s not responding.\n", SSDATA (display)); else error ("Display on %s not responding.\n", SSDATA (display)); } return Qnil; } DEFUN ("x-close-connection", Fx_close_connection, Sx_close_connection, 1, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); [NSApp terminate: NSApp]; return Qnil; } DEFUN ("x-display-list", Fx_display_list, Sx_display_list, 0, 0, 0, doc: /* SKIP: real doc in xfns.c. */) (void) { Lisp_Object result = Qnil; struct ns_display_info *ndi; for (ndi = x_display_list; ndi; ndi = ndi->next) result = Fcons (XCAR (ndi->name_list_element), result); return result; } DEFUN ("ns-hide-others", Fns_hide_others, Sns_hide_others, 0, 0, 0, doc: /* Hides all applications other than Emacs. */) (void) { check_window_system (NULL); [NSApp hideOtherApplications: NSApp]; return Qnil; } DEFUN ("ns-hide-emacs", Fns_hide_emacs, Sns_hide_emacs, 1, 1, 0, doc: /* If ON is non-nil, the entire Emacs application is hidden. Otherwise if Emacs is hidden, it is unhidden. If ON is equal to `activate', Emacs is unhidden and becomes the active application. */) (Lisp_Object on) { check_window_system (NULL); if (EQ (on, intern ("activate"))) { [NSApp unhide: NSApp]; [NSApp activateIgnoringOtherApps: YES]; } else if (NILP (on)) [NSApp unhide: NSApp]; else [NSApp hide: NSApp]; return Qnil; } DEFUN ("ns-emacs-info-panel", Fns_emacs_info_panel, Sns_emacs_info_panel, 0, 0, 0, doc: /* Shows the `Info' or `About' panel for Emacs. */) (void) { check_window_system (NULL); [NSApp orderFrontStandardAboutPanel: nil]; return Qnil; } DEFUN ("ns-font-name", Fns_font_name, Sns_font_name, 1, 1, 0, doc: /* Determine font PostScript or family name for font NAME. NAME should be a string containing either the font name or an XLFD font descriptor. If string contains `fontset' and not `fontset-startup', it is left alone. */) (Lisp_Object name) { char *nm; CHECK_STRING (name); nm = SSDATA (name); if (nm[0] != '-') return name; if (strstr (nm, "fontset") && !strstr (nm, "fontset-startup")) return name; return build_string (ns_xlfd_to_fontname (SSDATA (name))); } DEFUN ("ns-list-colors", Fns_list_colors, Sns_list_colors, 0, 1, 0, doc: /* Return a list of all available colors. The optional argument FRAME is currently ignored. */) (Lisp_Object frame) { Lisp_Object list = Qnil; NSEnumerator *colorlists; NSColorList *clist; if (!NILP (frame)) { CHECK_FRAME (frame); if (! FRAME_NS_P (XFRAME (frame))) error ("non-Nextstep frame used in `ns-list-colors'"); } block_input (); colorlists = [[NSColorList availableColorLists] objectEnumerator]; while ((clist = [colorlists nextObject])) { if ([[clist name] length] < 7 || [[clist name] rangeOfString: @"PANTONE"].location == 0) { NSEnumerator *cnames = [[clist allKeys] reverseObjectEnumerator]; NSString *cname; while ((cname = [cnames nextObject])) list = Fcons ([cname lispString], list); /* for (i = [[clist allKeys] count] - 1; i >= 0; i--) list = Fcons (build_string ([[[clist allKeys] objectAtIndex: i] UTF8String]), list); */ } } unblock_input (); return list; } DEFUN ("ns-list-services", Fns_list_services, Sns_list_services, 0, 0, 0, doc: /* List available Nextstep services by querying NSApp. */) (void) { #ifdef NS_IMPL_COCOA /* You can't get services like this in 10.6+. */ return Qnil; #else Lisp_Object ret = Qnil; NSMenu *svcs; check_window_system (NULL); svcs = [[NSMenu alloc] initWithTitle: @"Services"]; [NSApp setServicesMenu: svcs]; [NSApp registerServicesMenuSendTypes: ns_send_types returnTypes: ns_return_types]; [svcs setAutoenablesItems: NO]; ret = interpret_services_menu (svcs, Qnil, ret); return ret; #endif } DEFUN ("ns-perform-service", Fns_perform_service, Sns_perform_service, 2, 2, 0, doc: /* Perform Nextstep SERVICE on SEND. SEND should be either a string or nil. The return value is the result of the service, as string, or nil if there was no result. */) (Lisp_Object service, Lisp_Object send) { id pb; NSString *svcName; CHECK_STRING (service); check_window_system (NULL); svcName = [NSString stringWithLispString:service]; pb =[NSPasteboard pasteboardWithUniqueName]; ns_string_to_pasteboard (pb, send); if (NSPerformService (svcName, pb) == NO) Fsignal (Qquit, list1 (build_string ("service not available"))); if ([[pb types] count] == 0) return build_string (""); return ns_string_from_pasteboard (pb); } #ifdef NS_IMPL_COCOA /* Compile and execute the AppleScript SCRIPT and return the error status as function value. A zero is returned if compilation and execution is successful, in which case *RESULT is set to a Lisp string or a number containing the resulting script value. Otherwise, 1 is returned. */ static int ns_do_applescript (Lisp_Object script, Lisp_Object *result) { NSAppleEventDescriptor *desc; NSDictionary *errorDict; NSAppleEventDescriptor *returnDescriptor = NULL; NSAppleScript *scriptObject = [[NSAppleScript alloc] initWithSource: [NSString stringWithLispString:script]]; returnDescriptor = [scriptObject executeAndReturnError: &errorDict]; [scriptObject release]; *result = Qnil; if (returnDescriptor != NULL) { // successful execution if (kAENullEvent != [returnDescriptor descriptorType]) { *result = Qt; // script returned an AppleScript result if ((typeUnicodeText == [returnDescriptor descriptorType]) || #if defined (NS_IMPL_COCOA) (typeUTF16ExternalRepresentation == [returnDescriptor descriptorType]) || #endif (typeUTF8Text == [returnDescriptor descriptorType]) || (typeCString == [returnDescriptor descriptorType])) { desc = [returnDescriptor coerceToDescriptorType: typeUTF8Text]; if (desc) *result = [[desc stringValue] lispString]; } else { /* use typeUTF16ExternalRepresentation? */ // coerce the result to the appropriate ObjC type desc = [returnDescriptor coerceToDescriptorType: typeUTF8Text]; if (desc) *result = make_fixnum([desc int32Value]); } } } else { // no script result, return error return 1; } return 0; } /* Helper function called from sendEvent to run AppleScript from within the main event loop. */ void ns_run_ascript (void) { if (! NILP (as_script)) as_status = ns_do_applescript (as_script, as_result); as_script = Qnil; } DEFUN ("ns-do-applescript", Fns_do_applescript, Sns_do_applescript, 1, 1, 0, doc: /* Execute AppleScript SCRIPT and return the result. If compilation and execution are successful, the resulting script value is returned as a string, a number or, in the case of other constructs, t. In case the execution fails, an error is signaled. */) (Lisp_Object script) { Lisp_Object result; int status; NSEvent *nxev; struct input_event ev; CHECK_STRING (script); check_window_system (NULL); block_input (); as_script = script; as_result = &result; /* Executing AppleScript requires the event loop to run, otherwise errors aren't returned and executeAndReturnError hangs forever. Post an event that runs AppleScript and then start the event loop. The event loop is exited when the script is done. */ nxev = [NSEvent otherEventWithType: NSEventTypeApplicationDefined location: NSMakePoint (0, 0) modifierFlags: 0 timestamp: 0 windowNumber: [[NSApp mainWindow] windowNumber] context: [NSApp context] subtype: 0 data1: 0 data2: NSAPP_DATA2_RUNASSCRIPT]; [NSApp postEvent: nxev atStart: NO]; /* If there are other events, the event loop may exit. Keep running until the script has been handled. */ ns_init_events (&ev); while (! NILP (as_script)) [NSApp run]; ns_finish_events (); status = as_status; as_status = 0; as_result = 0; unblock_input (); if (status == 0) return result; else if (!STRINGP (result)) error ("AppleScript error %d", status); else error ("%s", SSDATA (result)); } #endif /* ========================================================================== Miscellaneous functions not called through hooks ========================================================================== */ /* called from frame.c */ struct ns_display_info * check_x_display_info (Lisp_Object frame) { return check_ns_display_info (frame); } void ns_set_scroll_bar_default_width (struct frame *f) { int wid = FRAME_COLUMN_WIDTH (f); FRAME_CONFIG_SCROLL_BAR_WIDTH (f) = NS_SCROLL_BAR_WIDTH_DEFAULT; FRAME_CONFIG_SCROLL_BAR_COLS (f) = (FRAME_CONFIG_SCROLL_BAR_WIDTH (f) + wid - 1) / wid; } void ns_set_scroll_bar_default_height (struct frame *f) { int height = FRAME_LINE_HEIGHT (f); FRAME_CONFIG_SCROLL_BAR_HEIGHT (f) = NS_SCROLL_BAR_WIDTH_DEFAULT; FRAME_CONFIG_SCROLL_BAR_LINES (f) = (FRAME_CONFIG_SCROLL_BAR_HEIGHT (f) + height - 1) / height; } /* Terms implement this instead of x-get-resource directly. */ const char * ns_get_string_resource (void *_rdb, const char *name, const char *class) { /* remove appname prefix; TODO: allow for !="Emacs" */ const char *res, *toCheck = class + (!strncmp (class, "Emacs.", 6) ? 6 : 0); check_window_system (NULL); if (inhibit_x_resources) /* --quick was passed, so this is a no-op. */ return NULL; res = ns_get_defaults_value (toCheck); return (const char *) (!res ? NULL : !c_strncasecmp (res, "YES", 3) ? "true" : !c_strncasecmp (res, "NO", 2) ? "false" : res); } /* ========================================================================== Lisp definitions that, for whatever reason, we can't alias as 'ns-XXX'. ========================================================================== */ DEFUN ("xw-color-defined-p", Fxw_color_defined_p, Sxw_color_defined_p, 1, 2, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object color, Lisp_Object frame) { NSColor * col; check_window_system (NULL); return ns_lisp_to_color (color, &col) ? Qnil : Qt; } DEFUN ("xw-color-values", Fxw_color_values, Sxw_color_values, 1, 2, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object color, Lisp_Object frame) { NSColor * col; EmacsCGFloat red, green, blue, alpha; check_window_system (NULL); CHECK_STRING (color); block_input (); if (ns_lisp_to_color (color, &col)) { unblock_input (); return Qnil; } [[col colorUsingDefaultColorSpace] getRed: &red green: &green blue: &blue alpha: &alpha]; unblock_input (); return list3i (lrint (red * 65535), lrint (green * 65535), lrint (blue * 65535)); } DEFUN ("xw-display-color-p", Fxw_display_color_p, Sxw_display_color_p, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { NSWindowDepth depth; NSString *colorSpace; check_ns_display_info (terminal); depth = [[[NSScreen screens] objectAtIndex:0] depth]; colorSpace = NSColorSpaceFromDepth (depth); return [colorSpace isEqualToString: NSDeviceWhiteColorSpace] || [colorSpace isEqualToString: NSCalibratedWhiteColorSpace] ? Qnil : Qt; } DEFUN ("x-display-grayscale-p", Fx_display_grayscale_p, Sx_display_grayscale_p, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { NSWindowDepth depth; check_ns_display_info (terminal); depth = [[[NSScreen screens] objectAtIndex:0] depth]; return NSBitsPerPixelFromDepth (depth) > 1 ? Qt : Qnil; } DEFUN ("x-display-pixel-width", Fx_display_pixel_width, Sx_display_pixel_width, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { struct ns_display_info *dpyinfo = check_ns_display_info (terminal); return make_fixnum (ns_display_pixel_width (dpyinfo)); } DEFUN ("x-display-pixel-height", Fx_display_pixel_height, Sx_display_pixel_height, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { struct ns_display_info *dpyinfo = check_ns_display_info (terminal); return make_fixnum (ns_display_pixel_height (dpyinfo)); } #ifdef NS_IMPL_COCOA /* Returns the name for the screen that OBJ represents, or NULL. Caller must free return value. */ static char * ns_get_name_from_ioreg (io_object_t obj) { char *name = NULL; NSDictionary *info = (NSDictionary *) IODisplayCreateInfoDictionary (obj, kIODisplayOnlyPreferredName); NSDictionary *names = [info objectForKey: [NSString stringWithUTF8String: kDisplayProductName]]; if ([names count] > 0) { NSString *n = [names objectForKey: [[names allKeys] objectAtIndex:0]]; if (n != nil) name = xstrdup ([n UTF8String]); } [info release]; return name; } /* Returns the name for the screen that DID came from, or NULL. Caller must free return value. */ static char * ns_screen_name (CGDirectDisplayID did) { char *name = NULL; #if MAC_OS_X_VERSION_MAX_ALLOWED >= 1090 #if MAC_OS_X_VERSION_MIN_REQUIRED < 1090 if (CGDisplayIOServicePort == NULL) #endif { mach_port_t masterPort; io_iterator_t it; io_object_t obj; /* CGDisplayIOServicePort is deprecated. Do it another (harder) way. Is this code OK for macOS < 10.9, and GNUstep? I suspect it is, in which case is it worth keeping the other method in here? */ if (IOMasterPort (MACH_PORT_NULL, &masterPort) != kIOReturnSuccess || IOServiceGetMatchingServices (masterPort, IOServiceMatching ("IONDRVDevice"), &it) != kIOReturnSuccess) return name; /* Must loop until we find a name. Many devices can have the same unit number (represents different GPU parts), but only one has a name. */ while (! name && (obj = IOIteratorNext (it))) { CFMutableDictionaryRef props; const void *val; if (IORegistryEntryCreateCFProperties (obj, &props, kCFAllocatorDefault, kNilOptions) == kIOReturnSuccess && props != nil && (val = CFDictionaryGetValue(props, @"IOFBDependentIndex"))) { unsigned nr = [(NSNumber *)val unsignedIntegerValue]; if (nr == CGDisplayUnitNumber (did)) name = ns_get_name_from_ioreg (obj); } CFRelease (props); IOObjectRelease (obj); } IOObjectRelease (it); } #if MAC_OS_X_VERSION_MIN_REQUIRED < 1090 else #endif #endif /* #if MAC_OS_X_VERSION_MAX_ALLOWED >= 1090 */ #if MAC_OS_X_VERSION_MIN_REQUIRED < 1090 name = ns_get_name_from_ioreg (CGDisplayIOServicePort (did)); #endif return name; } #endif /* NS_IMPL_COCOA */ static Lisp_Object ns_make_monitor_attribute_list (struct MonitorInfo *monitors, int n_monitors, int primary_monitor, const char *source) { Lisp_Object monitor_frames = make_nil_vector (n_monitors); Lisp_Object frame, rest; NSArray *screens = [NSScreen screens]; int i; FOR_EACH_FRAME (rest, frame) { struct frame *f = XFRAME (frame); if (FRAME_NS_P (f)) { NSView *view = FRAME_NS_VIEW (f); NSScreen *screen = [[view window] screen]; NSUInteger k; i = -1; for (k = 0; i == -1 && k < [screens count]; ++k) { if ([screens objectAtIndex: k] == screen) i = (int)k; } if (i > -1) ASET (monitor_frames, i, Fcons (frame, AREF (monitor_frames, i))); } } return make_monitor_attribute_list (monitors, n_monitors, primary_monitor, monitor_frames, source); } DEFUN ("ns-display-monitor-attributes-list", Fns_display_monitor_attributes_list, Sns_display_monitor_attributes_list, 0, 1, 0, doc: /* Return a list of physical monitor attributes on the X display TERMINAL. The optional argument TERMINAL specifies which display to ask about. TERMINAL should be a terminal object, a frame or a display name (a string). If omitted or nil, that stands for the selected frame's display. In addition to the standard attribute keys listed in `display-monitor-attributes-list', the following keys are contained in the attributes: source -- String describing the source from which multi-monitor information is obtained, \"NS\" is always the source." Internal use only, use `display-monitor-attributes-list' instead. */) (Lisp_Object terminal) { struct terminal *term = decode_live_terminal (terminal); NSArray *screens; NSUInteger i, n_monitors; struct MonitorInfo *monitors; Lisp_Object attributes_list = Qnil; CGFloat primary_display_height = 0; if (term->type != output_ns) return Qnil; screens = [NSScreen screens]; n_monitors = [screens count]; if (n_monitors == 0) return Qnil; monitors = xzalloc (n_monitors * sizeof *monitors); for (i = 0; i < [screens count]; ++i) { NSScreen *s = [screens objectAtIndex:i]; struct MonitorInfo *m = &monitors[i]; NSRect fr = [s frame]; NSRect vfr = [s visibleFrame]; short y, vy; #ifdef NS_IMPL_COCOA NSDictionary *dict = [s deviceDescription]; NSNumber *nid = [dict objectForKey:@"NSScreenNumber"]; CGDirectDisplayID did = [nid unsignedIntValue]; #endif if (i == 0) { primary_display_height = fr.size.height; y = (short) fr.origin.y; vy = (short) vfr.origin.y; } else { // Flip y coordinate as NS has y starting from the bottom. y = (short) (primary_display_height - fr.size.height - fr.origin.y); vy = (short) (primary_display_height - vfr.size.height - vfr.origin.y); } m->geom.x = (short) fr.origin.x; m->geom.y = y; m->geom.width = (unsigned short) fr.size.width; m->geom.height = (unsigned short) fr.size.height; m->work.x = (short) vfr.origin.x; // y is flipped on NS, so vy - y are pixels missing at the bottom, // and fr.size.height - vfr.size.height are pixels missing in total. // Pixels missing at top are // fr.size.height - vfr.size.height - vy + y. // work.y is then pixels missing at top + y. m->work.y = (short) (fr.size.height - vfr.size.height) - vy + y + y; m->work.width = (unsigned short) vfr.size.width; m->work.height = (unsigned short) vfr.size.height; #ifdef NS_IMPL_COCOA m->name = ns_screen_name (did); { CGSize mms = CGDisplayScreenSize (did); m->mm_width = (int) mms.width; m->mm_height = (int) mms.height; } #else // Assume 92 dpi as x-display-mm-height/x-display-mm-width does. m->mm_width = (int) (25.4 * fr.size.width / 92.0); m->mm_height = (int) (25.4 * fr.size.height / 92.0); #endif } // Primary monitor is always first for NS. attributes_list = ns_make_monitor_attribute_list (monitors, n_monitors, 0, "NS"); free_monitors (monitors, n_monitors); return attributes_list; } DEFUN ("x-display-planes", Fx_display_planes, Sx_display_planes, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { check_ns_display_info (terminal); return make_fixnum (NSBitsPerPixelFromDepth ([[[NSScreen screens] objectAtIndex:0] depth])); } DEFUN ("x-display-color-cells", Fx_display_color_cells, Sx_display_color_cells, 0, 1, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object terminal) { struct ns_display_info *dpyinfo = check_ns_display_info (terminal); /* We force 24+ bit depths to 24-bit to prevent an overflow. */ return make_fixnum (1 << min (dpyinfo->n_planes, 24)); } /* TODO: move to xdisp or similar */ static void compute_tip_xy (struct frame *f, Lisp_Object parms, Lisp_Object dx, Lisp_Object dy, int width, int height, int *root_x, int *root_y) { Lisp_Object left, top, right, bottom; NSPoint pt; NSScreen *screen; /* Start with user-specified or mouse position. */ left = Fcdr (Fassq (Qleft, parms)); top = Fcdr (Fassq (Qtop, parms)); right = Fcdr (Fassq (Qright, parms)); bottom = Fcdr (Fassq (Qbottom, parms)); if ((!FIXNUMP (left) && !FIXNUMP (right)) || (!FIXNUMP (top) && !FIXNUMP (bottom))) pt = [NSEvent mouseLocation]; else { /* Absolute coordinates. */ pt.x = FIXNUMP (left) ? XFIXNUM (left) : XFIXNUM (right); pt.y = (ns_display_pixel_height (FRAME_DISPLAY_INFO (f)) - (FIXNUMP (top) ? XFIXNUM (top) : XFIXNUM (bottom)) - height); } /* Find the screen that pt is on. */ for (screen in [NSScreen screens]) if (pt.x >= screen.frame.origin.x && pt.x < screen.frame.origin.x + screen.frame.size.width && pt.y >= screen.frame.origin.y && pt.y < screen.frame.origin.y + screen.frame.size.height) break; /* We could use this instead of the if above: if (CGRectContainsPoint ([screen frame], pt)) which would be neater, but it causes problems building on old versions of macOS and in GNUstep. */ /* Ensure in bounds. (Note, screen origin = lower left.) */ if (FIXNUMP (left) || FIXNUMP (right)) *root_x = pt.x; else if (pt.x + XFIXNUM (dx) <= screen.frame.origin.x) *root_x = screen.frame.origin.x; else if (pt.x + XFIXNUM (dx) + width <= screen.frame.origin.x + screen.frame.size.width) /* It fits to the right of the pointer. */ *root_x = pt.x + XFIXNUM (dx); else if (width + XFIXNUM (dx) <= pt.x) /* It fits to the left of the pointer. */ *root_x = pt.x - width - XFIXNUM (dx); else /* Put it left justified on the screen -- it ought to fit that way. */ *root_x = screen.frame.origin.x; if (FIXNUMP (top) || FIXNUMP (bottom)) *root_y = pt.y; else if (pt.y - XFIXNUM (dy) - height >= screen.frame.origin.y) /* It fits below the pointer. */ *root_y = pt.y - height - XFIXNUM (dy); else if (pt.y + XFIXNUM (dy) + height <= screen.frame.origin.y + screen.frame.size.height) /* It fits above the pointer. */ *root_y = pt.y + XFIXNUM (dy); else /* Put it on the top. */ *root_y = screen.frame.origin.y + screen.frame.size.height - height; } DEFUN ("x-show-tip", Fx_show_tip, Sx_show_tip, 1, 6, 0, doc: /* SKIP: real doc in xfns.c. */) (Lisp_Object string, Lisp_Object frame, Lisp_Object parms, Lisp_Object timeout, Lisp_Object dx, Lisp_Object dy) { int root_x, root_y; ptrdiff_t count = SPECPDL_INDEX (); struct frame *f; char *str; NSSize size; NSColor *color; Lisp_Object t; specbind (Qinhibit_redisplay, Qt); CHECK_STRING (string); str = SSDATA (string); f = decode_window_system_frame (frame); if (NILP (timeout)) timeout = make_fixnum (5); else CHECK_FIXNAT (timeout); if (NILP (dx)) dx = make_fixnum (5); else CHECK_FIXNUM (dx); if (NILP (dy)) dy = make_fixnum (-10); else CHECK_FIXNUM (dy); block_input (); if (ns_tooltip == nil) ns_tooltip = [[EmacsTooltip alloc] init]; else Fx_hide_tip (); t = gui_display_get_arg (NULL, parms, Qbackground_color, NULL, NULL, RES_TYPE_STRING); if (ns_lisp_to_color (t, &color) == 0) [ns_tooltip setBackgroundColor: color]; t = gui_display_get_arg (NULL, parms, Qforeground_color, NULL, NULL, RES_TYPE_STRING); if (ns_lisp_to_color (t, &color) == 0) [ns_tooltip setForegroundColor: color]; [ns_tooltip setText: str]; size = [ns_tooltip frame].size; /* Move the tooltip window where the mouse pointer is. Resize and show it. */ compute_tip_xy (f, parms, dx, dy, (int)size.width, (int)size.height, &root_x, &root_y); [ns_tooltip showAtX: root_x Y: root_y for: XFIXNUM (timeout)]; unblock_input (); return unbind_to (count, Qnil); } DEFUN ("x-hide-tip", Fx_hide_tip, Sx_hide_tip, 0, 0, 0, doc: /* SKIP: real doc in xfns.c. */) (void) { if (ns_tooltip == nil || ![ns_tooltip isActive]) return Qnil; [ns_tooltip hide]; return Qt; } /* Return geometric attributes of FRAME. According to the value of ATTRIBUTES return the outer edges of FRAME (Qouter_edges), the inner edges of FRAME, the root window edges of frame (Qroot_edges). Any other value means to return the geometry as returned by Fx_frame_geometry. */ static Lisp_Object frame_geometry (Lisp_Object frame, Lisp_Object attribute) { struct frame *f = decode_live_frame (frame); Lisp_Object fullscreen_symbol = Fframe_parameter (frame, Qfullscreen); bool fullscreen = (EQ (fullscreen_symbol, Qfullboth) || EQ (fullscreen_symbol, Qfullscreen)); int border = fullscreen ? 0 : f->border_width; int title_height = fullscreen ? 0 : FRAME_NS_TITLEBAR_HEIGHT (f); int native_width = FRAME_PIXEL_WIDTH (f); int native_height = FRAME_PIXEL_HEIGHT (f); int outer_width = native_width + 2 * border; int outer_height = native_height + 2 * border + title_height; int native_left = f->left_pos + border; int native_top = f->top_pos + border + title_height; int native_right = f->left_pos + outer_width - border; int native_bottom = f->top_pos + outer_height - border; int internal_border_width = FRAME_INTERNAL_BORDER_WIDTH (f); int tool_bar_height = FRAME_TOOLBAR_HEIGHT (f); int tool_bar_width = (tool_bar_height ? outer_width - 2 * internal_border_width : 0); /* Construct list. */ if (EQ (attribute, Qouter_edges)) return list4i (f->left_pos, f->top_pos, f->left_pos + outer_width, f->top_pos + outer_height); else if (EQ (attribute, Qnative_edges)) return list4i (native_left, native_top, native_right, native_bottom); else if (EQ (attribute, Qinner_edges)) return list4i (native_left + internal_border_width, native_top + tool_bar_height + internal_border_width, native_right - internal_border_width, native_bottom - internal_border_width); else return list (Fcons (Qouter_position, Fcons (make_fixnum (f->left_pos), make_fixnum (f->top_pos))), Fcons (Qouter_size, Fcons (make_fixnum (outer_width), make_fixnum (outer_height))), Fcons (Qexternal_border_size, (fullscreen ? Fcons (make_fixnum (0), make_fixnum (0)) : Fcons (make_fixnum (border), make_fixnum (border)))), Fcons (Qtitle_bar_size, Fcons (make_fixnum (0), make_fixnum (title_height))), Fcons (Qmenu_bar_external, Qnil), Fcons (Qmenu_bar_size, Fcons (make_fixnum (0), make_fixnum (0))), Fcons (Qtool_bar_external, FRAME_EXTERNAL_TOOL_BAR (f) ? Qt : Qnil), Fcons (Qtool_bar_position, FRAME_TOOL_BAR_POSITION (f)), Fcons (Qtool_bar_size, Fcons (make_fixnum (tool_bar_width), make_fixnum (tool_bar_height))), Fcons (Qinternal_border_width, make_fixnum (internal_border_width))); } DEFUN ("ns-frame-geometry", Fns_frame_geometry, Sns_frame_geometry, 0, 1, 0, doc: /* Return geometric attributes of FRAME. FRAME must be a live frame and defaults to the selected one. The return value is an association list of the attributes listed below. All height and width values are in pixels. `outer-position' is a cons of the outer left and top edges of FRAME relative to the origin - the position (0, 0) - of FRAME's display. `outer-size' is a cons of the outer width and height of FRAME. The outer size includes the title bar and the external borders as well as any menu and/or tool bar of frame. `external-border-size' is a cons of the horizontal and vertical width of FRAME's external borders as supplied by the window manager. `title-bar-size' is a cons of the width and height of the title bar of FRAME as supplied by the window manager. If both of them are zero, FRAME has no title bar. If only the width is zero, Emacs was not able to retrieve the width information. `menu-bar-external', if non-nil, means the menu bar is external (never included in the inner edges of FRAME). `menu-bar-size' is a cons of the width and height of the menu bar of FRAME. `tool-bar-external', if non-nil, means the tool bar is external (never included in the inner edges of FRAME). `tool-bar-position' tells on which side the tool bar on FRAME is and can be one of `left', `top', `right' or `bottom'. If this is nil, FRAME has no tool bar. `tool-bar-size' is a cons of the width and height of the tool bar of FRAME. `internal-border-width' is the width of the internal border of FRAME. */) (Lisp_Object frame) { return frame_geometry (frame, Qnil); } DEFUN ("ns-frame-edges", Fns_frame_edges, Sns_frame_edges, 0, 2, 0, doc: /* Return edge coordinates of FRAME. FRAME must be a live frame and defaults to the selected one. The return value is a list of the form (LEFT, TOP, RIGHT, BOTTOM). All values are in pixels relative to the origin - the position (0, 0) - of FRAME's display. If optional argument TYPE is the symbol `outer-edges', return the outer edges of FRAME. The outer edges comprise the decorations of the window manager (like the title bar or external borders) as well as any external menu or tool bar of FRAME. If optional argument TYPE is the symbol `native-edges' or nil, return the native edges of FRAME. The native edges exclude the decorations of the window manager and any external menu or tool bar of FRAME. If TYPE is the symbol `inner-edges', return the inner edges of FRAME. These edges exclude title bar, any borders, menu bar or tool bar of FRAME. */) (Lisp_Object frame, Lisp_Object type) { return frame_geometry (frame, ((EQ (type, Qouter_edges) || EQ (type, Qinner_edges)) ? type : Qnative_edges)); } DEFUN ("ns-set-mouse-absolute-pixel-position", Fns_set_mouse_absolute_pixel_position, Sns_set_mouse_absolute_pixel_position, 2, 2, 0, doc: /* Move mouse pointer to absolute pixel position (X, Y). The coordinates X and Y are interpreted in pixels relative to a position \(0, 0) of the selected frame's display. */) (Lisp_Object x, Lisp_Object y) { #ifdef NS_IMPL_COCOA /* GNUstep doesn't support CGWarpMouseCursorPosition, so none of this will work. */ struct frame *f = SELECTED_FRAME (); EmacsView *view = FRAME_NS_VIEW (f); NSScreen *screen = [[view window] screen]; NSRect screen_frame = [screen frame]; int mouse_x, mouse_y; NSScreen *primary_screen = [[NSScreen screens] objectAtIndex:0]; NSRect primary_screen_frame = [primary_screen frame]; CGFloat primary_screen_height = primary_screen_frame.size.height; if (FRAME_INITIAL_P (f) || !FRAME_NS_P (f)) return Qnil; int xval = check_integer_range (x, INT_MIN, INT_MAX); int yval = check_integer_range (y, INT_MIN, INT_MAX); mouse_x = screen_frame.origin.x + xval; if (screen == primary_screen) mouse_y = screen_frame.origin.y + yval; else mouse_y = (primary_screen_height - screen_frame.size.height - screen_frame.origin.y) + yval; CGPoint mouse_pos = CGPointMake(mouse_x, mouse_y); CGWarpMouseCursorPosition (mouse_pos); #endif /* NS_IMPL_COCOA */ return Qnil; } DEFUN ("ns-mouse-absolute-pixel-position", Fns_mouse_absolute_pixel_position, Sns_mouse_absolute_pixel_position, 0, 0, 0, doc: /* Return absolute position of mouse cursor in pixels. The position is returned as a cons cell (X . Y) of the coordinates of the mouse cursor position in pixels relative to a position (0, 0) of the selected frame's terminal. */) (void) { struct frame *f = SELECTED_FRAME (); EmacsView *view = FRAME_NS_VIEW (f); NSScreen *screen = [[view window] screen]; NSPoint pt = [NSEvent mouseLocation]; return Fcons(make_fixnum(pt.x - screen.frame.origin.x), make_fixnum(screen.frame.size.height - (pt.y - screen.frame.origin.y))); } DEFUN ("ns-show-character-palette", Fns_show_character_palette, Sns_show_character_palette, 0, 0, 0, doc: /* Show the macOS character palette. */) (void) { struct frame *f = SELECTED_FRAME (); EmacsView *view = FRAME_NS_VIEW (f); [NSApp orderFrontCharacterPalette:view]; return Qnil; } /* ========================================================================== Class implementations ========================================================================== */ @implementation EmacsFileDelegate /* -------------------------------------------------------------------------- Delegate methods for Open/Save panels -------------------------------------------------------------------------- */ - (BOOL)panel: (id)sender isValidFilename: (NSString *)filename { return YES; } - (BOOL)panel: (id)sender shouldShowFilename: (NSString *)filename { return YES; } - (NSString *)panel: (id)sender userEnteredFilename: (NSString *)filename confirmed: (BOOL)okFlag { return filename; } @end #endif /* Whether N bytes at STR are in the [1,127] range. */ static bool all_nonzero_ascii (unsigned char *str, ptrdiff_t n) { for (ptrdiff_t i = 0; i < n; i++) if (str[i] < 1 || str[i] > 127) return false; return true; } @implementation NSString (EmacsString) /* Make an NSString from a Lisp string. STRING must not be in an encoded form (e.g. UTF-8). */ + (NSString *)stringWithLispString:(Lisp_Object)string { /* Shortcut for the common case. */ if (all_nonzero_ascii (SDATA (string), SBYTES (string))) return [NSString stringWithCString: SSDATA (string) encoding: NSASCIIStringEncoding]; string = string_to_multibyte (string); /* Now the string is multibyte; convert to UTF-16. */ unichar *chars = xmalloc (4 * SCHARS (string)); unichar *d = chars; const unsigned char *s = SDATA (string); const unsigned char *end = s + SBYTES (string); while (s < end) { int c = string_char_advance (&s); /* We pass unpaired surrogates through, because they are typically handled fairly well by the NS libraries (displayed with distinct glyphs etc). */ if (c <= 0xffff) *d++ = c; else if (c <= 0x10ffff) { *d++ = 0xd800 + ((c - 0x10000) >> 10); *d++ = 0xdc00 + (c & 0x3ff); } else *d++ = 0xfffd; /* Not valid for UTF-16. */ } NSString *str = [NSString stringWithCharacters: chars length: d - chars]; xfree (chars); return str; } /* Make a Lisp string from an NSString. */ - (Lisp_Object)lispString { return build_string ([self UTF8String]); } @end /* ========================================================================== Lisp interface declaration ========================================================================== */ void syms_of_nsfns (void) { DEFSYM (Qfontsize, "fontsize"); DEFSYM (Qframe_title_format, "frame-title-format"); DEFSYM (Qicon_title_format, "icon-title-format"); DEFSYM (Qdark, "dark"); DEFSYM (Qlight, "light"); DEFVAR_LISP ("ns-icon-type-alist", Vns_icon_type_alist, doc: /* Alist of elements (REGEXP . IMAGE) for images of icons associated to frames. If the title of a frame matches REGEXP, then IMAGE.tiff is selected as the image of the icon representing the frame when it's miniaturized. If an element is t, then Emacs tries to select an icon based on the filetype of the visited file. The images have to be installed in a folder called English.lproj in the Emacs folder. You have to restart Emacs after installing new icons. Example: Install an icon Gnus.tiff and execute the following code (setq ns-icon-type-alist (append ns-icon-type-alist \\='((\"^\\\\*\\\\(Group\\\\*$\\\\|Summary \\\\|Article\\\\*$\\\\)\" . \"Gnus\")))) When you miniaturize a Group, Summary or Article frame, Gnus.tiff will be used as the image of the icon representing the frame. */); Vns_icon_type_alist = list1 (Qt); DEFVAR_LISP ("ns-version-string", Vns_version_string, doc: /* Toolkit version for NS Windowing. */); Vns_version_string = ns_appkit_version_str (); DEFVAR_BOOL ("ns-use-proxy-icon", ns_use_proxy_icon, doc: /* When non-nil display a proxy icon in the titlebar. Default is t. */); ns_use_proxy_icon = true; defsubr (&Sns_read_file_name); defsubr (&Sns_get_resource); defsubr (&Sns_set_resource); defsubr (&Sxw_display_color_p); /* this and next called directly by C code */ defsubr (&Sx_display_grayscale_p); defsubr (&Sns_font_name); defsubr (&Sns_list_colors); #ifdef NS_IMPL_COCOA defsubr (&Sns_do_applescript); #endif defsubr (&Sxw_color_defined_p); defsubr (&Sxw_color_values); defsubr (&Sx_server_max_request_size); defsubr (&Sx_server_vendor); defsubr (&Sx_server_version); defsubr (&Sx_display_pixel_width); defsubr (&Sx_display_pixel_height); defsubr (&Sns_display_monitor_attributes_list); defsubr (&Sns_frame_geometry); defsubr (&Sns_frame_edges); defsubr (&Sns_frame_list_z_order); defsubr (&Sns_frame_restack); defsubr (&Sns_set_mouse_absolute_pixel_position); defsubr (&Sns_mouse_absolute_pixel_position); defsubr (&Sns_show_character_palette); defsubr (&Sx_display_mm_width); defsubr (&Sx_display_mm_height); defsubr (&Sx_display_screens); defsubr (&Sx_display_planes); defsubr (&Sx_display_color_cells); defsubr (&Sx_display_visual_class); defsubr (&Sx_display_backing_store); defsubr (&Sx_display_save_under); defsubr (&Sx_create_frame); defsubr (&Sx_open_connection); defsubr (&Sx_close_connection); defsubr (&Sx_display_list); defsubr (&Sns_hide_others); defsubr (&Sns_hide_emacs); defsubr (&Sns_emacs_info_panel); defsubr (&Sns_list_services); defsubr (&Sns_perform_service); defsubr (&Sns_popup_font_panel); defsubr (&Sns_popup_color_panel); defsubr (&Sx_show_tip); defsubr (&Sx_hide_tip); as_status = 0; as_script = Qnil; staticpro (&as_script); as_result = 0; }