forked from sunrise-commander/sunrise-commander
-
Notifications
You must be signed in to change notification settings - Fork 1
/
sunrise-x-w32-addons.el
274 lines (228 loc) · 9.95 KB
/
sunrise-x-w32-addons.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
;;; sunrise-x-w32-addons --- MS-Windows-specific utilities for the Sunrise Commander File Manager
;; Copyright (C) 2011 José Alfredo Romero Latouche.
;; Author: José Alfredo Romero L. <escherdragon@gmail.com>
;; Štěpán Němec <stepnem@gmail.com>
;; Maintainer: José Alfredo Romero L. <escherdragon@gmail.com>
;; Created: 14 May 2011
;; Version: 1
;; RCS Version: $Rev: 374 $
;; Keywords: sunrise commander, w32, ms windows
;; URL: http://www.emacswiki.org/emacs/sunrise-x-w32-addons.el
;; Compatibility: GNU Emacs 23+
;; This file is *NOT* part of GNU Emacs.
;; 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 3 of the License, 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 de-
;; tails.
;; You should have received a copy of the GNU General Public License along with
;; this program. If not, see <http://www.gnu.org/licenses/>.
;;; Commentary:
;; This extension implements a listing of all Windows drives and special folders
;; inside the Sunrise Commander, as well as support for *.lnk shortcuts for all
;; file system operations inside Sunrise. Only standard Windows shortcuts are
;; currently supported.
;; *WARNING!* this extension is targeted at Emacs ports that run directly on the
;; Windows file system (like EmacsW32 and... are there any others?) if yours is
;; running on top of some simulation layer (like Cygwin does) that maps the file
;; system this code is of no use to you.
;; It was developed on EmacsW32 (version 23.1.50.1, patched) for Windows, during
;; the Hackergarten session at the GeeCon 2011 conference in Cracow. Thanks go
;; to Andreas Ames for beta-testing this code.
;;; Installation:
;; (These are generic installation instructions -- if you installed via ELPA you
;; don't need to follow them).
;; 1) Put this file somewhere in your Emacs `load-path'.
;; 2) Add a (require 'sunrise‐x‐w32-addons) expression to your .emacs file after
;; the (require 'sunrise‐commander) one.
;; 3) Evaluate the new expression, or reload your .emacs file, or restart Emacs.
;;; Usage:
;; * The "Windows Drives and Special Folders" pane can be accessed in two ways:
;; 1) by pressing "C-c w" anywhere in the file system, or
;; 2) by navigating "up" (J) from a top-level directory in any drive or network
;; share.
;; * Windows shortcuts are resolved automatically and, as long as a shortcut can
;; be resolved to an existing file, all operations (*INCLUDING DELETION!!!*) are
;; performed directly on that file. If you want to operate on shortcuts you must
;; first disable shortcut resolution by customizing the
;; `sr-w32-follow-shortcuts' flag and setting it to false.
;; * Virtual directories (i.e. directories containing a "target.lnk" shortcut to
;; another directory) are also dereferenced automatically. If you need to modify
;; the properties (Desktop.ini) of such a folder use the
;; `sr-w32-follow-shortcuts' flag as described above.
;; Enjoy ;-)
;;; Code:
(require 'sunrise-commander)
(defcustom sr-w32-follow-shortcuts t
"Controls the shortcut resolution mechanism.
When set, all operations executed on a Windows shortcut directly
affect the target of the shortcut."
:group 'sunrise
:type 'boolean)
(defvar sr-w32-local-map (let ((map (make-sparse-keymap)))
(set-keymap-parent map sr-virtual-mode-map)
(define-key map "s" 'ignore)
(define-key map "r" 'ignore)
(define-key map "l" 'ignore)
(define-key map "d" 'ignore)
map)
"Local keymap used inside the \"Windows Drives and Special Folders\" pane.")
(define-key sr-mode-map "\C-cw" 'sr-w32-virtual-entries)
(defadvice sr-dired-prev-subdir
(around sr-w32-advice-sr-dired-prev-subdir (&optional count))
"Bring up the drivers pane when navigating up from a topmost directory."
(if (sr-equal-dirs default-directory (expand-file-name ".."))
(sr-w32-virtual-entries)
ad-do-it))
(ad-activate 'sr-dired-prev-subdir)
(defadvice sr-find-file
(before sr-w32-advice-sr-find-file (filename &optional wildcards))
"Implement virtual folder resolution on Windows."
(when sr-w32-follow-shortcuts
(let ((info) (target (format "%s/target.lnk" filename)))
(if (file-readable-p target)
(setq info (sr-w32-resolve-lnk target)))
(if (< 0 (length info))
(setq filename info)))))
(ad-activate 'sr-find-file)
(defadvice dired-get-filename
(after sr-w32-advice-dired-get-filename (&optional LOCALP NO-ERROR))
"Implement standard Windows shortcut resolution."
(when sr-w32-follow-shortcuts
(let ((filename (or ad-return-value "")))
(if (string-match "\\.lnk\\'" filename)
(setq filename (sr-w32-resolve-lnk filename)))
(if (< 0 (length filename))
(setq ad-return-value filename)))))
(ad-activate 'dired-get-filename)
(defun sr-w32-goto-dir (dir)
"`sr-goto-dir' replacement for the \"Windows Drives and Special Folders\" pane."
(let ((sr-goto-dir-function nil))
(if (not (sr-equal-dirs dir default-directory))
(sr-goto-dir dir)
(sr-virtual-dismiss)
(sr-beginning-of-buffer))))
(defun sr-w32-resolve-lnk (link)
"Use the provided VBScript script to resolve standard Windows shortcuts."
(let* ((script (sr-w32-create-drivers-script))
(command (format "cscript /nologo \"%s\" /l \"%s\"" script link))
(info (shell-command-to-string command))
(info (replace-regexp-in-string "\\\\" "/" info))
(info (replace-regexp-in-string "\n" "" info)))
(if (file-exists-p info) info link)))
(defun sr-w32-virtual-entries(&optional ignore-auto no-confirm)
"Build a Sunrise pane containing all the Windows drives currently ready.
Also includes some selected special folders."
(interactive)
(let* ((script (sr-w32-create-drivers-script))
(command (format "cscript /nologo \"%s\"" script))
(info (car (read-from-string (sr-w32-execute-command command)))))
(sr-switch-to-clean-buffer
(generate-new-buffer-name "*W32 Drives & Folders*"))
(insert "Windows Drives and Special Folders: \n")
(insert "- \n") (sr-w32-entry-overlay (- (point) 3) (1- (point)))
(sr-w32-display-drives info)
(insert "- \n") (sr-w32-entry-overlay (- (point) 3) (1- (point)))
(sr-w32-display-folders info)
(sr-virtual-mode)
(sr-beginning-of-buffer)
(mapc 'make-local-variable '( revert-buffer-function
sr-goto-dir-function))
(setq revert-buffer-function 'sr-w32-virtual-entries
sr-goto-dir-function 'sr-w32-goto-dir)
(use-local-map sr-w32-local-map)))
(defun sr-w32-execute-command (command)
"Safely execute the given shell command and return its output as a string."
(condition-case description
(shell-command-to-string command)
(error
(progn
(sr-goto-dir "~")
(shell-command-to-string command)))))
(defun sr-w32-display-drives (info)
"Insert a list of all currently ready Windows drives into the current pane."
(let ((inhibit-read-only t))
(dolist (drive (cdr (assoc 'drives info)))
(insert (format "drwxrwxrwx 0 x x 0 0000-00-00 %s:/\n" drive))
(sr-w32-mask-drive))))
(defun sr-w32-mask-drive ()
"Remove unnecesary information from the listing of a drive."
(save-excursion
(forward-line -1)
(sr-w32-entry-overlay (point) (+ 30 (point)))))
(defun sr-w32-display-folders (info)
"Insert a list of Windows special folders into the current pane."
(dolist (folder (cdr (assoc 'folders info)))
(when (and (< 0 (length folder)) (file-directory-p folder))
(insert (format "drwxrwxrwx 0 x x 0 0000-00-00 %s\n" folder))
(sr-w32-mask-folder))))
(defun sr-w32-mask-folder ()
"Remove unnecesary details from the listing of a special folder."
(save-excursion
(forward-line -1)
(end-of-line)
(search-backward "/")
(sr-w32-entry-overlay (1+ (point)) (point-at-bol))))
(defun sr-w32-entry-overlay (start end)
"Create an invisible, tangible overlay from start to end."
(let ((overlay (make-overlay start end)))
(overlay-put overlay 'invisible t)
(overlay-put overlay 'before-string " ")))
(defun sr-w32-create-drivers-script ()
"Return the path of the VBScript file used for Windows-specific operations.
Creates it first if necessary."
(let* ((script-name "sunrise-x-w32-addons.vbs")
(script-dir (file-name-directory (symbol-file 'sunrise-x-w32-addons)))
(script-path (concat script-dir script-name)))
(unless (file-exists-p script-path)
(with-temp-buffer
(insert "Set objArgs = WScript.Arguments
If objArgs.Count = 0 Then
info()
Else
If objArgs(0) = \"/l\" Then
resolve_lnk(objArgs(1))
End If
End If
Function info()
Dim filesys, drv, drvcoll, w32info, shell, folder
Dim folders(7)
folders(0) = \"Desktop\"
folders(1) = \"Programs\"
folders(2) = \"MyDocuments\"
folders(3) = \"Favorites\"
folders(4) = \"PrintHood\"
folders(5) = \"NetHood\"
folders(6) = \"AllUsersDesktop\"
folders(7) = \"AllUsersPrograms\"
Set filesys = CreateObject(\"Scripting.FileSystemObject\")
Set drvcoll = filesys.Drives
w32info = \"((drives . (\"
For Each drv in drvcoll
If drv.IsReady Then
w32info = w32info & \"\"\"\" & drv.DriveLetter & \"\"\" \"
End If
Next
w32info = w32info & \")) (folders . (\"
Set shell = CreateObject(\"WScript.Shell\")
For Each folder in folders
folder = Replace(shell.SpecialFolders(folder), \"\\\", \"/\")
w32info = w32info & \"\"\"\" & folder & \"\"\" \"
Next
w32info = w32info & \")))\"
Wscript.Echo w32info
End Function
Function resolve_lnk(linkFile)
Set link = WScript.CreateObject(\"WScript.Shell\").CreateShortcut(linkFile)
WScript.Echo link.TargetPath
End Function")
(write-file script-path)))
script-path))
(defun sunrise-x-w32-addons-unload-function ()
(sr-ad-disable "^sr-w32-"))
(provide 'sunrise-x-w32-addons)
;;; sunrise-x-w32-addons.el ends here