This function runs the hook `focus-out-hook'."
(interactive "e")
(run-hooks 'focus-out-hook))
+
+(defun handle-move-frame (event)
+ "Handle a move-frame event.
+This function runs the abnormal hook `move-frame-functions'."
+ (interactive "e")
+ (let ((frame (posn-window (event-start event))))
+ (run-hook-with-args 'move-frame-functions frame)))
\f
;;;; Arrangement of frames at startup
for frames = (cdr (assq 'frames attributes))
if (memq frame frames) return attributes))
+(defun frame-size-changed-p (&optional frame)
+ "Return non-nil when the size of FRAME has changed.
+More precisely, return non-nil when the inner width or height of
+FRAME has changed since `window-size-change-functions' was run
+for FRAME."
+ (let* ((frame (window-normalize-frame frame))
+ (root (frame-root-window frame))
+ (mini (minibuffer-window frame))
+ (mini-height-before-size-change 0)
+ (mini-height 0))
+ ;; FRAME's minibuffer window counts iff it's on FRAME and FRAME is
+ ;; not a minibuffer-only frame.
+ (when (and (eq (window-frame mini) frame) (not (eq mini root)))
+ (setq mini-height-before-size-change
+ (window-pixel-height-before-size-change mini))
+ (setq mini-height (window-pixel-height mini)))
+ ;; Return non-nil when either the width of the root or the sum of
+ ;; the heights of root and minibuffer window changed.
+ (or (/= (window-pixel-width-before-size-change root)
+ (window-pixel-width root))
+ (/= (+ (window-pixel-height-before-size-change root)
+ mini-height-before-size-change)
+ (+ (window-pixel-height root) mini-height)))))
\f
;;;; Frame/display capabilities.
Lisp_Object
do_switch_frame (Lisp_Object frame, int track, int for_deletion, Lisp_Object norecord)
{
- struct frame *sf = SELECTED_FRAME ();
+ struct frame *sf = SELECTED_FRAME (), *f;
/* If FRAME is a switch-frame event, extract the frame we should
switch to. */
a switch-frame event to arrive after a frame is no longer live,
especially when deleting the initial frame during startup. */
CHECK_FRAME (frame);
- if (! FRAME_LIVE_P (XFRAME (frame)))
+ f = XFRAME (frame);
+ if (!FRAME_LIVE_P (f))
return Qnil;
-
- if (sf == XFRAME (frame))
+ else if (f == sf)
return frame;
/* If a frame's focus has been redirected toward the currently
#else /* ! 0 */
/* Instead, apply it only to the frame we're pointing to. */
#ifdef HAVE_WINDOW_SYSTEM
- if (track && FRAME_WINDOW_P (XFRAME (frame)))
+ if (track && FRAME_WINDOW_P (f))
{
Lisp_Object focus, xfocus;
- xfocus = x_get_focus_frame (XFRAME (frame));
+ xfocus = x_get_focus_frame (f);
if (FRAMEP (xfocus))
{
focus = FRAME_FOCUS_FRAME (XFRAME (xfocus));
/* Redirect frame focus also when FRAME has its minibuffer
window on the selected frame (see Bug#24500). */
|| (NILP (focus)
- && EQ (FRAME_MINIBUF_WINDOW (XFRAME (frame)),
- sf->selected_window)))
+ && EQ (FRAME_MINIBUF_WINDOW (f), sf->selected_window)))
Fredirect_frame_focus (xfocus, frame);
}
}
if (!for_deletion && FRAME_HAS_MINIBUF_P (sf))
resize_mini_window (XWINDOW (FRAME_MINIBUF_WINDOW (sf)), 1);
- if (FRAME_TERMCAP_P (XFRAME (frame)) || FRAME_MSDOS_P (XFRAME (frame)))
+ if (FRAME_TERMCAP_P (f) || FRAME_MSDOS_P (f))
{
- struct frame *f = XFRAME (frame);
struct tty_display_info *tty = FRAME_TTY (f);
Lisp_Object top_frame = tty->top_frame;
if (! FRAME_MINIBUF_ONLY_P (XFRAME (selected_frame)))
last_nonminibuf_frame = XFRAME (selected_frame);
- Fselect_window (XFRAME (frame)->selected_window, norecord);
+ Fselect_window (f->selected_window, norecord);
/* We want to make sure that the next event generates a frame-switch
event to the appropriate frame. This seems kludgy to me, but
to that frame. */)
(Lisp_Object event)
{
+ Lisp_Object value;
+
/* Preserve prefix arg that the command loop just cleared. */
kset_prefix_arg (current_kboard, Vcurrent_prefix_arg);
run_hook (Qmouse_leave_buffer_hook);
/* `switch-frame' implies a focus in. */
+ value = do_switch_frame (event, 0, 0, Qnil);
call1 (intern ("handle-focus-in"), event);
- return do_switch_frame (event, 0, 0, Qnil);
+ return value;
}
DEFUN ("selected-frame", Fselected_frame, Sselected_frame, 0, 0, 0,
promise that the terminal of the frame must be valid until we
have called the window-system-dependent frame destruction
routine. */
-
-
{
struct terminal *terminal;
block_input ();
Vmake_pointer_invisible = Qt;
DEFVAR_LISP ("focus-in-hook", Vfocus_in_hook,
- doc: /* Normal hook run when a frame gains input focus. */);
+ doc: /* Normal hook run when a frame gains input focus.
+The frame gaining focus is selected at the time this hook is run. */);
Vfocus_in_hook = Qnil;
DEFVAR_LISP ("focus-out-hook", Vfocus_out_hook,
- doc: /* Normal hook run when a frame loses input focus. */);
+ doc: /* Normal hook run when all frames lost input focus. */);
Vfocus_out_hook = Qnil;
+ DEFVAR_LISP ("move-frame-functions", Vmove_frame_functions,
+ doc: /* Functions run after a frame was moved.
+The functions are run with one arg, the frame that moved. */);
+ Vmove_frame_functions = Qnil;
+
DEFVAR_LISP ("delete-frame-functions", Vdelete_frame_functions,
doc: /* Functions run before deleting a frame.
The functions are run with one arg, the frame to be deleted.
kbd_fetch_ptr = event + 1;
}
#endif
+#if defined (HAVE_X11) || defined (HAVE_NTGUI)
+ else if (event->kind == MOVE_FRAME_EVENT)
+ {
+ /* Make an event (move-frame (FRAME)). */
+ obj = list2 (Qmove_frame, list1 (event->ie.frame_or_window));
+ kbd_fetch_ptr = event + 1;
+ }
+#endif
#ifdef HAVE_XWIDGETS
else if (event->kind == XWIDGET_EVENT)
{
obj = make_lispy_event (&event->ie);
kbd_fetch_ptr = event + 1;
}
+ else if (event->kind == SELECT_WINDOW_EVENT)
+ {
+ obj = list2 (Qselect_window, list1 (event->ie.frame_or_window));
+ kbd_fetch_ptr = event + 1;
+ }
else
{
/* If this event is on a different frame, return a switch-frame this
{SYMBOL_INDEX (Qfocus_in), SYMBOL_INDEX (Qfocus_in)},
{SYMBOL_INDEX (Qfocus_out), SYMBOL_INDEX (Qfocus_out)},
+ {SYMBOL_INDEX (Qmove_frame), SYMBOL_INDEX (Qmove_frame)},
{SYMBOL_INDEX (Qdelete_frame), SYMBOL_INDEX (Qdelete_frame)},
{SYMBOL_INDEX (Qiconify_frame), SYMBOL_INDEX (Qiconify_frame)},
{SYMBOL_INDEX (Qmake_frame_visible), SYMBOL_INDEX (Qmake_frame_visible)},
DEFSYM (Qswitch_frame, "switch-frame");
DEFSYM (Qfocus_in, "focus-in");
DEFSYM (Qfocus_out, "focus-out");
+ DEFSYM (Qmove_frame, "move-frame");
DEFSYM (Qdelete_frame, "delete-frame");
DEFSYM (Qiconify_frame, "iconify-frame");
DEFSYM (Qmake_frame_visible, "make-frame-visible");
"handle-focus-in");
initial_define_lispy_key (Vspecial_event_map, "focus-out",
"handle-focus-out");
+ initial_define_lispy_key (Vspecial_event_map, "move-frame",
+ "handle-move-frame");
}
/* Mark the pointers in the kboard objects.
FOCUS_OUT_EVENT,
+ /* Generated when a frame is moved. */
+ MOVE_FRAME_EVENT,
+
/* Generated when mouse moves over window not currently selected. */
SELECT_WINDOW_EVENT,
f = x_window_to_frame (dpyinfo, msg.msg.hwnd);
if (f && !FRAME_ICONIFIED_P (f))
- x_real_positions (f, &f->left_pos, &f->top_pos);
+ {
+ x_real_positions (f, &f->left_pos, &f->top_pos);
+ inev.kind = MOVE_FRAME_EVENT;
+ XSETFRAME (inev.frame_or_window, f);
+ }
check_visibility = 1;
break;
Lisp_Object functions = Vwindow_size_change_functions;
if (FRAME_WINDOW_CONFIGURATION_CHANGED (f)
+ /* Here we implicitly exclude the possibility that the height of
+ FRAME and its minibuffer window both change leaving the height
+ of FRAME's root window alone. */
|| window_size_changed (r))
{
while (CONSP (functions))
}
window_set_before_size_change_sizes (r);
+
+ if (FRAME_HAS_MINIBUF_P (f) && !FRAME_MINIBUF_ONLY_P (f))
+ /* Record size of FRAME's minibuffer window too. */
+ window_set_before_size_change_sizes
+ (XWINDOW (FRAME_MINIBUF_WINDOW (f)));
+
FRAME_WINDOW_CONFIGURATION_CHANGED (f) = false;
}
}
if (FRAME_GTK_OUTER_WIDGET (f)
&& gtk_widget_get_mapped (FRAME_GTK_OUTER_WIDGET (f)))
#endif
- x_real_positions (f, &f->left_pos, &f->top_pos);
+ {
+ int old_left = f->left_pos;
+ int old_top = f->top_pos;
+ Lisp_Object frame = Qnil;
+
+ XSETFRAME (frame, f);
+
+ x_real_positions (f, &f->left_pos, &f->top_pos);
+
+ if (old_left != f->left_pos || old_top != f->top_pos)
+ {
+ inev.ie.kind = MOVE_FRAME_EVENT;
+ XSETFRAME (inev.ie.frame_or_window, f);
+ }
+ }
+
#ifdef HAVE_X_I18N
if (FRAME_XIC (f) && (FRAME_XIC_STYLE (f) & XIMStatusArea))
modified_top += FRAME_X_OUTPUT (f)->move_offset_top;
}
+#ifdef USE_GTK
+ gtk_window_move (GTK_WINDOW (FRAME_GTK_OUTER_WIDGET (f)),
+ modified_left, modified_top);
+#else
XMoveWindow (FRAME_X_DISPLAY (f), FRAME_OUTER_WINDOW (f),
- modified_left, modified_top);
+ modified_left, modified_top);
+#endif
x_sync_with_move (f, f->left_pos, f->top_pos,
FRAME_DISPLAY_INFO (f)->wm_type == X_WMTYPE_UNKNOWN);