2184 lines
103 KiB
EmacsLisp
2184 lines
103 KiB
EmacsLisp
;;; frame-cmds.el --- Frame and window commands (interactive functions).
|
|
;;
|
|
;; Filename: frame-cmds.el
|
|
;; Description: Frame and window commands (interactive functions).
|
|
;; Author: Drew Adams
|
|
;; Maintainer: Drew Adams (concat "drew.adams" "@" "oracle" ".com")
|
|
;; Copyright (C) 1996-2021, Drew Adams, all rights reserved.
|
|
;; Created: Tue Mar 5 16:30:45 1996
|
|
;; Version: 0
|
|
;; Package-Requires: ((frame-fns "0"))
|
|
;; Last-Updated: Thu Mar 18 09:11:08 2021 (-0700)
|
|
;; By: dradams
|
|
;; Update #: 3187
|
|
;; URL: https://www.emacswiki.org/emacs/download/frame-cmds.el
|
|
;; Doc URL: https://emacswiki.org/emacs/FrameModes
|
|
;; Doc URL: https://www.emacswiki.org/emacs/OneOnOneEmacs
|
|
;; Doc URL: https://www.emacswiki.org/emacs/Frame_Tiling_Commands
|
|
;; Keywords: internal, extensions, mouse, frames, windows, convenience
|
|
;; Compatibility: GNU Emacs: 20.x, 21.x, 22.x, 23.x, 24.x, 25.x, 26.x
|
|
;;
|
|
;; Features that might be required by this library:
|
|
;;
|
|
;; `avoid', `backquote', `bytecomp', `cconv', `cl-lib',
|
|
;; `frame-fns', `macroexp', `misc-fns', `strings', `thingatpt',
|
|
;; `thingatpt+'.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;;; Commentary:
|
|
;;
|
|
;; Frame and window commands (interactive functions).
|
|
;;
|
|
;;
|
|
;; Summary:
|
|
;;
|
|
;; Load this library from your init file (~/.emacs or _emacs).
|
|
;; Add the suggested key bindings (below) to your init file.
|
|
;; Use `M-up|down|left|right' to move frames around incrementally.
|
|
;; Use `C-S-v', `M-S-v', `C-S-next', `C-S-prior' to move frames to
|
|
;; screen edges.
|
|
;; Use `C-M-up|down|left|right' to resize frames incrementally.
|
|
;; Use `C-x {', `C-x }', `C-x ^', and `C-u C-x ^' to resize windows
|
|
;; incrementally.
|
|
;; Use `C-M-z' or `C-x C-z' to iconify/hide all frames.
|
|
;; Use `C-M-z' in a lone frame to restore all frames.
|
|
;; Use `C-mouse-1' in the minibuffer to restore all frames.
|
|
;; Use `C-mouse-1' in Dired to mark/unmark a file.
|
|
;; Use `C-mouse-3' on the mode line to remove window from frame.
|
|
;; Use `tile-frames-horizontally', `-vertically' to tile frames.
|
|
;; Use `C-x o' to select `other-window' or `other-frame'.
|
|
;;
|
|
;; Commands to incrementally resize frames are `enlarge-frame' and
|
|
;; `enlarge-frame-horizontally'. Sarir Khamsi
|
|
;; [sarir.khamsi@raytheon.com] originally wrote `enlarge-frame',
|
|
;; which he called `sk-grow-frame'.
|
|
;;
|
|
;; Note on saving changes made with the commands defined here:
|
|
;;
|
|
;; Some of the commands defined here change frame properties.
|
|
;; You can save any changes you have made, by using Customize.
|
|
;; To visit a Customize buffer of all unsaved changes you have
|
|
;; made, use command `customize-customized'.
|
|
;;
|
|
;; Frame parameter changes, such as background color, can be saved
|
|
;; for future use by all frames or all frames of a certain
|
|
;; kind. For that, you must change the frame parameters of the
|
|
;; corresponding frame-alist variable.
|
|
;;
|
|
;; There is no single variable for saving changes to parameters of
|
|
;; the current frame. Instead, there are several different
|
|
;; frame-alist variables, which you can use to define different
|
|
;; kinds of frames. These include: `default-frame-alist',
|
|
;; `initial-frame-alist', and `special-display-frame-alist'. The
|
|
;; complete list of such frame alist variables is available using
|
|
;; function `frcmds-frame-alist-var-names', defined here.
|
|
;;
|
|
;; Example: Suppose you change the background color of a frame and
|
|
;; want to make that the default background color for new frames in
|
|
;; the future. You will need to update the value of variable
|
|
;; `default-frame-alist' to use the `background-color' parameter
|
|
;; setting of the changed frame.
|
|
;;
|
|
;; You can easily copy one or all parameter values from any given
|
|
;; frame to any frame alist (such as `default-frame-alist'), by
|
|
;; using the commands `set-frame-alist-parameter-from-frame' and
|
|
;; `set-all-frame-alist-parameters-from-frame'. Those commands are
|
|
;; defined here.
|
|
;;
|
|
;; NOTE: If you also use library `fit-frame.el', and you are on MS
|
|
;; Windows, then load that library before `frame-cmds.el'. The
|
|
;; commands `maximize-frame' and `restore-frame' defined here are
|
|
;; more general and non-Windows-specific than the commands of the
|
|
;; same name defined in `fit-frame.el'.
|
|
;;
|
|
;;
|
|
;; User options defined here:
|
|
;;
|
|
;; `available-screen-pixel-bounds', `clone-frame-parameters',
|
|
;; `enlarge-font-tries', `frame-config-register',
|
|
;; `frame-parameters-to-exclude',
|
|
;; `move-frame-wrap-within-display-flag'
|
|
;; `rename-frame-when-iconify-flag', `show-hide-show-function',
|
|
;; `window-mgr-title-bar-pixel-height'.
|
|
;;
|
|
;; Commands defined here:
|
|
;;
|
|
;; `clone-frame', `create-frame-tiled-horizontally',
|
|
;; `create-frame-tiled-vertically', `decrease-frame-transparency'
|
|
;; (Emacs 23+), `delete-1-window-frames-on',
|
|
;; `delete/iconify-window', `delete/iconify-windows-on',
|
|
;; `delete-other-frames', `delete-windows-for', `enlarge-font',
|
|
;; `enlarge-frame', `enlarge-frame-horizontally',
|
|
;; `enlarge/shrink-window', `enlarge/shrink-window-repeat',
|
|
;; `enlarge-window-repeat', `hide-everything', `hide-frame',
|
|
;; `iconify-everything', `iconify/map-frame', `iconify/show-frame',
|
|
;; `increase-frame-transparency' (Emacs 23+),
|
|
;; `jump-to-frame-config-register', `maximize-frame',
|
|
;; `maximize-frame-horizontally', `maximize-frame-vertically',
|
|
;; `mouse-iconify/map-frame', `mouse-iconify/show-frame',
|
|
;; `mouse-remove-window', `mouse-show-hide-mark-unmark',
|
|
;; `move-frame-down', `move-frame-left', `move-frame-right',
|
|
;; `move-frame-to-screen-bottom', `move-frame-to-screen-left',
|
|
;; `move-frame-to-screen-right', `move-frame-to-screen-top',
|
|
;; `move-frame-to-screen-top-left', `move-frame-up',
|
|
;; `name-all-frames-numerically', `name-frame-numerically',
|
|
;; `other-window-or-frame', `remove-window', `remove-windows-on',
|
|
;; `rename-frame', `rename-non-minibuffer-frame', `restore-frame',
|
|
;; `restore-frame-horizontally', `restore-frame-vertically',
|
|
;; `save-frame-config',
|
|
;; `set-all-frame-alist-parameters-from-frame',
|
|
;; `set-frame-alist-parameter-from-frame', `show-*Help*-buffer',
|
|
;; `show-a-frame-on', `show-buffer-menu', `show-frame',
|
|
;; `show-hide', `shrink-frame', `shrink-frame-horizontally',
|
|
;; `split-frame-horizontally', `split-frame-vertically',
|
|
;; `shrink-window-horizontally-repeat', `shrink-window-repeat',
|
|
;; `tell-customize-var-has-changed', `tile-frames',
|
|
;; `tile-frames-horizontally', `tile-frames-side-by-side',
|
|
;; `tile-frames-top-to-bottom', `tile-frames-vertically',
|
|
;; `toggle-max-frame', `toggle-max-frame-horizontally',
|
|
;; `toggle-max-frame-vertically', `toggle-menu-bar-for-frame'.
|
|
;;
|
|
;; Non-interactive functions defined here:
|
|
;;
|
|
;; `assq-delete-all' (Emacs 20), `butlast' (Emacs 20),
|
|
;; `frcmds-available-screen-pixel-bounds',
|
|
;; `frcmds-available-screen-pixel-height',
|
|
;; `frcmds-available-screen-pixel-width',
|
|
;; `frcmds-effective-screen-pixel-bounds',
|
|
;; `frcmds-enlarged-font-name', `frcmds-extra-pixels-width',
|
|
;; `frcmds-extra-pixels-height', `frcmds-frame-alist-var-names',
|
|
;; `frcmds-frame-parameter-names', `frcmds-frame-iconified-p',
|
|
;; `frcmds-frame-number', `frcmds-new-frame-position',
|
|
;; `frcmds-read-args-for-tiling',
|
|
;; `frcmds-read-buffer-for-delete-windows',
|
|
;; `frcmds-repeat-command', `frcmds-set-difference',
|
|
;; `frcmds-smart-tool-bar-pixel-height', `frcmds-split-frame-1',
|
|
;; `frcmds-tile-frames', `nbutlast' (Emacs 20).
|
|
;;
|
|
;; Error symbols defined here:
|
|
;;
|
|
;; `font-too-small', `font-size'.
|
|
;;
|
|
;;
|
|
;; ***** NOTE: The following EMACS PRIMITIVE has been ADVISED HERE:
|
|
;;
|
|
;; `delete-window' - If only one window in frame, `delete-frame'.
|
|
;;
|
|
;;
|
|
;; ***** NOTE: The following EMACS PRIMITIVE has been REDEFINED HERE:
|
|
;;
|
|
;; `delete-windows-on' -
|
|
;; 1) Reads buffer differently. Only buffers showing windows are candidates.
|
|
;; 2) Calls `delete-window', so this also deletes frames where
|
|
;; window showing the BUFFER is the only window.
|
|
;; (That's true also for vanilla Emacs 23+, but not before.)
|
|
;;
|
|
;;
|
|
;; Suggested key bindings:
|
|
;;
|
|
;; (global-set-key [(meta up)] 'move-frame-up)
|
|
;; (global-set-key [(meta down)] 'move-frame-down)
|
|
;; (global-set-key [(meta left)] 'move-frame-left)
|
|
;; (global-set-key [(meta right)] 'move-frame-right)
|
|
;; (global-set-key [(meta shift ?v)] 'move-frame-to-screen-top) ; like `M-v'
|
|
;; (global-set-key [(control shift ?v)] 'move-frame-to-screen-bottom) ; like `C-v'
|
|
;; (global-set-key [(control shift prior)] 'move-frame-to-screen-left) ; like `C-prior'
|
|
;; (global-set-key [(control shift next)] 'move-frame-to-screen-right) ; like `C-next'
|
|
;; (global-set-key [(control shift home)] 'move-frame-to-screen-top-left)
|
|
;; (global-set-key [(control meta down)] 'enlarge-frame)
|
|
;; (global-set-key [(control meta right)] 'enlarge-frame-horizontally)
|
|
;; (global-set-key [(control meta up)] 'shrink-frame)
|
|
;; (global-set-key [(control meta left)] 'shrink-frame-horizontally)
|
|
;; (global-set-key [remap enlarge-window-horizontally] 'enlarge-window-horizontally-repeat)
|
|
;; (global-set-key [remap shrink-window-horizontally] 'shrink-window-horizontally-repeat)
|
|
;; (global-set-key [remap enlarge-window] 'enlarge/shrink-window-repeat)
|
|
;; (global-set-key (kbd "C-M-S-<down>") 'increase-frame-transparency)
|
|
;; (global-set-key (kbd "C-M-S-<up>") 'decrease-frame-transparency)
|
|
;; (global-set-key [(control ?x) (control ?z)] 'iconify-everything)
|
|
;; (global-set-key [vertical-line S-down-mouse-1] 'iconify-everything)
|
|
;; (global-set-key [(control ?z)] 'iconify/show-frame)
|
|
;; (global-set-key [mode-line mouse-3] 'mouse-iconify/show-frame)
|
|
;; (global-set-key [mode-line C-mouse-3] 'mouse-remove-window)
|
|
;; (global-set-key [(control meta ?z)] 'show-hide)
|
|
;; (global-set-key [vertical-line C-down-mouse-1] 'show-hide)
|
|
;; (global-set-key [C-down-mouse-1] 'mouse-show-hide-mark-unmark)
|
|
;; (substitute-key-definition 'make-frame-command 'clone-frame global-map)
|
|
;; (substitute-key-definition 'delete-window 'remove-window global-map)
|
|
;; (define-key ctl-x-map "o" 'other-window-or-frame)
|
|
;; (define-key ctl-x-4-map "1" 'delete-other-frames)
|
|
;; (define-key ctl-x-5-map "h" 'show-*Help*-buffer)
|
|
;; (substitute-key-definition 'delete-window 'delete-windows-for global-map)
|
|
;; (define-key global-map "\C-xt." 'save-frame-config)
|
|
;; (define-key ctl-x-map "o" 'other-window-or-frame)
|
|
;;
|
|
;; (defalias 'doremi-prefix (make-sparse-keymap))
|
|
;; (defvar doremi-map (symbol-function 'doremi-prefix) "Keymap for Do Re Mi commands.")
|
|
;; (define-key global-map "\C-xt" 'doremi-prefix)
|
|
;; (define-key doremi-map "." 'save-frame-config)
|
|
;;
|
|
;; Customize the menu. Uncomment this to try it out.
|
|
;;
|
|
;; (defvar menu-bar-frames-menu (make-sparse-keymap "Frames"))
|
|
;; (define-key global-map [menu-bar frames]
|
|
;; (cons "Frames" menu-bar-frames-menu)))
|
|
;; (define-key menu-bar-frames-menu [set-all-params-from-frame]
|
|
;; '(menu-item "Set All Frame Parameters from Frame" set-all-frame-alist-parameters-from-frame
|
|
;; :help "Set frame parameters of a frame to their current values in frame"))
|
|
;; (define-key menu-bar-frames-menu [set-params-from-frame]
|
|
;; '(menu-item "Set Frame Parameter from Frame..." set-frame-alist-parameter-from-frame
|
|
;; :help "Set parameter of a frame alist to its current value in frame"))
|
|
;; (define-key menu-bar-frames-menu [separator-frame-1] '("--"))
|
|
;; (define-key menu-bar-frames-menu [tile-frames-vertically]
|
|
;; '(menu-item "Tile Frames Vertically..." tile-frames-vertically
|
|
;; :help "Tile all visible frames vertically"))
|
|
;; (define-key menu-bar-frames-menu [tile-frames-horizontally]
|
|
;; '(menu-item "Tile Frames Horizontally..." tile-frames-horizontally
|
|
;; :help "Tile all visible frames horizontally"))
|
|
;; (define-key menu-bar-frames-menu [separator-frame-2] '("--"))
|
|
;; (define-key menu-bar-frames-menu [toggle-max-frame-vertically]
|
|
;; '(menu-item "Toggle Max Frame Vertically" toggle-max-frame-vertically
|
|
;; :help "Maximize or restore the selected frame vertically"
|
|
;; :enable (frame-parameter nil 'restore-height)))
|
|
;; (define-key menu-bar-frames-menu [toggle-max-frame-horizontally]
|
|
;; '(menu-item "Toggle Max Frame Horizontally" toggle-max-frame-horizontally
|
|
;; :help "Maximize or restore the selected frame horizontally"
|
|
;; :enable (frame-parameter nil 'restore-width)))
|
|
;; (define-key menu-bar-frames-menu [toggle-max-frame]
|
|
;; '(menu-item "Toggle Max Frame" toggle-max-frame
|
|
;; :help "Maximize or restore the selected frame (in both directions)"
|
|
;; :enable (or (frame-parameter nil 'restore-width) (frame-parameter nil 'restore-height))))
|
|
;; (define-key menu-bar-frames-menu [maximize-frame-vertically]
|
|
;; '(menu-item "Maximize Frame Vertically" maximize-frame-vertically
|
|
;; :help "Maximize the selected frame vertically"))
|
|
;; (define-key menu-bar-frames-menu [maximize-frame-horizontally]
|
|
;; '(menu-item "Maximize Frame Horizontally" maximize-frame-horizontally
|
|
;; :help "Maximize the selected frame horizontally"))
|
|
;; (define-key menu-bar-frames-menu [maximize-frame]
|
|
;; '(menu-item "Maximize Frame" maximize-frame
|
|
;; :help "Maximize the selected frame (in both directions)"))
|
|
;; (define-key menu-bar-frames-menu [separator-frame-3] '("--"))
|
|
;; (define-key menu-bar-frames-menu [iconify-everything]
|
|
;; '(menu-item "Iconify All Frames" iconify-everything
|
|
;; :help "Iconify all frames of session at once"))
|
|
;; (define-key menu-bar-frames-menu [show-hide]
|
|
;; '(menu-item "Hide Frames / Show Buffers" show-hide
|
|
;; :help "Show, if only one frame visible; else hide.")))
|
|
;;
|
|
;; (defvar menu-bar-doremi-menu (make-sparse-keymap "Do Re Mi"))
|
|
;; (define-key global-map [menu-bar doremi]
|
|
;; (cons "Do Re Mi" menu-bar-doremi-menu))
|
|
;; (define-key menu-bar-doremi-menu [doremi-push-current-frame-config]
|
|
;; '("Save Frame Configuration" . save-frame-config))
|
|
;;
|
|
;; See also these files for other frame commands:
|
|
;;
|
|
;; `autofit-frame.el' - Automatically fit each frame to its
|
|
;; selected window. Uses `fit-frame.el'.
|
|
;;
|
|
;; `fit-frame.el' - 1) Fit a frame to its selected window.
|
|
;; 2) Incrementally resize a frame.
|
|
;;
|
|
;; `doremi-frm.el' - Incrementally adjust frame properties
|
|
;; using arrow keys and/or mouse wheel.
|
|
;;
|
|
;; `thumb-frm.el' - Shrink frames to a thumbnail size and
|
|
;; restore them again.
|
|
;;
|
|
;; `zoom-frm.el' - Zoom a frame or buffer, so that its text
|
|
;; appears larger or smaller.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;;; Change Log:
|
|
;;
|
|
;; 2020/09/13 dadams
|
|
;; Added: toggle-menu-bar-for-frame.
|
|
;; 2019/12/13 dadams
|
|
;; Added: frcmds-repeat-command, (enlarge|shrink)-window(-horizontally)-repeat,
|
|
;; enlarge/shrink-window(-repeat).
|
|
;; 2019/03/18 dadams
|
|
;; clone-frame: Use frame-geom-value-numeric.
|
|
;; 2019/03/03 dadams
|
|
;; Added: clone-frame-parameters.
|
|
;; clone-frame: Always select new frame. Augment current params with clone-frame-parameters.
|
|
;; 2019/03/02 dadama
|
|
;; clone-frame: Bind fit-frame-inhibit-fitting-flag to preserve current frame dimensions.
|
|
;; Return the new frame.
|
|
;; 2018/09/22 dadams
|
|
;; Moved to mouse+.el: tear-off-window(-if-not-alone).
|
|
;; 2018/09/21 dadams
|
|
;; tear-off-window: Use pop-to-buffer-same-window, not switch-to-buffer.
|
|
;; 2018/09/14 dadams
|
|
;; Added: clone-frame.
|
|
;; 2018/01/05 dadams
|
|
;; frcmds-available-screen-pixel-bounds:
|
|
;; Use display-monitor-attributes-list to compute, if option is nil.
|
|
;; frcmds-new-frame-position: Correct for use with multiple monitors.
|
|
;; 2018/01/02 dadams
|
|
;; available-screen-pixel-bounds: Change :type to allow nil.
|
|
;; 2017/10/22 dadams
|
|
;; remove-windows-on: Added optional arg ALL-FRAMES.
|
|
;; Just repeat get-buffer-window with ALL-FRAMES until no window.
|
|
;; delete/iconify-windows-on: Removed second arg to frames-on.
|
|
;; 2017/08/19 dadams
|
|
;; delete-window: Use with-selected-window for Emacs 22+.
|
|
;; Updated Emacs-Wiki URLs.
|
|
;; 2017/05/06 dadams
|
|
;; maximize-frame: Sidestep nil frame parameters.
|
|
;; 2017/02/07 dadams
|
|
;; Added decrease-frame-transparency, increase-frame-transparency. Suggest bind to C-M-up|down.
|
|
;; 2016/01/24 dadams
|
|
;; Added: tear-off-window, tear-off-window-if-not-alone.
|
|
;; 2015/08/14 dadams
|
|
;; tell-customize-var-has-changed: Use symbol-value, not eval.
|
|
;; 2014/12/09 dadams
|
|
;; Added: frcmds-frame-pixel-height.
|
|
;; frcmds-split-frame-1: Use frame-pixel-width and frcmds-frame-pixel-height, instead of working
|
|
;; with width and height frame parameters (char-based).
|
|
;; frcmds-tile-frames:
|
|
;; If Emacs 24.4+, use PIXELWISE arg with set-frame-size.
|
|
;; Otherwise: * Always subtract frcmds-extra-pixels-width.
|
|
;; * Do not subtract borders.
|
|
;; * Increment origin by one border-width.
|
|
;; 2014/12/07 dadams
|
|
;; Added: split-frame-horizontally, split-frame-vertically.
|
|
;; frcmds-tile-frames: Added optional args, so can tile within a rectangle.
|
|
;; create-frame-tiled-(horizontally|vertically): Keep same font size.
|
|
;; 2014/12/06 dadams
|
|
;; Added: create-frame-tiled-horizontally, create-frame-tiled-vertically.
|
|
;; Added aliases: tile-frames-side-by-side, tile-frames-top-to-bottom.
|
|
;; window-mgr-title-bar-pixel-height: Changed default value for ns to 50. Thx to Nate Eagleson.
|
|
;; 2014/10/15 dadams
|
|
;; window-mgr-title-bar-pixel-height: Added default value for ns (Next). Thx to Nate Eagleson.
|
|
;; 2014/10/13 dadams
|
|
;; Removed extra, empty Package-Requires.
|
|
;; 2014/07/21 dadams
|
|
;; Do not redefine delete-window - just advise it.
|
|
;; delete/iconify-window: Just use delete-window, not old-delete-window.
|
|
;; 2014/04/19 dadams
|
|
;; Added: frcmds-frame-number, name-all-frames-numerically, name-frame-numerically.
|
|
;; Renamed: available-screen-pixel-* to frcmds-available-screen-pixel-*,
|
|
;; enlarged-font-name to frcmds-enlarged-font-name,
|
|
;; extra-pixels-* to frcmds-extra-pixels-*,
|
|
;; frame-alist-var-names to frcmds-frame-alist-var-names,
|
|
;; frame-parameter-names to frcmds-frame-parameter-names,
|
|
;; frame-iconified-p to frcmds-frame-iconified-p,
|
|
;; new-frame-position to frcmds-new-frame-position,
|
|
;; read-args-for-tile-frames to frcmds-read-args-for-tiling,
|
|
;; read-buffer-for-delete-windows to frcmds-read-buffer-for-delete-windows,
|
|
;; frame-cmds-set-difference to frcmds-set-difference,
|
|
;; smart-tool-bar-pixel-height to frcmds-smart-tool-bar-pixel-height,
|
|
;; tile-frames to frcmds-tile-frames.
|
|
;; rename-non-minibuffer-frame: Pass OLD-NAME and NEW-NAME to rename-frame.
|
|
;; Group Frame-Commands: Added :prefix frcmds-.
|
|
;;
|
|
;; 2014/02/24 dadams
|
|
;; rename-frame, rename-non-minibuffer-frame: Fixed default buffer name for non-interactive.
|
|
;; 2013/09/21 dadams
|
|
;; maximize-frame: Apply frame-geom-value-numeric to new-* also. Bug report thx: Mike Fitzgerald.
|
|
;; 2013/07/21 dadams
|
|
;; Added Package-Requires to header, at least temporarily, but should not need to specify version.
|
|
;; 2013/07/12 dadams
|
|
;; set-frame-alist-parameter-from-frame: Use lax completion, so do not limit to known parameters.
|
|
;; frame-parameter-names: Updated for Emacs 24.
|
|
;; 2013/07/05 dadams
|
|
;; Added: move-frame-to-screen-top-left.
|
|
;; move-frame-to-screen-*: Read FRAME name in interactive spec.
|
|
;; 2013/07/04 dadams
|
|
;; show-hide-show-function: Use function-item instead of const for jump-to-frame-config-register.
|
|
;; 2013/05/15 dadams
|
|
;; Added error symbols font-too-small and font-size.
|
|
;; enlarged-font-name: Signal font-too-small error.
|
|
;; 2013/04/29 dadams
|
|
;; Added: deiconify-everything, (mouse-)iconify/show-frame (renamed (mouse-)iconify/map-frame).
|
|
;; iconify/show-frame: Negative prefix arg now deiconifies all.
|
|
;; 2013/03/12 dadams
|
|
;; maximize-frame: Corrected new-left, new-top.
|
|
;; Corrected arg to modify-frame-parameters - use frame-geom-value-numeric
|
|
;; Do not alias if function name is already fboundp.
|
|
;; toggle-max-frame-*: Use toggle-max-frame, not restore-frame (the alias).
|
|
;; toggle-max-frame: If no restore-* parameter then first maximize.
|
|
;; Condition last four parameters on orig-*, not restore-*.
|
|
;; 2013/02/06 dadams
|
|
;; move-frame-(up|down|left|right): Set N to 1 if nil.
|
|
;; 2013/01/17 dadams
|
|
;; Added: move-frame-to-screen-(top|bottom|left|right).
|
|
;; move-frame-(up|down|left|right): Redefined so prefix arg moves increments of char size.
|
|
;; 2012/02/29 dadams
|
|
;; Added, for Emacs 20 only: nbutlast, butlast. To avoid runtime load of cl.el.
|
|
;; Added frame-cmds-set-difference, to avoid runtime load of cl.el.
|
|
;; set-all-frame-alist-parameters-from-frame: Use frame-cmds-set-difference.
|
|
;; 2011/07/25 dadams
|
|
;; save-frame-config: Use fboundp, not featurep.
|
|
;; 2011/01/04 dadams
|
|
;; Removed autoload cookie from non-interactive function.
|
|
;; 2010/10/19 dadams
|
|
;; enlarge-font: Only do frame-update-faces if Emacs 20 (obsolete in 21).
|
|
;; 2010/06/04 dadams
|
|
;; Added: (toggle-max|restore)-frame(-horizontally|-vertically). Thx to Uday Reddy for suggestion.
|
|
;; Renamed max-frame to maximize-frame.
|
|
;; maximize-frame: Save original location & position params for later restoration.
|
|
;; 2010/05/25 dadams
|
|
;; Added: max-frame, maximize-frame-horizontally, maximize-frame-vertically.
|
|
;; 2009/10/02 dadams
|
|
;; delete-windows-on: Return nil. Make BUFFER optional: default is current buffer.
|
|
;; 2009/08/03 dadams
|
|
;; delete-window: Wrap with save-current-buffer. Thx to Larry Denenberg.
|
|
;; 2009/05/17 dadams
|
|
;; Updated to reflect thumb-frm.el name changes.
|
|
;; 2009/01/30 dadams
|
|
;; enlarge-font, enlarged-font-name, enlarge-font-tries:
|
|
;; Removed temporary workaround - Emacs 23 bug #119 was finally fixed.
|
|
;; 2009/01/01 dadams
|
|
;; Removed compile-time require of doremi-frm.el to avoid infinite recursion.
|
|
;; 2008/12/13 dadams
|
|
;; enlarge-font: Redefined for Emacs 23 - just use :height face attribute.
|
|
;; enlarge-font-tries, enlarged-font-name: Not used for Emacs 23.
|
|
;; 2008/10/31 dadams
|
|
;; Updated frame-parameter-names for Emacs 23.
|
|
;; 2008/07/29 dadams
|
|
;; Option available-screen-pixel-bounds: Use nil as default value.
|
|
;; available-screen-pixel-bounds: Redefined as the code that defined the option's default value.
|
|
;; Added: effective-screen-pixel-bounds - code taken from old available-screen-pixel-bounds,
|
|
;; but also convert frame geom value to numeric.
|
|
;; Everywhere:
|
|
;; Use effective-screen-pixel-bounds in place of available-screen-pixel-bounds function.
|
|
;; Use available-screen-pixel-bounds function instead of option.
|
|
;; available-screen-pixel-(width|height): Added optional INCLUDE-MINI-P arg.
|
|
;; new-frame-position: Call available-screen-pixel-(width|height) with arg.
|
|
;; save-frame-config: push-current-frame-config -> doremi-push-current-frame-config.
|
|
;; Soft-require doremi-frm.el when byte-compile.
|
|
;; 2008/06/02 dadams
|
|
;; Added: available-screen-pixel-bounds (option and function).
|
|
;; tile-frames, available-screen-pixel-(width|height):
|
|
;; Redefined to use available-screen-pixel-bounds. Thx to Nathaniel Cunningham for input.
|
|
;; 2008/05/29 dadams
|
|
;; Fixes for macOS by Nathaniel Cunningham and David Reitter:
|
|
;; window-mgr-title-bar-pixel-height, tile-frames, smart-tool-bar-pixel-height (added).
|
|
;; 2007/12/27 dadams
|
|
;; tile-frames: Restored border calculation, but using only external border.
|
|
;; Renamed window-mgr-*-width to window-mgr-*-height and changed default value from 32 to 27.
|
|
;; 2007/12/20 dadams
|
|
;; Added: frame-extra-pixels(width|height). Use in tile-frames. Thx to David Reitter.
|
|
;; frame-horizontal-extra-pixels: Changed default value from 30 to 32.
|
|
;; 2007/10/11 dadams
|
|
;; Added: assq-delete-all (for Emacs 20).
|
|
;; 2007/09/02 dadams
|
|
;; Added: available-screen-pixel-(width|height). Use in tile-frames, new-frame-position.
|
|
;; 2007/06/12 dadams
|
|
;; tile-frames: Corrected use of fboundp for thumbnail-frame-p.
|
|
;; 2007/05/27 dadams
|
|
;; enlarged-font-name:
|
|
;; Do nothing if null assq of ascii. Not sure what this means, but gets around Emacs 23 bug.
|
|
;; 2006/08/22 dadams
|
|
;; Added: delete-windows-for, read-buffer-for-delete-windows.
|
|
;; delete-windows-on: Use read-buffer-for-delete-windows.
|
|
;; Removed old-delete-windows-on (not used).
|
|
;; 2006/05/30 dadams
|
|
;; delete-windows-on: Return nil if buffer arg is nil. Thanks to Slawomir Nowaczyk.
|
|
;; 2006/01/07 dadams
|
|
;; Added :link for sending bug report.
|
|
;; 2006/01/06 dadams
|
|
;; Renamed group. Added :link.
|
|
;; 2006/01/04 dadams
|
|
;; Added: other-window-or-frame.
|
|
;; 2005/12/29 dadams
|
|
;; mouse-show-hide-mark-unmark: dired-mouse-mark/unmark -> diredp-mouse-mark/unmark.
|
|
;; 2005/12/13 dadams
|
|
;; Added: delete-other-frames.
|
|
;; 2005/11/18 dadams
|
|
;; enlarge-font: Try to increment or decrment further, testing for an existing font.
|
|
;; Added: enlarge-font-tries, enlarged-font-name.
|
|
;; 2005/10/03 dadams
|
|
;; Removed require of icomplete+.el (no longer redefines read-from-minibuffer).
|
|
;; 2005/07/03 dadams
|
|
;; Renamed: args-for-tile-frames to read-args-for-tile-frames.
|
|
;; 2005/06/19 dadams
|
|
;; tile-frames: Don't tile thumbnail frames.
|
|
;; 2005/05/29 dadams
|
|
;; Moved here from frame+.el and fit-frame.el: enlarge-frame*, shrink-frame*.
|
|
;; Added: move-frame-up|down|left|right, move-frame-wrap-within-display-flag,
|
|
;; new-frame-position.
|
|
;; 2005/05/28 dadams
|
|
;; show-a-frame-on: Use another-buffer as default for read-buffer, if available.
|
|
;; 2005/05/15 dadams
|
|
;; Renamed: minibuffer-frame to 1on1-minibuffer-frame.
|
|
;; 2005/05/10 dadams
|
|
;; remove-window: Removed definition; just defalias it to delete-window.
|
|
;; delete-window: (one-window-p) -> (one-window-p t).
|
|
;; set-frame-alist-parameter-from-frame: No longer use destructive fns.
|
|
;; 2005/01/19 dadams
|
|
;; set-all-frame-alist-parameters-from-frame:
|
|
;; Added really-all-p and use frame-parameters-to-exclude.
|
|
;; Added: frame-parameters-to-exclude, tell-customize-var-has-changed.
|
|
;; 2005/01/18 dadams
|
|
;; Added: set-all-frame-alist-parameters-from-frame, set-frame-alist-parameter-from-frame,
|
|
;; frame-alist-var-names, frame-parameter-names.
|
|
;; Added Note on saving changes.
|
|
;; 2005/01/08 dadams
|
|
;; Moved enlarge-font here from doremi-frm.el, where it was called doremi-grow-font.
|
|
;; 2005/01/04 dadams
|
|
;; Added rename-frame-when-iconify-flag.
|
|
;; Use it in iconify-everything, (mouse-)iconify/map-frame.
|
|
;; Added (defgroup frame-cmds).
|
|
;; 2004/12/23 dadams
|
|
;; frame-config-register, show-hide-show-function, window-mgr-title-bar-pixel-width:
|
|
;; Changed defvar to defcustom.
|
|
;; 2004/12/21 dadams
|
|
;; hide-everything, iconify-everything: bind thumbify-instead-of-iconify-flag to nil.
|
|
;; 2004/12/10 dadams
|
|
;; tile-frames: Change 15 to (frame-char-height fr) for scroll-bar-width.
|
|
;; tile-frames-*: Corrected doc strings for non-interactive case.
|
|
;; 2004/12/09 dadams
|
|
;; Changed compile-time require of strings to a soft require.
|
|
;; 2004/10/11 dadams
|
|
;; args-for-tile-frames: Fixed bug when non-existent frame in name history.
|
|
;; tile-frames: show-frame at end (for case where use prefix arg)
|
|
;; 2004/09/11 dadams
|
|
;; Moved to doremi-frm.el: frame-config-ring*, frame-config-wo-parameters,
|
|
;; push-frame-config.
|
|
;; 2004/09/07 dadams
|
|
;; Added: jump-to-frame-config-register, push-frame-config, save-frame-config.
|
|
;; 2004/09/01 dadams
|
|
;; Added: frame-config-register, show-hide-show-function,
|
|
;; jump-to-frame-config-register.
|
|
;; Rewrote to record frame config: iconify-everything, hide-everything.
|
|
;; Rewrote to use show-hide-show-function: show-hide.
|
|
;; 2004/03/22 dadams
|
|
;; Added: tile-frames, tile-frames-vertically, args-for-tile-frames.
|
|
;; Rewrote tile-frames-horizontally to use tile-frames.
|
|
;; 2004/03/19 dadams
|
|
;; Added tile-frames-horizontally.
|
|
;; 2000/11/27 dadams
|
|
;; hide-frame: fixed bug: Added get-a-frame for frame name read.
|
|
;; 2000/09/27 dadams
|
|
;; 1. Added: frame-iconified-p.
|
|
;; 2. remove-window: only make-frame-invisible if not iconified (HACK).
|
|
;; 1999/10/05 dadams
|
|
;; rename-frame: fixed bug if only 1 frame and old-name was a frame.
|
|
;; 1999/08/25 dadams
|
|
;; Added: hide-everything, show-buffer-menu, show-hide.
|
|
;; 1999/03/17 dadams
|
|
;; delete-1-window-frames-on: ensure a buffer object (not a name).
|
|
;; 1996/04/26 dadams
|
|
;; delete/iconify-windows-on, show-a-frame-on: Do nothing if null buffer.
|
|
;; 1996/03/12 dadams
|
|
;; delete/iconify-window: Unless one-window-p, do old-delete-window outside of
|
|
;; save-window-excursion.
|
|
;; 1996/03/08 dadams
|
|
;; 1. delete-windows-on: a. Fixed incorrect interactive spec (bad paren).
|
|
;; b. Second arg FRAME also provided interactively now.
|
|
;; 2. Added: delete/iconify-window, delete/iconify-windows-on.
|
|
;; 1996/02/27 dadams
|
|
;; show-frame: Call make-frame-visible.
|
|
;; 1996/02/09 dadams
|
|
;; Added show-*Help*-buffer.
|
|
;; 1996/01/30 dadams
|
|
;; 1. show-frame: Don't make-frame-visible. Done by raise-frame anyway.
|
|
;; 2. Added show-a-frame-on.
|
|
;; 1996/01/09 dadams
|
|
;; Added delete-windows-on and made it interactive.
|
|
;; 1996/01/08 dadams
|
|
;; Added rename-non-minibuffer-frame. Use in iconify-everything,
|
|
;; iconify/map-frame, mouse-iconify/map-frame.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;; This program 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 2, or (at your option)
|
|
;; any later version.
|
|
|
|
;; This program 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 this program; see the file COPYING. If not, write to
|
|
;; the Free Software Foundation, Inc., 51 Franklin Street, Fifth
|
|
;; Floor, Boston, MA 02110-1301, USA.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;;; Code:
|
|
|
|
(eval-when-compile (require 'cl)) ;; case, incf (plus, for Emacs 20: dolist, dotimes)
|
|
(require 'frame-fns) ;; frame-geom-value-cons, frame-geom-value-numeric, frames-on, get-frame-name,
|
|
;; get-a-frame, read-frame
|
|
(require 'strings nil t) ;; (no error if not found) read-buffer
|
|
(require 'misc-fns nil t) ;; (no error if not found) another-buffer
|
|
|
|
;; Don't require even to byte-compile, because doremi-frm.el soft-requires frame-cmds.el
|
|
;; (eval-when-compile (require 'doremi-frm nil t)) ;; (no error if not found)
|
|
;; ;; doremi-push-current-frame-config
|
|
|
|
;; Not required here, because this library requires `frame-cmds.el': `thumb-frm.el'.
|
|
;; However, `frame-cmds.el' soft-uses `thumfr-thumbnail-frame-p', which is defined
|
|
;; in `thumb-frm.el'.
|
|
|
|
;; Quiet byte-compiler.
|
|
(defvar 1on1-minibuffer-frame) ; In `oneonone.el'
|
|
(defvar frame-alpha-lower-limit) ; Emacs 23+
|
|
(defvar mac-tool-bar-display-mode)
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
|
|
;;; USER OPTIONS (VARIABLES) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(defgroup Frame-Commands nil
|
|
"Miscellaneous frame and window commands."
|
|
:group 'frames
|
|
:prefix "frcmds-"
|
|
:link `(url-link :tag "Send Bug Report"
|
|
,(concat "mailto:" "drew.adams" "@" "oracle" ".com?subject=\
|
|
frame-cmds.el bug: \
|
|
&body=Describe bug here, starting with `emacs -q'. \
|
|
Don't forget to mention your Emacs and library versions."))
|
|
:link '(url-link :tag "Other Libraries by Drew"
|
|
"https://www.emacswiki.org/emacs/DrewsElispLibraries")
|
|
:link '(url-link :tag "Download"
|
|
"https://www.emacswiki.org/emacs/download/frame-cmds.el")
|
|
:link '(url-link :tag "Description - `delete-window'"
|
|
"https://www.emacswiki.org/emacs/FrameModes")
|
|
:link '(url-link :tag "Description - Frame Renaming"
|
|
"https://www.emacswiki.org/emacs/FrameTitle")
|
|
:link '(url-link :tag "Description - Frame Resizing"
|
|
"https://www.emacswiki.org/emacs/Shrink-Wrapping_Frames")
|
|
:link '(url-link :tag "Description - Frame Customization"
|
|
"https://www.emacswiki.org/emacs/CustomizingAndSaving")
|
|
:link '(url-link :tag "Description - Frame Tiling"
|
|
"https://www.emacswiki.org/emacs/Frame_Tiling_Commands")
|
|
:link '(url-link :tag "Description - General"
|
|
"https://www.emacswiki.org/emacs/FrameModes")
|
|
:link '(emacs-commentary-link :tag "Commentary" "frame-cmds"))
|
|
|
|
(defcustom clone-frame-parameters (cons 30 30)
|
|
"Frame parameter settings that override those of the frame to clone.
|
|
The value can be an alist of frame parameters or a cons of two
|
|
integers, (LEFT-OFFSET . TOP-OFFSET).
|
|
|
|
The latter case sets parameters `left' and `top' of the new frame to
|
|
the `left' and `top' of the selected frame, offset by adding
|
|
LEFT-OFFSET and TOP-OFFSET to them, respectively."
|
|
:type '(choice
|
|
(cons :tag "Offset from current frame location"
|
|
(integer :tag "Left")
|
|
(integer :tag "Top"))
|
|
(alist :tag "Parameters to augment/replace those of current frame"
|
|
:key-type (symbol :tag "Parameter")))
|
|
:group 'Frame-Commands)
|
|
|
|
(defcustom rename-frame-when-iconify-flag t
|
|
"*Non-nil means frames are renamed when iconified.
|
|
The new name is the name of the current buffer."
|
|
:type 'boolean :group 'Frame-Commands)
|
|
|
|
(defcustom frame-config-register ?\C-l ; Control-L is the name of the register.
|
|
"*Character naming register for saving/restoring frame configuration."
|
|
:type 'character :group 'Frame-Commands)
|
|
|
|
(defcustom show-hide-show-function 'jump-to-frame-config-register
|
|
"*Function to show stuff that is hidden or iconified by `show-hide'.
|
|
Candidates include `jump-to-frame-config-register' and `show-buffer-menu'."
|
|
:type '(choice (function-item :tag "Restore frame configuration" jump-to-frame-config-register)
|
|
(function :tag "Another function"))
|
|
:group 'Frame-Commands)
|
|
|
|
;; Use `cond', not `case', for Emacs 20 byte-compiler.
|
|
(defcustom window-mgr-title-bar-pixel-height (cond ((eq window-system 'mac) 22)
|
|
;; For older versions of macOS, 40 might be better.
|
|
((eq window-system 'ns) 50)
|
|
(t 27))
|
|
"*Height of frame title bar provided by the window manager, in pixels.
|
|
You might alternatively call this constant the title-bar \"width\" or
|
|
\"thickness\". There is no way for Emacs to determine this, so you
|
|
must set it."
|
|
:type 'integer :group 'Frame-Commands)
|
|
|
|
(defcustom enlarge-font-tries 100
|
|
"*Number of times to try to change font-size, when looking for a font.
|
|
The font-size portion of a font name is incremented or decremented at
|
|
most this many times, before giving up and raising an error."
|
|
:type 'integer :group 'Frame-Commands)
|
|
|
|
(defcustom frame-parameters-to-exclude '((window-id) (buffer-list) (name) (title) (icon-name))
|
|
"*Parameters to exclude in `set-all-frame-alist-parameters-from-frame'.
|
|
An alist of the same form as that returned by `frame-parameters'.
|
|
The cdr of each alist element is ignored.
|
|
These frame parameters are not copied to the target alist."
|
|
:type '(repeat (cons symbol sexp)) :group 'Frame-Commands)
|
|
|
|
(defcustom move-frame-wrap-within-display-flag t
|
|
"*Non-nil means wrap frame movements within the display.
|
|
Commands `move-frame-up', `move-frame-down', `move-frame-left', and
|
|
`move-frame-right' then move the frame back onto the display when it
|
|
moves off of it.
|
|
If nil, you can move the frame as far off the display as you like."
|
|
:type 'boolean :group 'Frame-Commands)
|
|
|
|
(defcustom available-screen-pixel-bounds nil
|
|
"*Upper left and lower right of available screen space for tiling frames.
|
|
Integer list: (x0 y0 x1 y1), where (x0, y0) is the upper left position
|
|
and (x1, y1) is the lower right position. Coordinates are in pixels,
|
|
measured from the screen absolute origin, (0, 0), at the upper left.
|
|
|
|
If this is nil, then the available space is calculated. That should
|
|
give good results in most cases."
|
|
:type '(choice
|
|
(const :tag "Calculate automatically" nil)
|
|
(list :tag "List of (x0 y0 x1 y1)"
|
|
(integer :tag "X0 (upper left) - pixels from screen left")
|
|
(integer :tag "Y0 (upper left) - pixels from screen top")
|
|
(integer :tag "X1 (lower right) - pixels from screen left" )
|
|
(integer :tag "Y1 (lower right) - pixels from screen top")))
|
|
:group 'Frame-Commands)
|
|
|
|
|
|
|
|
;;; FUNCTIONS ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
;;;###autoload
|
|
(defun save-frame-config ()
|
|
"Save current frame configuration.
|
|
You can restore it with \\[jump-to-frame-config-register]."
|
|
(interactive)
|
|
(frame-configuration-to-register frame-config-register)
|
|
(when (fboundp 'doremi-push-current-frame-config) ; In `doremi-frm.el'.
|
|
(doremi-push-current-frame-config))
|
|
(message
|
|
(substitute-command-keys
|
|
(if (fboundp 'doremi-frame-configs) ; In `doremi-frm.el'.
|
|
(format "Use `\\[jump-to-frame-config-register]' (`C-x r j %c') or \
|
|
`\\[doremi-frame-configs]' to restore frames as before (undo)." frame-config-register)
|
|
"Use `\\[jump-to-frame-config-register]' to restore frames as before (undo)."))))
|
|
|
|
;;;###autoload
|
|
(defun jump-to-frame-config-register ()
|
|
"Restore frame configuration saved in `frame-config-register'."
|
|
(interactive)
|
|
(jump-to-register frame-config-register))
|
|
|
|
;;;###autoload
|
|
(defun deiconify-everything ()
|
|
"Deiconify any iconified frames."
|
|
(interactive)
|
|
(frame-configuration-to-register frame-config-register)
|
|
(dolist (frame (frame-list))
|
|
(when (eq 'icon (frame-visible-p frame)) (make-frame-visible frame))))
|
|
|
|
;;;###autoload
|
|
(defun iconify-everything ()
|
|
"Iconify all frames of session at once.
|
|
Remembers frame configuration in register `C-l' (Control-L).
|
|
To restore this frame configuration, use `\\[jump-to-register] C-l'."
|
|
(interactive)
|
|
(frame-configuration-to-register frame-config-register)
|
|
(let ((thumfr-thumbify-dont-iconify-flag nil)) ; Defined in `thumb-frm.el'.
|
|
(dolist (frame (visible-frame-list))
|
|
(when rename-frame-when-iconify-flag (rename-non-minibuffer-frame frame))
|
|
(iconify-frame frame))))
|
|
|
|
;;;###autoload
|
|
(defun hide-everything ()
|
|
"Hide all frames of session at once.
|
|
Iconify minibuffer frame; make all others invisible.
|
|
Remembers frame configuration in register `C-l' (Control-L).
|
|
To restore this frame configuration, use `\\[jump-to-register] C-l'."
|
|
(interactive)
|
|
(frame-configuration-to-register frame-config-register)
|
|
(let ((minibuf-frame-name (and (boundp '1on1-minibuffer-frame)
|
|
(cdr (assq 'name (frame-parameters
|
|
1on1-minibuffer-frame)))))
|
|
(thumfr-thumbify-dont-iconify-flag nil)) ; Defined in `thumb-frm.el'.
|
|
(dolist (frame (frame-list))
|
|
(if (eq minibuf-frame-name (cdr (assq 'name (frame-parameters frame))))
|
|
(iconify-frame frame) ; minibuffer frame
|
|
(make-frame-invisible frame t))))) ; other frames
|
|
|
|
;;;###autoload
|
|
(defun show-hide ()
|
|
"1 frame visible: `show-hide-show-function'; else: `hide-everything'.
|
|
This acts as a toggle between showing all frames and showing only an
|
|
iconified minibuffer frame."
|
|
(interactive)
|
|
(if (< (length (visible-frame-list)) 2) (funcall show-hide-show-function) (hide-everything)))
|
|
|
|
;;;###autoload
|
|
(defun show-buffer-menu ()
|
|
"Call `buffer-menu' after making all frames visible.
|
|
Useful after using `hide-everything' because of a Windows bug that
|
|
doesn't let you display frames that have been made visible after
|
|
being made invisible."
|
|
(interactive)
|
|
(let ((minibuf-frame-name (and (boundp '1on1-minibuffer-frame)
|
|
(cdr (assq 'name (frame-parameters 1on1-minibuffer-frame))))))
|
|
(dolist (frame (frame-list))
|
|
(if (eq minibuf-frame-name (cdr (assq 'name (frame-parameters frame))))
|
|
(make-frame-visible frame) ; minibuffer frame
|
|
(iconify-frame frame))) ; other frames
|
|
(buffer-menu)))
|
|
|
|
;;;###autoload
|
|
(defun mouse-show-hide-mark-unmark (event)
|
|
"In minibuffer: `show-hide'. In dired: mark/unmark; else: buffer menu."
|
|
(interactive "e")
|
|
(if (window-minibuffer-p (posn-window (event-start event)))
|
|
(show-hide)
|
|
(or (and (memq major-mode '(dired-mode vc-dired-mode))
|
|
(fboundp 'diredp-mouse-mark/unmark)
|
|
(diredp-mouse-mark/unmark event)) ; Return nil if not on a file or dir.
|
|
(mouse-buffer-menu event))))
|
|
|
|
;;;###autoload
|
|
(defalias 'iconify/map-frame 'iconify/show-frame) ; `.../map...' is the old name.
|
|
;;;###autoload
|
|
(defun iconify/show-frame (&optional all-action)
|
|
"Iconify selected frame if now shown. Show it if now iconified.
|
|
A non-negative prefix arg iconifies all shown frames.
|
|
A negative prefix arg deiconifies all iconified frames."
|
|
(interactive "P")
|
|
(cond ((not all-action)
|
|
(when rename-frame-when-iconify-flag (rename-non-minibuffer-frame))
|
|
(iconify-or-deiconify-frame))
|
|
((natnump (prefix-numeric-value all-action))
|
|
(iconify-everything))
|
|
(t
|
|
(deiconify-everything))))
|
|
|
|
;;;###autoload
|
|
(defalias 'mouse-iconify/map-frame 'mouse-iconify/show-frame) ; `.../map...' is the old name.
|
|
;;;###autoload
|
|
(defun mouse-iconify/show-frame (event)
|
|
"Iconify frame you click, if now shown. Show it if now iconified."
|
|
(interactive "e")
|
|
(select-window (posn-window (event-start event)))
|
|
(when rename-frame-when-iconify-flag (rename-non-minibuffer-frame))
|
|
(iconify-or-deiconify-frame))
|
|
|
|
;;;###autoload
|
|
(defun toggle-menu-bar-for-frame (&optional frame)
|
|
"Toggle showing the menu-bar for selected frame.
|
|
When called from Lisp, optional arg FRAME is used instead of the
|
|
selected frame."
|
|
(interactive)
|
|
(let ((now (or (frame-parameter frame 'menu-bar-lines) 0)))
|
|
(set-frame-parameter frame 'menu-bar-lines (if (zerop now) 1 0))))
|
|
|
|
|
|
|
|
;; ADVISE ORIGINAL (built-in):
|
|
;;
|
|
;; If WINDOW is the only one in its frame, `delete-frame'.
|
|
(defadvice delete-window (around delete-frame-if-one-win activate)
|
|
"If WINDOW is the only one in its frame, then `delete-frame' too."
|
|
(if (fboundp 'with-selected-window) ; Emacs 22+
|
|
(with-selected-window
|
|
(or (ad-get-arg 0) (selected-window))
|
|
(if (one-window-p t) (delete-frame) ad-do-it))
|
|
(save-current-buffer
|
|
(select-window (or (ad-get-arg 0) (selected-window)))
|
|
(if (one-window-p t) (delete-frame) ad-do-it))))
|
|
|
|
;;;###autoload
|
|
(defun delete-windows-for (&optional buffer)
|
|
"`delete-window' or prompt for buffer and delete its windows.
|
|
With no prefix arg, delete the selected window.
|
|
With a prefix arg, prompt for a buffer and delete all windows, on any
|
|
frame, that show that buffer."
|
|
(interactive (list (and current-prefix-arg (frcmds-read-buffer-for-delete-windows))))
|
|
(if buffer (delete-windows-on buffer) (delete-window)))
|
|
|
|
|
|
|
|
;; REPLACES ORIGINAL in `window.el' (built-in prior to Emacs 24.5):
|
|
;;
|
|
;; 1) Use `read-buffer' in interactive spec.
|
|
;; 2) Do not raise an error if BUFFER is a string that does not name a buffer.
|
|
;; 3) Call `delete-window', so if you use the advised `delete-window' here then this also deletes
|
|
;; frames where window showing the BUFFER is the only window.
|
|
;;
|
|
;;;###autoload
|
|
(defun delete-windows-on (&optional buffer frame)
|
|
"Delete windows showing BUFFER.
|
|
Optional arg BUFFER defaults to the current buffer.
|
|
|
|
Optional second arg FRAME controls which frames are considered.
|
|
If nil or omitted, delete all windows showing BUFFER in any frame.
|
|
If t, delete only windows showing BUFFER in the selected frame.
|
|
If `visible', delete all windows showing BUFFER in any visible frame.
|
|
If a frame, delete only windows showing BUFFER in that frame.
|
|
|
|
Interactively, FRAME depends on the prefix arg, as follows:
|
|
Without a prefix arg (prefix = nil), FRAME is nil (all frames).
|
|
With prefix arg >= 0, FRAME is t (this frame only).
|
|
With prefix arg < 0, FRAME is `visible' (all visible frames)."
|
|
(interactive
|
|
(list (frcmds-read-buffer-for-delete-windows)
|
|
(and current-prefix-arg
|
|
(or (natnump (prefix-numeric-value current-prefix-arg)) 'visible))))
|
|
(unless buffer (setq buffer (current-buffer))) ; Like Emacs 23+ - unlike Emacs 21-22.
|
|
|
|
;; `get-buffer-window' interprets FRAME oppositely for t and nil, so switch.
|
|
(setq frame (if (eq t frame) nil (if (eq nil frame) t frame)))
|
|
(let (win)
|
|
;; Vanilla Emacs version raises an error if BUFFER is a string that does not name a buffer.
|
|
;; We do not raise an error - we do nothing.
|
|
(and (get-buffer buffer)
|
|
(while (setq win (get-buffer-window buffer frame)) (delete-window win))
|
|
nil))) ; Return nil always, like vanilla Emacs.
|
|
|
|
(defun frcmds-read-buffer-for-delete-windows ()
|
|
"Read buffer name for delete-windows commands.
|
|
Only displayed buffers are completion candidates."
|
|
(completing-read "Delete windows on buffer: "
|
|
(let ((all-bufs (buffer-list))
|
|
(cand-bufs ()))
|
|
(dolist (buf all-bufs)
|
|
(when (get-buffer-window buf t)
|
|
(push (list (buffer-name buf)) cand-bufs)))
|
|
cand-bufs)
|
|
nil t nil 'minibuffer-history (buffer-name (current-buffer)) t))
|
|
|
|
(defsubst frcmds-frame-iconified-p (frame)
|
|
"Return non-nil if FRAME is `frame-live-p' and `frame-visible-p'."
|
|
(and (frame-live-p frame) (eq (frame-visible-p frame) 'icon)))
|
|
|
|
;; (defun remove-window (&optional window)
|
|
;; "Remove WINDOW from the display. Default is `selected-window'.
|
|
;; If WINDOW is the only one in its frame, then:
|
|
;; If WINDOW is dedicated to its buffer, then make its frame invisible.
|
|
;; Otherwise, delete its frame (as well as the window)."
|
|
;; (interactive)
|
|
;; (setq window (or window (selected-window)))
|
|
;; (select-window window)
|
|
;; (if (and (window-dedicated-p (selected-window))
|
|
;; (one-window-p t))
|
|
;; (let ((fr (selected-frame)))
|
|
;; ;; HACK because of Emacs bug: `raise-frame' won't raise a frame
|
|
;; ;; that was first iconified and then made invisible.
|
|
;; ;; So, here we don't make an iconified frame invisible.
|
|
;; (unless (frcmds-frame-iconified-p fr)
|
|
;; (make-frame-invisible fr)))
|
|
;; (delete-window)))
|
|
|
|
;; REMOVED old definition, above, because of problems with invisible
|
|
;; *Completions* frame when use completion window with subsequent args
|
|
;; to a command. Just use `delete-window' now, which deletes frame if
|
|
;; `one-window-p'. Use a `defalias' because its easier than replacing
|
|
;; all my calls to `remove-window' with `delete-window'.
|
|
;;
|
|
;;;###autoload
|
|
(defalias 'remove-window 'delete-window)
|
|
|
|
;;;###autoload
|
|
(defun remove-windows-on (buffer &optional all-frames)
|
|
"Remove all windows showing BUFFER.
|
|
This calls `remove-window' on each window showing BUFFER.
|
|
|
|
When called from Lisp, optional arg ALL-FRAMES controls which frames
|
|
are considered. See `get-buffer-window' for its interpretation."
|
|
(interactive
|
|
(list (read-buffer "Remove all windows showing buffer: " (current-buffer) 'existing)
|
|
t))
|
|
(setq buffer (get-buffer buffer)) ; Convert to buffer.
|
|
(when buffer ; Do nothing if null BUFFER.
|
|
(let (win) (while (setq win (get-buffer-window buffer all-frames)) (remove-window win)))))
|
|
|
|
;;;###autoload
|
|
(defun mouse-remove-window (event)
|
|
"Remove the window you click on. (This calls `remove-window'.)
|
|
This command must be bound to a mouse click."
|
|
(interactive "e")
|
|
(mouse-minibuffer-check event)
|
|
(remove-window (posn-window (event-start event))))
|
|
|
|
;;;###autoload
|
|
(defun delete/iconify-window (&optional window frame-p)
|
|
"Delete or iconify WINDOW (default: `selected-window').
|
|
If WINDOW is the only one in its frame (`one-window-p'), then optional
|
|
arg FRAME-P determines the behavior regarding the frame, as follows:
|
|
If FRAME-P is nil, then the frame is deleted (with the window).
|
|
If FRAME-P is t, then the frame is iconified.
|
|
If FRAME-P is a symbol naming a function, the function is applied
|
|
to WINDOW as its only arg.
|
|
If the result is nil, then the frame is deleted.
|
|
If the result is non-nil, then the frame is iconified.
|
|
If FRAME-P is anything else, then behavior is as if FRAME-P were the
|
|
symbol `window-dedicated-p': the frame is iconified if
|
|
WINDOW is dedicated, otherwise the frame is deleted.
|
|
|
|
Interactively, FRAME-P depends on the prefix arg, as follows:
|
|
Without a prefix arg (prefix = nil), FRAME-P is `window-dedicated-p'.
|
|
With prefix arg < 0, FRAME-P is t. The frame is iconified.
|
|
With prefix arg >= 0, FRAME-P is nil. The frame is deleted."
|
|
(interactive
|
|
(list nil (if current-prefix-arg
|
|
(not (natnump (prefix-numeric-value current-prefix-arg)))
|
|
'window-dedicated-p)))
|
|
(setq window (or window (selected-window)))
|
|
(let ((one-win-p t))
|
|
(save-window-excursion
|
|
(select-window window)
|
|
(if (one-window-p)
|
|
(if frame-p
|
|
(if (eq t frame-p)
|
|
(iconify-frame)
|
|
(unless (and (symbolp frame-p) (fboundp frame-p))
|
|
(setq frame-p 'window-dedicated-p))
|
|
(if (funcall frame-p window) (iconify-frame) (delete-frame)))
|
|
(delete-frame)) ; Default.
|
|
(setq one-win-p nil)))
|
|
;; Do this outside `save-window-excursion'.
|
|
(unless one-win-p (delete-window window))))
|
|
|
|
;;;###autoload
|
|
(defun delete/iconify-windows-on (buffer &optional frame frame-p)
|
|
"For each window showing BUFFER: delete it or iconify its frame.
|
|
\(This calls `delete/iconify-window' on each window showing BUFFER.)
|
|
|
|
Optional second arg FRAME controls which frames are considered.
|
|
If nil or omitted, treat all windows showing BUFFER in any frame.
|
|
If t, treat only windows showing BUFFER in the selected frame.
|
|
If `visible', treat all windows showing BUFFER in any visible frame.
|
|
If a frame, treat only windows showing BUFFER in that frame.
|
|
|
|
Optional third arg FRAME-P controls what to do with one-window frames.
|
|
If FRAME-P is nil, then one-window frames showing BUFFER are deleted.
|
|
If FRAME-P is t, then one-window frames are iconified.
|
|
If FRAME-P is a symbol naming a function, the function is applied
|
|
to each window showing buffer in a frame by itself.
|
|
If the result is nil, then the frame is deleted.
|
|
If the result is non-nil, then the frame is iconified.
|
|
If FRAME-P is anything else, then behavior is as if FRAME-P were the
|
|
symbol `window-dedicated-p': One-window frames are
|
|
iconified if window is dedicated, else they are deleted.
|
|
|
|
Interactively, FRAME is nil, and FRAME-P depends on the prefix arg:
|
|
Without a prefix arg (prefix = nil), FRAME-P is `window-dedicated-p'.
|
|
With prefix arg < 0, FRAME-P is t. The frame is iconified.
|
|
With prefix arg >= 0, FRAME-P is nil. The frame is deleted."
|
|
(interactive
|
|
(list (read-buffer "Delete windows on buffer: " (current-buffer) 'existing)
|
|
nil
|
|
(if current-prefix-arg
|
|
(not (natnump (prefix-numeric-value current-prefix-arg)))
|
|
'window-dedicated-p)))
|
|
(setq buffer (get-buffer buffer)) ; Convert to buffer.
|
|
(when buffer ; Do nothing if null BUFFER.
|
|
;; `get-buffer-window' interprets FRAME oppositely for t and nil, so switch.
|
|
(setq frame (if (eq t frame) nil (if (eq nil frame) t frame)))
|
|
(dolist (fr (frames-on buffer))
|
|
(delete/iconify-window (get-buffer-window buffer frame) frame-p))))
|
|
|
|
;;;###autoload
|
|
(defun clone-frame (&optional frame no-clone)
|
|
"Make and select a new frame with the same parameters as FRAME.
|
|
With a prefix arg, don't clone - just call `make-frame-command'.
|
|
Return the new frame.
|
|
|
|
FRAME defaults to the selected frame. The frame is created on the
|
|
same terminal as FRAME. If the terminal is a text-only terminal then
|
|
also select the new frame."
|
|
(interactive "i\nP")
|
|
(if no-clone
|
|
(make-frame-command)
|
|
(let* ((fit-frame-inhibit-fitting-flag t)
|
|
(clone-frame-parameters (if (and clone-frame-parameters
|
|
(not (consp (car clone-frame-parameters))))
|
|
`((left . ,(+ (car clone-frame-parameters)
|
|
(or (frame-geom-value-numeric
|
|
'left (frame-parameter frame 'left))
|
|
0)))
|
|
(top . ,(+ (cdr clone-frame-parameters)
|
|
(or (frame-geom-value-numeric
|
|
'top (frame-parameter frame 'top))
|
|
0))))
|
|
clone-frame-parameters))
|
|
(default-frame-alist (append clone-frame-parameters (frame-parameters frame)))
|
|
(new-fr (make-frame)))
|
|
(select-frame new-fr)
|
|
new-fr)))
|
|
|
|
;;;###autoload
|
|
(defun rename-frame (&optional old-name new-name all-named)
|
|
"Rename a frame named OLD-NAME to NEW-NAME.
|
|
Prefix arg non-nil means rename all frames named OLD-NAME to NEWNAME.
|
|
OLD-NAME may be a frame, its name, or nil. Default is `selected-frame'.
|
|
NEW-NAME is a string or nil. Default NEW-NAME is current `buffer-name'."
|
|
(interactive
|
|
(list (read-frame (concat "Rename " (and current-prefix-arg "all ")
|
|
"frame" (and current-prefix-arg "s named") ": ")
|
|
nil t) ; Default = selected. Must exist.
|
|
(read-from-minibuffer "Rename to (new name): " (cons (buffer-name) 1))
|
|
current-prefix-arg))
|
|
(setq old-name (or old-name (get-frame-name)) ; Batch defaults from current.
|
|
new-name (or new-name (buffer-name (window-buffer (frame-selected-window)))))
|
|
;; Convert to frame if string.
|
|
(let ((fr (get-a-frame old-name)))
|
|
(if all-named
|
|
(while fr
|
|
(modify-frame-parameters fr (list (cons 'name new-name)))
|
|
(setq fr (get-a-frame old-name))) ; Get another.
|
|
(when (string= (get-frame-name fr) (get-frame-name))
|
|
(setq fr (selected-frame)))
|
|
(modify-frame-parameters fr (list (cons 'name new-name))))))
|
|
|
|
;;;###autoload
|
|
(defun rename-non-minibuffer-frame (&optional old-name new-name all-named)
|
|
"Unless OLD-NAME names the minibuffer frame, use `rename-frame'
|
|
to rename a frame named OLD-NAME to NEW-NAME.
|
|
Prefix arg non-nil means rename all frames named OLD-NAME to NEW-NAME.
|
|
OLD-NAME may be a frame, its name, or nil. Default is `selected-frame'.
|
|
NEW-NAME is a string or nil. Default NEW-NAME is current `buffer-name'."
|
|
(interactive
|
|
(list (read-frame (concat "Rename " (and current-prefix-arg "all ")
|
|
"frame" (and current-prefix-arg "s named") ": ")
|
|
nil t) ; Default = selected. Must exist.
|
|
(read-from-minibuffer "Rename to (new name): " (cons (buffer-name) 1))
|
|
current-prefix-arg))
|
|
(setq old-name (or old-name (get-frame-name)) ; Batch defaults from current.
|
|
new-name (or new-name (buffer-name (window-buffer (frame-selected-window)))))
|
|
(let ((fr (get-a-frame old-name))) ; Convert to frame if string.
|
|
(if (and (boundp '1on1-minibuffer-frame)
|
|
(eq (cdr (assq 'name (frame-parameters 1on1-minibuffer-frame)))
|
|
(cdr (assq 'name (frame-parameters fr)))))
|
|
(and (interactive-p)
|
|
(error "Use `rename-frame' if you really want to rename minibuffer frame"))
|
|
(rename-frame old-name new-name))))
|
|
|
|
;;;###autoload
|
|
(defun name-all-frames-numerically (&optional startover)
|
|
"Rename all frames to numerals in 1,2,3...
|
|
With optional arg STARTOVER (prefix arg, interactively), rename all
|
|
starting over from 1. Otherwise, numbering continues from the highest
|
|
existing frame number."
|
|
(interactive "P")
|
|
(when startover
|
|
(dolist (fr (frame-list))
|
|
(rename-non-minibuffer-frame fr (format "a%s" (frame-parameter fr 'name)))))
|
|
(mapc #'name-frame-numerically (frame-list)))
|
|
|
|
;;;###autoload
|
|
(defun name-frame-numerically (&optional frame frames)
|
|
"Name FRAME (default, selected frame) to a numeral in 1,2,3...
|
|
If FRAME's name is already such a numeral, do nothing.
|
|
Else:
|
|
Rename it to a numeral one greater than the max numeric frame name.
|
|
Rename any other frames to numerals also.
|
|
|
|
To automatically name new frames numerically, you can do this in your
|
|
init file:
|
|
|
|
(add-hook 'after-make-frame-functions 'name-frame-numerically)"
|
|
(interactive)
|
|
(setq frame (or frame (selected-frame))
|
|
frames (or frames (list frame)))
|
|
(let ((onum (frcmds-frame-number frame))
|
|
onums max)
|
|
(unless onum
|
|
(dolist (fr (frcmds-set-difference (frame-list) frames))
|
|
(unless (eq fr frame)
|
|
(name-frame-numerically fr (cons fr frames))))
|
|
(setq onums (delq nil (mapcar #'frcmds-frame-number (frame-list)))
|
|
max (if onums (apply #'max onums) 0))
|
|
(rename-non-minibuffer-frame frame (number-to-string (1+ max))))))
|
|
|
|
(defun frcmds-frame-number (frame)
|
|
"Return FRAME's number, or nil if its name is not a numeral 1,2,3..."
|
|
(let ((num (string-to-number (frame-parameter frame 'name))))
|
|
(and (wholenump num) (not (zerop num)) num)))
|
|
|
|
;;;###autoload
|
|
(defun show-frame (frame)
|
|
"Make FRAME visible and raise it, without selecting it.
|
|
FRAME may be a frame or its name."
|
|
(interactive (list (read-frame "Frame to make visible: ")))
|
|
(setq frame (get-a-frame frame))
|
|
(make-frame-visible frame)
|
|
(raise-frame frame))
|
|
|
|
;;;###autoload
|
|
(defun hide-frame (frame &optional prefix)
|
|
"Make FRAME invisible. Like `make-frame-invisible', but reads frame name.
|
|
Non-nil PREFIX makes it invisible even if all other frames are invisible."
|
|
(interactive (list (read-frame "Frame to make invisible: ")))
|
|
(make-frame-invisible (get-a-frame frame) prefix))
|
|
|
|
;;;###autoload
|
|
(defun show-a-frame-on (buffer)
|
|
"Make visible and raise a frame showing BUFFER, if there is one.
|
|
Neither the frame nor the BUFFER are selected.
|
|
BUFFER may be a buffer or its name (a string)."
|
|
(interactive
|
|
(list (read-buffer "Show a frame showing buffer: "
|
|
(if (fboundp 'another-buffer) ; Defined in `misc-fns.el'.
|
|
(another-buffer nil t)
|
|
(other-buffer (current-buffer)))
|
|
'existing)))
|
|
(when buffer ; Do nothing if null BUFFER.
|
|
(let ((fr (car (frames-on buffer)))) (when fr (show-frame fr)))))
|
|
|
|
;;;###autoload
|
|
(defun show-*Help*-buffer ()
|
|
"Raise a frame showing buffer *Help*, without selecting it."
|
|
(interactive) (show-a-frame-on "*Help*"))
|
|
|
|
;;;###autoload
|
|
(defun delete-1-window-frames-on (buffer)
|
|
"Delete all visible 1-window frames showing BUFFER."
|
|
(interactive
|
|
(list (read-buffer "Delete all visible 1-window frames showing buffer: "
|
|
(current-buffer) 'existing)))
|
|
(setq buffer (get-buffer buffer))
|
|
(save-excursion
|
|
(when (buffer-live-p buffer) ; Do nothing if dead buffer.
|
|
(dolist (fr (frames-on buffer)) ; Is it better to search through
|
|
(save-window-excursion ; `frames-on' or `get-buffer-window-list'?
|
|
(select-frame fr)
|
|
(when (one-window-p t fr) (delete-frame)))))))
|
|
|
|
;;;###autoload
|
|
(defun delete-other-frames (&optional frame)
|
|
"Delete all frames except FRAME (default: selected frame).
|
|
Interactively, use a prefix arg (`\\[universal-argument]') to be prompted for FRAME."
|
|
(interactive (list (if current-prefix-arg
|
|
(get-a-frame (read-frame "Frame to make invisible: "))
|
|
(selected-frame))))
|
|
(when frame
|
|
(dolist (fr (frame-list))
|
|
(unless (eq fr frame) (condition-case nil (delete-frame fr) (error nil))))))
|
|
|
|
;;;###autoload
|
|
(defun maximize-frame-horizontally (&optional frame)
|
|
"Maximize selected frame horizontally."
|
|
(interactive (list (selected-frame)))
|
|
(maximize-frame 'horizontal frame))
|
|
|
|
;;;###autoload
|
|
(defun maximize-frame-vertically (&optional frame)
|
|
"Maximize selected frame vertically."
|
|
(interactive (list (selected-frame)))
|
|
(maximize-frame 'vertical frame))
|
|
|
|
;;;###autoload
|
|
(defun maximize-frame (&optional direction frame)
|
|
"Maximize selected frame horizontally, vertically, or both.
|
|
With no prefix arg, maximize both directions.
|
|
With a non-negative prefix arg, maximize vertically.
|
|
With a negative prefix arg, maximize horizontally.
|
|
|
|
In Lisp code:
|
|
DIRECTION is the direction: `horizontal', `vertical', or `both'.
|
|
FRAME is the frame to maximize."
|
|
(interactive (list (if current-prefix-arg
|
|
(if (natnump (prefix-numeric-value current-prefix-arg))
|
|
'vertical
|
|
'horizontal)
|
|
'both)))
|
|
(unless frame (setq frame (selected-frame)))
|
|
(unless direction (setq direction 'both))
|
|
(let (;; Size of a frame that uses all of the available screen area,
|
|
;; but leaving room for a minibuffer frame at bottom of display.
|
|
(fr-pixel-width (frcmds-available-screen-pixel-width))
|
|
(fr-pixel-height (frcmds-available-screen-pixel-height))
|
|
(fr-origin (if (eq direction 'horizontal)
|
|
(car (frcmds-effective-screen-pixel-bounds))
|
|
(cadr (frcmds-effective-screen-pixel-bounds))))
|
|
(orig-left (or (frame-parameter frame 'left) 0))
|
|
(orig-top (or (frame-parameter frame 'top) 0))
|
|
(orig-width (or (frame-parameter frame 'width) 0))
|
|
(orig-height (or (frame-parameter frame 'height) 0)))
|
|
(let* ((bord-width (cdr (assq 'border-width (frame-parameters frame))))
|
|
(borders (if bord-width (* 2 bord-width) 0))
|
|
(new-left (if (memq direction '(horizontal both)) 0 orig-left))
|
|
(new-top (if (memq direction '(vertical both)) 0 orig-top))
|
|
;; Subtract borders, scroll bars, & title bar, then convert pixel sizes to char sizes.
|
|
(new-width (if (memq direction '(horizontal both))
|
|
(/ (- fr-pixel-width borders (frcmds-extra-pixels-width frame))
|
|
(frame-char-width frame))
|
|
orig-width))
|
|
(new-height (if (memq direction '(vertical both))
|
|
(- (/ (- fr-pixel-height borders
|
|
(frcmds-extra-pixels-height frame)
|
|
window-mgr-title-bar-pixel-height
|
|
(frcmds-smart-tool-bar-pixel-height))
|
|
(frame-char-height frame))
|
|
;; Subtract menu bar unless on Carbon Emacs (menu bar not in the frame).
|
|
(if (eq window-system 'mac)
|
|
0
|
|
(cdr (assq 'menu-bar-lines (frame-parameters frame)))))
|
|
orig-height)))
|
|
(modify-frame-parameters
|
|
frame
|
|
`((left . ,new-left)
|
|
(width . ,new-width)
|
|
(top . ,new-top)
|
|
(height . ,new-height)
|
|
;; If we actually changed a parameter, record the old one for restoration.
|
|
,(and new-left
|
|
(/= (frame-geom-value-numeric 'left orig-left)
|
|
(frame-geom-value-numeric 'left new-left))
|
|
(cons 'restore-left orig-left))
|
|
,(and new-top
|
|
(/= (frame-geom-value-numeric 'top orig-top)
|
|
(frame-geom-value-numeric 'top new-top))
|
|
(cons 'restore-top orig-top))
|
|
,(and new-width
|
|
(/= (frame-geom-value-numeric 'width orig-width)
|
|
(frame-geom-value-numeric 'width new-width))
|
|
(cons 'restore-width orig-width))
|
|
,(and new-height
|
|
(/= (frame-geom-value-numeric 'height orig-height)
|
|
(frame-geom-value-numeric 'height new-height))
|
|
(cons 'restore-height orig-height)))))
|
|
(show-frame frame)
|
|
(incf fr-origin (if (eq direction 'horizontal) fr-pixel-width fr-pixel-height))))
|
|
|
|
;;;###autoload
|
|
(unless (fboundp 'restore-frame-horizontally)
|
|
(defalias 'restore-frame-horizontally 'toggle-max-frame-horizontally))
|
|
;;;###autoload
|
|
(defun toggle-max-frame-horizontally (&optional frame)
|
|
"Toggle maximization of FRAME horizontally.
|
|
If used once, this restores the frame. If repeated, it maximizes.
|
|
This affects the `left' and `width' frame parameters.
|
|
|
|
FRAME defaults to the selected frame."
|
|
(interactive (list (selected-frame)))
|
|
(toggle-max-frame 'horizontal frame))
|
|
|
|
;;;###autoload
|
|
(unless (fboundp 'restore-frame-vertically)
|
|
(defalias 'restore-frame-vertically 'toggle-max-frame-vertically))
|
|
;;;###autoload
|
|
(defun toggle-max-frame-vertically (&optional frame)
|
|
"Toggle maximization of FRAME vertically.
|
|
If used once, this restores the frame. If repeated, it maximizes.
|
|
This affects the `top' and `height' frame parameters.
|
|
|
|
FRAME defaults to the selected frame."
|
|
(interactive (list (selected-frame)))
|
|
(toggle-max-frame 'vertical frame))
|
|
|
|
;;;###autoload
|
|
(unless (fboundp 'restore-frame) (defalias 'restore-frame 'toggle-max-frame))
|
|
;;;###autoload
|
|
(defun toggle-max-frame (&optional direction frame)
|
|
"Toggle maximization of FRAME horizontally, vertically, or both.
|
|
Reverses or (if restored) repeats the effect of the Emacs maximize
|
|
commands. Does not restore from maximization effected outside Emacs.
|
|
|
|
With no prefix arg, toggle both directions.
|
|
With a non-negative prefix arg, toggle only vertically.
|
|
With a negative prefix arg, toggle horizontally.
|
|
|
|
When toggling both directions, each is toggled from its last maximize
|
|
or restore state. This means that using this after
|
|
`maximize-frame-horizontally', `maximize-frame-vertically',
|
|
`toggle-max-frame-horizontally', or `toggle-max-frame-vertically' does
|
|
not necessarily just reverse the effect of that command.
|
|
|
|
In Lisp code:
|
|
DIRECTION is the direction: `horizontal', `vertical', or `both'.
|
|
FRAME is the frame to change. It defaults to the selected frame."
|
|
(interactive (list (if current-prefix-arg
|
|
(if (natnump (prefix-numeric-value current-prefix-arg))
|
|
'vertical
|
|
'horizontal)
|
|
'both)))
|
|
(unless frame (setq frame (selected-frame)))
|
|
(unless direction (setq direction 'both))
|
|
(let ((restore-left (frame-parameter frame 'restore-left))
|
|
(restore-top (frame-parameter frame 'restore-top))
|
|
(restore-width (frame-parameter frame 'restore-width))
|
|
(restore-height (frame-parameter frame 'restore-height))
|
|
(orig-left (frame-parameter frame 'left))
|
|
(orig-top (frame-parameter frame 'top))
|
|
(orig-width (frame-parameter frame 'width))
|
|
(orig-height (frame-parameter frame 'height))
|
|
(horiz (memq direction '(horizontal both)))
|
|
(vert (memq direction '(vertical both))))
|
|
(case direction
|
|
(both (unless (and restore-left restore-width restore-top restore-height)
|
|
(maximize-frame 'both frame)))
|
|
(vertical (unless (and restore-top restore-height) (maximize-frame-vertically frame)))
|
|
(horizontal (unless (and restore-left restore-width) (maximize-frame-horizontally frame))))
|
|
(modify-frame-parameters
|
|
frame `(,(and horiz restore-left (cons 'left restore-left))
|
|
,(and horiz restore-width (cons 'width restore-width))
|
|
,(and vert restore-top (cons 'top restore-top))
|
|
,(and vert restore-height (cons 'height restore-height))
|
|
,(and horiz orig-left (cons 'restore-left orig-left))
|
|
,(and horiz orig-width (cons 'restore-width orig-width))
|
|
,(and vert orig-top (cons 'restore-top orig-top))
|
|
,(and vert orig-height (cons 'restore-height orig-height)))))
|
|
(show-frame frame))
|
|
|
|
;;;###autoload
|
|
(defalias 'tile-frames-side-by-side 'tile-frames-horizontally)
|
|
;;;###autoload
|
|
(defun tile-frames-horizontally (&optional frames)
|
|
"Tile frames horizontally (side by side).
|
|
Interactively:
|
|
With prefix arg, you are prompted for names of two frames to tile.
|
|
With no prefix arg, all visible frames are tiled, except a
|
|
standalone minibuffer frame, if any.
|
|
If called from a program, all frames in list FRAMES are tiled."
|
|
(interactive (and current-prefix-arg (frcmds-read-args-for-tiling)))
|
|
(frcmds-tile-frames 'horizontal frames))
|
|
|
|
;;;###autoload
|
|
(defalias 'tile-frames-top-to-bottom 'tile-frames-vertically)
|
|
;;;###autoload
|
|
(defun tile-frames-vertically (&optional frames)
|
|
"Tile frames vertically (stacking from the top of the screen downward).
|
|
Interactively:
|
|
With prefix arg, you are prompted for names of two frames to tile.
|
|
With no prefix arg, all visible frames are tiled, except a
|
|
standalone minibuffer frame, if any.
|
|
If called from a program, all frames in list FRAMES are tiled."
|
|
(interactive (and current-prefix-arg (frcmds-read-args-for-tiling)))
|
|
(frcmds-tile-frames 'vertical frames))
|
|
|
|
;;;###autoload
|
|
(defun create-frame-tiled-horizontally ()
|
|
"Horizontally tile screen with selected frame and a copy.
|
|
The same character size is used for the new frame."
|
|
(interactive)
|
|
(let* ((fr1 (selected-frame))
|
|
(font1 (frame-parameter fr1 'font))
|
|
(fr2 (make-frame-command)))
|
|
(save-selected-window (select-frame fr2) (set-frame-font font1))
|
|
(frcmds-tile-frames 'horizontal (list fr1 fr2))))
|
|
|
|
;;;###autoload
|
|
(defun create-frame-tiled-vertically ()
|
|
"Vertically tile screen with selected frame and a copy.
|
|
The same character size is used for the new frame."
|
|
(interactive)
|
|
(let* ((fr1 (selected-frame))
|
|
(font1 (frame-parameter fr1 'font))
|
|
(fr2 (make-frame-command)))
|
|
(frcmds-tile-frames 'vertical (list fr1 fr2))))
|
|
|
|
;;;###autoload
|
|
(defun split-frame-horizontally (num)
|
|
"Horizontally split the selected frame.
|
|
With a prefix arg, create that many new frames.
|
|
The same character size is used for the new frames."
|
|
(interactive "p")
|
|
(frcmds-split-frame-1 'horizontal num))
|
|
|
|
;;;###autoload
|
|
(defun split-frame-vertically (num)
|
|
"Vertically split the selected frame.
|
|
With a prefix arg, create that many new frames.
|
|
The same character size is used for the new frames."
|
|
(interactive "p")
|
|
(frcmds-split-frame-1 'vertical num))
|
|
|
|
(defun frcmds-split-frame-1 (direction num)
|
|
"Helper for `split-frame-horizontally' and `split-frame-vertically'.
|
|
DIRECTION is `horizontal' or `vertical'.
|
|
NUM is the desired number of new frames to create."
|
|
(let* ((fr1 (selected-frame))
|
|
(font1 (frame-parameter fr1 'font))
|
|
(x-min (frame-geom-value-numeric 'left (frame-parameter fr1 'left)))
|
|
(y-min (frame-geom-value-numeric 'top (frame-parameter fr1 'top)))
|
|
(wid (frame-pixel-width fr1))
|
|
(hght (frcmds-frame-pixel-height fr1))
|
|
(frames (list fr1))
|
|
fr)
|
|
(dotimes (ii num)
|
|
(setq fr (make-frame-command))
|
|
(save-selected-window (select-frame fr) (set-frame-font font1))
|
|
(push fr frames))
|
|
(frcmds-tile-frames direction frames x-min y-min wid hght)))
|
|
|
|
(defun frcmds-frame-pixel-height (frame)
|
|
"Pixel height of FRAME, including the window-manager title bar and menu-bar.
|
|
For the title bar, `window-mgr-title-bar-pixel-height' is used.
|
|
For the menu-bar, the frame char size is multiplied by frame parameter
|
|
`menu-bar-lines'. But that parameter does not take into account
|
|
menu-bar wrapping."
|
|
(+ window-mgr-title-bar-pixel-height
|
|
(frame-pixel-height frame)
|
|
(if (not (eq window-system 'x))
|
|
0
|
|
(+ (* (frame-char-height frame)
|
|
(cdr (assq 'menu-bar-lines (frame-parameters frame))))))))
|
|
|
|
(defun frcmds-tile-frames (direction frames &optional x-min-pix y-min-pix pix-width pix-height)
|
|
"Tile visible frames horizontally or vertically, depending on DIRECTION.
|
|
Arg DIRECTION is `horizontal' or `vertical' (meaning side by side or
|
|
above and below, respectively).
|
|
|
|
Arg FRAMES is the list of frames to tile. If nil, then tile all visible
|
|
frames (except a standalone minibuffer frame, if any).
|
|
|
|
The optional args cause tiling to be limited to the bounding rectangle
|
|
they specify. X-MIN-PIX and Y-MIN-PIX are the `left' and `top' screen
|
|
pixel positions of the rectangle. X-PIX-WIDTH and Y-PIX-HEIGHT are
|
|
the pixel width and height of the rectangle."
|
|
(let ((visible-frames (or frames
|
|
(filtered-frame-list ; Get visible frames, except minibuffer.
|
|
#'(lambda (fr)
|
|
(and (eq t (frame-visible-p fr))
|
|
(or (not (fboundp 'thumfr-thumbnail-frame-p))
|
|
(not (thumfr-thumbnail-frame-p fr)))
|
|
(or (not (boundp '1on1-minibuffer-frame))
|
|
(not (eq (cdr (assq 'name (frame-parameters
|
|
1on1-minibuffer-frame)))
|
|
(cdr (assq 'name (frame-parameters fr)))))))))))
|
|
;; Size of a frame that uses all of the available screen area,
|
|
;; but leaving room for a minibuffer frame at bottom of display.
|
|
(fr-pixel-width (or pix-width (frcmds-available-screen-pixel-width)))
|
|
(fr-pixel-height (or pix-height (frcmds-available-screen-pixel-height)))
|
|
(fr-origin (if (eq direction 'horizontal)
|
|
(or x-min-pix (car (frcmds-effective-screen-pixel-bounds)))
|
|
(or y-min-pix (cadr (frcmds-effective-screen-pixel-bounds))))))
|
|
(case direction ; Size of frame in pixels.
|
|
(horizontal (setq fr-pixel-width (/ fr-pixel-width (length visible-frames))))
|
|
(vertical (setq fr-pixel-height (/ fr-pixel-height (length visible-frames))))
|
|
(otherwise (error "`frcmds-tile-frames': DIRECTION must be `horizontal' or `vertical'")))
|
|
(dolist (fr visible-frames)
|
|
(if (or (> emacs-major-version 24)
|
|
(and (= emacs-major-version 24) (> emacs-minor-version 3)))
|
|
(let ((frame-resize-pixelwise t))
|
|
(set-frame-size
|
|
fr
|
|
;; Subtract scroll bars, & title bar.
|
|
(- fr-pixel-width (frcmds-extra-pixels-width fr))
|
|
(- fr-pixel-height
|
|
window-mgr-title-bar-pixel-height
|
|
(if pix-height 0 (frcmds-smart-tool-bar-pixel-height fr))
|
|
(if (not (eq window-system 'x)) ; Menu bar for X is not in the frame.
|
|
0
|
|
(* (frame-char-height fr) (cdr (assq 'menu-bar-lines (frame-parameters fr))))))
|
|
'PIXELWISE))
|
|
(set-frame-size
|
|
fr
|
|
;; Subtract scroll bars, & title bar, then convert pixel sizes to char sizes.
|
|
(/ (- fr-pixel-width
|
|
(frcmds-extra-pixels-width fr))
|
|
(frame-char-width fr))
|
|
(/ (- fr-pixel-height
|
|
(frcmds-extra-pixels-height fr)
|
|
window-mgr-title-bar-pixel-height
|
|
(if pix-height 0 (frcmds-smart-tool-bar-pixel-height fr))
|
|
(if (not (eq window-system 'x)) ; Menu bar for X is not in the frame.
|
|
0
|
|
(* (frame-char-height fr) (cdr (assq 'menu-bar-lines (frame-parameters fr))))))
|
|
(frame-char-height fr))))
|
|
(set-frame-position fr
|
|
(if (eq direction 'horizontal) fr-origin (or x-min-pix 0))
|
|
(if (eq direction 'horizontal) (or y-min-pix 0) fr-origin))
|
|
(show-frame fr)
|
|
;; Move over the width or height of one frame, and add one border width.
|
|
(incf fr-origin (+ (or (cdr (assq 'border-width (frame-parameters fr))) 0)
|
|
(if (eq direction 'horizontal) fr-pixel-width fr-pixel-height))))))
|
|
|
|
(defun frcmds-extra-pixels-width (frame)
|
|
"Pixel difference between FRAME total width and its text area width."
|
|
(- (frame-pixel-width frame) (* (frame-char-width frame) (frame-width frame))))
|
|
|
|
(defun frcmds-extra-pixels-height (frame)
|
|
"Pixel difference between FRAME total height and its text area height."
|
|
(- (frame-pixel-height frame) (* (frame-char-height frame) (frame-height frame))))
|
|
|
|
(defun frcmds-smart-tool-bar-pixel-height (&optional frame)
|
|
"Pixel height of macOS smart tool bar."
|
|
(if (and (boundp 'mac-tool-bar-display-mode) (> (frame-parameter frame 'tool-bar-lines) 0))
|
|
(if (eq mac-tool-bar-display-mode 'icons) 40 56)
|
|
0))
|
|
|
|
(defun frcmds-read-args-for-tiling ()
|
|
"Read two frame names and return a list of their frames.
|
|
Return value can be used as second argument for `frcmds-tile-frames'."
|
|
(list
|
|
(list
|
|
;; Note: `read-frame' puts selected-frame name at front of `frame-name-history'.
|
|
(get-a-frame (read-frame "Tile two frames - First frame: " nil t))
|
|
;; Get next visible frame. For default (prompt) value:
|
|
;; If there is another visible frame in `frame-name-history', use next such.
|
|
;; Else if there is another visible frame in internal frame list, use next such.
|
|
;; Else use selected frame. (`frame-name-history' is defined in `frame.el'.)
|
|
(get-a-frame
|
|
(read-frame
|
|
"Second frame: "
|
|
(let ((fr-names (cdr frame-name-history))
|
|
(visible-p nil)
|
|
(fr nil))
|
|
(while (and (not fr) fr-names) ; While no visible frame found and still fr-names to check.
|
|
(setq fr (car fr-names) ; Name
|
|
fr (get-a-frame fr) ; Frame
|
|
fr (and fr (eq t (frame-visible-p fr)) fr) ; Visible frame
|
|
fr-names (cdr fr-names)))
|
|
|
|
;; If no visible frames in history, besides selected-frame,
|
|
;; then get next visible frame (not its name) from internal frame list.
|
|
(unless fr
|
|
(setq fr (selected-frame))
|
|
(while (and (not visible-p)
|
|
(setq fr (next-frame fr))
|
|
(not (equal fr (selected-frame)))) ; equal => no other found.
|
|
(setq visible-p (eq t (frame-visible-p fr)))))
|
|
fr)
|
|
t)))))
|
|
|
|
(defun frcmds-available-screen-pixel-bounds ()
|
|
"Returns a value of the same form as option `available-screen-pixel-bounds'.
|
|
This represents the currently available screen area."
|
|
(or available-screen-pixel-bounds ; Use the option value, if available.
|
|
(if (fboundp 'mac-display-available-pixel-bounds) ; macOS-specific.
|
|
(mac-display-available-pixel-bounds)
|
|
(if (fboundp 'display-monitor-attributes-list) ; Emacs 24.4+
|
|
(let ((attss (display-monitor-attributes-list))
|
|
(x0 most-positive-fixnum)
|
|
(y0 most-positive-fixnum)
|
|
(x1 0)
|
|
(y1 0)
|
|
geom)
|
|
(dolist (atts attss)
|
|
(setq geom (cdr (assq 'geometry atts))
|
|
x0 (min x0 (nth 0 geom))
|
|
y0 (min y0 (nth 1 geom))
|
|
x1 (max x1 (nth 2 geom))
|
|
;; Use `max' for the height too, but it does not account for taskbar etc.
|
|
y1 (max y1 (nth 3 geom))))
|
|
(list x0 y0 x1 y1))
|
|
;; Punt. Assume only one monitor.
|
|
(list 0 0 (x-display-pixel-width) (x-display-pixel-height))))))
|
|
|
|
; Emacs 20 doesn't have `butlast'. Define it to avoid requiring `cl.el' at runtime. From `subr.el'.
|
|
(unless (fboundp 'butlast)
|
|
(defun nbutlast (list &optional n)
|
|
"Modifies LIST to remove the last N elements."
|
|
(let ((m (length list)))
|
|
(or n (setq n 1))
|
|
(and (< n m) (progn (when (> n 0) (setcdr (nthcdr (- (1- m) n) list) ()))
|
|
list))))
|
|
|
|
(defun butlast (list &optional n)
|
|
"Return a copy of LIST with the last N elements removed."
|
|
(if (and n (<= n 0)) list (nbutlast (copy-sequence list) n))))
|
|
|
|
(defun frcmds-effective-screen-pixel-bounds ()
|
|
"Upper left and lower right of available screen space for tiling frames.
|
|
This is `frcmds-available-screen-pixel-bounds', possibly adjusted to
|
|
allow for the standalone minibuffer frame provided by `oneonone.el'."
|
|
(if (boundp '1on1-minibuffer-frame)
|
|
(append (butlast (frcmds-available-screen-pixel-bounds))
|
|
(list (frame-geom-value-numeric 'top (cdr (assq 'top (frame-parameters
|
|
1on1-minibuffer-frame))))))
|
|
(frcmds-available-screen-pixel-bounds)))
|
|
|
|
(defun frcmds-available-screen-pixel-width (&optional include-mini-p)
|
|
"Width of the usable screen, in pixels.
|
|
Non-nil optional argument `include-mini-p' means include the space
|
|
occupied by a standalone minibuffer, if any."
|
|
(let ((bounds (if include-mini-p
|
|
(frcmds-available-screen-pixel-bounds)
|
|
(frcmds-effective-screen-pixel-bounds))))
|
|
(- (caddr bounds) (car bounds)))) ; X1 - X0
|
|
|
|
(defun frcmds-available-screen-pixel-height (&optional include-mini-p)
|
|
"Height of the usable screen, in pixels.
|
|
Non-nil optional argument `include-mini-p' means include the
|
|
space occupied by a standalone minibuffer, if any."
|
|
(let ((bounds (if include-mini-p
|
|
(frcmds-available-screen-pixel-bounds)
|
|
(frcmds-effective-screen-pixel-bounds))))
|
|
(- (cadddr bounds) (cadr bounds)))) ; Y1 - Y0
|
|
|
|
;; Same as `zz-repeat-command' in `zones.el'.
|
|
;;
|
|
(defun frcmds-repeat-command (command) ; Emacs 22+
|
|
"Repeat COMMAND."
|
|
(require 'repeat) ; Define its vars before we let-bind them.
|
|
(let ((repeat-previous-repeated-command command)
|
|
(repeat-message-function #'ignore)
|
|
(last-repeatable-command 'repeat))
|
|
(repeat nil)))
|
|
|
|
;;;###autoload
|
|
(defun enlarge-window-horizontally-repeat () ; Emacs 22+
|
|
"Enlarge selected window horizontally by one column.
|
|
You can repeat this by hitting the last key again..."
|
|
(interactive)
|
|
(require 'repeat)
|
|
(frcmds-repeat-command 'enlarge-window-horizontally))
|
|
|
|
;;;###autoload
|
|
(defun shrink-window-horizontally-repeat () ; Emacs 22+
|
|
"Srhink selected window horizontally by one column.
|
|
You can repeat this by hitting the last key again..."
|
|
(interactive)
|
|
(require 'repeat)
|
|
(frcmds-repeat-command 'shrink-window-horizontally))
|
|
|
|
;;;###autoload
|
|
(defun enlarge-window-repeat () ; Emacs 22+
|
|
"Enlarge selected window vertically by one line.
|
|
You can repeat this by hitting the last key again..."
|
|
(interactive)
|
|
(require 'repeat)
|
|
(frcmds-repeat-command 'enlarge-window))
|
|
|
|
;;;###autoload
|
|
(defun shrink-window-repeat () ; Emacs 22+
|
|
"Shrink selected window vertically by one line.
|
|
You can repeat this by hitting the last key again..."
|
|
(interactive)
|
|
(require 'repeat)
|
|
(frcmds-repeat-command 'shrink-window))
|
|
|
|
;;;###autoload
|
|
(defun enlarge/shrink-window (arg) ; Emacs 22+
|
|
"Enlarge selected window vertically by one line.
|
|
With a prefix arg, shrink it."
|
|
(interactive "P")
|
|
(if arg (shrink-window 1) (enlarge-window 1)))
|
|
|
|
;;;###autoload
|
|
(defun enlarge/shrink-window-repeat () ; Emacs 22+
|
|
"Enlarge selected window vertically by one line.
|
|
With a prefix arg, shrink it.
|
|
You can repeat this by hitting the last key again..."
|
|
(interactive)
|
|
(require 'repeat)
|
|
(frcmds-repeat-command 'enlarge/shrink-window))
|
|
|
|
;; Inspired by `sk-grow-frame' from Sarir Khamsi [sarir.khamsi@raytheon.com]
|
|
;;;###autoload
|
|
(defun enlarge-frame (&optional increment frame) ; Suggested binding: `C-M-down'.
|
|
"Increase the height of FRAME (default: selected-frame) by INCREMENT.
|
|
INCREMENT is in lines (characters).
|
|
Interactively, it is given by the prefix argument."
|
|
(interactive "p")
|
|
(set-frame-height frame (+ (frame-height frame) increment)))
|
|
|
|
;;;###autoload
|
|
(defun enlarge-frame-horizontally (&optional increment frame) ; Suggested binding: `C-M-right'.
|
|
"Increase the width of FRAME (default: selected-frame) by INCREMENT.
|
|
INCREMENT is in columns (characters).
|
|
Interactively, it is given by the prefix argument."
|
|
(interactive "p")
|
|
(set-frame-width frame (+ (frame-width frame) increment)))
|
|
|
|
;;;###autoload
|
|
(defun shrink-frame (&optional increment frame) ; Suggested binding: `C-M-up'.
|
|
"Decrease the height of FRAME (default: selected-frame) by INCREMENT.
|
|
INCREMENT is in lines (characters).
|
|
Interactively, it is given by the prefix argument."
|
|
(interactive "p")
|
|
(set-frame-height frame (- (frame-height frame) increment)))
|
|
|
|
;;;###autoload
|
|
(defun shrink-frame-horizontally (&optional increment frame) ; Suggested binding: `C-M-left'.
|
|
"Decrease the width of FRAME (default: selected-frame) by INCREMENT.
|
|
INCREMENT is in columns (characters).
|
|
Interactively, it is given by the prefix argument."
|
|
(interactive "p")
|
|
(set-frame-width frame (- (frame-width frame) increment)))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-down (&optional n frame) ; Suggested binding: `M-down'.
|
|
"Move selected frame down.
|
|
Move it N times `frame-char-height', where N is the prefix arg.
|
|
In Lisp code, FRAME is the frame to move."
|
|
(interactive "p")
|
|
(unless n (setq n 1))
|
|
(setq n (* n (frame-char-height frame)))
|
|
(modify-frame-parameters frame (list (list 'top '+ (frcmds-new-frame-position frame 'top n)))))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-up (&optional n frame) ; Suggested binding: `M-up'.
|
|
"Move selected frame up.
|
|
Same as `move-frame-down', except movement is up."
|
|
(interactive "p")
|
|
(unless n (setq n 1))
|
|
(move-frame-down (- n)))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-right (&optional n frame) ; Suggested binding: `M-right'.
|
|
"Move frame to the right.
|
|
Move it N times `frame-char-width', where N is the prefix arg.
|
|
In Lisp code, FRAME is the frame to move."
|
|
(interactive "p")
|
|
(unless n (setq n 1))
|
|
(setq n (* n (frame-char-width frame)))
|
|
(modify-frame-parameters frame (list (list 'left '+ (frcmds-new-frame-position frame 'left n)))))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-left (&optional n frame) ; Suggested binding: `M-left'.
|
|
"Move frame to the left.
|
|
Same as `move-frame-right', except movement is to the left."
|
|
(interactive "p")
|
|
(unless n (setq n 1))
|
|
(move-frame-right (- n)))
|
|
|
|
;; Helper function.
|
|
(defun frcmds-new-frame-position (frame type incr)
|
|
"Return the new TYPE position of FRAME, incremented by INCR.
|
|
TYPE is `left' or `top'.
|
|
INCR is the increment to use when changing the position."
|
|
(let* ((f-dim (if (eq 'left type) (frame-pixel-width frame) (frame-pixel-height frame)))
|
|
(f-min (cadr (frame-geom-value-cons type (cdr (assq type (frame-parameters frame))))))
|
|
(f-max (+ f-min f-dim))
|
|
(new-f-min (+ incr f-min))
|
|
(new-f-max (+ incr f-max))
|
|
(d-bnds (frcmds-available-screen-pixel-bounds))
|
|
(d-min (if (eq 'left type) (nth 0 d-bnds) (nth 1 d-bnds)))
|
|
(d-max (if (eq 'left type) (nth 2 d-bnds) (nth 3 d-bnds))))
|
|
(if (not move-frame-wrap-within-display-flag)
|
|
new-f-min
|
|
(when (< new-f-max d-min) (setq new-f-min d-max))
|
|
(when (> new-f-min d-max) (setq new-f-min (- d-min (- f-max f-min))))
|
|
new-f-min)))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-to-screen-top (arg &optional frame) ; Suggested binding: `M-S-v'.
|
|
"Move FRAME (default: selected-frame) to the top of the screen.
|
|
With a prefix arg, offset it that many char heights from the top."
|
|
(interactive (list (if current-prefix-arg
|
|
(* (frame-char-height) (prefix-numeric-value current-prefix-arg))
|
|
0)
|
|
(get-a-frame (read-frame "Frame: " nil 'EXISTING))))
|
|
(modify-frame-parameters frame `((top . ,arg))))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-to-screen-bottom (arg &optional frame) ; Suggested binding: `C-S-v'.
|
|
"Move FRAME (default: selected-frame) to the bottom of the screen.
|
|
With a prefix arg, offset it that many char heights from the bottom."
|
|
(interactive (list (if current-prefix-arg
|
|
(* (frame-char-height) (prefix-numeric-value current-prefix-arg))
|
|
0)
|
|
(get-a-frame (read-frame "Frame: " nil 'EXISTING))))
|
|
(let* ((borders (* 2 (cdr (assq 'border-width (frame-parameters frame)))))
|
|
(avail-height (- (/ (- (frcmds-available-screen-pixel-height) borders
|
|
(frcmds-extra-pixels-height frame)
|
|
window-mgr-title-bar-pixel-height
|
|
(frcmds-smart-tool-bar-pixel-height))
|
|
(frame-char-height frame))
|
|
;; Subtract menu bar unless on Carbon Emacs (menu bar not in the frame).
|
|
(if (eq window-system 'mac)
|
|
0
|
|
(cdr (assq 'menu-bar-lines (frame-parameters frame)))))))
|
|
(modify-frame-parameters frame `((top . ,(- (+ avail-height arg)))))))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-to-screen-left (arg &optional frame) ; Suggested binding: `C-S-prior'.
|
|
"Move FRAME (default: selected-frame) to the left side of the screen.
|
|
With a prefix arg, offset it that many char widths from the left."
|
|
(interactive (list (if current-prefix-arg
|
|
(* (frame-char-width) (prefix-numeric-value current-prefix-arg))
|
|
0)
|
|
(get-a-frame (read-frame "Frame: " nil 'EXISTING))))
|
|
(modify-frame-parameters frame `((left . ,arg))))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-to-screen-right (arg &optional frame) ; Suggested binding: `C-S-next'.
|
|
"Move FRAME (default: selected-frame) to the right side of the screen.
|
|
With a prefix arg, offset it that many char widths from the right."
|
|
(interactive (list (if current-prefix-arg
|
|
(* (frame-char-width) (prefix-numeric-value current-prefix-arg))
|
|
0)
|
|
(get-a-frame (read-frame "Frame: " nil 'EXISTING))))
|
|
(modify-frame-parameters
|
|
frame ; Hard-code 7 here - what does it depend on?
|
|
`((left . ,(- (x-display-pixel-width) (+ (frame-pixel-width) 7 arg))))))
|
|
|
|
;;;###autoload
|
|
(defun move-frame-to-screen-top-left (arg &optional frame) ; Suggested binding: `C-S-home'.
|
|
"Move FRAME (default: selected-frame) to the top and left of the screen.
|
|
With a prefix arg, offset it that many char widths from the edges.
|
|
|
|
Note: You can use this command to move an off-screen (thus not
|
|
visible) frame back onto the screen."
|
|
(interactive (list (if current-prefix-arg
|
|
(* (frame-char-width) (prefix-numeric-value current-prefix-arg))
|
|
0)
|
|
(get-a-frame (read-frame "Frame: " nil 'EXISTING))))
|
|
(modify-frame-parameters frame '((top . ,arg) (left . ,arg))))
|
|
|
|
|
|
(when (> emacs-major-version 22) ; Emacs 23+
|
|
|
|
(defun decrease-frame-transparency (&optional n frame) ; Suggested binding: `C-M-S-up'.
|
|
"Decrease the transparency of the selected frame.
|
|
Decrease it by N percent, where N is the prefix arg.
|
|
In Lisp code, FRAME is the frame to move."
|
|
(interactive "p")
|
|
(unless n (setq n 1))
|
|
(let* ((now (or (frame-parameter frame 'alpha) 100))
|
|
(new (+ now n)))
|
|
(when (> new 100) (setq new frame-alpha-lower-limit))
|
|
(when (< new frame-alpha-lower-limit) (setq new 100))
|
|
(set-frame-parameter frame 'alpha new)))
|
|
|
|
(defun increase-frame-transparency (&optional n frame) ; Suggested binding: `C-M-S-down'.
|
|
"Increase the transparency of the selected frame.
|
|
Same as `decrease-frame-transparency', except increase."
|
|
(interactive "p")
|
|
(unless n (setq n 1))
|
|
(decrease-frame-transparency (- n) frame))
|
|
)
|
|
|
|
|
|
;; This does not work 100% well. For instance, set frame font to
|
|
;; "-raster-Terminal-normal-r-normal-normal-12-90-96-96-c-50-ms-oemlatin", then decrease font size.
|
|
;; The next smaller existing font on my machine is
|
|
;; "-raster-Terminal-normal-r-normal-normal-11-*-96-96-c-*-ms-oemlatin". Decrease size again.
|
|
;; Next smaller font is "-raster-Terminal-bold-r-normal-normal-5-37-96-96-c-60-ms-oemlatin". Notice
|
|
;; the switch to bold from regular. Cannot decrease any more. Increase size. Next larger font is
|
|
;; "-raster-Terminal-bold-r-normal-normal-8-*-96-96-c-*-ms-oemlatin". Can no longer increase size.
|
|
;;
|
|
;;;###autoload
|
|
(defun enlarge-font (&optional increment frame)
|
|
"Increase size of font in FRAME by INCREMENT.
|
|
Interactively, INCREMENT is given by the prefix argument.
|
|
Optional FRAME parameter defaults to current frame."
|
|
(interactive "p")
|
|
(setq frame (or frame (selected-frame)))
|
|
(let ((fontname (cdr (assq 'font (frame-parameters frame))))
|
|
(count enlarge-font-tries))
|
|
(setq fontname (frcmds-enlarged-font-name fontname frame increment))
|
|
(while (and (not (x-list-fonts fontname)) (wholenump (setq count (1- count))))
|
|
(setq fontname (frcmds-enlarged-font-name fontname frame increment)))
|
|
(unless (x-list-fonts fontname) (error "Cannot change font size"))
|
|
(modify-frame-parameters frame (list (cons 'font fontname)))
|
|
;; Update faces that want a bold or italic version of the default font.
|
|
(when (< emacs-major-version 21) (frame-update-faces frame))))
|
|
|
|
;;; This was a workaround hack for an Emacs 23 bug (#119, aka #1562).
|
|
;;; This works OK, but it is not as refined as the version I use, and it does not work for
|
|
;;; older Emacs versions.
|
|
;;;
|
|
;;; (when (> emacs-major-version 22)
|
|
;;; (defun enlarge-font (&optional increment frame)
|
|
;;; "Increase size of font in FRAME by INCREMENT.
|
|
;;; Interactively, INCREMENT is given by the prefix argument.
|
|
;;; Optional FRAME parameter defaults to current frame."
|
|
;;; (interactive "p")
|
|
;;; (setq frame (or frame (selected-frame)))
|
|
;;; (set-face-attribute
|
|
;;; 'default frame :height (+ (* 10 increment)
|
|
;;; (face-attribute 'default :height frame 'default)))))
|
|
|
|
|
|
|
|
|
|
|
|
;;; Define error symbols `font-too-small' and `font-size', and their error conditions and messages.
|
|
;;;
|
|
;;; You can use these to handle an error of trying to make the font too small.
|
|
;;; See library `thumb-frm.el', command `thumfr-thumbify-frame'.
|
|
;;;
|
|
(put 'font-too-small 'error-conditions '(error font-size font-too-small))
|
|
(put 'font-too-small 'error-message "Font size is too small")
|
|
|
|
(put 'font-size 'error-conditions '(error font-size))
|
|
(put 'font-size 'error-message "Bad font size")
|
|
|
|
(defun frcmds-enlarged-font-name (fontname frame increment)
|
|
"FONTNAME, after enlarging font size of FRAME by INCREMENT.
|
|
FONTNAME is the font of FRAME."
|
|
(when (query-fontset fontname)
|
|
(let ((ascii (assq 'ascii (aref (fontset-info fontname frame) 2))))
|
|
(when ascii (setq fontname (nth 2 ascii)))))
|
|
(let ((xlfd-fields (x-decompose-font-name fontname)))
|
|
(unless xlfd-fields (error "Cannot decompose font name"))
|
|
(let ((new-size (+ (string-to-number (aref xlfd-fields xlfd-regexp-pixelsize-subnum))
|
|
increment)))
|
|
(unless (> new-size 0) (signal 'font-too-small (list new-size)))
|
|
(aset xlfd-fields xlfd-regexp-pixelsize-subnum (number-to-string new-size)))
|
|
;; Set point size & width to "*", so frame width will adjust to new font size
|
|
(aset xlfd-fields xlfd-regexp-pointsize-subnum "*")
|
|
(aset xlfd-fields xlfd-regexp-avgwidth-subnum "*")
|
|
(x-compose-font-name xlfd-fields)))
|
|
|
|
;;;###autoload
|
|
(defun set-frame-alist-parameter-from-frame (alist parameter &optional frame)
|
|
"Set PARAMETER of frame alist ALIST to its current value in FRAME.
|
|
FRAME defaults to the selected frame. ALIST is a variable (symbol)
|
|
whose value is an alist of frame parameters."
|
|
(interactive
|
|
(let ((symb (or (and (fboundp 'symbol-nearest-point)
|
|
(symbol-nearest-point))
|
|
(symbolp (variable-at-point))))
|
|
(enable-recursive-minibuffers t))
|
|
(list (intern (completing-read
|
|
"Frame alist to change (variable): "
|
|
(frcmds-frame-alist-var-names) nil t nil nil 'default-frame-alist t))
|
|
(intern (completing-read "Parameter to set:" ; Lax completion - not just known parameters.
|
|
(frcmds-frame-parameter-names) nil nil nil nil 'left t))
|
|
(get-a-frame (read-frame "Frame to copy parameter value from: " nil t)))))
|
|
(unless (boundp alist)
|
|
(error "Not a defined Emacs variable: `%s'" alist))
|
|
(set alist (assq-delete-all parameter (copy-alist (eval alist))))
|
|
(set alist (cons (assq parameter (frame-parameters frame)) (eval alist)))
|
|
(tell-customize-var-has-changed alist))
|
|
|
|
;;; Standard Emacs 21+ function, defined here for Emacs 20.
|
|
(unless (fboundp 'assq-delete-all)
|
|
(defun assq-delete-all (key alist)
|
|
"Delete from ALIST all elements whose car is `eq' to KEY.
|
|
Return the modified alist.
|
|
Elements of ALIST that are not conses are ignored."
|
|
(while (and (consp (car alist)) (eq (car (car alist)) key)) (setq alist (cdr alist)))
|
|
(let ((tail alist)
|
|
tail-cdr)
|
|
(while (setq tail-cdr (cdr tail))
|
|
(if (and (consp (car tail-cdr)) (eq (car (car tail-cdr)) key))
|
|
(setcdr tail (cdr tail-cdr))
|
|
(setq tail tail-cdr))))
|
|
alist))
|
|
|
|
;; Define this to avoid requiring `cl.el' at runtime. Same as `icicle-set-difference'.
|
|
(defun frcmds-set-difference (list1 list2 &optional key)
|
|
"Combine LIST1 and LIST2 using a set-difference operation.
|
|
Optional arg KEY is a function used to extract the part of each list
|
|
item to compare.
|
|
|
|
The result list contains all items that appear in LIST1 but not LIST2.
|
|
This is non-destructive; it makes a copy of the data if necessary, to
|
|
avoid corrupting the original LIST1 and LIST2."
|
|
(if (or (null list1) (null list2))
|
|
list1
|
|
(let ((keyed-list2 (and key (mapcar key list2)))
|
|
(result ()))
|
|
(while list1
|
|
(unless (if key
|
|
(member (funcall key (car list1)) keyed-list2)
|
|
(member (car list1) list2))
|
|
(setq result (cons (car list1) result)))
|
|
(setq list1 (cdr list1)))
|
|
result)))
|
|
|
|
;;;###autoload
|
|
(defun set-all-frame-alist-parameters-from-frame (alist &optional frame really-all-p)
|
|
"Set frame parameters of ALIST to their current values in FRAME.
|
|
Unless optional argument REALLY-ALL-P (prefix arg) is non-nil, the
|
|
frame parameters in list `frame-parameters-to-exclude' are
|
|
excluded: they are not copied from FRAME to ALIST.
|
|
ALIST is a variable (symbol) whose value is an alist of frame parameters.
|
|
FRAME defaults to the selected frame."
|
|
(interactive
|
|
(let ((symb (or (and (fboundp 'symbol-nearest-point)
|
|
(symbol-nearest-point))
|
|
(symbolp (variable-at-point))))
|
|
(enable-recursive-minibuffers t))
|
|
(list (intern (completing-read
|
|
"Frame alist to change (variable): "
|
|
(frcmds-frame-alist-var-names) nil t nil nil 'default-frame-alist t))
|
|
(get-a-frame (read-frame "Frame to copy parameter values from: " nil t))
|
|
current-prefix-arg)))
|
|
(unless (boundp alist)
|
|
(error "Not a defined Emacs variable: `%s'" alist))
|
|
(set alist (frcmds-set-difference (frame-parameters frame)
|
|
(and (not really-all-p) frame-parameters-to-exclude)
|
|
#'car))
|
|
(tell-customize-var-has-changed alist))
|
|
|
|
(defun frcmds-frame-alist-var-names ()
|
|
"Return an alist of all variable names that end in \"frame-alist\".
|
|
The CAR of each list item is a string variable name.
|
|
The CDR is nil."
|
|
(let ((vars ()))
|
|
(mapatoms (lambda (sym) (and (boundp sym)
|
|
(setq sym (symbol-name sym))
|
|
(string-match "frame-alist$" sym)
|
|
(push (list sym) vars))))
|
|
vars))
|
|
|
|
(defun frcmds-frame-parameter-names ()
|
|
"Return an alist of all available frame-parameter names.
|
|
These are the documented, out-of-the-box (predefined) parameters.
|
|
The CAR of each list item is a string parameter name.
|
|
The CDR is nil."
|
|
(let ((params '(("auto-lower")
|
|
("auto-raise")
|
|
("background-color")
|
|
("background-mode")
|
|
("border-color")
|
|
("border-width")
|
|
("buffer-list")
|
|
("buffer-predicate")
|
|
("cursor-color")
|
|
("cursor-type")
|
|
("display")
|
|
("display-type")
|
|
("font")
|
|
("foreground-color")
|
|
("height")
|
|
("horizontal-scroll-bars")
|
|
("icon-left")
|
|
("icon-name")
|
|
("icon-top")
|
|
("icon-type")
|
|
("internal-border-width")
|
|
("left")
|
|
("menu-bar-lines")
|
|
("minibuffer")
|
|
("mouse-color")
|
|
("name")
|
|
("scroll-bar-width")
|
|
("title")
|
|
("top")
|
|
("unsplittable")
|
|
("user-position")
|
|
("vertical-scroll-bars")
|
|
("visibility")
|
|
("width")
|
|
("window-id"))))
|
|
(when (> emacs-major-version 20)
|
|
(setq params (nconc params '(("fullscreen")
|
|
("left-fringe")
|
|
("line-spacing")
|
|
("outer-window-id")
|
|
("right-fringe")
|
|
("screen-gamma")
|
|
("scroll-bar-background")
|
|
("scroll-bar-foreground")
|
|
("tool-bar-lines")
|
|
("tty-color-mode")
|
|
("wait-for-wm")))))
|
|
(when (> emacs-major-version 21)
|
|
(setq params (nconc params '(("user-size")))))
|
|
(when (> emacs-major-version 22)
|
|
(setq params (nconc params '(("alpha")
|
|
("display-environment-variable")
|
|
("font-backend")
|
|
("sticky")
|
|
("term-environment-variable")))))
|
|
(when (> emacs-major-version 23)
|
|
(setq params (nconc params '(("explicit-name")
|
|
("tool-bar-position")))))
|
|
params))
|
|
|
|
;;;###autoload
|
|
(defun tell-customize-var-has-changed (variable)
|
|
"Tell Customize to recognize that VARIABLE has been set (changed).
|
|
VARIABLE is a symbol that names a user option."
|
|
(interactive "vVariable: ")
|
|
(put variable 'customized-value (list (custom-quote (symbol-value variable)))))
|
|
|
|
;;;###autoload
|
|
(defun other-window-or-frame (arg)
|
|
"`other-frame', if `one-window-p'; otherwise, `other-window'."
|
|
(interactive "p")
|
|
(if (one-window-p) (other-frame arg) (other-window arg)))
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(provide 'frame-cmds)
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;; frame-cmds.el ends here
|