;;; ztree-dir.el --- Text mode directory tree -*- lexical-binding: t; -*- ;; Copyright (C) 2013-2021 Free Software Foundation, Inc. ;; ;; Author: Alexey Veretennikov ;; ;; Created: 2013-11-11 ;; ;; Keywords: files tools ;; URL: https://github.com/fourier/ztree ;; Compatibility: GNU Emacs 24.x ;; ;; This file is part of GNU Emacs. ;; ;; GNU Emacs is free software: you can redistribute it and/or modify ;; it under the terms of the GNU General Public License as published by ;; the Free Software Foundation, either version 3 of the License, or ;; (at your option) any later version. ;; ;; GNU Emacs is distributed in the hope that it will be useful, ;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;; GNU General Public License for more details. ;; ;; You should have received a copy of the GNU General Public License ;; along with GNU Emacs. If not, see . ;; ;;; Commentary: ;; ;; Add the following to your .emacs file: ;; ;; (push (substitute-in-file-name "path-to-ztree-directory") load-path) ;; (require 'ztree-dir) ;; ;; Call the ztree interactive function: ;; M-x ztree-dir ;; Open/close directories with double-click, Enter or Space keys ;; ;;; Issues: ;; ;;; TODO: ;; 1) Add some file-handling and marking abilities ;; ;;; Code: (require 'ztree-util) (require 'ztree-view) (require 'ztree-protocol) (eval-when-compile (require 'cl-lib)) ;; ;; Constants ;; (defconst ztree-hidden-files-regexp "^\\." "Hidden files regexp. By default all filest starting with dot `.', including . and ..") ;; ;; Configurable variables ;; (defvar ztree-dir-move-focus nil "Defines if move focus to opened window on hard-action command (RETURN) on a file.") (defvar-local ztree-dir-filter-list (list ztree-hidden-files-regexp) "List of regexp file names to filter out. By default paths starting with dot (like .git) are ignored. One could add own filters in the following way: (setq-default ztree-dir-filter-list (cons \"^.*\\.pyc\" ztree-dir-filter-list)) ") (defvar-local ztree-dir-show-filtered-files nil "Show or not files from the filtered list.") ;; ;; Faces ;; (defface ztreep-header-face '((((type tty pc) (class color)) :foreground "lightblue" :weight bold) (((background dark)) (:height 1.2 :foreground "lightblue" :weight bold)) (t :height 1.2 :foreground "darkblue" :weight bold)) "*Face used for the header in Ztree buffer." :group 'Ztree :group 'font-lock-highlighting-faces) (defvar ztreep-header-face 'ztreep-header-face) (define-minor-mode ztreedir-mode "A minor mode for displaying the directory trees in text mode." ;; initial value nil ;; modeline name " Dir" ;; The minor mode keymap `( (,(kbd "H") . ztree-dir-toggle-show-filtered-files) (,(kbd ">") . ztree-dir-narrow-to-dir) (,(kbd "<") . ztree-dir-widen-to-parent) (,(kbd "d") . ztree-dir-open-dired-at-point))) ;; ;; File bindings to the directory tree control ;; (defun ztree-insert-buffer-header () "Insert the header to the ztree buffer." (let ((start (point))) (insert "Directory tree") (insert "\n") (insert "==============") (set-text-properties start (point) '(face ztreep-header-face))) (insert "\n")) (defun ztree-file-not-hidden (filename) "Determines if the file with FILENAME should be visible." (let ((name (ztree-file-short-name filename))) (and (not (or (string= name ".") (string= name ".."))) (or ztree-dir-show-filtered-files (not (cl-find-if (lambda (rx) (string-match rx name)) ztree-dir-filter-list)))))) (defun ztree-find-file (node hard) "Find the file at NODE. If HARD is non-nil, the file is opened in another window. Otherwise, the ztree window is used to find the file." (when (and (stringp node) (file-readable-p node)) (cond ((and hard ztree-dir-move-focus) (find-file-other-window node)) (hard (save-selected-window (find-file-other-window node))) (t (find-file node))))) (defun ztree-dir-toggle-show-filtered-files () "Toggle visibility of the filtered files." (interactive) (setq ztree-dir-show-filtered-files (not ztree-dir-show-filtered-files)) (message (concat (if ztree-dir-show-filtered-files "Show" "Hide") " filtered files")) (ztree-refresh-buffer)) (defun ztree-dir-directory-files (path) "Return the list of files/directories for the given PATH." ;; remove . and .. from the list of files to avoid infinite ;; recursion (cl-remove-if (lambda (x) (string-match-p "/\\.\\.?$" x)) (directory-files path 'full))) (defun ztree-dir-change-directory (node) "Change the start node to NODE and update current directory." (ztree-change-start-node node) (setq default-directory node)) (defun ztree-dir-narrow-to-dir () "Interactive command to narrow the current directory buffer. The buffer is narrowed to the directory under the cursor. If the cursor is on a file, the buffer is narrowed to the parent directory." (interactive) (let* ((line (line-number-at-pos)) (node (ztree-find-node-in-line line)) (parent (ztree-get-parent-for-line line))) (if (file-directory-p node) (ztree-dir-change-directory node) (when parent (ztree-dir-change-directory (ztree-find-node-in-line parent)))))) (defun ztree-dir-widen-to-parent () "Interactive command to widen the current directory buffer to parent. The buffer is widened to the parent of the directory of the current buffer. This allows to jump to the parent directory if this directory is one level up of the opened." (interactive) (let* ((node ztree-start-node) (parent (file-name-directory (directory-file-name node)))) (when parent (ztree-dir-change-directory parent)))) (defun ztree-dir-open-dired-at-point () "If the point is on a directory, open DIRED with this directory. Otherwise open DIRED with the parent directory" (interactive) (let* ((line (line-number-at-pos)) (node (ztree-find-node-in-line line)) (parent (ztree-get-parent-for-line line))) (cond ((and node (file-directory-p node)) (dired node)) (parent (dired (ztree-find-node-in-line parent)))))) ;; ;; Implementation of the ztree-protocol ;; (cl-defmethod ztree-node-visible-p ((file string)) "Return T if the NODE shall be visible." (ztree-file-not-hidden file)) (cl-defmethod ztree-node-short-name ((file string)) "Return the short name for a node." (ztree-file-short-name file)) (cl-defmethod ztree-node-expandable-p ((file string)) "Return T if the node is expandable." (file-directory-p file)) (cl-defmethod ztree-node-equal ((file1 string) (file2 string)) "Equality function for NODE1 and NODE2. Return T if nodes are equal" (string-equal file1 file2)) (cl-defmethod ztree-node-children ((file string)) "Return a list of NODE children" (ztree-dir-directory-files file)) (cl-defmethod ztree-node-action ((file string) hard) "Perform an action when the Return is pressed on a NODE." (ztree-find-file file hard)) ;; for ztree-node-side, ztree-node-face, ztree-node-left-short-name ;; and ztree-node-right-short-name use default implementations ;; ;; Entry point ;; ;;;###autoload (defun ztree-dir (path) "Create an interactive buffer with the directory tree of the PATH given." (interactive "DDirectory: ") (when (and (file-exists-p path) (file-directory-p path)) (let ((buf-name (concat "*Directory " path " tree*"))) (ztree-view buf-name #'ztree-insert-buffer-header (expand-file-name (substitute-in-file-name path)) #'ztreedir-mode nil)))) (provide 'ztree-dir) ;;; ztree-dir.el ends here