>>>>> Stefan Monnier <monn...@iro.umontreal.ca> writes:
>> [Incompatibility] >> 1. directory local variables >> Currently, directory local variables are prepared in a entry like >> (latex-mode (reftex-label-alist . (("equation" ?e "eq:%f-" "\\eqref{%s}") >> ("align" ?e "eq:%f-" "\\eqref{%s}") >> ("gather" ?e "eq:%f-" "\\eqref{%s}"))))) >> in .dir-locals.el. In this feature branch, mode name is `LaTeX-mode', >> thus this entry is ignored. >> >> The facility of major-mode-remap-list still does not handle this >> situation. >> >> We can announce this fact and ask users to rewrite their >> .dir-locals.el, but that can't be done if the user doesn't have write >> access for that particular .dir-locals.el. (Maybe we can ignore such >> corner cases?) > We should fix this for Emacs-30. > [ And then add some backward compatibility hack (presumably using > an advice for when AUCTeX is used on Emacs<30). ] I attach my tentative proposal for Emacs<30, using an advice.
>From 5447814da30ffe9e9e06095fd567e8169df5fb64 Mon Sep 17 00:00:00 2001 From: Ikumi Keita <ik...@ikumi.que.jp> Date: Sat, 21 Oct 2023 21:16:36 +0900 Subject: [PATCH] Add ugly hack to accept dir local vars with former mode names * tex.el (TeX-hack-former-dir-local-variables): New customize option. (TeX--hack-dir-locals-collect-variables): New function to advice `dir-locals-collect-variables'. When `major-mode' is one of major modes which have former name, pull in directory local variables declared under former name. Take into account the parent mode of the former mode as well. --- tex.el | 61 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 61 insertions(+) diff --git a/tex.el b/tex.el index d4fded73..72b39e9b 100644 --- a/tex.el +++ b/tex.el @@ -3875,6 +3875,67 @@ Run after mode hooks and file local variables application." ;;;###autoload (put 'TeX-mode 'auctex-function-definition (symbol-function 'TeX-mode)) +;; COMPATIBILITY for Emacs<30 +;; hack for directory local variables +(defcustom TeX-hack-former-dir-local-variables nil + "If non-nil, apply dir local vars under former mode names. +For example, directory local variables prepared for `latex-mode' +will be available in `LaTeX-mode' buffers. +It is highly discouraged to enable this option. Instead, please +rewrite mode names in .dir-locals.el whenever possible." + :group 'TeX-misc + :type 'boolean) + +(advice-add 'dir-locals-collect-variables :around + #'TeX--hack-dir-locals-collect-variables) +(defun TeX--hack-dir-locals-collect-variables (f &rest args) + ;; Call `dir-locals-collect-variables' once. + (let ((result (apply f args)) + entry former former-parent orig-former orig-former-parent) + (when (and TeX-hack-former-dir-local-variables + (setq entry + (cdr + (assq major-mode + ;; MAJOR-MODE FORMER FORMER-PARENT + '((plain-TeX-mode plain-tex-mode nil) + (LaTeX-mode latex-mode nil) + (docTeX-mode doctex-mode latex-mode) + (Texinfo-mode texinfo-mode nil) + (ConTeXt-mode context-mode nil) + (AmSTeX-mode ams-tex-mode nil) + (japanese-plain-TeX-mode japanese-plain-tex-mode plain-tex-mode) + (japanese-LaTeX-mode japanese-latex-mode latex-mode)))))) + (setq former (pop entry) ; former AUCTeX mode name, e.g. latex-mode + former-parent (pop entry)) ; former parent mode name + (setq orig-former-parent (get former 'derived-mode-parent)) + (put former 'derived-mode-parent former-parent) + ;; Very ugly hack to deceive `derived-mode-p'. Suppose + ;; - aaa-mode is an alias of bbb-mode + ;; - major-mode value is aaa-mode + ;; Then (derived-mode-p 'aaa-mode) returns nil on Emacs 28 and + ;; later, which makes `dir-locals-collect-variables' ignore a + ;; directory local variable entry for aaa-mode and look for an + ;; entry for bbb-mode instead. Hence we temporally make + ;; aaa-mode an alias of aaa-mode itself. :-( + (setq orig-former (symbol-function former)) + (fset former former) + (unwind-protect + (let ((major-mode former)) + ;; Append dir local vars with former mode names to return value. + ;; We need the next `setq' because `result' can be nil. + (setq result (nconc result (apply f args)))) + ;; Restore original state. + (put former 'derived-mode-parent orig-former-parent) + (fset former orig-former))) + result)) +;; The above treatment fails to pick the "correct" value when +;; .dir-locals.el assigns different value for the same variable +;; between latex-mode and japanese-latex-mode. In that case, the +;; value for latex-mode is picked up even in japanese-latex-mode +;; buffer, but I couldn't find a workaround. This is just a makeshift +;; for users who don't rewrite .dir-locals.el after all, thus let us +;; provide this as instrument with limitation. + ;;; Hilighting ;; FIXME: It's likely that `hilit-patterns-alist' is much obsolete. -- 2.42.0
>> (a-2) Now `TeX-add-local-master' adds entry of new mode names such as >> %%% Local Variables: >> %%% mode: LaTeX <-- not `latex' >> %%% End: > I don't like this very much: IMO the `mode:` cookie should really not specify > the "mode" to use but rather describe the file-type (Emacs could/should > aim to support "mime types" kind of names), and then which mode to use > for each file-type is decided based on the user's configuration. I understand your preferance, but I think such discrimination can't be much rigorous in practical situations. For example, ConTeXt mode doesn't have overlapped mode, so it's most natural to have %%% mode: ConTeXt ; In my opinion, having both %%% mode: context and additional entry in major-mode-remap-alist is spurious implementation. Then "%%% mode: ConTeXt" v.s. "%%% mode: latex" doesn't look nice to me. I prefer "%%% mode: ConTeXt" and "%%% mode: LaTeX", which look coherent. > Maybe in addition to that, we need to be able to say in `dir-locals.el` > that a setting applies only for exactly this major mode, and not > its derivatives, maybe with a syntax like: > ((= latex-mode) (reftex-label-alist > . (("equation" ?e "eq:%f-" "\\eqref{%s}") > ("align" ?e "eq:%f-" "\\eqref{%s}") > ("gather" ?e "eq:%f-" "\\eqref{%s}")))) Looks reasonable. Best regards, Ikumi Keita #StandWithUkraine #StopWarInUkraine