怎么使用原生 Android 版本的 Emacs
原生安卓版本的 Emacs,令安卓上的编程出现了很大的想象力。或许未来,在安卓上编程,会成为更方便的事情。以前,大家都绕道,用 Termux 安装 Emacs,终究是隔了一层。现在,gnu 本身发布了原生安卓版本的 Emacs,可以从 FDroid 分流下载。但是安装后,你会发现你遇到了重重难题。
不迷路
怎么让自己不迷路,知道自己在哪里?
# 查看 HOME 环境变量
M-: (getenv "HOME")
# 示例输出:"/data/data/org.gnu.emacs/files"
# 查看 ~ 的展开路径
M-: (expand-file-name "~")
# 示例输出:"/data/data/org.gnu.emacs/files"
# 查看工作起始目录
M-: default-directory
# 示例输出:"~/"
# M-: 运行命令
# 注意 default 别写成 defualt
# 评估变量不用括号
# 调用函数要括号
# 输出就算一样,也是各有所用,并不重复
# 可以单独设置
文件夹
怎么让安卓版本的 Emacs 访问你的安卓里的文件夹呢?
# 授予管理所有文件的权限
M-x android-request-storage-access
# 授予管理某个文件夹的权限
M-x android-request-directory-access
设置工作目录
(setq default-directory "/sdcard/Emacs/")
M-x setenv RET HOME RET /sdcard/Emacs
;; ======== 固定 new-file 目录为 /storage/sdcard0/Emacs/ =========
(defvar my-android-emacs-dir "/storage/sdcard0/Emacs/"
"默认用于新建文件和文件选择的目录(Android)。")
(setq-default default-directory (expand-file-name my-android-emacs-dir))
(add-hook 'emacs-startup-hook
(lambda ()
(let ((d (expand-file-name my-android-emacs-dir)))
(dolist (buf (buffer-list))
(with-current-buffer buf
(setq default-directory d))))))
(defun my-android-find-file-minibuffer-setup ()
(when (memq this-command '(find-file find-file-other-window))
(setq default-directory (expand-file-name my-android-emacs-dir))))
(add-hook 'minibuffer-setup-hook #'my-android-find-file-minibuffer-setup)
;; 强力:为 read-file-name 提供默认目录(覆盖大多数 UI 框架)
(defun my-android-read-file-name-wrapper (orig prompt &optional dir default-filename mustmatch initial)
(let ((dir (or dir (expand-file-name my-android-emacs-dir))))
(apply orig prompt dir default-filename mustmatch initial)))
(advice-add 'read-file-name :around #'my-android-read-file-name-wrapper)
;; 权限检查命令
(defun my-android-ensure-emacs-dir-access ()
(interactive)
(let ((d (expand-file-name my-android-emacs-dir)))
(if (file-directory-p d)
(message "目录可访问:%s" d)
(if (fboundp 'android-request-directory-access)
(message "请运行 M-x android-request-directory-access 并选择 %s 以授权。" d)
(message "目录不可访问:%s。请在 系统 -> 应用 -> 特殊应用访问 -> All files access 打开 Emacs 并授予“文件与媒体”权限。" d)))))
(global-set-key (kbd "C-c e d") #'my-android-ensure-emacs-dir-access)
;; ================================================================
字体
怎么让原生安卓版本的 Emacs 识别并正常显示中文字体,而不是 16 进制的字符块呢?
首先,去下载某个字体。
其次,把它复制进原生安卓版本的 Emacs 的工作目录下的 fonts 文件夹。
最后,在 ~/.emacs.d/init.el 里调用该字体。
(prefer-coding-system 'utf-8)
(set-default-coding-systems 'utf-8-unix)
(set-language-environment "UTF-8")
(set-face-attribute 'default nil :height 300)
(set-fontset-font t 'han "Noto Sans SC:pixelsize=50")
私有目录,一般是 /data/data/org.gnu.emacs/files/fonts。
M-x copy-file RET /sdcard/Downloads/NotoSansSC-Regular.ttf RET ~/fonts/NotoSansSC-Regular.ttf RET
debug
;; 把整段粘进 *scratch* 并评估,或放到 init.el
(require 'cl-lib)
(defun my-android-debug-info ()
"Gather Android/`default-directory' debug info and show it in *emacs-android-debug* buffer.
Run `M-x my-android-debug-info' and copy the buffer content to paste here."
(interactive)
(let* ((buf (get-buffer-create "*emacs-android-debug*"))
(dir (if (and (boundp 'my-android-emacs-dir) my-android-emacs-dir)
(expand-file-name my-android-emacs-dir)
"/storage/sdcard0/Emacs/")))
(with-current-buffer buf
(setq buffer-read-only nil)
(erase-buffer)
(insert (format "===== emacs-android-debug (timestamp: %s) =====\n\n"
(current-time-string)))
;; basic vars
(insert (format "my-android-emacs-dir bound: %s\n"
(if (boundp 'my-android-emacs-dir) "yes" "no")))
(when (boundp 'my-android-emacs-dir)
(insert (format "my-android-emacs-dir value: %s\n" my-android-emacs-dir)))
(insert (format "resolved my-android-emacs-dir: %s\n\n" dir))
;; env
(insert "=== Environment ===\n")
(dolist (v '("HOME" "ANDROID_DATA" "EXTERNAL_STORAGE" "EMACS"))
(insert (format "%-18s = %s\n" v (or (getenv v) ""))))
(insert "\n")
;; home/dirs and current
(insert "=== Home / Directories ===\n")
(insert (format "getenv \"HOME\": %s\n" (or (getenv "HOME") "")))
(insert (format "expand-file-name \"~\": %s\n" (expand-file-name "~")))
(insert (format "default-directory (current buffer): %s\n\n" default-directory))
;; hooks and minibuffer
(insert "=== Minibuffer & Hooks ===\n")
(insert (format "minibuffer-setup-hook length: %d\n" (length minibuffer-setup-hook)))
(insert (format "emacs-startup-hook length: %d\n" (length emacs-startup-hook)))
(insert (format "my-android-find-file-minibuffer-setup present in minibuffer-setup-hook?: %s\n"
(if (memq #'my-android-find-file-minibuffer-setup minibuffer-setup-hook) "yes" "no")))
(insert "\n")
;; buffers
(insert "=== Buffers and their default-directory ===\n")
(dolist (b (buffer-list))
(with-current-buffer b
(insert (format "%-30s -> %s\n" (buffer-name b) default-directory))))
(insert "\n")
;; features commonly related to file UIs
(insert "=== Common UI features loaded ===\n")
(dolist (f '(ivy helm vertico counsel selectrum))
(insert (format "%-10s : %s\n" f (if (featurep f) "loaded" "not loaded"))))
(insert "\n")
;; read-file-name info
(insert "=== read-file-name / advice info ===\n")
(insert (format "read-file-name-function: %s\n" read-file-name-function))
(insert (format "symbol property 'advice on read-file-name: %s\n"
(if (get 'read-file-name 'advice) "present" "none")))
(insert (format "symbol-function of read-file-name: %s\n\n" (symbol-function 'read-file-name)))
;; filesystem checks
(insert "=== Filesystem checks for resolved my-android-emacs-dir ===\n")
(insert (format "file-exists-p: %s\n" (file-exists-p dir)))
(insert (format "file-directory-p: %s\n" (file-directory-p dir)))
(insert "directory-files (attempt, up to first 20 entries):\n")
(condition-case err
(let ((lst (directory-files dir nil nil t)))
(insert (format " total entries: %d\n" (length lst)))
(dolist (e (cl-subseq lst 0 (min 20 (length lst))))
(insert (format " %s\n" e))))
(error (insert (format " ERROR listing directory: %s\n" err))))
(insert "\n")
;; android helpers
(insert "=== Android helper functions availability ===\n")
(insert (format "android-request-directory-access fboundp: %s\n"
(if (fboundp 'android-request-directory-access) "yes" "no")))
(insert (format "android-request-storage-access fboundp: %s\n"
(if (fboundp 'android-request-storage-access) "yes" "no")))
(insert "\n")
(insert "===== End of report =====\n")
(goto-char (point-min))
;; make read-only for safety and show it
(view-mode 1)
(display-buffer buf))))