home *** CD-ROM | disk | FTP | other *** search
/ Geek Gadgets 1 / ADE-1.bin / ade-dist / emacs-19.28-src.tgz / tar.out / fsf / emacs / lisp / ediff.el < prev    next >
Text File  |  1996-09-28  |  171KB  |  4,465 lines

  1. ;;; ediff.el --- a visual interface to diff & patch
  2. ;;; Copyright (C) 1994 Free Software Foundation, Inc.
  3.  
  4. ;; Author: Michael Kifer <kifer@cs.sunysb.edu>
  5. ;; Created: February 2, 1994
  6. ;; Version: 1.65c
  7. ;; Keywords: comparing, merging, patching, version control.
  8.  
  9. ;; This file is part of GNU Emacs.
  10.  
  11. ;; GNU Emacs is free software; you can redistribute it and/or modify
  12. ;; it under the terms of the GNU General Public License as published by
  13. ;; the Free Software Foundation; either version 2, or (at your option)
  14. ;; any later version.
  15.  
  16. ;; GNU Emacs is distributed in the hope that it will be useful,
  17. ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  19. ;; GNU General Public License for more details.
  20.  
  21. ;; You should have received a copy of the GNU General Public License
  22. ;; along with GNU Emacs; see the file COPYING.  If not, write to
  23. ;; the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  24.  
  25.  
  26. ;;; Commentary:
  27. ;;  ----------
  28.  
  29. ;; Never read those diff outputs again!
  30. ;; Apply patch selectively, like a pro!
  31.  
  32. ;; This package provides a convenient way of simultaneous browsing through
  33. ;; the differences between a pair of files or buffers.  The two files being
  34. ;; compared (file-A and file-B) are shown in separate windows (side by
  35. ;; side, one above the another, or in separate frames), and the differences
  36. ;; are highlighted as you step through them.  You can also copy difference
  37. ;; regions from one buffer to another (and recover old differences if you
  38. ;; change your mind).
  39.  
  40. ;; In addition, Ediff can apply a patch to a file and then let you step
  41. ;; though both files, the patched and the original one, simultaneously,
  42. ;; difference-by-difference.  You can even apply a patch right out of a
  43. ;; mail buffer, i.e., patches received by mail don't even have to be saved.
  44. ;; Since Ediff lets you copy differences between buffers, you can, in
  45. ;; effect, apply patches selectively (i.e., you can copy a difference
  46. ;; region from file.orig to file, thereby undoing any particular patch that
  47. ;; you don't like).
  48.  
  49. ;; This package builds upon the ideas borrowed from emerge.el.  It is still
  50. ;; using half a dozen of functions defined there. Several other Ediff's
  51. ;; functions are adaptations from emerge.el. Ediff requires, at least, 
  52. ;; Version 5 of emerge.el. This version comes with standard distributions
  53. ;; of Emacs and Lemacs.  Make sure you don't have some stray old copy of
  54. ;; Emerge on your load path.
  55.  
  56. ;; Ediff is complimentary to Emerge.  While Emerge is primarily intended
  57. ;; for merging of files, Ediff is by far superior for browsing through
  58. ;; files compared via diff and for patching files with patch.
  59. ;; Furthermore, Ediff is more convenient even for merging, when one of the
  60. ;; files is a designated output.  This situation arises while patching
  61. ;; files or when comparing an old version of a file with a newer version
  62. ;; (in such cases, it is often desirable to selectively revert some
  63. ;; portions of the new file to its old state).
  64.  
  65. ;; Ediff also supports version control via vc.el (in the standard
  66. ;; distribution of Emacs 19) and rcs.el. The latter is a package written by 
  67. ;; Sebastian Kremer <sk@thp.Uni-Koeln.DE>, which is available in
  68. ;;
  69. ;;         ftp.cs.buffalo.edu:pub/Emacs/rcs.tar.Z
  70. ;;         ftp.uni-koeln.de:/pub/gnu/emacs/rcs.tar.Z
  71. ;;
  72.  
  73.  
  74. ;; Window configuration:
  75. ;; ----------------------
  76.  
  77. ;; By default, Ediff sets things up in one frame, splitting it between a
  78. ;; small control window and the two windows for file-A and file-B.  The
  79. ;; split between these latter windows can be horizontal or vertical, which
  80. ;; can be changed interactively by hitting 's' while the cursor is in the
  81. ;; control window.
  82. ;;
  83. ;; In a multi-frame situation, Ediff would work as follows.  When it starts,
  84. ;; it will place the control window in the frame that was selected at the
  85. ;; time of the invocation.  If file-A or file-B is seen in one of the
  86. ;; frames, Ediff will leave it there.  If a file (A/B) is not visible in any
  87. ;; frame, Ediff will arrange that it will share a frame with the control
  88. ;; window. (If none of the files is visible, then both will share the
  89. ;; control window frame.) The same algorithm works when you hit 'c'
  90. ;; (ediff-recenter), 'p' (ediff-previous-difference), 'n', etc.
  91. ;;
  92. ;; Thus, you can compare files in one frame or in different frames.
  93. ;; The former is done by default, while the latter can be achieved by
  94. ;; arranging files A and B to be seen in different frames.  Ediff
  95. ;; respects these arrangements, automatically adapting itself to 
  96. ;; the multi-frame mode.
  97.  
  98.  
  99. ;; To those who like to go where noone has gone before:
  100.  
  101. ;;  Ediff lets the user run multiple sessions at once, i.e., you can invoke
  102. ;;  Ediff on different functions several times in a row, without exiting
  103. ;;  the previous Ediff sessions. Different sessions may even operate on the
  104. ;;  same pair of files.  So, in principle, it is possible to do, say,
  105. ;;  pairwise comparison of three (or more) different files.  Each session
  106. ;;  would have its own Ediff Control Panel and all the regarding a
  107. ;;  particular session is local to the associated control panel buffer.
  108. ;;  You can switch between sessions by suspending one session and then
  109. ;;  switching to another control panel. (Different control panel buffers
  110. ;;  are distinguished by a numerical suffix, e.g., Ediff Control Panel<3>.)
  111. ;;  Thus, if you would like to compare three files pairwise, you can do
  112. ;;  this by preparing three different frames, each with its own control
  113. ;;  window.  (This would require a very wide screen, and I never claimed
  114. ;;  that such 3-way comparison is very easy to do.)
  115. ;;
  116. ;; If you need to conduct multiple Ediff sessions on the same file, one
  117. ;; thing should be kept in mind: each time you invoke Ediff on a buffer that
  118. ;; already participates in another Ediff session, that buffer should not
  119. ;; have any ASCII Ediff flags in it. (Highlighting with faces is OK.)  If
  120. ;; flags are not removed, difference overlays won't be set correctly
  121. ;; for the second invocation of Ediff.  The simplest way to remove ASCII
  122. ;; flags from an Ediff buffer is to hit `h' and thus switch to highlighting
  123. ;; with faces (unhighlighting on a dumb terminal).
  124.  
  125. ;;;  Remote and Compressed Files
  126. ;;   ---------------------------
  127.  
  128. ;;  Ediff will work with remote files, compressed files, and, probably,
  129. ;;  with encrypted files. (I have only tested it with ange-ftp.el,
  130. ;;  jka-compr.el, and uncompress.el.) This means that you can compare files
  131. ;;  residing on another machine, or you can apply a patch to a file on
  132. ;;  another machine (even the patch itself can be a remote file!). This is
  133. ;;  provided that you access remote files via the usual find-file command
  134. ;;  using the syntax acceptable to ange-ftp (see Emacs Manual).
  135. ;;
  136. ;;  The files to be compared or patched can be compressed or be both
  137. ;;  compressed and remote. To be able to handle compressed files, you
  138. ;;  should use uncompress.el or jka-compr.el. Ediff is designed so that it
  139. ;;  will interface to these packages. When patching compressed or remote
  140. ;;  files, Ediff doesn't rename the source file into source-file-name.orig
  141. ;;  (unlike what `patch' would usually do). Instead, the source file
  142. ;;  retains its name and the result of applying the patch is placed in a
  143. ;;  temporary file that has the suffix `.patched'.  Generally, this applies
  144. ;;  to files that are handled using black magic, such as special file
  145. ;;  handlers (ange-ftp and some compression and encryption packages all
  146. ;;  use this method).
  147. ;;
  148. ;;  Regular files are treated by `patch' in the usual manner, i.e., the
  149. ;;  original is renamed into source-name.orig and the result of the patch
  150. ;;  is placed into the file source-name.
  151.  
  152.  
  153. ;;; Remarks: 
  154. ;;  -------
  155.  
  156. ;;  1. Ediff is heavily dependent on the new features of Emacs 19.
  157. ;;     It won't run under Emacs 18 at all.
  158. ;;  2. If running Lucid Emacs, Ediff requires at least version 19.9.
  159. ;;  3. The function vc-ediff requires the version of vc.el that comes with
  160. ;;     Emacs 19.22 and later.
  161.  
  162.  
  163. ;;; Installation and use:
  164. ;;  --------------------
  165.  
  166. ;; You can invoke Ediff interactively using the following functions:
  167. ;;
  168. ;;          ediff-buffers                        - compare buffers
  169. ;;          ediff                                - alias for ediff-files)
  170. ;;          ediff-files                        - compare files
  171. ;;          ediff-patch-file                     - patch file then compare
  172. ;;          epatch                               - alias for ediff-patch-file
  173. ;;          ediff-patch-buffer                     - patch buffer then compare
  174. ;;          epatch-buffer                        - alias for ediff-patch-buffer
  175. ;;          vc-ediff                              - compare buffer & version
  176. ;;                                                 using vc.el package
  177. ;;                                                 (Emacs 19.22 and up).
  178. ;;          rcs-ediff                             - same using rcs.el; rcs.el
  179. ;;                                                 is not part of the
  180. ;;                                                 standard Emacs distribution.
  181. ;;
  182. ;;
  183. ;; To use Ediff, put this in your .emacs file:
  184. ;;
  185. ;;  (autoload 'ediff-buffers "ediff" "Visual interface to diff" t)
  186. ;;  (autoload 'ediff  "ediff"  "Visual interface to diff" t)
  187. ;;  (autoload 'ediff-files "ediff" "Visual interface to diff" t)
  188. ;;  (autoload 'epatch  "ediff"  "Visual interface to patch" t)
  189. ;;  (autoload 'ediff-patch-file "ediff" "Visual interface to patch" t)
  190. ;;  (autoload 'ediff-patch-buffer "ediff" "Visual interface to patch" t)
  191. ;;  (autoload 'epatch-buffer "ediff" "Visual interface to patch" t)
  192. ;;  (autoload 'vc-ediff "ediff"
  193. ;;                      "Interface to diff & version control via vc.el" t) 
  194. ;;  (autoload 'rcs-ediff "ediff"
  195. ;;                       "Interface to diff & version control via rcs.el" t)
  196. ;;
  197. ;;
  198. ;; If you want Ediff to be loaded from the very beginning, you should have
  199. ;;
  200. ;;  (require 'ediff)
  201. ;;
  202. ;; in your .emacs file.  This way it is also easier to figure out changes
  203. ;; to the default Ediff setting, if such changes become necessary --- see
  204. ;; Customization.
  205. ;;
  206.  
  207. ;;; Compilation
  208. ;;  -----------
  209. ;;
  210. ;; When you byte-compile Ediff, you will get some warnings about functions
  211. ;; being undefined.  These can be safely ignored.
  212. ;;
  213. ;;   Important:
  214. ;;   =========
  215. ;;
  216. ;;    If you are using advice.el (directly or indirectly, via one of the
  217. ;;    other packages), Ediff may not compile properly.  In this case, you
  218. ;;    should do:
  219. ;;
  220. ;;          M-x ad-deactivate-all RET
  221. ;;
  222. ;;        M-x byte-compile-file RET ediff.el RET
  223. ;;
  224. ;;        M-x ad-activate-all RET
  225. ;;
  226. ;;    This precaution will not be needed starting with GNU Emacs 19.23 and
  227. ;;    Lucid Emacs 19.10, due to fixing a bug in advice.el.
  228.  
  229. ;;; Customization:
  230. ;;  -------------
  231.  
  232. ;; Hooks:
  233. ;; -----
  234. ;; If you don't like the default setting, you can change it through the
  235. ;; various variables and hooks.  In particular, the following hooks are
  236. ;; available: 
  237.  
  238. ;;        ediff-load-hooks
  239. ;;          ediff-before-setup-windows-hooks
  240. ;;          ediff-startup-hooks
  241. ;;          ediff-select-hooks
  242. ;;          ediff-unselect-hooks
  243. ;;          ediff-suspend-hooks
  244. ;;          ediff-quit-hooks
  245. ;;          ediff-prepare-buffer-hooks
  246.  
  247. ;; The hooks in ediff-load-hooks can be used to change defaults after Ediff
  248. ;; is loaded.
  249. ;; The hooks in ediff-before-setup-windows-hooks, ediff-suspend-hooks, and
  250. ;; ediff-quit-hooks can be used to save and then restore whatever window
  251. ;; configuration you want.  However, make sure you understand what you are
  252. ;; doing.  Many variables that drive Ediff are local to the different
  253. ;; Ediff Control Panel buffers.  Take a look at ediff-default-suspend-hook and
  254. ;; ediff-default-quit-hook to see what's involved.
  255. ;; The hooks in ediff-prepare-buffer-hooks are executed for each Ediff buffer
  256. ;; (A and B) right after these buffers are arranged.
  257. ;;
  258. ;;
  259. ;; Selective browsing: Control over stepping through difference regions
  260. ;; ----------------------------------------------- 
  261. ;;
  262. ;; Sometimes it is convenient to be able to step through only some
  263. ;; difference regions, those that satisfy certain conditions and to ignore
  264. ;; all others. The commands `#f' and `#h' let the user specify regular
  265. ;; expressions to control the way Ediff skips to the next or previous
  266. ;; difference. Typing `#f' lets one specify a pair of regular expressions,
  267. ;; regexp-A and regexp-B.
  268. ;; Ediff will then start stepping only through those difference regions where
  269. ;; the region in buffer A matches regexp-A and the region in buffer B
  270. ;; matches regexp-B.
  271. ;; Similarly, using `#h', one specifies expressions that match difference
  272. ;; regions to be ignored while stepping through the differences. That is, if
  273. ;; the buf A part matches regexp-A and the buf B part matches regexp B then
  274. ;; the region will be ignored by ediff-next-difference and
  275. ;; ediff-previous-difference commands.
  276. ;;
  277. ;;  Hitting `#f' and `#h' toggles this feature on/off.
  278. ;;
  279. ;; Note that selective browsing affects only ediff-next-difference and
  280. ;; ediff-previous-difference, i.e., the commands invoked by typing n/SPC
  281. ;; and p/DEL. You can still jump directly (using `j' or `ga/gb') to any
  282. ;; numbered  difference. Also, it should be understood, that #f and #h do
  283. ;; not change the position of the point in the buffers. The effect of these
  284. ;; commands is seen only when the user types `n' or `p', i.e., when
  285. ;; Ediff is told to jump to the next or previous difference.
  286. ;;
  287. ;; Users can supply their own functions that specify how Ediff should do
  288. ;; selective browsing. To change the default Ediff function, add a function to
  289. ;; ediff-load-hooks which will do the following assignments:
  290. ;;
  291. ;;      (fset ediff-hide-regexp-matches 'your-hide-function) 
  292. ;;      (fset ediff-focus-on-regexp-matches 'your-focus-function)
  293. ;;
  294. ;; Useful hints: To specify a regexp that matches everything, don't simply
  295. ;; type RET in response to a prompt. Typing RET tells Ediff to accept the
  296. ;; default value, which may not be what you want. Instead, one should enter
  297. ;; something like `^' or `$' --- which would match every line. 
  298. ;;
  299. ;; If the user doesn't remember if selective browsing is in effect and
  300. ;; which regexps are being used, the status command, `i', will supply
  301. ;; the requisite information.
  302. ;;
  303. ;; In addition to the ability to ignore regions that match regular
  304. ;; expressions, Ediff can be ordered to start skipping over certain
  305. ;; `inessential' regions. This is controlled by the variable
  306. ;;
  307. ;;      ediff-ignore-similar-regions
  308. ;;
  309. ;; which, if set to t, will cause Ediff to skip over difference regions
  310. ;; that has been found similar, i.e., where the only differences are those
  311. ;; in the white space and newlines.
  312. ;;
  313. ;; Note: In order for this feature to work, auto-refining of difference
  314. ;; regions must be on, since otherwise Ediff won't know if there are no
  315. ;; fine differences between regions. Under X, auto-refining is a default,
  316. ;; but it is nixed on a dumb terminal or in an Xterm window. Therefore, in
  317. ;; a non-windowing environment, the user must explicitly turn
  318. ;; auto-refining on (e.g., by typing `@').
  319. ;;
  320. ;; CAUTION: If many inessential regions appear in a row, Ediff may take a
  321. ;; long time to jump to the next region because it has to compute fine
  322. ;; differences of all intermediate regions.
  323. ;;
  324. ;;
  325. ;; Highlighting difference regions
  326. ;; -------------------------------
  327. ;; The second group of Ediff variables that could be changed, if you so
  328. ;; wish, is: 
  329. ;;
  330. ;;          ediff-before-flag-eol
  331. ;;          ediff-after-flag-eol
  332. ;;          ediff-before-flag-mol
  333. ;;          ediff-after-flag-mol
  334. ;;
  335. ;;          ediff-current-diff-face-A
  336. ;;          ediff-current-diff-face-B
  337. ;;          ediff-fine-diff-face-A
  338. ;;          ediff-fine-diff-face-B
  339. ;;          ediff-even-diff-face-A
  340. ;;          ediff-even-diff-face-B
  341. ;;          ediff-odd-diff-face-A
  342. ;;          ediff-odd-diff-face-B
  343. ;
  344. ;; The first four are ASCII strings that mark the beginning and the end of
  345. ;; the differences found in file-A and file-B. Ediff uses different flags
  346. ;; to highlight regions that begin/end at the beginning of a line or in a
  347. ;; middle of a line.
  348.  
  349. ;; The rest are the faces used to highlight text on X displays.  On X
  350. ;; displays, Ediff uses ediff-current-diff-face-A and
  351. ;; ediff-current-diff-face-B to highlight the current difference regions.
  352. ;; The faces ediff-fine-diff-face-A and ediff-fine-diff-face-B
  353. ;; are used to show the fine differences between the current differences
  354. ;; regions in buffer A and B.
  355. ;; Other (non-current) difference regions are displayed in alternating
  356. ;; faces: ediff-even/odd-diff-face-A/B.   The odd and the even
  357. ;; faces are actually identical on monochrome displays, because it is
  358. ;; rather poor in what you can do on such a display. So, I chose to use
  359. ;; italics to highlight other differences. Any ideas would be welcome.
  360. ;; There are two ways to change the default setting for highlighting faces:
  361. ;; either change the variables, as in
  362. ;;
  363. ;; (setq ediff-current-diff-face-A 'bold-italic)
  364. ;;
  365. ;; or
  366. ;;
  367. ;; (setq ediff-current-diff-face-A
  368. ;;       (copy-face 'bold-italic 'ediff-current-diff-face-A))
  369. ;;
  370. ;; or by selectively modifying the defaults:
  371. ;;
  372. ;; (add-hook 'ediff-load-hooks
  373. ;;   (function (lambda () 
  374. ;;                (set-face-foreground ediff-current-diff-face-B "blue")
  375. ;;                (set-face-background ediff-current-diff-face-B "red")
  376. ;;                (make-face-italic ediff-current-diff-face-B))))
  377. ;;
  378. ;; You may also want to take a look at how the above faces are defined in
  379. ;; Ediff. 
  380. ;;
  381. ;; Note: it is not recommended to use `internal-get-face' (or `get-face' in
  382. ;;       Lucid) when defining faces for Ediff, since this may cause
  383. ;;       problems when there are several frames with different font sizes.
  384. ;;       Instead, use copy-face or set/make-face-* as shown above.
  385. ;;
  386. ;; The last group of variables in this group,
  387. ;;
  388. ;;          ediff-want-faces
  389. ;;          ediff-highlight-all-diffs
  390. ;;          ediff-want-default-menus
  391. ;;
  392. ;; indicate whether---on a window system---you want differences to be
  393. ;; marked using ASCII strings (like on a dumb terminal) or using colors and
  394. ;; highlighting. If ediff-want-faces is t, then highlighting with faces is
  395. ;; used. Normally, Ediff highlights all differences, but the selected
  396. ;; difference is highlighted more visibly. You can cycle through various
  397. ;; modes of highlighting by hitting `h'. By default, Ediff starts in the
  398. ;; mode where all difference regions are highlighted. If you prefer to
  399. ;; start in the mode where unselected differences are not highlighted, you
  400. ;; should set ediff-highlight-all-diffs to nil. 
  401. ;; You will still be able to turn on highlighting of all differences by
  402. ;; hitting `h'.
  403. ;; The variable `ediff-want-default-menus', if true, will cause Ediff to
  404. ;; set up a pair of menues in the menu bar, so you can invoke it from there.
  405. ;; If you don't like the look of the default menus, set this variable to
  406. ;; nil and design your own menus.
  407. ;;
  408. ;; If you plan on changing these variables, they must be set
  409. ;; BEFORE ediff.el is loaded. 
  410. ;;
  411. ;; Note: Ediff lets you switch between the two types of highlighting.  That
  412. ;; is you can switch, interactively, from highlighting using faces to
  413. ;; highlighting using ASCII flags, and back.  Of course, toggling has
  414. ;; effect only on a window system.  On a dumb terminal or in an xterm
  415. ;; window, the only available option is highlighting with ASCII flags.
  416. ;;
  417. ;; Refining difference regions
  418. ;; ---------------------------
  419. ;; Ediff has variables that control the way fine differences are
  420. ;; highlighted. This feature lets the user highlight the exact words that 
  421. ;; make the difference regions in buffer A and B different. This process
  422. ;; ignores spaces, tabs, and newlines.
  423. ;;
  424. ;;          ediff-auto-refine
  425. ;;          ediff-auto-refine-limit
  426. ;;
  427. ;; By default, `ediff-auto-refine' is `'on', which means that fine differences
  428. ;; within regions will be highlighted automatically. On a slow system, this
  429. ;; feature may be undesirable. In any case, the user can always toggle
  430. ;; auto-refining on/off/nix by hitting `@'. When auto-refining is off, fine
  431. ;; differences will be shown only for regions for which these differences
  432. ;; have been computed and saved before. If auto-refining is nixed, fine
  433. ;; differences will not be shown at all. Hitting `*' will compute and
  434. ;; display fine differences for the current difference region regardless of
  435. ;; whether auto-refining is on, off, or nixed. 
  436. ;; If auto-refining is on, the variable `ediff-auto-refine-limit' limits
  437. ;; the size of the regions to be auto-refined. This variable guards against
  438. ;; possible slow-down that may be caused by an extraordinary large
  439. ;; difference region. However, the user can always force refining by
  440. ;; hitting `*'.
  441. ;;
  442. ;;          ediff-fine-diff-program
  443. ;;          ediff-fine-diff-options
  444. ;;          ediff-forward-word-function
  445. ;;
  446. ;; These variables let the user control how fine differences are computed.
  447. ;; `ediff-fine-diff-program' is diff, by default. However, you can use
  448. ;; any function as long as it produces output consistent with diff.
  449. ;; `ediff-forward-word-function' is a lisp function that determines how the
  450. ;; current difference region is split into words. (Fine diferences are
  451. ;; computed by first splitting the current difference region into words and
  452. ;; then passing this along to `ediff-fine-diff-program'. For the default
  453. ;; wordify function, `ediff-forward-word', a word is a string consisting of
  454. ;; letters, `-', or `_', or a string consisting of symbols that are neither
  455. ;; space, nor a letter.)
  456. ;;
  457. ;; Patch and diff programs
  458. ;; -----------------------
  459. ;; The next group of variables determines the programs to be used for
  460. ;; applying patches and for computing the main difference regions (not the
  461. ;; fine difference regions):
  462. ;;
  463. ;;          ediff-patch-program
  464. ;;          ediff-patch-options
  465. ;;          ediff-diff-program
  466. ;;          ediff-diff-options
  467. ;;
  468. ;; These specify the functions that produce differences and do patching.
  469. ;; The *-options variables specify which options to pass to these programs.
  470. ;; It is unlikely that you would want to change these.  One possible
  471. ;; exception is when you may want to generate differences with context
  472. ;; lines in order to send a patch file through email.  Then, you might want
  473. ;; to set ediff-diff-options to '-c'. Sometimes, you may also want to tell
  474. ;; diff to ignore spaces and such. Use the option '-w' for that. Diff
  475. ;; has several other useful options (type 'man diff' to find out).
  476. ;;
  477. ;; The output from diff is found in *ediff-diff* buffer, which you can save.
  478. ;; However, using Ediff for producing a diff output makes sense only if you
  479. ;; also intend to use Ediff to browse through the diff'ed files before
  480. ;; sending the patch.  This is because diff.el, which also comes with
  481. ;; Emacs, is much faster in yielding the output of diff  (Ediff is a big
  482. ;; gun, if used for this simple purpose).
  483. ;;
  484. ;; Mode line
  485. ;; ---------
  486. ;;
  487. ;; When Ediff is running, the mode line of Ediff Control Panel buffer
  488. ;; displays the current difference being displayed and the total number of
  489. ;; difference regions in the two files. 
  490. ;;
  491. ;; The mode line of the buffers being compared displays the type of the
  492. ;; buffer (`A:' or `B:') and (usually) the file name. Ediff is trying to be
  493. ;; intelligent in choosing mode line buffer identification. In particular,
  494. ;; it works well with uniquify.el and mode-line.el packages (which improve
  495. ;; on the default way in which Emacs displays buffer identification).
  496. ;; If you don't like the way Ediff identifies its buffers, there is always
  497. ;; ediff-prepare-buffer-hooks, which can be used to modify the mode line.
  498. ;;
  499. ;; Miscellaneous
  500. ;; -------------
  501. ;; The last set of variables that can be modified is
  502. ;;
  503. ;;          ediff-split-window-function
  504. ;;          ediff-use-last-dir
  505. ;;          ediff-no-help-in-control-buffer
  506. ;;
  507. ;; ediff-split-window-function controls the way you want the window be
  508. ;; split between file-A and file-B.  It defaults to vertical split, but you
  509. ;; can set it to 'split-window-horizontally, if you want.  Ediff lets you
  510. ;; toggle the way windows are split, so you can try different settings
  511. ;; interactively.  Note: if file-A and file-B are in different frames,
  512. ;; windows are not split, regardless of the value
  513. ;; ediff-split-window-function.  Instead, other windows on these frames are
  514. ;; deleted and Ediff starts displaying file-A and file-B using these two
  515. ;; frames, one file per frame.  You can then switch to one-frame mode
  516. ;; simply by hiding the file-A/B buffer that is displayed on a frame other
  517. ;; than the control-window frame.
  518. ;;
  519. ;; Note that if Ediff sees that the two buffers it compares are residing in
  520. ;; separate frames, it assumes that the user wants them to be so displayed
  521. ;; and stops splitting windows.  Instead, it will arrange each buffer to
  522. ;; occupy its own frame (possibly shared with Ediff's help window).
  523. ;;
  524. ;; The variable ediff-use-last-dir controls the way Ediff presents the
  525. ;; default directory when it prompts the user for files to compare.  If nil,
  526. ;; Ediff will use the default directory of the current buffer when it
  527. ;; prompts the user for file names.  Otherwise, it will use the
  528. ;; directories it had previously used for file-A and file-B. 
  529. ;;
  530. ;; The ediff-no-help-in-control-buffer, if set to t, makes C-h behave like
  531. ;; the DEL key, i.e., it will move you back to the previous difference
  532. ;; rather than invoking help.  This is useful when, in an xterm window or on
  533. ;; a dumb terminal, the Backspace key is bound to C-h and is positioned
  534. ;; more conveniently than the DEL key.
  535.  
  536.  
  537. ;;; Commands
  538. ;;  --------
  539.  
  540. ;; All Ediff commands are displayed in a help window, unless you hit '?' to
  541. ;; shrink it to just one line.  You can redisplay the help window by hitting
  542. ;; '?' again.
  543. ;;
  544. ;; Many Ediff commands take numeric prefix arguments.  For instance, if you
  545. ;; hit a number, N, and then 'j' (ediff-jump-to-difference), Ediff will
  546. ;; take you to Nth difference.  Hitting a number, N, and then 'ab'
  547. ;; (ediff-diff-to-diff) will copy Nth difference from buffer A to buffer B.
  548. ;; Hitting 'ba' does copying in the other direction.
  549. ;; Likewise, a number, N, followed by 'ra' will restore the Nth difference
  550. ;; region in buffer A (if it was previously saved as a result of copying
  551. ;; from B to A). 
  552. ;;
  553. ;; Without the prefix argument, all commands operate on the current
  554. ;; difference region.
  555. ;;
  556. ;; The total number of differences and the current difference number are
  557. ;; always displayed in the mode line of the control window. 
  558. ;;
  559. ;; If, after making changes to buffers A and B, you decide to save them, it
  560. ;; is recommended to use `ediff-save-buffer', which is bound to `wa' and
  561. ;; `wb' (`wa will save buffer A and `wb' saves buffer B).
  562. ;;
  563. ;; Typing `wf' will also save the diff output in a file. 
  564.  
  565. ;;; Display Modes
  566. ;;  -------------
  567.  
  568. ;; Ediff can display files in one frame, stacked side-by-side or one on top
  569. ;; of another; or it can display the files in different frames.  When you
  570. ;; start Ediff, it assumes a 1-frame mode.  You can toggle the side-by-side
  571. ;; and one-on-top-of-another displays by simply hitting 's'.
  572. ;;
  573. ;; Ediff switches to the multi-frame mode when:
  574. ;;
  575. ;;  1. file-A and file-B are in different frames (you have to put them into
  576. ;;     different frames manually); or
  577. ;;  2. Ediff Control Panel is visible in one frame and one other file (A
  578. ;;     or B) is visible in another frame.  If, say, fileA is visible in a
  579. ;;     different frame than Ediff Control Panel, fileB doesn't have to be
  580. ;;     visible.  If it is, Ediff will continue displaying fileB in the frame
  581. ;;     where it was visible before.  If it isn't then Ediff will arrange for
  582. ;;     fileB to share a frame with Ediff Control Panel.
  583. ;;
  584. ;;  If all three buffers are in separate frames, Ediff will switch to a
  585. ;;  3-frame mode.  If Ediff buffers are currently visible only in two
  586. ;;  frames, Ediff will work in a 2-frame mode.  In this mode, one of the
  587. ;;  frames will be shared by Ediff Control Panel and file-A or file-B
  588. ;;  (whichever is appropriate).
  589.  
  590.  
  591. ;;; Bugs:
  592. ;;  ----
  593.  
  594. ;;  1. The undo command doesn't restore deleted regions well. That is, if
  595. ;;  you delete all characters in a difference region and then invoke
  596. ;;  `undo', the reinserted text will most likely be reinserted outside of
  597. ;;  what Ediff thinks is the current difference region. (This problem
  598. ;;  doesn't seem to exist with Lucid Emacs.)
  599. ;;
  600. ;;  If at any point you feel that difference regions are no longer correct,
  601. ;;  you can hit '!' to recompute the differences.
  602.  
  603. ;;  2. Emacs 19.xx, where xx < 23, has several bugs related to overlays and
  604. ;;  faces. Somethimes, these may cause highlighting of the refinements or
  605. ;;  of the unselected differences to disappear. Hitting `!' will bring them
  606. ;;  back.  In version 19.23 and later, these problems no longer occur.
  607.  
  608. ;;  3. On a monochrome display, the repertoire of faces with which to
  609. ;;  highlight fine differences is limited. By default, Ediff is using
  610. ;;  underlining. However, if the region is already underlied by some other
  611. ;;  overlays, there is no simple way to temporarily remove that residual
  612. ;;  underlining. This problem occurs when a buffer is highlighted with
  613. ;;  hilit19.el or font-lock.el packages. If this residual highlighting gets
  614. ;;  in the way, you can do the following. Both font-lock.el and hilit19.el
  615. ;;  provide commands for unhighlighting buffers. You can either place these
  616. ;;  commands in `ediff-prepare-buffer-hooks' (which will unhighlight every
  617. ;;  buffer used by Ediff) or you can execute them interactively, at any time
  618. ;;  and on any buffer.
  619.  
  620. ;;  4. In Lucid Emacs (statically linked with Motif libraries), emerge.el
  621. ;;  and hence ediff.el won't start, unless you set (setq scrollbar-width 0).
  622. ;;  This is a Motif-related bug, I was told.
  623.  
  624.  
  625. ;;; Change Log:
  626. ;;  ----------
  627.  
  628. ;; Thu Feb  3, 1994 
  629.  
  630. ;;     Added ediff-read-file-name, which is a stub that takes care of Lemacs
  631. ;;     versions of Emerge. (Thanks to Alastair Burt <burt@dfki.uni-kl.de>.)
  632. ;;
  633. ;;     Fixed a bug in ediff-setup-windows that caused control window to
  634. ;;     appear in a wrong place when split-window-keep-point is nil
  635. ;;     (Thanks to Kevin Broadey <KevinB@bartley.demon.co.uk>.)
  636. ;;
  637. ;;     Added mechanism for using faces instead of before/after flags.  This
  638. ;;     looks much better on an X display, especially on a color one.
  639. ;;     (Thanks to Boris Goldowsky <boris@cs.rochester.edu> for the code
  640. ;;     that led to ediff-highlight-diff.
  641. ;;     Also, thanks to Kevin Esler <esler@ch.hp.com> for suggestions
  642. ;;     regarding highlighting differences on X displays.)
  643. ;;
  644. ;;     Added functions to apply patches.
  645. ;;     (Thanks to Kevin Broadey <KevinB@bartley.demon.co.uk> for this
  646. ;;     suggestion.)
  647.  
  648. ;; Fri Feb  4, 1994 
  649.  
  650. ;;     Added mechanism for toggling vertical/horizontal window split.
  651. ;;     (Inspired by a suggestion from Allan Gottlieb
  652. ;;     <gottlieb@allan.ultra.nyu.edu> -- thanks.)
  653. ;;
  654. ;;     Added mechanism for toggling between highlighting using faces and
  655. ;;     highlighting using ASCII flags.
  656. ;;
  657. ;;     Fixed a problem with undo.  Now, Ediff has smartened up and doesn't
  658. ;;     keep undo info on ASCII flags inserted in buffer-A and buffer-B.
  659. ;;     So, if you edit the files while browsing through them, undo behaves
  660. ;;     as you would expect, i.e., faces/flags don't get in the way.
  661.  
  662. ;; Sun Feb  6, 1994 
  663.  
  664. ;;     Added horizontal scrolling.  Added ediff-position-region to ensure
  665. ;;     that difference regions in buffer-A and buffer-B are aligned with
  666. ;;     each other.  Disabled ediff-toggle-split when buffers are displayed
  667. ;;     in different frames.
  668.  
  669. ;; Mon Feb  7, 1994
  670.  
  671. ;;     Added toggle-window help (Suggested by Boris Goldowsky
  672. ;;     <boris@cs.rochester.edu>.)
  673. ;;     Added functions to copy differences from one buffer to another and to
  674. ;;     recover old differences.
  675. ;;     Added prefix arguments to ediff-next-difference and
  676. ;;     ediff-previous-difference.
  677.  
  678. ;; Tue Feb  8, 1994
  679.  
  680. ;;     Replaced text properties with overlays.  Fixed ediff-setup-windows.
  681. ;;     Added ediff-save-buffer to local-write-file-hooks to prevent user
  682. ;;     from saving corrupted states. (Thanks to <boris@cs.rochester.edu>
  683. ;;     for suggestion.)  Instead, Ediff now has a pair of functions for
  684. ;;     safe saving of buffers. 
  685. ;;     Changed ediff-read-file-name to be more intuitive on ediff-files.
  686. ;;     Added ediff-prepare-buffer-hooks. (Thanks to Kevin Esler
  687. ;;     <esler@ch.hp.com> for the idea.)
  688.  
  689. ;; Wed Feb  9, 1994
  690.  
  691. ;;     Cleanups in ediff-patch-file.  Protected ediff-copy-diff against
  692. ;;     a bug that Emacs has in kill-region.
  693.  
  694. ;; Thu Feb 10, 1994
  695.  
  696. ;;     Added support for Lemacs. (Thanks to Alastair Burt
  697. ;;     <burt@dfki.uni-kl.de> for coercing Ediff into working under Lemacs.)
  698. ;;     Added ediff-kill-buffer-carefully and other suggestions by Boris
  699. ;;     Goldowsky <boris@cs.rochester.edu>.
  700. ;;     Refined the protection against interference with highlighting caused
  701. ;;     by Hilit19.  Added the variable ediff-third-party-highlighting.
  702. ;;     Added mechanisn for unhighlighting regions highlighted with Hilit19
  703. ;;     before hightlighting them with Ediff's overlays. (And for
  704. ;;     rehighlighting them with Hilit19, when the current difference moves on.)
  705.  
  706. ;; Sun Feb 13, 1994
  707.  
  708. ;;     Added ediff-place-flags-in-buffer and ediff-remote-exit, which are
  709. ;;     modifications of Emerge's similar functions.  The difference is that
  710. ;;     in Ediff they make ediff-before-flag and ediff-after-flag into
  711. ;;     read-only regions, so the user can't change them by mistake.
  712. ;;
  713. ;;     Adopted a suggestion by Boris Goldowsky <boris@cs.rochester.edu>
  714. ;;     that led to a more elegant treatment of faces.
  715. ;;
  716. ;;     Added protection against interference with Font-Lock highlighting
  717. ;;     similar to that of Hilit19's protection.
  718.  
  719. ;; Tue Feb 15, 1994
  720.  
  721. ;;     Deleted spurious (auto-save-mode 1) in ediff-control-buffer, which
  722. ;;     was causing this buffer to be auto-saved for no good reason.
  723. ;;     Added read-only protection to ediff-before/after-flags in Lemacs.
  724. ;;     (Thanks to Alastair Burt <burt@dfki.uni-kl.de> for help in testing.)
  725.  
  726. ;; Wed Feb 16, 1994
  727.  
  728. ;;     Further fixes in the Lemacs part.  Changed highlighted region in
  729. ;;     ediff-highlight-diff so that an extra character will be highlighted
  730. ;;     only if a difference is empty (thereby allowing the user to see where an
  731. ;;     insertion or a deletion has taken place).
  732. ;;
  733. ;;     Simplified interaction with other highlighting packages by giving
  734. ;;     Ediff overlays the highest priority. (Taking a cue from
  735. ;;     ediff-highlight-diff-lemacs written by Alastair Burt
  736. ;;     <burt@dfki.uni-kl.de>.) Zapped ediff-third-party-highlighting
  737. ;;     variable and hooks that were previously used to
  738. ;;     unhighlight/rehighlight buffers when hilit19/font-lock are on.
  739.  
  740. ;; Fri Feb 18, 1994
  741.  
  742. ;;     Added a bit more sophistication to ediff-read-file-name.  Now,
  743. ;;     ediff-files remembers both, the file-A and the file-B directories.
  744. ;;     They are offered as defaults when ediff-use-last-dir is set to t.
  745.  
  746. ;; Fri Feb 22, 1994
  747.  
  748. ;;     Added ediff-before-change-guard to remove ASCII highlighting when
  749. ;;     the user attempts to change buffer-A/B.  This is needed because
  750. ;;     otherwise the undo info may become screwed up in those buffers.
  751. ;;     Hitting 'h' (ediff-toggle-hilit) on a dumb terminal will toggle
  752. ;;     between ASCII highlighting and no highlighting.
  753.  
  754. ;; Fri Feb 24, 1994
  755.  
  756. ;;     Fixed problems with multiple Ediff sessions running simultaneously.
  757.  
  758. ;; Tue Mar 1, 1994
  759.  
  760. ;;     Added vc-ediff, the Ediff interface to vc.el. (Thanks to Eric
  761. ;;     Freudenthal <freudent@jan.ultra.nyu.edu> for contributing this
  762. ;;     function.) 
  763.  
  764. ;; Sun Mar 6, 1994
  765.  
  766. ;;     Added rcs-ediff, an Ediff interface to RCS via rcs.el. (Thanks to
  767. ;;     Alastair Burt  <burt@dfki.uni-kl.de>.)
  768. ;;     Some minor improvements.
  769.  
  770. ;; Tue March 15, 1994
  771.  
  772. ;;     Fixed a buglet in defining ediff-current-diff-face-A/B.
  773. ;;     (Thanks to Job Ganzevoort  <Job.Ganzevoort@cwi.nl>.) 
  774.  
  775. ;; Tue March 22, 1994
  776.  
  777. ;;     Fixed a bug with ediffing narrowed buffers, reported by Kevin
  778. ;;     Broadey <KevinB@bartley.demon.co.uk>.
  779. ;;     Made Ediff to work with files that have incomplete last line.
  780. ;;     Made Ediff execute diff and patch using Bourne Shell, which
  781. ;;     should eliminate problems with $prompt that some people had.
  782.  
  783. ;; Thu March 24, 1994
  784.  
  785. ;;     Achieved quadratic speedup in the size of the file by replacing the
  786. ;;     slow goto-line by forward-line.  Ediff is now *much* faster than
  787. ;;     Emerge on large files.  Converted demarkation of difference regions
  788. ;;     from markers to overlays.  This will later allow us to highlight all
  789. ;;     diffs, not just the current one.
  790.  
  791. ;; Wed March 30, 1994
  792.  
  793. ;;     Under X, Ediff now highlights all differences in dim colors and the
  794. ;;     current difference in bright colors. Improved Lucid Emacs support.
  795.  
  796. ;; Thu March 31, 1994
  797.  
  798. ;;     Changed toggle hilit to cycle through 3 states: highlighting all
  799. ;;     diffs, highlighting only the current diff, and highlighting using
  800. ;;     ASCII flags.
  801. ;;     Added support for difference regions that are not full lines.
  802.  
  803. ;; Fri April 1, 1994
  804.  
  805. ;;     Fixed bugs related to writing buffers A and B.
  806. ;;     Added commands 'ga', 'gb' to jump directly to the closest diff in
  807. ;;     buffer A and B, respectively.
  808.  
  809. ;; Fri April 11, 1994
  810.  
  811. ;;     Added `ediff-recompute-diffs', a function that lets the user recompute
  812. ;;     difference regions after extensive editing done to buffers A and B
  813. ;;     (bound to `!').
  814.  
  815. ;; Wed April 13, 1994
  816.  
  817. ;;     Added the new feature: refining the current difference region.
  818. ;;     This would highlight the precise differences between the regions in
  819. ;;     buffer A and B. (A way to implement this was suggested by Boris
  820. ;;     Goldowsky <boris@cs.rochester.edu>.)
  821. ;;
  822. ;;     Fixed Ediff to be immune to several different versions of rcs.el
  823. ;;     that are currently in distribution.
  824.  
  825. ;; Thu April 14, 1994
  826.  
  827. ;;     Ediff now respects X resources for the faces it uses. It no longer
  828. ;;     barks when the colormap has no colors it is using; or when face
  829. ;;     fonts can't be italicized, etc.
  830.  
  831. ;; Fri April 15, 1994
  832.  
  833. ;;     Changed `ediff-setup-windows' to minimize the need to delete and
  834. ;;     create windows. Now jumps faster from diff to diff.
  835.  
  836. ;; Sat April 16, 1994
  837.  
  838. ;;     Added Ediff to the File menu on the menu bar (FSF's version).
  839.  
  840. ;; Mon April 18, 1994
  841.  
  842. ;;     Fixed to work with OS/2's PM-Emacs.
  843.  
  844. ;; Thu April 21, 1994
  845.  
  846. ;;     Lemacs' menus added (thanks to Alastair Burt for the help).
  847.  
  848. ;; Wed April 28, 1994
  849.  
  850. ;;     Fixed ediff-leave-window-config (thanks to Norbert Kiesel 
  851. ;;     <norbert@i3.informatik.rwth-aachen.de>), ediff-shell and
  852. ;;     ediff-protect-metachars (thanks to Richard Stanton
  853. ;;     <stanton@haas.berkeley.edu>). Made access to difference
  854. ;;     overlays structure-independent, making it less bug-prone.
  855. ;;     Patched ediff-read-file-name to work more intuitively with directory
  856. ;;     names (thanks to Kevin Broadey <KevinB@bartley.demon.co.uk>).
  857.  
  858. ;; Mon May 2, 1994
  859.  
  860. ;;     Added `ediff-frame-has-menubar' to guard against the possibility that
  861. ;;     the current frame has no menu bar.
  862.  
  863. ;; Fri May 6, 1994
  864.  
  865. ;;     Fixed buglet in vc-ediff (thanks to Ray Nickson <nickson@cs.uq.oz.au>).
  866.  
  867. ;; Wed May 18, 1994
  868.  
  869. ;;     Modified ediff-read-file-name to not put long file names in the
  870. ;;     default prompt area, as suggested by KevinB@bartley.demon.co.uk.
  871. ;;     Applied patch supplied by burt@dfki.uni-kl.de, fixing a problem with
  872. ;;     ediff-diff-to-diff in Lemacs.
  873.  
  874. ;; Tue May 31, 1994
  875.  
  876. ;;     Added ediff-forward-word-function (as suggested by Job Ganzevoort
  877. ;;     <Job.Ganzevoort@cwi.nl>). Modified ediff-default-quit-hook so it
  878. ;;     will clean things up in a more satisfactory way.
  879.  
  880. ;; Thu Jun 2, 1994
  881.  
  882. ;;     Added `ediff-toggle-regexp-match', which allows the user to step
  883. ;;     through only those difference regions that match some regexp; or,
  884. ;;     vice versa, to skip over regions that match a regexp. (This feature
  885. ;;     was suggested by Andy Scott <ascott@pcocd2.intel.com>.)
  886. ;;     Added ediff-eval-in-buffer, which is a modified emerge-eval-in-buffer.
  887. ;;     The function ediff-status-info, bound to `i', now replaces and extends
  888. ;;     ediff-file-names and ediff-line-numbers, which were bound to `f'
  889. ;;     and `i', respectively.
  890.  
  891. ;; Wed Jun 8, 1994
  892.  
  893. ;;     Made `ediff-frame-has-menubar' into a function; copied
  894. ;;     `emerge-defvar-local' and turned it into `ediff-defvar-local' 
  895. ;;     This is supposed to make the same ediff.elc file work for both Emacs
  896. ;;     and Lucid Emacs, at least, if compiled under Lucid Emacs. (Thanks
  897. ;;     to Eric Eide <eeide@asylum.cs.utah.edu>.)
  898.  
  899. ;; Wed Jun 10, 1994
  900.  
  901. ;;     Improved `ediff-read-file-name' and `ediff-buffers' so they are now
  902. ;;     providing more intuitive defaults. Modified `ediff-read-file-name'
  903. ;;     so it won't cause problems under OS/2.
  904.  
  905. ;; Fri Jun 24, 1994
  906.  
  907. ;;     Modified ediff-find-file, ediff-files-internal, and made
  908. ;;     emerge-verify-file-buffer into ediff-verify-file-buffer so that
  909. ;;     Ediff will work correctly with remote and compressed
  910. ;;     files. (Suggested by Sandy Rutherford <sandy@ibm550.sissa.it>.)
  911.  
  912. ;; Fri Jun 28, 1994
  913.  
  914. ;;     Fixed ediff-patch-files to work with remote and compressed files.
  915.  
  916. ;; Wed July 20, 1994
  917.  
  918. ;;     Changed menu bar items per RMS's suggestion. Changed odd/even faces
  919. ;;     in Lemacs to italic.  Changed ediff-*-face-* variables so that they
  920. ;;     will contain names of faces instead of the face internal
  921. ;;     representation.  (Copy-face works better with face names than with
  922. ;;     face internal representation.  With face internal representation, if
  923. ;;     a face vector mentions a font explicitly, copy-face may attempt to
  924. ;;     copy this font, which would cause an error if the font has a wrong
  925. ;;     size for one of the existing frames.)  Improved the way
  926. ;;     mode-line-buffer-identification is set in ediff-setup so that Ediff
  927. ;;     will accommodate the way buffers are identified in mode-line.el and
  928. ;;     uniquify.el.
  929.  
  930. ;; Fri August 5, 1994
  931.  
  932. ;;     Ediff can now automatically skip over regions that differ only in
  933. ;;     the white space and line breaks. This is controled with the variable
  934. ;;     `ediff-ignore-similar-regions' and can be toggled on/off by typing
  935. ;;     `##'.
  936.  
  937. ;; Mon August 8, 1994
  938.  
  939. ;;     If ediff-save-buffer is invoked with `wf', it'll save the diff
  940. ;;     output in a file.
  941.  
  942.  
  943. ;;; Acknowledgements:
  944.  
  945. ;; Special thanks to Alastair Burt <burt@dfki.uni-kl.de>, Kevin Esler
  946. ;; <esler@ch.hp.com>, Kevin Broadey <KevinB@bartley.demon.co.uk>, 
  947. ;; Harald Boegeholz <hwb@machnix.mathematik.uni-stuttgart.de>,
  948. ;; Robert Estes <estes@ece.ucdavis.edu>, Eric Eide <eeide@asylum.cs.utah.edu>,
  949. ;; Eric Freudenthal  <freudent@jan.ultra.nyu.edu>, Job Ganzevoort 
  950. ;; <Job.Ganzevoort@cwi.nl>, Boris Goldowsky <boris@cs.rochester.edu>,
  951. ;; Allan Gottlieb <gottlieb@allan.ultra.nyu.edu>, Xiaoli Huang
  952. ;; <hxl@epic.com>, irvine@lks.csi.com, jaffe@chipmunk.cita.utoronto.ca,
  953. ;; David Karr, <dkarr@nmo.gtegsc.com>, Norbert Kiesel 
  954. ;; <norbert@i3.informatik.rwth-aachen.de>, Heinz Knutzen
  955. ;; <hk@informatik.uni-kiel.d400.de>, Martin Maechler
  956. ;; <maechler@stat.math.ethz.ch>, Richard Mlynarik <mly@adoc.xerox.com>,
  957. ;; Ray Nickson <nickson@cs.uq.oz.au>, Sandy Rutherford
  958. ;; <sandy@ibm550.sissa.it>,  Andy Scott <ascott@pcocd2.intel.com>,
  959. ;; Richard Stallman <rms@gnu.ai.mit.edu>, Richard Stanton 
  960. ;; <stanton@haas.berkeley.edu>, Peter Stout <Peter_Stout@cs.cmu.edu>
  961. ;; for contributing ideas, patches, and bug reports. 
  962. ;;
  963. ;; Thanks also to many others who felt obliged to drop a thanks note.
  964.  
  965.  
  966.  
  967. ;;; Code:
  968.  
  969. (require 'emerge) ;; Ediff uses some functions defined in emerge.el
  970.  
  971.  
  972. ;;; Macros
  973. (defmacro ediff-if-lucid ()
  974.   (` (string-match "Lucid" emacs-version)))
  975.  
  976. (defmacro ediff-odd-p (arg)
  977.   (` (eq (logand (, arg) 1) 1)))
  978.  
  979. (defmacro ediff-buffer-live-p (buf)
  980.   (` (and (, buf) (get-buffer (, buf)) (buffer-name (get-buffer (, buf))))))
  981.  
  982. (defmacro ediff-get-buffer (arg)
  983.   (` (cond ((eq (, arg) 'A) ediff-A-buffer)
  984.        ((eq (, arg) 'B) ediff-B-buffer)
  985.        )
  986.   ))
  987.        
  988. (defmacro ediff-char-to-buftype (arg)
  989.   (` (cond ((eq (, arg) ?a) 'A)
  990.        ((eq (, arg) ?b) 'B)
  991.        )
  992.   ))
  993.   
  994. (defmacro ediff-get-difference (n)
  995.   (` (aref ediff-difference-vector (, n))))
  996.   
  997. ;; tell if it has been previously found that the region doesn't
  998. ;; contain diffs other than the white space and newlines
  999. ;; The argument, N, is the diff region number used by Ediff to index the
  1000. ;; diff vector. It is 1 less than the number seen by the user.
  1001. (defmacro ediff-no-fine-diffs (n)
  1002.   (` (aref (ediff-get-difference n) 3)))
  1003.   
  1004. (defmacro ediff-get-diff-overlay-from-vector (vec buf-type)
  1005.   (` (aref (, vec)
  1006.        (cond ((eq (, buf-type) 'A) 0)
  1007.          ((eq (, buf-type) 'B) 1)
  1008.          )
  1009.        )))
  1010.   
  1011. (defmacro ediff-get-diff-overlay (n buf-type)  
  1012.   (` (ediff-get-diff-overlay-from-vector
  1013.       (ediff-get-difference (, n))
  1014.       (, buf-type))))
  1015.  
  1016. (defmacro ediff-get-fine-diff-vector-from-vec (current-diff-vec)
  1017.   (` (aref (, current-diff-vec) 2)))
  1018.       
  1019. (defmacro ediff-set-fine-diff-vector (n fine-vec)
  1020.   (` (aset (ediff-get-difference (, n)) 2 (, fine-vec))))
  1021.  
  1022. ;; if flag is t, puts a mark on diff region saying that 
  1023. ;; the differences are in white space only. If flag is nil,
  1024. ;; the region is marked as essential (i.e., differences are
  1025. ;; not just in the white space and newlines.)
  1026. (defmacro ediff-mark-diff-as-space-only (n flag)
  1027.   (` (aset (ediff-get-difference (, n)) 3 (, flag))))
  1028.   
  1029. (defmacro ediff-get-fine-diff-vector (n)
  1030.   (` (ediff-get-fine-diff-vector-from-vec (ediff-get-difference (, n)))))
  1031.   
  1032.   
  1033. (defmacro ediff-defvar-local (var value doc) 
  1034.   "Defines SYMBOL as an advertised local variable.  
  1035. Performs a defvar, then executes `make-variable-buffer-local' on
  1036. the variable.  Also sets the `preserved' (Emacs) or `permanent-local' (Lucid)
  1037. property, so that `kill-all-local-variables' (called by major-mode setting
  1038. commands) won't destroy Ediff control variables.
  1039.  
  1040. This is a merge of `emerge-defvar-local' for Emacs and Lucid Emacs.  It is
  1041. needed to make the same ediff.elc work under both Emacsen."
  1042.   (` (progn
  1043.        (defvar (, var) (, value) (, doc))
  1044.        (make-variable-buffer-local '(, var))
  1045.        (put '(, var)
  1046.         (if (ediff-if-lucid) 'permanent-local 'preserved)
  1047.         t))))
  1048.     
  1049. (defmacro ediff-eval-in-buffer (buffer &rest forms)
  1050.   "Macro to switch to BUFFER, evaluate FORMS, returns to original buffer.
  1051. Differs from `save-excursion' in that it doesn't save the point and mark.
  1052. This is essentially `emerge-eval-in-buffer' with the test for live buffers."
  1053.   (` (let ((StartBuffer (current-buffer)))
  1054.        (if (ediff-buffer-live-p (, buffer))
  1055.        (unwind-protect
  1056.            (progn
  1057.          (set-buffer (, buffer))
  1058.          (,@ forms))
  1059.          (set-buffer StartBuffer))
  1060.      (beep 1)
  1061.      (message "You seem to have killed an essential Ediff buffer---quit!"))
  1062.        )))
  1063.  
  1064.  
  1065. ;;; Functions
  1066.  
  1067. (defun ediff-mode ()
  1068.   "Ediff mode is used by the Ediff file-difference package.
  1069. It is entered only through one of the following commands:
  1070.     `ediff'
  1071.     `ediff-files'
  1072.     `ediff-buffers'
  1073.     `epatch'
  1074.     `ediff-patch-file'
  1075.     `ediff-patch-buffer'
  1076.     `epatch-buffer'
  1077.     `vc-ediff'
  1078.     `rcs-ediff'
  1079.  
  1080. Commands:
  1081. \\{ediff-mode-map}"
  1082.   (interactive)
  1083.   (kill-all-local-variables)
  1084.   (setq major-mode 'ediff-mode)
  1085.   (setq mode-name "Ediff"))
  1086.  
  1087.  
  1088. ;; Hook variables
  1089.  
  1090. (defvar ediff-before-setup-windows-hooks nil
  1091.   "*Hooks to run before Ediff sets its own window config. 
  1092. This can be used to save the previous window config, which can be restored
  1093. on ediff-quit or ediff-suspend.") 
  1094. (defvar ediff-startup-hooks nil
  1095.   "*Hooks to run in the control buffer after Ediff has been set up.")
  1096. (defvar ediff-select-hooks nil
  1097.   "*Hooks to run after a difference has been selected.")
  1098. (defvar ediff-unselect-hooks nil
  1099.   "*Hooks to run after a difference has been unselected.")
  1100. (defvar ediff-prepare-buffer-hooks  nil
  1101.   "*Hooks called after buffers A and B are set up.")
  1102. (defvar ediff-load-hooks nil
  1103.   "*Hook run after Ediff is loaded.  Can be used to change defaults.")
  1104.  
  1105. (defvar ediff-suspend-hooks (list 'ediff-default-suspend-hook)
  1106.   "*Hooks to run in the Ediff control buffer when Ediff is suspended.")
  1107. (defvar ediff-quit-hooks (list 'ediff-default-quit-hook)
  1108.   "*Hooks to run in the Ediff control buffer after finishing Ediff.") 
  1109.  
  1110. (make-variable-buffer-local 'local-write-file-hooks)
  1111. (make-variable-buffer-local 'before-change-function)
  1112.  
  1113. ;; Help messages
  1114.  
  1115. (defconst ediff-help-message-long
  1116.   "    Moving around      |     Toggling features     |       Miscellaneous
  1117. =======================|===========================|===========================
  1118. p,DEL -previous diff   |     s -vert/horiz split   | ab/ba -copy diff A->B/B->A
  1119. n,SPC -next diff       |     h -hiliting           | ra/rb -restore diff in A/B
  1120.     j -jump to diff    |     @ -auto-refining      |     * -refine diff
  1121. ga/gb -to point in A/B |---------------------------|     ! -recompute diffs
  1122.     c -recenter        |    ## -skip whitespace    |---------------------------
  1123.   v/V -scroll up/down  | #f/#h -focus/hide regions | wa/wb -save buf A/B
  1124.   </> -scroll lft/rght |   A/B -read-only buf A/B  |    wf -save diff output
  1125. =======================|===========================|===========================
  1126.     i -status info     |     ? -toggle help window |   z/q -suspend/quit Ediff"
  1127.   )
  1128.               
  1129. (defconst ediff-help-message-short
  1130.   "                             ? - toggle help window")
  1131.  
  1132. (defvar ediff-help-message ediff-help-message-long
  1133.   "*The actual help message.")
  1134.  
  1135. ;; diff stuff.
  1136. (defvar ediff-diff-program "diff"
  1137.   "*Name of the program that compares two files.")
  1138. (defvar ediff-diff-options ""  
  1139.   "*Options to pass to `ediff-diff-program'. 
  1140. If diff\(1\) is used as `ediff-diff-program', then the most useful options are
  1141. `-w', to ignore space, and `-i', to ignore case of letters.")
  1142.   
  1143. ;; Fine differences 
  1144. (defvar ediff-forward-word-function 'ediff-forward-word
  1145.   "*Function to call to move to the next word.
  1146. Used for splitting difference regions into individual words.")
  1147.  
  1148. (defvar ediff-fine-diff-program "diff"
  1149.   "*Name of the program that compares the current diff regions for fine differences.
  1150.   
  1151. This program should produce output in the format of diff. One could
  1152. possibly use `spiff' here if appropriate options are set.")
  1153.  
  1154. (defvar ediff-fine-diff-options ""  
  1155.   "*Options to pass to `ediff-fine-diff-program'.
  1156. If diff\(1\) is used as `ediff-diff-program', then the most useful options are
  1157. `-w', to ignore space, and `-i', to ignore case of letters.")
  1158.   
  1159. (defvar ediff-whitespace " \n\t\C-j"
  1160.   "*White space. Used to split strings into words.")
  1161.  
  1162. (defvar ediff-word-1 "a-zA-Z---_`'.?!:"
  1163.   "*Characters constituting type 1 words.
  1164.   
  1165. Ediff is using a very simple schema for splitting text into words, which is
  1166. used to determine fine differences between regions. There are two types of
  1167. words. One consists entirely out of characters in `ediff-word-1'  and
  1168. another out of characters matching `ediff-word-1'.")
  1169.  
  1170. (defvar ediff-word-2 "^a-zA-Z---_`'.?!: \t\n\C-j"
  1171.   "*Characters matching this regexp constitute words of type 2.
  1172. See `ediff-word-1' for more details.")  
  1173.  
  1174. ;; Selective browsing
  1175.  
  1176. (ediff-defvar-local ediff-skip-diff-region-function 'ediff-show-all-diffs
  1177.   "Function that determines the next/previous diff region to show.
  1178. Should return t for regions to be ignored and nil otherwise.
  1179. This function gets a region number as an argument. The region number
  1180. is the one used internally by Ediff. It is 1 less than the number seen
  1181. by the user.")
  1182.  
  1183. (ediff-defvar-local ediff-regexp-focus-A ""
  1184.   "Regexp that determines buf A regions to focus on when skipping to diff.")
  1185. (ediff-defvar-local ediff-regexp-focus-B ""
  1186.   "Regexp that determines buf B regions to focus on when skipping to diff.")
  1187.   
  1188. (ediff-defvar-local ediff-regexp-hide-A ""
  1189.   "Regexp that determines buf A regions to ignore when skipping to diff.")
  1190. (ediff-defvar-local ediff-regexp-hide-B ""
  1191.   "Regexp that determines buf B regions to ignore when skipping to diff.")  
  1192.   
  1193.  
  1194. ;; Support for patches 
  1195.  
  1196. (defvar ediff-patch-program "patch"
  1197.   "*Name of the program that applies patches.")
  1198. (defvar ediff-patch-options ""
  1199.   "*Options to pass to ediff-patch-program.")
  1200.   
  1201. (defvar ediff-shell (cond ((eq system-type 'emx) "cmd") ;; OS/2
  1202.               (t  "sh")) ;; unix
  1203.   "*The shell used to run diff and patch.  If user's .profile or
  1204. .cshrc files are set up correctly, any shell will do.  However, some people
  1205. set $prompt or other things incorrectly, which leads to undesirable output
  1206. messages.  These may cause Ediff to fail.  In such a case, set ediff-shell
  1207. to a shell that you are not using or, better, fix your shell's startup file.")
  1208.   
  1209. (defvar ediff-diff-ok-lines-regexp  
  1210.   "^\\([0-9,]+[acd][0-9,]+$\\|[<>] \\|---\\|Warning:\\)"
  1211.   "*Regexp that matches normal output lines from `ediff-diff-program'.
  1212. This is mostly lifted from Emerge, except that Ediff also considers the
  1213. 'Missing newline' message to be 'normal output.'
  1214. Lines that do not match are assumed to be error messages.")
  1215.  
  1216. (defvar ediff-fine-diff-ok-lines-regexp  
  1217.   "^\\([0-9,]+[acd][0-9,]+$\\|[<>] \\|---\\|Warning:\\)"
  1218.   "*Regexp that matches normal output lines from `ediff-fine-diff-program'.
  1219. This is mostly lifted from Emerge, except that Ediff also considers the
  1220. 'Missing newline' message to be 'normal output.'
  1221. Lines that do not match are assumed to be error messages.")
  1222.  
  1223. (defvar ediff-match-diff-line (let ((x "\\([0-9]+\\)\\(\\|,\\([0-9]+\\)\\)"))
  1224.                 (concat "^" x "\\([acd]\\)" x "$"))
  1225.   "*Pattern to match lines produced by diff that describe differences.")
  1226.   
  1227. (defvar ediff-patch-buf nil
  1228.   "The buffer of the patch file.")
  1229. (defvar ediff-patch-diagnostics nil
  1230.   "The buffer where patch would display its diagnostics.")
  1231.   
  1232.  
  1233. ;; Copying diffs betw buffers.    
  1234.  
  1235. (ediff-defvar-local ediff-killed-diffs-alist nil
  1236.   "A list of killed diffs. 
  1237. A diff is saved here if it is replaced by a diff
  1238. from another buffer.  This alist has the form:
  1239. \((num (A . diff) (B . diff)) ...),
  1240. where A or B parts may be missing.")
  1241.  
  1242.  
  1243. ;; Highlighting
  1244. (defvar ediff-before-flag-bol
  1245.   ;"vvvvvvvvvvvvvvvv---- ediff ----vvvvvvvvvvvvvvv\n"
  1246.   ">>--->>>\n"
  1247.   "*Flag placed above the highlighted block of differences. 
  1248. Must end with newline.  Must be set before Ediff is loaded.")
  1249. (defvar ediff-after-flag-bol
  1250.   ;"^^^^^^^^^^^^^^^^---- ediff ----^^^^^^^^^^^^^^^\n"
  1251.   "<<<---<<\n"
  1252.   "*Flag placed below the highlighted block of differences.
  1253. Must end with newline.  Must be set before Ediff is loaded.")
  1254.  
  1255. (defvar ediff-before-flag-mol ">>--->>>"
  1256.   "*Like ediff-before-flag, used when a difference starts in mid-line.")
  1257. (defvar ediff-after-flag-mol "<<<---<<"
  1258.   "*Like ediff-after-flag, used when a difference starts in mid-line.")
  1259.  
  1260. (ediff-defvar-local ediff-before-flag-A nil
  1261.   "This is the actual ASCII before-flag in effect in buffer A.
  1262. It is either `ediff-before-flag-mol' or `ediff-before-flag-bol' depending
  1263. on whether the selected difference region starts in the middle of a line 
  1264. or at the beginning of a line.")
  1265. (ediff-defvar-local ediff-after-flag-A nil
  1266.   "This is the actual ASCII after-flag in effect in buffer A.
  1267. It is either `ediff-before-flag-mol' or `ediff-before-flag-bol' depending
  1268. on whether the selected difference region starts in the middle of a line 
  1269. or at the beginning of a line.")
  1270. (ediff-defvar-local ediff-before-flag-B nil
  1271.   "This is the actual ASCII before-flag in effect in buffer B.
  1272. It is either `ediff-before-flag-mol' or `ediff-before-flag-bol' depending
  1273. on whether the selected difference region starts in the middle of a line 
  1274. or at the beginning of a line.")
  1275. (ediff-defvar-local ediff-after-flag-B nil
  1276.   "This is the actual ASCII after-flag in effect in buffer B.
  1277. It is either `ediff-before-flag-mol' or `ediff-before-flag-bol' depending
  1278. on whether the selected difference region starts in the middle of a line 
  1279. or at the beginning of a line.")
  1280.  
  1281.   
  1282. (ediff-defvar-local ediff-want-faces t
  1283.   "If t, differences are highlighted using faces on a window system.
  1284. If nil, they are highlighted using ASCII flags, ediff-before-flag
  1285. and ediff-after-flag.  On a non-window system, differences are always
  1286. highlighted using ASCII flags.
  1287.  
  1288. This variable can be set either in .emacs or toggled interactively, using
  1289. ediff-toggle-hilit.") 
  1290.  
  1291. (ediff-defvar-local ediff-want-default-menus t
  1292.   "If t, Ediff will set up menus in the menu bar.
  1293. This variable must be set before Ediff is loaded. If you don't like the
  1294. look of the default menus, set this variable to nil and make your own
  1295. menus.")  
  1296.  
  1297. (ediff-defvar-local ediff-auto-refine (if window-system 'on 'nix)
  1298.   "If `'on', Ediff auto-highlights fine diffs for the current diff region.
  1299. If `off', auto-highlighting is not used. If `'nix', no fine diffs are shown
  1300. at all, unless the user force-refines the region by hitting `*'.
  1301.  
  1302. This variable can be set either in .emacs or toggled interactively, using
  1303. ediff-toggle-hilit.") 
  1304.  
  1305. (ediff-defvar-local ediff-ignore-similar-regions nil
  1306.   "*If t, skip over difference regions that differ only in the white space and line breaks.")
  1307.  
  1308. (ediff-defvar-local ediff-auto-refine-limit 700
  1309.   "Auto-refine only those regions that are smaller than this number of bytes.")
  1310.  
  1311. (ediff-defvar-local ediff-highlight-all-diffs t
  1312.   "If nil, only the selected differences are highlighted.
  1313. This variable can be set either in .emacs or toggled interactively, using
  1314. ediff-toggle-hilit.") 
  1315.  
  1316. (ediff-defvar-local ediff-highlighting-style nil
  1317.   "A var local to each control panel buffer.
  1318. Indicates highlighting style in effect for this buffer: `face', `ascii',
  1319. nil -- temporarily unhighlighted, `off' -- turned off \(on a dumb
  1320. terminal only\).")
  1321.  
  1322.   
  1323.  
  1324. ;; Variables that control each Ediff session.  They are local to the
  1325. ;; control buffer. 
  1326.  
  1327. ;; Mode variables
  1328. (ediff-defvar-local ediff-A-buffer nil
  1329.   "The buffer in which the A variant is stored.")
  1330. (ediff-defvar-local ediff-B-buffer nil
  1331.   "The buffer in which the B variant is stored.")
  1332. (ediff-defvar-local ediff-control-buffer nil
  1333.   "The control buffer of ediff. ")
  1334.  
  1335. ;(ediff-defvar-local ediff-control-buffer-suffix nil
  1336. ;  "The suffix of the control buffer name. ")
  1337.   
  1338. (ediff-defvar-local ediff-control-window nil
  1339.   "The control window.")
  1340. (ediff-defvar-local ediff-window-config-saved ""
  1341.   "Ediff's window configuration.
  1342. Used to minimize the need to rearrange windows.")
  1343.  
  1344.  
  1345. (ediff-defvar-local ediff-A-buffer-values nil
  1346.   "Keeps working values of ediff-saved-variables for ediff-A-buffer.")
  1347. (ediff-defvar-local ediff-B-buffer-values nil
  1348.   "Keeps working values of ediff-saved-variables for ediff-B-buffer.")
  1349.  
  1350. (ediff-defvar-local ediff-A-buffer-values-setup nil
  1351.   "Remembers ediff-saved-variables for ediff-A-buffer as they were at setup.")
  1352. (ediff-defvar-local ediff-B-buffer-values-setup nil
  1353.   "Remembers ediff-saved-variables for ediff-B-buffer as they were at setup.")
  1354.  
  1355. (ediff-defvar-local ediff-difference-vector nil
  1356.   "Vector of differences between the variants. 
  1357. Each difference is represented by a vector of two overlays.  The first
  1358. overlays the difference section in the A buffer and the second overlays the
  1359. diff in the B buffer. If a difference section is empty, the corresponding
  1360. overlay's endpoints coincide. ")
  1361.  
  1362. (ediff-defvar-local ediff-current-difference -1
  1363.   "The difference that is currently selected.")
  1364. (ediff-defvar-local ediff-number-of-differences nil
  1365.   "Number of differences found.")
  1366.   
  1367. (ediff-defvar-local ediff-diff-buffer nil
  1368.   "Buffer containing the output of diff, which is used by Ediff to step
  1369. through files.")
  1370. (ediff-defvar-local ediff-fine-diff-buffer nil
  1371.   "Buffer used for diff-style fine differences between regions.")
  1372. (ediff-defvar-local ediff-tmp-buffer nil
  1373.   "Temporary buffer used for computing fine differences.")
  1374. (ediff-defvar-local ediff-error-buffer nil
  1375.   "Buffer containing the output of diff when diff returns errors.")
  1376.  
  1377. (ediff-defvar-local ediff-this-buffer-control-sessions  nil
  1378.   "List of ediff control panels associated with each buffer A/B.")
  1379.  
  1380. (defvar ediff-disturbed-overlays nil
  1381.   "List of difference overlays disturbed by working with the current diff.")
  1382.   
  1383. (defvar ediff-shaded-overlay-priority  100
  1384.   "Priority of non-selected overlays.")
  1385.  
  1386.  
  1387. (if (ediff-if-lucid)
  1388.     (progn
  1389.       (fset 'ediff-overlayp (symbol-function 'extentp))
  1390.       (fset 'ediff-make-overlay (symbol-function 'make-extent))
  1391.       (fset 'ediff-delete-overlay (symbol-function 'delete-extent))
  1392.       (fset 'ediff-overlay-put (symbol-function 'set-extent-property))
  1393.       (fset 'ediff-move-overlay (symbol-function 'set-extent-endpoints))
  1394.       (fset 'ediff-overlay-start (symbol-function 'extent-start-position))
  1395.       (fset 'ediff-overlay-end (symbol-function 'extent-end-position))
  1396.       (fset 'ediff-overlay-get (symbol-function 'extent-property)))
  1397.   (fset 'ediff-overlayp (symbol-function 'overlayp))
  1398.   (fset 'ediff-make-overlay (symbol-function 'make-overlay))
  1399.   (fset 'ediff-delete-overlay (symbol-function 'delete-overlay))
  1400.   (fset 'ediff-overlay-put (symbol-function 'overlay-put))
  1401.   (fset 'ediff-move-overlay (symbol-function 'move-overlay))
  1402.   (fset 'ediff-overlay-start (symbol-function 'overlay-start))
  1403.   (fset 'ediff-overlay-end (symbol-function 'overlay-end))
  1404.   (fset 'ediff-overlay-get (symbol-function 'overlay-get)))
  1405.   
  1406. (if window-system
  1407.     (if (ediff-if-lucid)
  1408.     (progn
  1409.       (fset 'ediff-select-frame (symbol-function 'select-screen))
  1410.       (fset 'ediff-window-frame (symbol-function 'window-screen))
  1411.       (fset 'ediff-display-color-p (symbol-function 'x-color-display-p))
  1412.       (fset 'ediff-valid-color-p (symbol-function 'x-valid-color-name-p))
  1413.       (fset 'ediff-get-face (symbol-function 'get-face)))
  1414.       (fset 'ediff-window-frame (symbol-function 'window-frame))
  1415.       (fset 'ediff-select-frame (symbol-function 'select-frame))
  1416.       (fset 'ediff-display-color-p (symbol-function 'x-display-color-p))
  1417.       
  1418.       ;; This is a temporary fix for OS/2 users
  1419.       ;; pm-win.el in PM-Emacs should be fixed.
  1420.       (if (eq window-system 'pm)
  1421.       (fset 'ediff-valid-color-p 
  1422.         (function (lambda (color) (assoc color pm-color-alist))))
  1423.     (fset 'ediff-valid-color-p (symbol-function 'x-color-defined-p))
  1424.     )
  1425.     
  1426.       (fset 'ediff-get-face (symbol-function 'internal-get-face)))
  1427.   ;; not a window system
  1428.   (fset 'ediff-window-frame (function (lambda (wind) (if wind 1 nil)) ))
  1429.   (fset 'ediff-select-frame (symbol-function 'identity))
  1430.   (fset 'ediff-make-current-diff-overlay (function (lambda (type) nil)))
  1431.   (fset 'ediff-unhighlight-diffs-totally (function (lambda () nil))))
  1432.   
  1433.  
  1434. (if (not window-system)
  1435.     ()
  1436.   (defun ediff-set-face (ground face color)
  1437.     "Sets face foreground/background."
  1438.     (if (ediff-valid-color-p color)
  1439.     (if (eq ground 'foreground)
  1440.         (set-face-foreground face color)
  1441.       (set-face-background face color))
  1442.       (cond ((memq face
  1443.            '(ediff-current-diff-face-A ediff-current-diff-face-B))
  1444.          (copy-face 'highlight face))
  1445.         ((memq face
  1446.            '(ediff-fine-diff-face-A ediff-fine-diff-face-B))
  1447.          (copy-face 'secondary-selection face)
  1448.          (set-face-underline-p face t))
  1449.         ((memq face
  1450.            '(ediff-odd-diff-face-A ediff-odd-diff-face-B
  1451.              ediff-even-diff-face-A ediff-even-diff-face-B))
  1452.          (copy-face 'secondary-selection face)))))
  1453.       
  1454.   (defvar ediff-current-diff-face-A
  1455.     (progn
  1456.       (make-face 'ediff-current-diff-face-A)
  1457.       (or (face-differs-from-default-p 'ediff-current-diff-face-A)
  1458.       (cond ((ediff-display-color-p)
  1459.          (ediff-set-face
  1460.           'foreground 'ediff-current-diff-face-A "firebrick")
  1461.          (ediff-set-face
  1462.           'background 'ediff-current-diff-face-A "pale green"))
  1463.         (t
  1464.          (if (ediff-if-lucid)
  1465.              (copy-face 'modeline 'ediff-current-diff-face-A)
  1466.            (copy-face 'highlight 'ediff-current-diff-face-A))
  1467.          )))
  1468.       'ediff-current-diff-face-A)
  1469.       ;;(ediff-get-face 'ediff-current-diff-face-A))
  1470.     "Face for highlighting the selected difference in buffer A.")
  1471.  
  1472.   (defvar ediff-current-diff-face-B
  1473.     (progn
  1474.       (make-face 'ediff-current-diff-face-B)
  1475.       (or (face-differs-from-default-p 'ediff-current-diff-face-B)
  1476.       (cond ((ediff-display-color-p)
  1477.          (ediff-set-face
  1478.           'foreground 'ediff-current-diff-face-B "DarkOrchid")
  1479.          (ediff-set-face
  1480.           'background 'ediff-current-diff-face-B "Yellow"))
  1481.         (t 
  1482.          (if (ediff-if-lucid)
  1483.              (copy-face 'modeline 'ediff-current-diff-face-B)
  1484.            (copy-face 'highlight 'ediff-current-diff-face-B))
  1485.          )))
  1486.       'ediff-current-diff-face-B)
  1487.       ;;(ediff-get-face 'ediff-current-diff-face-B))
  1488.     "Face for highlighting the selected difference in buffer B.")
  1489.  
  1490.   (defvar ediff-fine-diff-face-A
  1491.     (progn
  1492.       (make-face 'ediff-fine-diff-face-A)
  1493.       (or (face-differs-from-default-p 'ediff-fine-diff-face-A)
  1494.       (cond ((ediff-display-color-p)
  1495.          (ediff-set-face 'foreground 'ediff-fine-diff-face-A
  1496.                  "Navy")
  1497.          (ediff-set-face 'background 'ediff-fine-diff-face-A
  1498.                  "sky blue"))
  1499.         (t (set-face-underline-p 'ediff-fine-diff-face-A t))))
  1500.       'ediff-fine-diff-face-A)
  1501.       ;;(ediff-get-face 'ediff-fine-diff-face-A))
  1502.     "Face for highlighting the refinement of the selected diff in buffer A.")
  1503.  
  1504.   (defvar ediff-fine-diff-face-B
  1505.     (progn
  1506.       (make-face 'ediff-fine-diff-face-B)
  1507.       (or (face-differs-from-default-p 'ediff-fine-diff-face-B)
  1508.       (cond ((ediff-display-color-p)
  1509.          (ediff-set-face 'foreground 'ediff-fine-diff-face-B "Black")
  1510.          (ediff-set-face 'background 'ediff-fine-diff-face-B "cyan"))
  1511.         (t (set-face-underline-p 'ediff-fine-diff-face-B t))))
  1512.       'ediff-fine-diff-face-B)
  1513.       ;;(ediff-get-face 'ediff-fine-diff-face-B))
  1514.     "Face for highlighting the refinement of the selected diff in buffer B.")
  1515.  
  1516.  
  1517.   (defvar ediff-even-diff-face-A
  1518.     (progn
  1519.       (make-face 'ediff-even-diff-face-A)
  1520.       (or (face-differs-from-default-p 'ediff-even-diff-face-A)
  1521.       (cond ((ediff-display-color-p)
  1522.          (ediff-set-face
  1523.           'foreground 'ediff-even-diff-face-A "black")
  1524.          (ediff-set-face
  1525.           'background 'ediff-even-diff-face-A "light grey"))
  1526.         (t 
  1527.          (copy-face 'italic 'ediff-even-diff-face-A))))
  1528.       'ediff-even-diff-face-A)
  1529.       ;;(ediff-get-face 'ediff-even-diff-face-A))
  1530.     "Face used to highlight even-numbered differences in buffer A.")
  1531.       
  1532.   (defvar ediff-even-diff-face-B
  1533.     (progn
  1534.       (make-face 'ediff-even-diff-face-B)
  1535.       (or (face-differs-from-default-p 'ediff-even-diff-face-B)
  1536.       (cond ((ediff-display-color-p)
  1537.          (ediff-set-face
  1538.           'foreground 'ediff-even-diff-face-B "White")
  1539.          (ediff-set-face
  1540.           'background 'ediff-even-diff-face-B "Gray"))
  1541.         (t 
  1542.          (copy-face 'italic 'ediff-even-diff-face-B))))
  1543.       'ediff-even-diff-face-B)
  1544.       ;;(ediff-get-face 'ediff-even-diff-face-B))
  1545.     "Face used to highlight even-numbered differences in buffer B.")
  1546.   
  1547.   (defvar ediff-odd-diff-face-A
  1548.     (progn
  1549.       (make-face 'ediff-odd-diff-face-A)
  1550.       (or (face-differs-from-default-p 'ediff-odd-diff-face-A)
  1551.       (cond ((ediff-display-color-p)
  1552.          (ediff-set-face
  1553.           'foreground 'ediff-odd-diff-face-A "White")
  1554.          (ediff-set-face
  1555.           'background 'ediff-odd-diff-face-A "Gray"))
  1556.         (t 
  1557.          (copy-face 'italic 'ediff-odd-diff-face-A))))
  1558.       'ediff-odd-diff-face-A)
  1559.       ;;(ediff-get-face 'ediff-odd-diff-face-A))
  1560.     "Face used to highlight odd-numbered differences in buffer A.")
  1561.       
  1562.   (defvar ediff-odd-diff-face-B
  1563.     (progn
  1564.       (make-face 'ediff-odd-diff-face-B)
  1565.       (or (face-differs-from-default-p 'ediff-odd-diff-face-B)
  1566.       (cond ((ediff-display-color-p)
  1567.          (ediff-set-face
  1568.           'foreground 'ediff-odd-diff-face-B "Black")
  1569.          (ediff-set-face
  1570.           'background 'ediff-odd-diff-face-B "light grey"))
  1571.         (t 
  1572.          (copy-face 'italic 'ediff-odd-diff-face-B))))
  1573.       'ediff-odd-diff-face-B)
  1574.       ;;(ediff-get-face 'ediff-odd-diff-face-B))
  1575.     "Face used to highlight odd-numbered differences in buffer B.")
  1576.  
  1577.   ;; Create *-var faces. These are the actual faces used to highlight
  1578.   ;; odd-numbered difference regions.
  1579.   ;; They are used as follows: when highlighting is turned on,
  1580.   ;; ediff-odd/even-diff-face-A/B are copied
  1581.   ;; into ediff-odd/even-diff-face-A/B-var, and all odd/even overlays become
  1582.   ;; highlighted. When highlighting is turned off, then the face 'default is
  1583.   ;; copied into ediff-odd/even-diff-face-A/B-var, thereby unhighlighting all
  1584.   ;; difference regions.
  1585.   (make-face 'ediff-even-diff-face-A-var)
  1586.   (make-face 'ediff-even-diff-face-B-var)
  1587.   (make-face 'ediff-odd-diff-face-A-var)
  1588.   (make-face 'ediff-odd-diff-face-B-var)
  1589.   
  1590.   ;; initialize *-var faces
  1591.   (defun ediff-init-var-faces ()
  1592.     (copy-face (if (and ediff-want-faces ediff-highlight-all-diffs)
  1593.            ediff-even-diff-face-A 'default)
  1594.            'ediff-even-diff-face-A-var)
  1595.     (copy-face (if (and ediff-want-faces ediff-highlight-all-diffs)
  1596.            ediff-even-diff-face-B 'default)
  1597.            'ediff-even-diff-face-B-var)
  1598.     (copy-face (if (and ediff-want-faces ediff-highlight-all-diffs)
  1599.            ediff-odd-diff-face-A 'default)
  1600.            'ediff-odd-diff-face-A-var)
  1601.     (copy-face (if (and ediff-want-faces ediff-highlight-all-diffs)
  1602.            ediff-odd-diff-face-B 'default)
  1603.            'ediff-odd-diff-face-B-var))
  1604.       
  1605.  
  1606.   ;;; Overlays
  1607.  
  1608.   (ediff-defvar-local ediff-current-diff-overlay-A nil
  1609.     "Overlay specifying the current difference region in buffer A.")
  1610.   (ediff-defvar-local ediff-current-diff-overlay-B nil
  1611.     "Overlay specifying the current difference region in buffer B.")
  1612.   
  1613.   (defun ediff-make-current-diff-overlay (type)
  1614.       (let ((overlay (if (eq type 'A)
  1615.              'ediff-current-diff-overlay-A
  1616.                'ediff-current-diff-overlay-B))
  1617.         (buffer (ediff-get-buffer type))
  1618.         (face (if (eq type 'A) 
  1619.               (face-name ediff-current-diff-face-A)
  1620.             (face-name ediff-current-diff-face-B))))
  1621.     (set overlay (ediff-make-overlay (point-max) (point-max) buffer))
  1622.     (ediff-overlay-put (eval overlay) 'face face)
  1623.     (ediff-overlay-put (eval overlay) 'ediff ediff-control-buffer)
  1624.     ))
  1625.     
  1626.   ;; Compute priority of ediff overlay.
  1627.   (defun ediff-highest-priority (start end buffer)
  1628.     (let ((pos (max 1 (1- start)))
  1629.       ovr-list)
  1630.       (if (ediff-if-lucid)
  1631.       (1+ ediff-shaded-overlay-priority)
  1632.     (ediff-eval-in-buffer
  1633.      buffer
  1634.      (while (< pos (min (point-max) (1+ end)))
  1635.        (setq ovr-list (append (overlays-at pos) ovr-list))
  1636.        (setq pos (next-overlay-change pos)))
  1637.      (1+ (eval
  1638.           (cons '+
  1639.             (mapcar (function
  1640.                  (lambda (ovr)
  1641.                    (if ovr
  1642.                    (or (ediff-overlay-get ovr 'priority) 0)
  1643.                  0)))
  1644.                 ovr-list)
  1645.             )))
  1646.      ))))
  1647.  
  1648. )  ; end of window-system-only code.
  1649.   
  1650.   
  1651.  
  1652. ;;; Misc
  1653.  
  1654. (defvar ediff-split-window-function 'split-window-vertically
  1655.   "*The function to split the main window between buffer-A and buffer-B.
  1656. You can set it to be split horizontally instead of the
  1657. default vertical split by setting this variable to
  1658. 'split-window-horizontally.  You can also have your own function for fancy
  1659. splits.  This variable has no effect when buffer-A and buffer-B are shown in
  1660. different frames.  In this case, Ediff will use those frames to display
  1661. these buffers.")
  1662.          
  1663. (defconst ediff-saved-variables
  1664.   '(buffer-read-only
  1665.     buffer-auto-save-file-name)
  1666.   "Buffer-local variables saved and restored during an Ediff session.")
  1667.  
  1668. (defconst ediff-working-values '(nil nil)
  1669.   "Values to be assigned to `ediff-saved-variables' during diff.")
  1670.   
  1671. (defvar ediff-use-last-dir nil
  1672.   "*If t, Ediff uses previous directory as default when reading file name.")
  1673.   
  1674. (defvar ediff-no-help-in-control-buffer nil
  1675.   "*Non-nil means C-h should not invoke Emacs help in control buffer.
  1676. Instead, C-h jumps to previous difference.")
  1677.   
  1678. (defvar ediff-temp-file-prefix
  1679.   (let ((env (or (getenv "TMPDIR")
  1680.          (getenv "TMP")
  1681.          (getenv "TEMP")))
  1682.     d)
  1683.     (setq d (if (and env (> (length env) 0))
  1684.         env
  1685.           "/tmp"))
  1686.     (if (= (aref d (1- (length d))) ?/)
  1687.     (setq d (substring d 0 -1)))
  1688.     (concat d "/ediff"))
  1689.   "*Prefix to put on Ediff temporary file names.
  1690. Do not start with `~/' or `~user-name/'.")  
  1691.  
  1692. (defvar ediff-temp-file-mode 384    ; u=rw only
  1693.   "*Mode for Ediff temporary files.")
  1694.   
  1695. (ediff-defvar-local ediff-temp-file-A nil
  1696.   "Temporary file used for refining difference regions in buffer B.")
  1697. (ediff-defvar-local ediff-temp-file-B nil
  1698.   "Temporary file used for refining difference regions in buffer B.")
  1699.   
  1700. (defvar ediff-last-dir-A nil
  1701.   "Last directory used by an Ediff command for file-A.")
  1702. (defvar ediff-last-dir-B nil
  1703.   "Last directory used by an Ediff command for file-B.")
  1704. (defvar ediff-last-dir-patch nil
  1705.   "Last directory used by an Ediff command for file to patch.")
  1706.   
  1707. ;; Build keymaps
  1708.  
  1709. (defvar ediff-mode-map nil
  1710.   "Local keymap used in Ediff mode.")
  1711.   
  1712. (defun ediff-frame-has-menubar ()
  1713.   (if (ediff-if-lucid)
  1714.       current-menubar
  1715.     (< 0 (cdr (assq 'menu-bar-lines (frame-parameters (selected-frame)))))
  1716.     ))
  1717.  
  1718. ;;; This is split in three parts to avoid
  1719. ;;; making a line in loaddefs.el that is too long for patch.
  1720. ;;; Note that autoload.el currently looks for cookies
  1721. ;;; only at top level in the file.
  1722. ;;; So I moved these to top level.  But the conditionals on
  1723. ;;; purify-flag make these no-ops when you load ediff.
  1724. ;;; They only do something in loaddefs.el.
  1725. ;;;###autoload
  1726. (if purify-flag
  1727.     (progn
  1728.       (defvar menu-bar-epatch-menu (make-sparse-keymap "Epatch"))
  1729.       (fset 'menu-bar-epatch-menu (symbol-value 'menu-bar-epatch-menu))
  1730.       (defvar menu-bar-ediff-menu (make-sparse-keymap "Ediff"))
  1731.       (fset 'menu-bar-ediff-menu (symbol-value 'menu-bar-ediff-menu))))
  1732.  
  1733.  
  1734. ;;;###autoload
  1735. (if purify-flag
  1736.     (progn
  1737.       (define-key menu-bar-ediff-menu [rcs-ediff]
  1738.     '("File with a version via RCS ..." . rcs-ediff))
  1739.       (define-key menu-bar-ediff-menu [vc-ediff]
  1740.     '("File with a version via VC ..." . vc-ediff))
  1741.       (define-key menu-bar-ediff-menu [ediff-buffers]
  1742.     '("Buffers ..." . ediff-buffers))
  1743.       (define-key menu-bar-ediff-menu [ediff-files]
  1744.     '("Files ..." . ediff-files))))
  1745.  
  1746. ;;;###autoload
  1747. (if purify-flag
  1748.     (progn
  1749.       (define-key menu-bar-epatch-menu [ediff-patch-buffer]
  1750.     '("To a Buffer ..." . ediff-patch-buffer))
  1751.       (define-key menu-bar-epatch-menu [ediff-patch-file]
  1752.     '("To a File ..." . ediff-patch-file))))
  1753.     
  1754. (define-key menu-bar-file-menu [epatch]
  1755.   '("Apply Patch" . menu-bar-epatch-menu))
  1756. (define-key menu-bar-file-menu [ediff]
  1757.   '("Compare" . menu-bar-ediff-menu))
  1758.  
  1759. (if (and window-system ediff-want-default-menus (ediff-frame-has-menubar)
  1760.      (ediff-if-lucid))
  1761.     (progn  ;; Lucid menu bars
  1762.       (defvar ediff-menu
  1763.     '(""
  1764.       ["Files ..."  ediff-files t]
  1765.       ["Buffers ..." ediff-buffers t]
  1766.       ["File with a version via VC ..."  vc-ediff t]
  1767.       ["File with a version via RCS ..."  rcs-ediff t]))
  1768.       (defvar epatch-menu
  1769.     '(""
  1770.       ["To a file ..."  ediff-patch-file t]
  1771.       ["To a buffer ..." ediff-patch-buffer t]))
  1772.       (add-menu '("File") "Compare" 
  1773.         ediff-menu
  1774.         "New Screen")
  1775.       (add-menu '("File") "Apply Patch" 
  1776.         epatch-menu
  1777.         "New Screen")
  1778.       ;; Displays as a solid horizontal line 
  1779.       (add-menu-item '("File") "---" nil nil "New Screen")))
  1780.  
  1781.  
  1782.  
  1783. (defun ediff-setup-keymap ()
  1784.   "Set up the keymap used in the control buffer of Ediff."
  1785.   (setq ediff-mode-map (make-sparse-keymap))
  1786.   (suppress-keymap ediff-mode-map)
  1787.   
  1788.   (define-key ediff-mode-map "p" 'ediff-previous-difference)
  1789.   (define-key ediff-mode-map "\C-?" 'ediff-previous-difference)
  1790.   (define-key ediff-mode-map "\C-h" (if ediff-no-help-in-control-buffer
  1791.                     'ediff-previous-difference nil))
  1792.   (define-key ediff-mode-map "n" 'ediff-next-difference)
  1793.   (define-key ediff-mode-map " " 'ediff-next-difference)
  1794.   (define-key ediff-mode-map "j" 'ediff-jump-to-difference)
  1795.   (define-key ediff-mode-map "g"  nil)
  1796.   (define-key ediff-mode-map "ga" 'ediff-jump-to-difference-at-point)
  1797.   (define-key ediff-mode-map "gb" 'ediff-jump-to-difference-at-point)
  1798.   (define-key ediff-mode-map "q" 'ediff-quit)
  1799.   (define-key ediff-mode-map "z" 'ediff-suspend)
  1800.   (define-key ediff-mode-map "c" 'ediff-recenter)
  1801.   (define-key ediff-mode-map "s" 'ediff-toggle-split)
  1802.   (define-key ediff-mode-map "h" 'ediff-toggle-hilit)
  1803.   (define-key ediff-mode-map "@" 'ediff-toggle-autorefine)
  1804.   (define-key ediff-mode-map "v" 'ediff-scroll-up)
  1805.   (define-key ediff-mode-map "\C-v" 'ediff-scroll-up)
  1806.   (define-key ediff-mode-map "^" 'ediff-scroll-down)
  1807.   (define-key ediff-mode-map "\M-v" 'ediff-scroll-down)
  1808.   (define-key ediff-mode-map "V" 'ediff-scroll-down)
  1809.   (define-key ediff-mode-map "<" 'ediff-scroll-left)
  1810.   (define-key ediff-mode-map ">" 'ediff-scroll-right)
  1811.   (define-key ediff-mode-map "i" 'ediff-status-info)
  1812.   (define-key ediff-mode-map "?" 'ediff-toggle-help)
  1813.   (define-key ediff-mode-map "!" 'ediff-recompute-diffs)
  1814.   (define-key ediff-mode-map "*" 'ediff-make-fine-diffs)
  1815.   (define-key ediff-mode-map "a"  nil)
  1816.   (define-key ediff-mode-map "ab" 'ediff-diff-to-diff)
  1817.   (define-key ediff-mode-map "b"  nil)
  1818.   (define-key ediff-mode-map "ba" 'ediff-diff-to-diff)
  1819.   (define-key ediff-mode-map "r"  nil)
  1820.   (define-key ediff-mode-map "ra" 'ediff-restore-diff)
  1821.   (define-key ediff-mode-map "rb" 'ediff-restore-diff)
  1822.   (define-key ediff-mode-map "#"  nil)
  1823.   (define-key ediff-mode-map "#h"  'ediff-toggle-regexp-match)
  1824.   (define-key ediff-mode-map "#f"  'ediff-toggle-regexp-match)
  1825.   (define-key ediff-mode-map "##"  'ediff-toggle-skip-similar)
  1826.   (define-key ediff-mode-map "o"   nil)
  1827.   (define-key ediff-mode-map "A"  'ediff-toggle-read-only)
  1828.   (define-key ediff-mode-map "B"  'ediff-toggle-read-only)
  1829.   (define-key ediff-mode-map "w"   nil)
  1830.   (define-key ediff-mode-map "wa"  'ediff-save-buffer)
  1831.   (define-key ediff-mode-map "wb"  'ediff-save-buffer)
  1832.   (define-key ediff-mode-map "wf"  'ediff-save-buffer)
  1833.   (define-key ediff-mode-map "k"   nil)
  1834.   (define-key ediff-mode-map "kkk" 'ediff-reload-keymap) ;; for debugging
  1835.   ;; Allow ediff-mode-map to be referenced indirectly
  1836.   (fset 'ediff-mode-map ediff-mode-map))
  1837.  
  1838.  
  1839. ;;; Setup functions
  1840.  
  1841. (defun ediff-find-file (file-var buffer &optional last-dir hooks-var)
  1842.   "Visit FILE and arrange its buffer to Ediff's liking. 
  1843. FILE is actually a variables symbol that must contain a true file name.
  1844. BUFFER is a variable symbol, which will get the buffer object into which
  1845. FILE is read.  LAST-DIR is the directory variable symbol where FILE's
  1846. directory name should be returned. HOOKS is a variable symbol that will be
  1847. assigned the hook to be executed after `ediff-strartup' is finished.
  1848. `ediff-find-file' arranges that the temp files it might create will be
  1849. deleted. 
  1850. Arguments: (file buffer &optional last-dir hooks)"
  1851.   (let* ((file (eval file-var))
  1852.      (file-magic (find-file-name-handler file 'find-file-noselect)))
  1853.     (if (not (file-readable-p file))
  1854.     (error "File `%s' does not exist or is not readable" file))
  1855.   
  1856.     ;; Record the directory of the file
  1857.     (if last-dir
  1858.     (set last-dir (expand-file-name (file-name-directory file))))
  1859.     
  1860.     ;; Setup the buffer
  1861.     (set buffer (find-file-noselect file))
  1862.   
  1863.     (ediff-eval-in-buffer 
  1864.      (eval buffer)
  1865.      (widen)  ;; Make sure the entire file is seen
  1866.      (cond (file-magic  ;; file has handler, such as jka-compr-handler or
  1867.                     ;; ange-ftp-hook-function--arrange for temp file
  1868.         (ediff-verify-file-buffer 'magic)
  1869.         (setq file (ediff-make-temp-file))
  1870.         (set hooks-var (cons (` (lambda () (delete-file (, file))))
  1871.                  (eval hooks-var))))
  1872.        ;; file processed via auto-mode-alist, a la uncompress.el
  1873.        ((not (equal (file-truename file)
  1874.             (file-truename (buffer-file-name))))
  1875.         (setq file (ediff-make-temp-file))
  1876.         (set hooks-var (cons (` (lambda () (delete-file (, file))))
  1877.                  (eval hooks-var))))
  1878.        (t ;; plain file---just check that the file matches the buffer
  1879.          (ediff-verify-file-buffer))))
  1880.     (set file-var file)))
  1881.  
  1882. (defun ediff-files-internal (file-A file-B &optional startup-hooks)
  1883.   (let (buffer-A buffer-B)
  1884.     (message "Reading file %s ... " file-A)(sit-for .5)
  1885.     (ediff-find-file 'file-A 'buffer-A 'ediff-last-dir-A 'startup-hooks)
  1886.     (message "Reading file %s ... " file-B)(sit-for .5)
  1887.     (ediff-find-file 'file-B 'buffer-B 'ediff-last-dir-B 'startup-hooks)
  1888.     (ediff-setup buffer-A file-A buffer-B file-B startup-hooks)))
  1889.   
  1890. (defun ediff-get-patch-buffer (dir)
  1891.   "Obtain patch buffer.  If patch is already in a buffer---use it.
  1892. Else, read patch file into a new buffer."
  1893.   (if (y-or-n-p "Is the patch file already in a buffer? ")
  1894.       (setq ediff-patch-buf
  1895.         (get-buffer (read-buffer "Patch buffer name: " nil t))) ;must match
  1896.     (setq ediff-patch-buf
  1897.       (find-file-noselect (read-file-name "Patch file name: " dir))))
  1898.   (ediff-eval-in-buffer
  1899.    ediff-patch-buf
  1900.    (toggle-read-only 1))
  1901.   (setq ediff-patch-diagnostics
  1902.     (get-buffer-create "*ediff patch diagnostics*"))
  1903.   (ediff-eval-in-buffer
  1904.    ediff-patch-diagnostics
  1905.    (insert-buffer ediff-patch-buf))
  1906.   )
  1907.  
  1908. ;; Start up Ediff on two files
  1909. (defun ediff-setup (buffer-A file-A buffer-B file-B startup-hooks)
  1910.   (setq file-A (expand-file-name file-A))
  1911.   (setq file-B (expand-file-name file-B))
  1912.   (let* ((control-buffer-name 
  1913.       (emerge-unique-buffer-name "Ediff Control Panel" ""))
  1914.      (control-buffer (ediff-eval-in-buffer
  1915.               buffer-A
  1916.               (get-buffer-create control-buffer-name))))
  1917.     (ediff-eval-in-buffer
  1918.      control-buffer
  1919.      (ediff-mode) ;; in control buffer only
  1920.      (setq buffer-read-only nil)
  1921.      (setq ediff-A-buffer buffer-A)
  1922.      (setq ediff-B-buffer buffer-B)
  1923.      (setq ediff-control-buffer control-buffer)
  1924. ;    (setq ediff-control-buffer-suffix
  1925. ;       (if (string-match "<[0-9]*>" control-buffer-name)
  1926. ;           (substring control-buffer-name
  1927. ;              (match-beginning 0) (match-end 0))
  1928. ;         "<1>"))
  1929.      (setq ediff-error-buffer (get-buffer-create (emerge-unique-buffer-name
  1930.                           "*ediff-errors" "*")))
  1931.      (ediff-remember-buffer-characteristics t) ;; remember at setup
  1932.      
  1933.      (ediff-set-keys)
  1934.      (setq ediff-difference-vector (ediff-setup-diff-regions file-A file-B))
  1935.      (setq ediff-number-of-differences (length ediff-difference-vector))
  1936.      (setq ediff-current-difference -1)
  1937.      (ediff-make-current-diff-overlay 'A)
  1938.      (ediff-make-current-diff-overlay 'B)
  1939.      (if window-system
  1940.      (ediff-init-var-faces))
  1941.      (run-hooks 'ediff-before-setup-windows-hooks)
  1942.      (ediff-setup-windows buffer-A buffer-B control-buffer t)
  1943.          
  1944.      ;; all these must be inside ediff-eval-in-buffer control-buffer,
  1945.      ;; since these vars are local to control-buffer
  1946.      ;; These won't run if there are errors in diff
  1947.      (ediff-eval-in-buffer
  1948.       ediff-A-buffer
  1949.       (add-hook 'local-write-file-hooks 'ediff-block-write-file)
  1950.       (setq before-change-function 'ediff-before-change-guard)
  1951.       ;; add control-buffer to the list of sessions
  1952.       (or (memq control-buffer ediff-this-buffer-control-sessions)
  1953.       (setq ediff-this-buffer-control-sessions
  1954.         (cons control-buffer ediff-this-buffer-control-sessions)))
  1955.       (setq mode-line-buffer-identification
  1956.         (cons "A: "
  1957.           (if (string-match "\\(^ \\|^[^ \t]*: \\)"
  1958.                     (car mode-line-buffer-identification))
  1959.               (cons (substring (car mode-line-buffer-identification)
  1960.                        (match-end 0))
  1961.                 (cdr mode-line-buffer-identification))
  1962.             mode-line-buffer-identification)))
  1963.       (run-hooks 'ediff-prepare-buffer-hooks))
  1964.      (ediff-eval-in-buffer
  1965.       ediff-B-buffer
  1966.       (add-hook 'local-write-file-hooks 'ediff-block-write-file)
  1967.       (setq before-change-function 'ediff-before-change-guard)
  1968.       ;; add control-buffer to the list of sessions
  1969.       (or (memq control-buffer ediff-this-buffer-control-sessions)
  1970.       (setq ediff-this-buffer-control-sessions
  1971.         (cons control-buffer ediff-this-buffer-control-sessions)))
  1972.       (setq mode-line-buffer-identification
  1973.         (cons "B: "
  1974.           (if (string-match "\\(^ \\|^[^ \t]*: \\)"
  1975.                     (car mode-line-buffer-identification))
  1976.               (cons (substring (car mode-line-buffer-identification)
  1977.                        (match-end 0))
  1978.                 (cdr mode-line-buffer-identification))
  1979.             mode-line-buffer-identification)))
  1980.       (run-hooks 'ediff-prepare-buffer-hooks))
  1981.  
  1982.      (ediff-eval-in-buffer control-buffer
  1983.                 (run-hooks 'startup-hooks 'ediff-startup-hooks)
  1984.                 (setq buffer-read-only t)))))
  1985.  
  1986. ;; Generate the difference vector and overlays for the two files
  1987. ;; With optional arg `refine', create refining difference regions
  1988. (defun ediff-setup-diff-regions (file-A file-B 
  1989.                     &optional use-old refine-region
  1990.                               diff-program diff-options
  1991.                           diff-ok-lines-regexp)
  1992.                           
  1993.   (setq diff-program (or diff-program ediff-diff-program)
  1994.     diff-options (or diff-options ediff-diff-options)
  1995.     diff-ok-lines-regexp
  1996.                  (or diff-ok-lines-regexp ediff-diff-ok-lines-regexp))
  1997.              
  1998.   (or use-old (setq ediff-diff-buffer 
  1999.             (get-buffer-create
  2000.              (emerge-unique-buffer-name "*ediff-diff" "*"))
  2001.             ediff-fine-diff-buffer
  2002.             (get-buffer-create
  2003.              (emerge-unique-buffer-name "*ediff-fine-diff" "*"))
  2004.             ))
  2005.   (ediff-eval-in-buffer
  2006.    (if refine-region ediff-fine-diff-buffer ediff-diff-buffer)
  2007.    (erase-buffer)
  2008.    ;; shell-command tends to display old shell command buffers even when it
  2009.    ;; puts output in another buffer---probably an Emacs bug.
  2010.    (ediff-kill-buffer-carefully "*Shell Command Output*")
  2011.    (let ((shell-file-name ediff-shell))
  2012.      (if refine-region
  2013.      (message "Refining difference region %d ..." (1+ refine-region))
  2014.        (message "Computing differences ...")(sit-for .5))
  2015.      (shell-command
  2016.       (format "%s %s %s %s"
  2017.           diff-program diff-options
  2018.           (ediff-protect-metachars file-A)
  2019.           (ediff-protect-metachars file-B))
  2020.       t)
  2021.      ))
  2022.   
  2023.    
  2024.   (if refine-region
  2025.       (progn
  2026.     (ediff-prepare-error-list diff-ok-lines-regexp ediff-fine-diff-buffer)
  2027.     (message "Refining difference region %d ... Done." (1+ refine-region))
  2028.     (ediff-convert-diffs-to-overlays-refine
  2029.      ediff-A-buffer ediff-B-buffer
  2030.      (ediff-extract-diffs ediff-fine-diff-buffer)
  2031.      refine-region))
  2032.     (ediff-prepare-error-list diff-ok-lines-regexp ediff-diff-buffer)
  2033.     (message "Computing differences ... Done.")(sit-for .5)
  2034.     (ediff-convert-diffs-to-overlays
  2035.      ediff-A-buffer ediff-B-buffer
  2036.      (ediff-extract-diffs ediff-diff-buffer
  2037.               ediff-A-buffer ediff-B-buffer))))
  2038.   
  2039.     
  2040. (defun ediff-prepare-error-list (ok-regexp diff-buff)
  2041.     (ediff-eval-in-buffer
  2042.      ediff-error-buffer
  2043.      (erase-buffer)
  2044.      (insert-buffer diff-buff)
  2045.      (delete-matching-lines ok-regexp)))
  2046.  
  2047. ;;; Function to start Ediff by patching a file
  2048.  
  2049. ;;;###autoload
  2050. (defun ediff-patch-file (source-filename &optional startup-hooks)
  2051.   "Run Ediff by patching FILE-TP-PATCH."
  2052.   (interactive 
  2053.    (list (ediff-read-file-name "File to patch"
  2054.                    (if ediff-use-last-dir
  2055.                    ediff-last-dir-patch
  2056.                  default-directory)
  2057.                    nil)))
  2058.   
  2059.   (ediff-get-patch-buffer (file-name-directory source-filename))
  2060.   (let* ((backup-extension 
  2061.       ;; if the user specified a -b option, extract the backup
  2062.       ;; extension from there; else use `.orig'
  2063.       (substring ediff-patch-options
  2064.              (if (string-match "-b[ \t]+" ediff-patch-options)
  2065.              (match-end 0) 0)
  2066.              (if (string-match "-b[ \t]+[^ \t]+" ediff-patch-options)
  2067.              (match-end 0) 0)))
  2068.      (backup-extension (if (string= backup-extension "")
  2069.                    "orig" backup-extension))
  2070.      (shell-file-name ediff-shell)
  2071.      ;; ediff-find-file may use a temp file to do the patch
  2072.      ;; so, we save source-filename and true-source-filename as a var
  2073.      ;; that initially is source-filename but may be changed to a temp
  2074.      ;; file for the purpose of patching.
  2075.      (true-source-filename source-filename)
  2076.      (target-filename source-filename)
  2077.      target-buf buf-to-patch file-name-magic-p)
  2078.                     
  2079.     ;; Make a temp file, if source-filename has a magic file handler (or if
  2080.     ;; it is handled via auto-mode-alist and similar magic).
  2081.     ;; Check if there is a buffer visiting source-filename and if they are in
  2082.     ;; synch; arrange for the deletion of temp file.
  2083.     (ediff-find-file 'true-source-filename 'buf-to-patch
  2084.              'ediff-last-dir-patch 'startup-hooks)
  2085.  
  2086.     ;; Check if source file name has triggered black magic, such as file name
  2087.     ;; handlers or auto mode alist, and make a note of it.
  2088.     (setq file-name-magic-p (not (equal (file-truename true-source-filename)
  2089.                     (file-truename source-filename))))
  2090.     
  2091.     (ediff-eval-in-buffer
  2092.      ediff-patch-diagnostics
  2093.      (message "Applying patch ... ")(sit-for .5)
  2094.      ;; always pass patch the -f option, so it won't ask any questions
  2095.      (shell-command-on-region 
  2096.       (point-min) (point-max)
  2097.       (format "%s -f %s %s"
  2098.           ediff-patch-program ediff-patch-options
  2099.           (expand-file-name true-source-filename))
  2100.       t))
  2101.     (message "Applying patch ... Done.")(sit-for .5)
  2102.     (switch-to-buffer ediff-patch-diagnostics)
  2103.     (sit-for 0) ;; synchronize
  2104.     
  2105.     (or (file-exists-p (concat true-source-filename "." backup-extension))
  2106.     (error "Patch failed or didn't modify the original file."))
  2107.   
  2108.     ;; If black magic is involved, apply patch to a temp copy of the
  2109.     ;; file. Otherwise, apply patch to the orig copy.
  2110.     ;; If patch is applied to temp copy, we name the result
  2111.     ;; ***.patched. The orig file name isn't changed, and the temp copy of
  2112.     ;; the original is later deleted.
  2113.     ;; Without magic, the original file is renamed (usually into
  2114.     ;; old-name.orig) and the result of patching will have the 
  2115.     ;; same name as the original.
  2116.     (if (not file-name-magic-p)
  2117.     (ediff-eval-in-buffer
  2118.      buf-to-patch
  2119.      (set-visited-file-name (concat source-filename "." backup-extension))
  2120.      (set-buffer-modified-p nil))
  2121.       (setq target-filename (concat true-source-filename ".patched"))
  2122.       (rename-file true-source-filename target-filename t)
  2123.       
  2124.       ;; arrange that the temp copy of orig will be deleted
  2125.       (rename-file (concat true-source-filename "." backup-extension)
  2126.            true-source-filename t))
  2127.     
  2128.     ;; make orig buffer read-only
  2129.     (setq startup-hooks (cons 'ediff-toggle-read-only-A startup-hooks))
  2130.     ;; set up a buf for the patched file
  2131.     (setq target-buf (find-file-noselect target-filename))
  2132.     
  2133.     (ediff-buffers buf-to-patch target-buf startup-hooks)
  2134.   
  2135.     (bury-buffer ediff-patch-diagnostics)
  2136.     (message "Patch diagnostics available in buffer %s."
  2137.          (buffer-name ediff-patch-diagnostics))))
  2138.  
  2139. (defalias 'epatch 'ediff-patch-file)
  2140. (defalias 'epatch-buffer 'ediff-patch-buffer)
  2141.  
  2142. ;;; Function to start Ediff on files
  2143.  
  2144. ;;;###autoload
  2145. (defun ediff-files (file-A file-B &optional startup-hooks)
  2146.   "Run Ediff on a pair files, FILE-A and FILE-B."
  2147.   (interactive
  2148.    (let (f)
  2149.      (list (setq f (ediff-read-file-name "File A to compare" 
  2150.                      (if ediff-use-last-dir
  2151.                          ediff-last-dir-A
  2152.                        default-directory)
  2153.                      nil))
  2154.        (ediff-read-file-name "File B to compare" 
  2155.                  (if ediff-use-last-dir
  2156.                      ediff-last-dir-B  nil)
  2157.                  f)
  2158.        )))
  2159.   (ediff-files-internal file-A 
  2160.             (if (file-directory-p file-B)
  2161.                 (expand-file-name
  2162.                  (file-name-nondirectory file-A) file-B)
  2163.               file-B)
  2164.             startup-hooks))
  2165.  
  2166.  
  2167. (defalias 'ediff 'ediff-files)
  2168.  
  2169.  
  2170. ;;; Function to start Ediff on buffers
  2171.  
  2172. ;;;###autoload
  2173. (defun ediff-buffers (buffer-A buffer-B &optional startup-hooks)
  2174.   "Run Ediff on a pair of buffers, BUFFER-A and BUFFER-B."
  2175.   (interactive 
  2176.    (list (read-buffer "Buffer A to compare: " (current-buffer) t)
  2177.      (read-buffer "Buffer B to compare: "
  2178.               (progn
  2179.             ;; realign buffers so that two visible bufs will be
  2180.             ;; at the top
  2181.             (save-window-excursion (other-window 1))
  2182.             (other-buffer (current-buffer) t))
  2183.               t)))
  2184.   (if (not (ediff-buffer-live-p buffer-A))
  2185.       (error "Buffer %S doesn't exist." buffer-A))
  2186.   (if (not (ediff-buffer-live-p buffer-B))
  2187.       (error "Buffer %S doesn't exist." buffer-B))
  2188.       
  2189.   (let (file-A file-B)
  2190.     (ediff-eval-in-buffer
  2191.      buffer-A
  2192.      (setq file-A (ediff-make-temp-file)))
  2193.     (ediff-eval-in-buffer
  2194.      buffer-B
  2195.      (setq file-B (ediff-make-temp-file)))
  2196.     (ediff-setup (get-buffer buffer-A) file-A
  2197.          (get-buffer buffer-B) file-B
  2198.          (cons (` (lambda ()
  2199.                 (delete-file (, file-A))
  2200.                 (delete-file (, file-B))))
  2201.                startup-hooks)
  2202.          )))
  2203.           
  2204. ;;;###autoload
  2205. (defun ediff-patch-buffer (buffer-name &optional startup-hooks)          
  2206.   "Run Ediff by patching BUFFER-NAME."
  2207.   (interactive "bBuffer to patch: ")
  2208.   
  2209.   (let* ((file-buffer (get-buffer buffer-name))
  2210.      (file-name (if file-buffer (buffer-file-name  file-buffer))))
  2211.     (if (not file-name)
  2212.     (error "Buffer %s doesn't exist or doesn't visit any file.  Why patch?"
  2213.            file-name))
  2214.     
  2215.     (ediff-patch-file file-name startup-hooks)))
  2216.       
  2217.  
  2218. ;;; Versions Control functions      
  2219.       
  2220. ;;;###autoload
  2221. (defun vc-ediff (rev)
  2222. ;; Note: this function will work only with Emacs 19.22 and higher.
  2223.   "Run ediff on version REV of the current buffer in another window.
  2224. If the current buffer is named `F', the version is named `F.~REV~'.
  2225. If `F.~REV~' already exists, it is used instead of being re-created."
  2226.   (interactive "sVersion to ediff with (default is the latest version): ")
  2227.   (or (featurep 'vc)
  2228.       (if (locate-library "vc") ;; if vc.el is available
  2229.       (progn
  2230.         (require 'vc-hooks) 
  2231.         (define-key vc-prefix-map "=" 'vc-ediff))
  2232.     (error "The VC package is apparently not installed.")))
  2233.   (let ((newvers (current-buffer))
  2234.     )
  2235.     (vc-version-other-window rev)
  2236.     ;; current-buffer is supposed to contain the old version in another
  2237.     ;; window
  2238.     (ediff-buffers newvers (current-buffer)) 
  2239.     ))
  2240.     
  2241. (defun rcs-ediff-view-revision (&optional rev)
  2242.   "View previous RCS revision of current file.
  2243. With prefix argument, prompts for a revision name." 
  2244.   (interactive (list (if current-prefix-arg 
  2245.              (read-string "Revision: "))))
  2246.   (let* ((filename (buffer-file-name (current-buffer)))
  2247.      (switches (append '("-p")
  2248.                (if rev (list (concat "-r" rev)) nil)))
  2249.      (buff (concat (file-name-nondirectory filename) ".~" rev "~")))
  2250.     (message "Working...")
  2251.     (setq filename (expand-file-name filename))
  2252.     (with-output-to-temp-buffer buff
  2253.       (let ((output-buffer (ediff-rcs-get-output-buffer filename buff)))
  2254.     (delete-windows-on output-buffer)
  2255.     (save-excursion
  2256.       (set-buffer output-buffer)
  2257.       (apply 'call-process "co" nil t nil
  2258.          ;; -q: quiet (no diagnostics)
  2259.          (append switches rcs-default-co-switches
  2260.              (list "-q" filename))))) 
  2261.       (message "")
  2262.       buff)))    
  2263.       
  2264. (defun ediff-rcs-get-output-buffer (file name)
  2265.   ;; Get a buffer for RCS output for FILE, make it writable and clean it up.
  2266.   ;; Optional NAME is name to use instead of `*RCS-output*'.
  2267.   ;; This is a modified version from rcs.el v1.1. I use it here to make
  2268.   ;; Ediff immune to changes in rcs.el
  2269.   (let* ((default-major-mode 'fundamental-mode);; no frills!
  2270.      (buf (get-buffer-create name)))
  2271.     (save-excursion
  2272.       (set-buffer buf)
  2273.       (setq buffer-read-only nil
  2274.         default-directory (file-name-directory (expand-file-name file)))
  2275.       (erase-buffer))
  2276.     buf))
  2277.  
  2278. ;;;###autoload
  2279. (defun rcs-ediff (&optional rev)
  2280.   "Run Ediff on the current buffer, comparing it with previous RCS revision.  
  2281. With prefix argument, prompts for revision name." 
  2282.   (interactive (list (if current-prefix-arg 
  2283.              (read-string "Revision: "))))
  2284.   (or (featurep 'rcs)
  2285.       (if (locate-library "rcs")      
  2286.       (progn
  2287.         (require 'rcs)
  2288.         (global-set-key "\C-cD" 'rcs-ediff))
  2289.     (error "The RCS package is apparently not installed.")))
  2290.   (let ((newvers (current-buffer))
  2291.     (oldvers (rcs-ediff-view-revision rev)))
  2292.     (ediff-buffers newvers oldvers)
  2293.     ))
  2294.  
  2295.  
  2296.  
  2297.  
  2298. ;; Select the lowest window on the frame.
  2299. (defun ediff-select-lowest-window ()
  2300.   (let* ((lowest-window (selected-window))
  2301.      (bottom-edge (car (cdr (cdr (cdr (window-edges))))))
  2302.          (last-window (previous-window))
  2303.          (window-search t))
  2304.     (while window-search
  2305.       (let* ((this-window (next-window))
  2306.              (next-bottom-edge (car (cdr (cdr (cdr 
  2307.                                                (window-edges this-window)))))))
  2308.         (if (< bottom-edge next-bottom-edge)
  2309.             (progn
  2310.               (setq bottom-edge next-bottom-edge)
  2311.               (setq lowest-window this-window)))
  2312.  
  2313.         (select-window this-window)
  2314.         (if (eq last-window this-window)
  2315.             (progn
  2316.               (select-window lowest-window)
  2317.               (setq window-search nil)))))))
  2318.  
  2319. ;;; Common setup routines
  2320.  
  2321. ;; Set up the window configuration.  If POS is given, set the points to
  2322. ;; the beginnings of the buffers.
  2323. (defun ediff-setup-windows (buffer-A buffer-B control-buffer &optional pos)
  2324.   ;; Make sure we are not in the minibuffer window when we try to delete
  2325.   ;; all other windows.
  2326.   (if (eq (selected-window) (minibuffer-window))
  2327.       (other-window 1))
  2328.   (or (ediff-leave-window-config control-buffer)
  2329.       (progn
  2330.     (delete-other-windows)
  2331.     (switch-to-buffer control-buffer)
  2332.     (ediff-refresh-mode-line)
  2333.   
  2334.     (ediff-arrange-buffer buffer-A buffer-B (current-buffer) pos)
  2335.     (ediff-arrange-buffer buffer-B buffer-A (current-buffer) pos)
  2336.     ;; ediff-arrange-buffer always leaves in ctl buffer
  2337.     ;; setup ctl wind if it is not set.
  2338.     (ediff-setup-control-window)
  2339.   
  2340.     ;; If diff reports errors, show them then quit.
  2341.     (if (/= 0 (ediff-eval-in-buffer ediff-error-buffer (buffer-size)))
  2342.         (let ((diff-output-buf  ediff-diff-buffer))
  2343.           (switch-to-buffer ediff-error-buffer)
  2344.           (ediff-kill-buffer-carefully control-buffer)
  2345.           (error "Errors found in diff output.  Diff output buffer is %s"
  2346.              diff-output-buf))))))
  2347.  
  2348.  
  2349. ;; Arranges goal-buf on the screen.
  2350. (defun ediff-arrange-buffer (goal-buf other-buf ctl-buf &optional pos)
  2351.   (let* ((ctl-wind (get-buffer-window ctl-buf t))
  2352.      (goal-wind (get-buffer-window goal-buf t))
  2353.      (other-wind (get-buffer-window other-buf t))
  2354.      (ctl-frame (ediff-window-frame ctl-wind))
  2355.      (goal-frame (if goal-wind (ediff-window-frame goal-wind)))
  2356.      (other-frame (if other-wind (ediff-window-frame other-wind)))
  2357.      (ctl-frame-shared (or (eq ctl-frame goal-frame)
  2358.                    (eq ctl-frame other-frame))))
  2359.               
  2360.     (cond ((and goal-frame (not (eq goal-wind other-wind)))
  2361.                 ;; goal buffer is visible and we are not comparing file
  2362.             ;; against itself (by mistake).
  2363.             ;; Note:  goal-frame != ctl-frame, as we deleted other
  2364.             ;; windows  on ctl-frame.
  2365.                   (ediff-select-frame goal-frame)
  2366.               (select-window goal-wind)
  2367.               (delete-other-windows)) 
  2368.                                           
  2369.       ;; goal-buf invisible, ctl-frame has only ctl-buf
  2370.       ;; then put goal-buf on ctl-frame
  2371.       ((null ctl-frame-shared)
  2372.               (ediff-select-frame ctl-frame)
  2373.               (split-window-vertically)
  2374.               (ediff-select-lowest-window)
  2375.               (setq ctl-wind (selected-window))
  2376.               (switch-to-buffer ctl-buf)
  2377.               (ediff-setup-control-window)
  2378.               (other-window 1)
  2379.               (switch-to-buffer goal-buf)) ; goal-buf set
  2380.       ;; goal-buf invisible, ctl-frame has ctl-buf and other-buf
  2381.       ;; So, put everything in one frame
  2382.       (other-frame   ;; share with the other buf
  2383.                    (ediff-select-frame ctl-frame)
  2384.                (select-window other-wind)
  2385.                (funcall ediff-split-window-function)
  2386.                (other-window 1)
  2387.                (switch-to-buffer goal-buf))
  2388.       (t ;; debug
  2389.        (error "Funny window combination (Ediff bug?)")))
  2390.  
  2391.       (if pos
  2392.       (goto-char (point-min)))
  2393.       
  2394.       (ediff-select-frame ctl-frame)
  2395.       (select-window ctl-wind)
  2396.       (switch-to-buffer ctl-buf)))
  2397.       
  2398. ;; This function assumes that we are in the window where control buffer is
  2399. ;; to reside.
  2400. (defun ediff-setup-control-window ()
  2401.   "Set up window for control buffer."
  2402.   (erase-buffer)
  2403.   (insert ediff-help-message)
  2404.   (shrink-window-if-larger-than-buffer)
  2405.   (setq ediff-control-window (selected-window))
  2406.   (setq ediff-window-config-saved
  2407.     (format "%S%S%S%S"
  2408.         ediff-control-window
  2409.         (get-buffer-window ediff-A-buffer t)
  2410.         (get-buffer-window ediff-B-buffer t)
  2411.         ediff-split-window-function))
  2412.   (goto-char (point-min))
  2413.   (skip-chars-forward ediff-whitespace))
  2414.   
  2415. (defun ediff-leave-window-config (control-buf)
  2416.   (and (eq control-buf (current-buffer))
  2417.        (/= (buffer-size) 0)
  2418.        (ediff-eval-in-buffer
  2419.     control-buf
  2420.     (string= ediff-window-config-saved
  2421.          (format "%S%S%S%S"
  2422.              (get-buffer-window ediff-control-buffer t)
  2423.              (get-buffer-window ediff-A-buffer t)
  2424.              (get-buffer-window ediff-B-buffer t)
  2425.              ediff-split-window-function)))))
  2426.       
  2427.  
  2428. ;; Set up the keymap in the control buffer
  2429. (defun ediff-set-keys ()
  2430.   "Set up Ediff keymap, if necessary."
  2431.   (if (null ediff-mode-map)
  2432.       (ediff-setup-keymap))
  2433.   (use-local-map ediff-mode-map))
  2434.   
  2435. ;; Reload Ediff keymap.  For debugging only.
  2436. (defun ediff-reload-keymap ()
  2437.   (interactive)
  2438.   (setq ediff-mode-map nil)
  2439.   (ediff-set-keys))
  2440.  
  2441. (defun ediff-before-change-guard (start end)
  2442.   "If buffer is highlighted with ASCII flags, remove highlighting.
  2443. Arguments, START and END are not used, but are provided
  2444. because this is required by `before-change-function'."
  2445.   (let (rehighlight-key)
  2446.     (save-window-excursion
  2447.       (mapcar
  2448.        (function
  2449.     (lambda (buf)
  2450.       (ediff-eval-in-buffer
  2451.        buf
  2452.        (if (eq ediff-highlighting-style 'ascii)
  2453.            (progn
  2454.          (ediff-unselect-and-select-difference
  2455.           ediff-current-difference 
  2456.           'unselect-only 'no-recenter)
  2457.          (setq rehighlight-key
  2458.                (substitute-command-keys "\\[ediff-recenter]"))
  2459.          )))))
  2460.        ediff-this-buffer-control-sessions)
  2461.       (if rehighlight-key
  2462.       (error 
  2463.        "ASCII flags removed. You can edit now. Hit %S to rehighlight."
  2464.        rehighlight-key))
  2465.      )))
  2466.      
  2467. (defun ediff-recompute-diffs ()
  2468.   "Recompute difference regions in buffers A and B."
  2469.   (interactive)
  2470.   (let ((point-A (ediff-eval-in-buffer ediff-A-buffer (point)))
  2471.     (point-B (ediff-eval-in-buffer ediff-B-buffer (point)))
  2472.     file-A file-B)
  2473.     (ediff-unselect-and-select-difference -1)
  2474.     (ediff-eval-in-buffer
  2475.      ediff-A-buffer
  2476.      (setq file-A (ediff-make-temp-file)))
  2477.     (ediff-eval-in-buffer
  2478.      ediff-B-buffer
  2479.      (setq file-B (ediff-make-temp-file)))
  2480.     (ediff-clear-diff-vector ediff-difference-vector 'fine-diffs-also)
  2481.     (setq ediff-killed-diffs-alist nil) ; saved kills will no longer be valid
  2482.                                         ; after recompute
  2483.     (setq ediff-difference-vector
  2484.       (ediff-setup-diff-regions file-A file-B 'use-old))
  2485.     (setq ediff-number-of-differences (length ediff-difference-vector))
  2486.     (delete-file file-A)
  2487.     (delete-file file-B)
  2488.     (ediff-eval-in-buffer ediff-A-buffer (goto-char point-A))
  2489.     (ediff-jump-to-difference (ediff-diff-at-point 'A))
  2490.     (beep 1)
  2491.     (if (y-or-n-p 
  2492.     "Ediff is at last posn in buff A. Stay (or goto last posn in B)? ")
  2493.     ()
  2494.       (ediff-eval-in-buffer ediff-B-buffer (goto-char point-B))
  2495.       (ediff-jump-to-difference (ediff-diff-at-point 'B)))
  2496.     (message "")
  2497.     ))
  2498.  
  2499. (defun ediff-remember-buffer-characteristics (&optional arg)
  2500.   "Record certain properties of the buffers being compared.
  2501. Must be called in the control buffer.  Saves `read-only', `modified',
  2502. and `auto-save' properties in buffer local variables.  Turns off
  2503. `auto-save-mode'.  These properties are restored via a call to
  2504. `ediff-restore-buffer-characteristics'."
  2505.  
  2506.   ;; remember and alter buffer characteristics
  2507.   (set  (if arg 'ediff-A-buffer-values-setup 'ediff-A-buffer-values)
  2508.     (ediff-eval-in-buffer
  2509.      ediff-A-buffer
  2510.      (prog1
  2511.          (emerge-save-variables ediff-saved-variables)
  2512.        (emerge-restore-variables ediff-saved-variables
  2513.                      ediff-working-values))))
  2514.   (set  (if arg 'ediff-B-buffer-values-setup 'ediff-B-buffer-values)
  2515.     (ediff-eval-in-buffer
  2516.      ediff-B-buffer
  2517.      (prog1
  2518.          (emerge-save-variables ediff-saved-variables)
  2519.        (emerge-restore-variables ediff-saved-variables
  2520.                      ediff-working-values)))))
  2521.  
  2522. (defun ediff-restore-buffer-characteristics (&optional arg)
  2523.   "Restores properties saved by `ediff-remember-buffer-characteristics'."
  2524.   (let ((A-values (if arg ediff-A-buffer-values-setup ediff-A-buffer-values))
  2525.     (B-values (if arg ediff-B-buffer-values-setup ediff-B-buffer-values)))
  2526.     
  2527.     (ediff-eval-in-buffer ediff-A-buffer
  2528.               (emerge-restore-variables ediff-saved-variables
  2529.                             A-values))
  2530.     (ediff-eval-in-buffer ediff-B-buffer
  2531.               (emerge-restore-variables ediff-saved-variables
  2532.                             B-values))))
  2533.  
  2534.  
  2535. ;; If optional A-buffer and B-buffer are given, then construct a vector of
  2536. ;; diff using point values. Otherwise, use line offsets.
  2537. (defun ediff-extract-diffs (diff-buffer &optional A-buffer B-buffer)
  2538.   (let (diff-list
  2539.     (a-prev 1) ;; this is needed to set the first diff line correctly
  2540.     (b-prev 1))
  2541.  
  2542.     (if (and A-buffer B-buffer)
  2543.     (progn    ;; reset point in buffers A and B
  2544.       (ediff-eval-in-buffer
  2545.        A-buffer
  2546.        (goto-char (point-min)))
  2547.       (ediff-eval-in-buffer
  2548.        B-buffer
  2549.        (goto-char (point-min)))))
  2550.     
  2551.     (ediff-eval-in-buffer
  2552.      diff-buffer
  2553.      (goto-char (point-min))
  2554.      (while (re-search-forward ediff-match-diff-line nil t)
  2555.        (let* ((a-begin (string-to-int (buffer-substring (match-beginning 1)
  2556.                             (match-end 1))))
  2557.           (a-end  (let ((b (match-beginning 3))
  2558.                 (e (match-end 3)))
  2559.             (if b
  2560.                 (string-to-int (buffer-substring b e))
  2561.               a-begin)))
  2562.           (diff-type (buffer-substring (match-beginning 4) (match-end 4)))
  2563.           (b-begin (string-to-int (buffer-substring (match-beginning 5)
  2564.                             (match-end 5))))
  2565.           (b-end (let ((b (match-beginning 7))
  2566.                (e (match-end 7)))
  2567.                (if b
  2568.                (string-to-int (buffer-substring b e))
  2569.              b-begin)))
  2570.           a-begin-pt a-end-pt b-begin-pt b-end-pt)
  2571.      ;; fix the beginning and end numbers, because diff is somewhat
  2572.      ;; strange about how it numbers lines
  2573.      (if (string-equal diff-type "a")
  2574.          (setq b-end (1+ b-end)
  2575.            a-begin (1+ a-begin)
  2576.            a-end a-begin)
  2577.        (if (string-equal diff-type "d")
  2578.            (setq a-end (1+ a-end)
  2579.              b-begin (1+ b-begin)
  2580.              b-end b-begin)
  2581.          ;; (string-equal diff-type "c")
  2582.          (setq a-end (1+ a-end)
  2583.            b-end (1+ b-end))))
  2584.      (if (and A-buffer B-buffer)
  2585.          (progn    ;; computing main diff vector
  2586.            ;; convert to relative line numbers    
  2587.            (ediff-eval-in-buffer
  2588.         A-buffer
  2589.         (forward-line (- a-begin a-prev))
  2590.         (setq a-begin-pt (point))
  2591.         (forward-line (- a-end a-begin))
  2592.         (setq a-end-pt (point)
  2593.               a-prev a-end))
  2594.            (ediff-eval-in-buffer
  2595.         B-buffer
  2596.         (forward-line (- b-begin b-prev))
  2597.         (setq b-begin-pt (point))
  2598.         (forward-line (- b-end b-begin))
  2599.         (setq b-end-pt (point)
  2600.               b-prev b-end))
  2601.            (setq diff-list 
  2602.              (nconc diff-list (list (vector a-begin-pt a-end-pt
  2603.                             b-begin-pt b-end-pt)))))
  2604.        ;; computing refinement vector
  2605.        (setq  diff-list 
  2606.           (nconc diff-list (list (vector (- a-begin a-prev)
  2607.                          (- a-end a-begin)
  2608.                          (- b-begin b-prev)
  2609.                          (- b-end b-begin))))
  2610.           a-prev a-end
  2611.           b-prev b-end))
  2612.           
  2613.      ))) ;; end ediff-eval-in-buffer
  2614.     diff-list
  2615.     ))
  2616.     
  2617. (defun ediff-convert-diffs-to-overlays (A-buffer B-buffer diff-list)
  2618.   (let* ((current-diff -1)
  2619.      (total-diffs (length diff-list))
  2620. ;     (control-buffer-suffix ediff-control-buffer-suffix)
  2621.      diff-overlay-list list-element
  2622.      a-begin a-end b-begin b-end
  2623.      a-overlay b-overlay)
  2624.  
  2625.     (while diff-list
  2626.       (setq current-diff (1+ current-diff)
  2627.         list-element (car diff-list)
  2628.         a-begin      (aref list-element 0)
  2629.         a-end      (aref list-element 1)
  2630.         b-begin      (aref list-element 2)
  2631.         b-end      (aref list-element 3))
  2632.         
  2633.       ;; Put overlays at appropriate places in buffers
  2634.       (setq a-overlay (ediff-make-overlay a-begin a-end A-buffer))
  2635.       ;; Priorities of overlays should be equal in all ediff control
  2636.       ;; panels buffers. Otherwise it won't work due to Emacs
  2637.       ;; bug---insert-in-front-hooks will be called 
  2638.       ;; only on behalf of the buffer with higher priority.
  2639.       (ediff-overlay-put a-overlay 'priority ediff-shaded-overlay-priority)
  2640.       (ediff-overlay-put a-overlay 'ediff-diff-num current-diff)
  2641.       (ediff-overlay-put a-overlay
  2642.              'insert-in-front-hooks '(ediff-insert-in-front))
  2643. ;     (ediff-overlay-put a-overlay
  2644. ;             'ediff-control-buffer control-buffer-suffix)
  2645.       (ediff-overlay-put a-overlay 
  2646.              'face (if (ediff-odd-p current-diff) ;; odd diff
  2647.                    'ediff-odd-diff-face-A-var
  2648.                  'ediff-even-diff-face-A-var))
  2649.              
  2650.       (setq b-overlay (ediff-make-overlay b-begin b-end B-buffer))
  2651.       (ediff-overlay-put b-overlay 'priority ediff-shaded-overlay-priority)
  2652.       (ediff-overlay-put b-overlay 'ediff-diff-num current-diff)
  2653.       (ediff-overlay-put b-overlay
  2654.              'insert-in-front-hooks '(ediff-insert-in-front))
  2655. ;     (ediff-overlay-put b-overlay
  2656. ;             'ediff-control-buffer control-buffer-suffix)
  2657.       (ediff-overlay-put b-overlay 
  2658.              'face (if (ediff-odd-p current-diff) ;; odd diff
  2659.                    'ediff-odd-diff-face-B-var
  2660.                  'ediff-even-diff-face-B-var))
  2661.                  
  2662.       (if (ediff-if-lucid) ;; chars inserted at end will be inside extent
  2663.       (progn
  2664.         (ediff-overlay-put a-overlay
  2665.                    'ediff-marker 
  2666.                    (move-marker (make-marker) a-begin A-buffer))
  2667.         (ediff-overlay-put b-overlay
  2668.                    'ediff-marker 
  2669.                    (move-marker (make-marker) b-begin B-buffer))
  2670.         (ediff-overlay-put a-overlay 'end-open nil)
  2671.         (ediff-overlay-put b-overlay 'end-open nil)))
  2672.              
  2673.       ;; record all overlays for this difference
  2674.       (setq diff-overlay-list
  2675.         (nconc diff-overlay-list
  2676.            (list (vector a-overlay b-overlay nil nil)))
  2677.         diff-list
  2678.         (cdr diff-list))
  2679.       (message "Processing diff region %d of %d"
  2680.            current-diff total-diffs)
  2681.       ) ;; while
  2682.     ;; this is just to avoid confusing the user with diff num < total-diffs
  2683.     (message "Processing diff region %d of %d"
  2684.            (1+ current-diff) total-diffs)
  2685.     ;; convert the list of difference information into a vector for
  2686.     ;; fast access
  2687.     (apply 'vector diff-overlay-list)))
  2688.  
  2689.  
  2690.  
  2691. ;;; Commands
  2692.  
  2693. (defun ediff-recenter (&optional no-rehighlight)
  2694.   "Bring the highlighted region of all buffers A and B into view.
  2695. Reestablish the default three-window display."
  2696.   (interactive)
  2697.   (setq ediff-disturbed-overlays nil) ;; clear after use
  2698.   (let (buffer-read-only)
  2699.     (if (and (ediff-buffer-live-p ediff-A-buffer)
  2700.          (ediff-buffer-live-p ediff-B-buffer))
  2701.     (ediff-setup-windows
  2702.      ediff-A-buffer ediff-B-buffer ediff-control-buffer)))
  2703.   ;; Redisplay whatever buffers are showing, if there is a selected difference
  2704.   (if (and (ediff-buffer-live-p ediff-A-buffer)
  2705.        (ediff-buffer-live-p ediff-B-buffer)
  2706.        (>= ediff-current-difference 0)
  2707.        (< ediff-current-difference ediff-number-of-differences))
  2708.       (let* ( ;; context must be saved before switching to windows A/B
  2709.          (buffer-A ediff-A-buffer)
  2710.          (buffer-B ediff-B-buffer)
  2711.          (wind (selected-window))
  2712.          (control-buf ediff-control-buffer)
  2713.          (before-flag-shift-A (if (eq ediff-highlighting-style 'ascii)
  2714.                       (1- (length ediff-before-flag-A))
  2715.                     0))
  2716.          (after-flag-shift-A (if (eq ediff-highlighting-style 'ascii)
  2717.                      (1- (length ediff-after-flag-A))
  2718.                    0))
  2719.          (before-flag-shift-B (if (eq ediff-highlighting-style 'ascii)
  2720.                       (1- (length ediff-before-flag-B))
  2721.                     0))
  2722.          (after-flag-shift-B (if (eq ediff-highlighting-style 'ascii)
  2723.                      (1- (length ediff-after-flag-B))
  2724.                    0))
  2725.          (window-A (get-buffer-window buffer-A t))
  2726.          (window-B (get-buffer-window buffer-B t)))
  2727.     
  2728.     (or no-rehighlight
  2729.         (ediff-operate-on-flags 'insert))
  2730.         
  2731.     (if window-A (progn
  2732.                (select-window window-A)
  2733.                (ediff-position-region
  2734.             (- (ediff-get-diff-posn 'A 'beg nil control-buf)
  2735.                before-flag-shift-A)
  2736.             (+ (ediff-get-diff-posn 'A 'end nil control-buf)
  2737.                after-flag-shift-A)
  2738.             (ediff-get-diff-posn 'A 'beg nil control-buf))))
  2739.     (if window-B (progn
  2740.                (select-window window-B)
  2741.                (ediff-position-region
  2742.             (- (ediff-get-diff-posn 'B 'beg nil control-buf)
  2743.                before-flag-shift-B)
  2744.             (+ (ediff-get-diff-posn 'B 'end nil control-buf)
  2745.                after-flag-shift-B)
  2746.             (ediff-get-diff-posn 'B 'beg nil control-buf))))
  2747.     (select-window wind))))
  2748.     
  2749. (defun ediff-toggle-split ()
  2750.   "Toggle vertical/horizontal window split. 
  2751. Does nothing if file-A and file-B are in different frames."
  2752.   (interactive)
  2753.   (let* ((wind-A (get-buffer-window ediff-A-buffer t))
  2754.      (wind-B (get-buffer-window ediff-B-buffer t))
  2755.      (frame-A (if wind-A (ediff-window-frame wind-A)))
  2756.      (frame-B (if wind-B (ediff-window-frame wind-B))))
  2757.     (if (eq frame-A frame-B)
  2758.     (setq ediff-split-window-function
  2759.           (if (eq ediff-split-window-function 'split-window-vertically)
  2760.           'split-window-horizontally
  2761.         'split-window-vertically))
  2762.       (message "No splitting: Buffers A and B are in different frames."))
  2763.     (ediff-recenter 'no-rehighlight)))
  2764.   
  2765. (defun ediff-toggle-hilit ()
  2766.   "Switch between highlighting using ASCII flags and highlighting using faces.
  2767. On a dumb terminal, switches between ASCII highlighting and no highlighting." 
  2768.   (interactive)
  2769.   (if (not window-system)
  2770.       (if (eq ediff-highlighting-style 'ascii)
  2771.       (progn
  2772.         (message "ASCII highlighting flags removed.")
  2773.         (ediff-unselect-and-select-difference ediff-current-difference
  2774.                           'unselect-only)
  2775.         (setq ediff-highlighting-style 'off))
  2776.     (ediff-unselect-and-select-difference ediff-current-difference
  2777.                           'select-only))
  2778.     (ediff-unselect-and-select-difference ediff-current-difference
  2779.                       'unselect-only)
  2780.     ;; cycle through highlighting
  2781.     (cond ((and ediff-want-faces ediff-highlight-all-diffs)
  2782.        (message "Unhighlighting unselected difference regions.")
  2783.        (setq ediff-highlight-all-diffs nil))
  2784.       (ediff-want-faces
  2785.        (message "Highlighting with ASCII flags.")
  2786.        (setq ediff-want-faces nil))
  2787.       (t
  2788.        (message "Re-highlighting all difference regions.")
  2789.        (setq ediff-want-faces t
  2790.          ediff-highlight-all-diffs t)))
  2791.          
  2792.     (if (and ediff-want-faces ediff-highlight-all-diffs)
  2793.     (if (not (face-differs-from-default-p 'ediff-odd-diff-face-A-var))
  2794.         (progn
  2795.           (copy-face ediff-odd-diff-face-A 'ediff-odd-diff-face-A-var)
  2796.           (copy-face ediff-odd-diff-face-B 'ediff-odd-diff-face-B-var)
  2797.           (copy-face ediff-even-diff-face-A 'ediff-even-diff-face-A-var)
  2798.           (copy-face ediff-even-diff-face-B 'ediff-even-diff-face-B-var)))
  2799.       (copy-face 'default 'ediff-odd-diff-face-A-var)
  2800.       (copy-face 'default 'ediff-odd-diff-face-B-var)
  2801.       (copy-face 'default 'ediff-even-diff-face-A-var)
  2802.       (copy-face 'default 'ediff-even-diff-face-B-var))
  2803.     
  2804.     (ediff-unselect-and-select-difference
  2805.      ediff-current-difference 'select-only))
  2806.   (ediff-operate-on-flags 'insert)
  2807.   )
  2808.   
  2809. (defun ediff-toggle-autorefine ()
  2810.   "Toggle auto-refine mode."
  2811.   (interactive)
  2812.   (cond ((eq ediff-auto-refine 'nix)
  2813.      (setq ediff-auto-refine 'on)
  2814.      (ediff-make-fine-diffs ediff-current-difference 'noforce)
  2815.      (message "Auto-refining is ON."))
  2816.     ((eq ediff-auto-refine 'on)
  2817.      (message "Auto-refining is OFF.")
  2818.      (setq ediff-auto-refine 'off))
  2819.     (t
  2820.      (ediff-set-fine-diff-properties ediff-current-difference 'default)
  2821.      (message "Refinements are HIDDEN.")
  2822.      (setq ediff-auto-refine 'nix))
  2823.     ))
  2824.   
  2825. (defun ediff-toggle-help ()
  2826.   "Toggle short/long help message."
  2827.   (interactive)
  2828.   (let (buffer-read-only)
  2829.     (erase-buffer)
  2830.     (if (string= ediff-help-message ediff-help-message-long)
  2831.     (setq ediff-help-message ediff-help-message-short)
  2832.       (setq ediff-help-message ediff-help-message-long)))
  2833.   (setq ediff-window-config-saved "") ;; force redisplay
  2834.   (ediff-recenter 'no-rehighlight))
  2835.   
  2836.   
  2837. (defun ediff-toggle-read-only-A ()
  2838.   "Used as a startup hook to set `.orig' patch file read-only."
  2839.   (let ((last-command-char ?A))
  2840.     (ediff-toggle-read-only)))
  2841.   
  2842. (defun ediff-toggle-read-only ()
  2843.   "Toggles buffer-read-only for buffer buffers A and B."
  2844.   (interactive)
  2845.   (ediff-eval-in-buffer
  2846.    (if (eq last-command-char ?A) ediff-A-buffer ediff-B-buffer)
  2847.    (setq buffer-read-only (null buffer-read-only))))
  2848.  
  2849. ;;; Window scrolling operations
  2850. ;; These operations are designed to scroll all three windows the same amount,
  2851. ;; so as to keep the text in them aligned.
  2852.  
  2853. ;; Perform some operation on the two file windows (if they are showing).
  2854. ;; Catches all errors on the operation in the A and B windows.
  2855. ;; Usually, errors come from scrolling off the
  2856. ;; beginning or end of the buffer, and this gives nice nice error messages.
  2857. (defun ediff-operate-on-windows (operation arg)
  2858.   (let* ((buffer-A ediff-A-buffer)
  2859.      (buffer-B ediff-B-buffer)
  2860.      (wind (selected-window))
  2861.      (window-A (get-buffer-window buffer-A t))
  2862.      (window-B (get-buffer-window buffer-B t)))
  2863.       (if window-A (progn
  2864.              (select-window window-A)
  2865.              (condition-case nil
  2866.              (funcall operation arg)
  2867.                (error))))
  2868.       (if window-B (progn
  2869.              (select-window window-B)
  2870.              (condition-case nil
  2871.              (funcall operation arg)
  2872.                (error))))
  2873.       (select-window wind)
  2874.                ))
  2875.  
  2876. (defun ediff-scroll-up (&optional arg)
  2877.   "Scroll up buffers A and B, if they are in windows.
  2878. With optional argument ARG, scroll ARG lines; otherwise scroll by nearly
  2879. the height of window-A."
  2880.   (interactive "P")
  2881.   (ediff-operate-on-windows
  2882.    'scroll-up 
  2883.    ;; calculate argument to scroll-up
  2884.    ;; if there is an explicit argument
  2885.    (if (and arg (not (equal arg '-)))
  2886.        ;; use it
  2887.        (prefix-numeric-value arg)
  2888.      ;; if not, see if we can determine a default amount (the window height)
  2889.      (let* ((window-A (get-buffer-window ediff-A-buffer t))
  2890.         (window-B (get-buffer-window ediff-B-buffer t))
  2891.         default-amount)
  2892.        (if (or (null window-A) (null window-B))
  2893.        (setq default-amount 0)
  2894.      (setq default-amount 
  2895.            (- (min (window-height window-A) (window-height window-B))
  2896.           1 next-screen-context-lines)))
  2897.        ;; the window was found
  2898.        (if arg
  2899.        ;; C-u as argument means half of default amount
  2900.        (/ default-amount 2)
  2901.      ;; no argument means default amount
  2902.      default-amount)))))
  2903.  
  2904. (defun ediff-scroll-down (&optional arg)
  2905.   "Scroll down buffers A and B, if they are in windows.
  2906. With optional argument ARG, scroll ARG lines; otherwise scroll by nearly
  2907. the height of window-A."
  2908.   (interactive "P")
  2909.   (ediff-operate-on-windows
  2910.    'scroll-down
  2911.    ;; calculate argument to scroll-down
  2912.    ;; if there is an explicit argument
  2913.    (if (and arg (not (equal arg '-)))
  2914.        ;; use it
  2915.        (prefix-numeric-value arg)
  2916.      ;; if not, see if we can determine a default amount (the window height)
  2917.      (let* ((window-A (get-buffer-window ediff-A-buffer t))
  2918.         (window-B (get-buffer-window ediff-B-buffer t))
  2919.         default-amount)
  2920.        (if (or (null window-A) (null window-B))
  2921.        (setq default-amount 0)
  2922.      (setq default-amount 
  2923.            (- (min (window-height window-A) (window-height window-B))
  2924.           1 next-screen-context-lines)))
  2925.        ;; the window was found
  2926.        (if arg
  2927.        ;; C-u as argument means half of default amount
  2928.        (/ default-amount 2)
  2929.      ;; no argument means default amount
  2930.      default-amount)))))
  2931.  
  2932. (defun ediff-scroll-left (&optional arg)
  2933.   "Scroll left buffer-A and buffer-B, if they are in windows.
  2934. If an argument is given, that is how many columns are scrolled, else nearly
  2935. the width of the A and B windows."
  2936.   (interactive "P")
  2937.   (ediff-operate-on-windows
  2938.    'scroll-left
  2939.    ;; calculate argument to scroll-left
  2940.    ;; if there is an explicit argument
  2941.    (if (and arg (not (equal arg '-)))
  2942.        ;; use it
  2943.        (prefix-numeric-value arg)
  2944.      ;; if not, see if we can determine a default amount
  2945.      ;; (half the window width)
  2946.      (if (null ediff-control-window)
  2947.      ;; no control window, use nil
  2948.      nil
  2949.        (let ((default-amount
  2950.            (- (/ (window-width ediff-control-window) 2) 3)))
  2951.      ;; the window was found
  2952.      (if arg
  2953.          ;; C-u as argument means half of default amount
  2954.          (/ default-amount 2)
  2955.        ;; no argument means default amount
  2956.        default-amount))))))
  2957.  
  2958. (defun ediff-scroll-right (&optional arg)
  2959.   "Scroll right buffer-A and buffer-B, if they are in windows.
  2960. If an argument is given, that is how many columns are scrolled, else nearly
  2961. the width of the A and B windows."
  2962.   (interactive "P")
  2963.   (ediff-operate-on-windows
  2964.    'scroll-right
  2965.    ;; calculate argument to scroll-right
  2966.    ;; if there is an explicit argument
  2967.    (if (and arg (not (equal arg '-)))
  2968.        ;; use it
  2969.        (prefix-numeric-value arg)
  2970.      ;; if not, see if we can determine a default amount
  2971.      ;; (half the window width)
  2972.      (if (null ediff-control-window)
  2973.      ;; no control window, use nil
  2974.      nil
  2975.        (let ((default-amount
  2976.            (- (/ (window-width ediff-control-window) 2) 3)))
  2977.      ;; the window was found
  2978.      (if arg
  2979.          ;; C-u as argument means half of default amount
  2980.          (/ default-amount 2)
  2981.        ;; no argument means default amount
  2982.        default-amount))))))
  2983.  
  2984. (defun ediff-position-region (beg end pos)
  2985.   "This is a variation on `emerge-position-region'. 
  2986. The difference is that it always tries to align difference regions in
  2987. buffer-A and buffer-B, so that it will be easier to compare them."
  2988.   (set-window-start (selected-window) beg)
  2989.   (if (pos-visible-in-window-p end)
  2990.       ;; Determine the number of lines that the region occupies
  2991.       (let ((lines 0))
  2992.     (while (> end (progn
  2993.             (move-to-window-line lines)
  2994.             (point)))
  2995.       (setq lines (1+ lines)))
  2996.     ;; And position the beginning on the right line
  2997.     (goto-char beg)
  2998.     (recenter (/ (1+ (max (- (1- (window-height (selected-window)))
  2999.                  lines)
  3000.                   1)
  3001.              )
  3002.              2))))
  3003.   (goto-char pos)
  3004.   )
  3005.  
  3006.  
  3007. (defun ediff-next-difference (&optional arg)
  3008.   "Advance to the next difference. 
  3009. With a prefix argument, go back that many differences."
  3010.   (interactive "P")
  3011.   (if (< ediff-current-difference ediff-number-of-differences)
  3012.       (let ((n (min ediff-number-of-differences
  3013.             (+ ediff-current-difference (if arg arg 1))))
  3014.         (buffer-read-only nil))
  3015.         
  3016.     (while (and (< n ediff-number-of-differences)
  3017.             (funcall ediff-skip-diff-region-function n))
  3018.       (setq n (1+ n)))
  3019.     
  3020.     (ediff-unselect-and-select-difference n)
  3021.     ;; possibly skip inessential difference regions
  3022.     (while (and ediff-ignore-similar-regions
  3023.             (< n ediff-number-of-differences)
  3024.             (ediff-no-fine-diffs n))
  3025.       (setq n (1+ n))
  3026.       (ediff-unselect-and-select-difference n))
  3027.     ) ;; let
  3028.     (error "At end of the difference list.")))
  3029.  
  3030. (defun ediff-previous-difference (&optional arg)
  3031.   "Go to the previous difference. 
  3032. With a prefix argument, go back that many differences."
  3033.   (interactive "P")
  3034.   (if (> ediff-current-difference -1)
  3035.       (let ((n (max -1 (- ediff-current-difference (if arg arg 1))))
  3036.         (buffer-read-only nil))
  3037.         
  3038.     (while (and (funcall ediff-skip-diff-region-function n)
  3039.             (> n -1))
  3040.       (setq n (1- n)))
  3041.     (ediff-unselect-and-select-difference n)
  3042.     ;; possibly skip inessential difference regions
  3043.     (while (and ediff-ignore-similar-regions
  3044.             (> n -1)
  3045.             (ediff-no-fine-diffs n))
  3046.       (setq n (1- n))
  3047.       (ediff-unselect-and-select-difference n))
  3048.     ) ;; let
  3049.     (error "At beginning of the difference list.")))
  3050.  
  3051. (defun ediff-jump-to-difference (difference-number)
  3052.   "Go to the difference specified as a prefix argument."
  3053.   (interactive "p")
  3054.   (let ((buffer-read-only nil))
  3055.     (setq difference-number (1- difference-number))
  3056.     (if (and (>= difference-number -1)
  3057.          (< difference-number (1+ ediff-number-of-differences)))
  3058.     (ediff-unselect-and-select-difference difference-number)
  3059.       (error "Bad difference number"))))
  3060.       
  3061. (defun ediff-jump-to-difference-at-point ()
  3062.   "Go to the difference closest to the point in buffer A or B.
  3063. If this command is invoked via `\\[ediff-jump-to-difference-at-point]' 
  3064. then the point in buffer B is used.
  3065. Otherwise, buffer A's point is used."
  3066.   (interactive)
  3067.   (let ((buffer-read-only nil)
  3068.     (buf-type (ediff-char-to-buftype last-command-char)))
  3069.     (ediff-jump-to-difference (ediff-diff-at-point buf-type))))
  3070.     
  3071.       
  3072. ;; find region "most related to the current point position      
  3073.  
  3074. (defun ediff-diff-at-point (buf-type)
  3075.   (let ((buffer (ediff-get-buffer buf-type))
  3076.     (ctl-buffer ediff-control-buffer)
  3077.     (max-dif-num (1- ediff-number-of-differences))
  3078.     (diff-no -1)
  3079.     (prev-beg 0)
  3080.     (prev-end 0)
  3081.     (beg 0)
  3082.     (end 0))
  3083.     
  3084.     (ediff-eval-in-buffer
  3085.      buffer
  3086.      (while (and (or (< (point) prev-beg) (> (point) beg))
  3087.          (< diff-no max-dif-num))
  3088.        (setq diff-no (1+ diff-no))
  3089.        (setq prev-beg beg
  3090.          prev-end end)
  3091.        (setq beg (ediff-get-diff-posn buf-type 'beg diff-no ctl-buffer)
  3092.          end (ediff-get-diff-posn buf-type 'end diff-no ctl-buffer))
  3093.        )
  3094.       
  3095.      (if (< (abs (- (point) prev-end))
  3096.         (abs (- (point) beg)))
  3097.      diff-no
  3098.        (1+ diff-no))  ;; jump-to-diff works with diff nums higher by 1
  3099.      )))
  3100.  
  3101. ;;; Copying diffs.
  3102.  
  3103. (defun ediff-diff-to-diff (arg)
  3104.   "Copy buffer-A'th diff to buffer B.
  3105. If numerical prefix argument, copy this diff specified in the arg.
  3106. Otherwise, copy the difference given by `ediff-current-difference'." 
  3107.   (interactive "P")
  3108.   (if arg
  3109.       (ediff-jump-to-difference arg))
  3110.   (let* ((key1 (aref (this-command-keys) 0))
  3111.      (key2 (aref (this-command-keys) 1))
  3112.      (char1 (if (ediff-if-lucid) (event-key key1) key1))
  3113.      (char2 (if (ediff-if-lucid) (event-key key2) key2)))
  3114.     (ediff-copy-diff ediff-current-difference
  3115.              (ediff-char-to-buftype char1)
  3116.              (ediff-char-to-buftype char2))
  3117.     (ediff-recenter 'no-rehighlight)))
  3118.  
  3119.  
  3120. (defun ediff-copy-diff (n from-buf-type to-buf-type)
  3121.   "Copy diff N from FROM-BUF-TYPE \(given as 'A or 'B\) to TO-BUF-TYPE."
  3122.   (let* ((to-buf (ediff-get-buffer to-buf-type))
  3123.      (from-buf (ediff-get-buffer from-buf-type))
  3124.      (ctrl-buf ediff-control-buffer)
  3125.      reg-to-copy reg-to-delete
  3126.      reg-to-delete-beg reg-to-delete-end)
  3127.     
  3128.     (ediff-operate-on-flags 'remove)
  3129.     (setq reg-to-delete-beg
  3130.       (ediff-get-diff-posn to-buf-type 'beg n ctrl-buf))
  3131.     (setq reg-to-delete-end
  3132.       (ediff-get-diff-posn to-buf-type 'end n ctrl-buf))
  3133.     (setq reg-to-copy (ediff-eval-in-buffer
  3134.                from-buf
  3135.                (buffer-substring (ediff-get-diff-posn
  3136.                       from-buf-type 'beg n ctrl-buf)
  3137.                      (ediff-get-diff-posn
  3138.                       from-buf-type 'end n ctrl-buf))))
  3139.     (setq reg-to-delete (ediff-eval-in-buffer
  3140.              to-buf
  3141.              (buffer-substring reg-to-delete-beg
  3142.                        reg-to-delete-end)))
  3143.     (setq ediff-disturbed-overlays nil) ;; clear before use
  3144.     
  3145.     (if (string= reg-to-delete reg-to-copy)
  3146.     (progn
  3147.       (ding)
  3148.       (message
  3149.        "Diff regions %d are identical in buffers %S and %S. Nothing copied." 
  3150.        (1+ n) from-buf-type to-buf-type))
  3151.     
  3152.       ;; seems ok to copy
  3153.       (if (ediff-test-save-region n to-buf-type)
  3154.         (condition-case conds
  3155.         (let (inhibit-read-only)
  3156.           (ediff-eval-in-buffer
  3157.            to-buf
  3158.            ;; to prevent flags from interfering if buffer is writable
  3159.            (setq inhibit-read-only (null buffer-read-only))
  3160.            (let ((before-change-function nil))
  3161.              (goto-char reg-to-delete-end)
  3162.              (insert-before-markers reg-to-copy)
  3163.              (if (ediff-if-lucid)
  3164.              (progn
  3165.                (ediff-collect-extents-lucid reg-to-delete-beg)
  3166.                (if (> reg-to-delete-end reg-to-delete-beg)
  3167.                    (progn
  3168.                  (kill-region reg-to-delete-beg
  3169.                           reg-to-delete-end) 
  3170.                  (if (string= reg-to-copy "")
  3171.                      (ediff-adjust-disturbed-extents-lucid
  3172.                       reg-to-delete-beg)))))
  3173.                (if (> reg-to-delete-end reg-to-delete-beg)
  3174.                (kill-region reg-to-delete-beg reg-to-delete-end)
  3175.              (ediff-move-disturbed-overlays reg-to-delete-beg)))
  3176.              ))
  3177.           (ediff-save-diff-region n to-buf-type reg-to-delete))
  3178.           (error (message "%s %s"
  3179.                   (car conds)
  3180.                   (mapconcat 'prin1-to-string (cdr conds) " "))
  3181.              (beep 1))))
  3182.       )
  3183.     (ediff-operate-on-flags 'insert)
  3184.     ))
  3185.      
  3186. (defun ediff-save-diff-region (n buf-type reg)
  3187.   "Save Nth diff of buffer BUF-TYPE \(`A' or `B'\).
  3188. That is to say, the Nth diff on the `ediff-killed-diffs-alist'.  REG
  3189. is the region to save.  It is redundant here, but is passed anyway, for
  3190. convenience."
  3191.  
  3192.   (let* ((n-th-diff-saved (assoc n ediff-killed-diffs-alist))
  3193.      (this-buf-n-th-diff-saved (assoc buf-type (cdr n-th-diff-saved))))
  3194.      
  3195.     (if this-buf-n-th-diff-saved
  3196.     ;; either nothing saved for n-th diff and buffer or we OK'ed
  3197.     ;; overriding
  3198.     (setcdr this-buf-n-th-diff-saved reg)
  3199.       (if n-th-diff-saved ;; n-th diff saved, but for another buffer
  3200.       (nconc n-th-diff-saved  (list (cons buf-type reg)))
  3201.     (setq ediff-killed-diffs-alist  ;; create record for n-th diff
  3202.           (cons (list n (cons buf-type reg))
  3203.             ediff-killed-diffs-alist))))
  3204.     (message "Saved diff region #%d for buffer %S. To recover hit 'r%s'."
  3205.          (1+ n) buf-type
  3206.          (downcase (symbol-name buf-type)))))
  3207.     
  3208. (defun ediff-test-save-region (n buf-type)
  3209.   "Test if saving Nth difference region of buffer BUF-TYPE is possible."
  3210.   (let* ((n-th-diff-saved (assoc n ediff-killed-diffs-alist))
  3211.      (this-buf-n-th-diff-saved (assoc buf-type (cdr n-th-diff-saved))))
  3212.      
  3213.     (if this-buf-n-th-diff-saved
  3214.     (if (yes-or-no-p
  3215.          (format 
  3216.           "You've previously copied diff region %d to buffer %S. Confirm."
  3217.           (1+ n) buf-type))
  3218.         t
  3219.       (error "Quit."))
  3220.       t)))
  3221.       
  3222. (defun ediff-pop-diff (n buf-type)
  3223.   "Pop last killed Nth diff region from buffer BUF-TYPE."
  3224.   (let* ((n-th-record (assoc n ediff-killed-diffs-alist))
  3225.      (saved-rec (assoc buf-type (cdr n-th-record)))
  3226.      (buf (ediff-get-buffer buf-type))
  3227.      saved-diff reg-beg reg-end recovered)
  3228.     
  3229.     (if (cdr saved-rec)
  3230.     (setq saved-diff (cdr saved-rec))
  3231.       (if (> ediff-number-of-differences 0)
  3232.       (error "Nothing saved for diff %d in buffer %S." (1+ n) buf-type)
  3233.     (error "No differences found.")))
  3234.     
  3235.     (ediff-operate-on-flags 'remove)
  3236.     
  3237.     (setq reg-beg (ediff-get-diff-posn buf-type 'beg n ediff-control-buffer))
  3238.     (setq reg-end (ediff-get-diff-posn buf-type 'end n ediff-control-buffer))
  3239.     (setq ediff-disturbed-overlays nil) ;; clear before use
  3240.     
  3241.     (condition-case conds
  3242.     (ediff-eval-in-buffer
  3243.      buf
  3244.      (let ((inhibit-read-only (null buffer-read-only))
  3245.            (before-change-function nil))
  3246.        (goto-char reg-end)
  3247.        (insert-before-markers saved-diff)
  3248.        
  3249.        (if (ediff-if-lucid)
  3250.            (progn
  3251.          (ediff-collect-extents-lucid reg-beg)
  3252.          (if (> reg-end reg-beg)
  3253.              (progn
  3254.                (kill-region reg-beg reg-end)
  3255.                (if (string= saved-diff "")
  3256.                (ediff-adjust-disturbed-extents-lucid reg-beg)))))
  3257.          (if (> reg-end reg-beg)
  3258.          (kill-region reg-beg reg-end)
  3259.            (ediff-move-disturbed-overlays reg-beg)))
  3260.          
  3261.        (setq recovered t)
  3262.        ))
  3263.       (error (message "%s %s"
  3264.               (car conds)
  3265.               (mapconcat 'prin1-to-string (cdr conds) " "))
  3266.          (beep 1)))
  3267.     
  3268.     (ediff-operate-on-flags 'insert)
  3269.     (if recovered
  3270.     (progn
  3271.       (setq  n-th-record (delq saved-rec n-th-record))
  3272.       (message "Restored diff region %d in buffer %S." (1+ n) buf-type)))
  3273.     ))
  3274.       
  3275. (defun ediff-restore-diff  (arg)
  3276.   "Restore ARGth diff from `ediff-killed-diffs-alist'.
  3277. ARG is a prefix argument.  If ARG is nil, restore current-difference."
  3278.   (interactive "P")
  3279.   (if arg
  3280.       (ediff-jump-to-difference arg))
  3281.   (ediff-pop-diff ediff-current-difference 
  3282.           (ediff-char-to-buftype last-command-char))
  3283.   (ediff-recenter 'no-rehighlight))
  3284.   
  3285. (defun ediff-toggle-regexp-match ()
  3286.   "Toggle focus on difference regions that match a regexp or hide those diffs."
  3287.   (interactive)
  3288.   (let (regexp-A regexp-B)
  3289.     (cond
  3290.      ((or (and (eq ediff-skip-diff-region-function 'ediff-focus-on-regexp-matches)
  3291.            (eq last-command-char ?f))
  3292.       (and (eq ediff-skip-diff-region-function 'ediff-hide-regexp-matches)
  3293.            (eq last-command-char ?h)))
  3294.       (message "Selective browsing by regexp turned off.")
  3295.       (setq ediff-skip-diff-region-function 'ediff-show-all-diffs))
  3296.      ((eq last-command-char ?h)
  3297.       (setq ediff-skip-diff-region-function 'ediff-hide-regexp-matches
  3298.         regexp-A 
  3299.         (read-string
  3300.          (format 
  3301.           "Ignore A-regions matching this regexp (default \"%s\"): "
  3302.           (regexp-quote ediff-regexp-hide-A)))
  3303.         regexp-B
  3304.         (read-string
  3305.          (format 
  3306.           "Ignore B-regions matching this regexp (default \"%s\"): "
  3307.           (regexp-quote ediff-regexp-hide-B))))
  3308.       (message "Hide difference regions matching regexp.")
  3309.       (or (string= regexp-A "") (setq ediff-regexp-hide-A regexp-A))
  3310.       (or (string= regexp-B "") (setq ediff-regexp-hide-B regexp-B)))
  3311.      ((eq last-command-char ?f)
  3312.       (setq ediff-skip-diff-region-function 'ediff-focus-on-regexp-matches
  3313.         regexp-A 
  3314.         (read-string
  3315.          (format 
  3316.           "Focus on A-regions matching this regexp (default \"%s\"): "
  3317.           (regexp-quote ediff-regexp-focus-A)))
  3318.         regexp-B
  3319.         (read-string
  3320.          (format 
  3321.           "Focus on B-regions matching this regexp (default \"%s\"): "
  3322.           (regexp-quote ediff-regexp-focus-B))))
  3323.       (message "Focus on difference regions matching regexp.")
  3324.       (or (string= regexp-A "") (setq ediff-regexp-focus-A regexp-A))
  3325.       (or (string= regexp-B "") (setq ediff-regexp-focus-B regexp-B))))))
  3326.       
  3327. (defun ediff-toggle-skip-similar ()
  3328.   (interactive)
  3329.   (setq ediff-ignore-similar-regions (not ediff-ignore-similar-regions))
  3330.   (if ediff-ignore-similar-regions
  3331.       (message "Skipping over regions that differ only in white space & line breaks.")
  3332.     (message "Skipping over white-space differences turned off.")))
  3333.   
  3334. (defun ediff-show-all-diffs (n)
  3335.   "Don't skip difference regions."
  3336.   nil)
  3337.   
  3338. (defun ediff-focus-on-regexp-matches (n)
  3339.   "Focus on diffs that match regexp `ediff-regexp-focus-A/B'.
  3340. Regions to be ignored according to this function are those where   
  3341. buf A region doesn't match `ediff-regexp-focus-A' and buf B region
  3342. doesn't match `ediff-regexp-focus-B'.
  3343. This function returns nil if the region number N (specified as
  3344. an argument) is not to be ignored and t if region N is to be ignored.
  3345.  
  3346. N is a region number used by Ediff internally. It is 1 less
  3347. the number seen by the user."
  3348.   (if (and (>= n 0) (< n ediff-number-of-differences))
  3349.       (let* ((ctl-buf ediff-control-buffer)
  3350.          (regex-A ediff-regexp-focus-A)
  3351.          (regex-B ediff-regexp-focus-B)
  3352.          (reg-A-match (ediff-eval-in-buffer
  3353.                ediff-A-buffer
  3354.                (goto-char (ediff-get-diff-posn 'A 'beg n ctl-buf))
  3355.                (re-search-forward
  3356.                 regex-A
  3357.                 (ediff-get-diff-posn 'A 'end n ctl-buf)
  3358.                 t)))
  3359.          (reg-B-match (ediff-eval-in-buffer
  3360.                ediff-B-buffer
  3361.                (goto-char (ediff-get-diff-posn 'B 'beg n ctl-buf))
  3362.                (re-search-forward
  3363.                 regex-B
  3364.                 (ediff-get-diff-posn 'B 'end n ctl-buf)
  3365.                 t))))
  3366.     (not (and reg-A-match reg-B-match)))))
  3367.   
  3368. (defun ediff-hide-regexp-matches (n)  
  3369.   "Hide diffs that match regexp `ediff-regexp-hide-A/B'.
  3370. Regions to be ignored are those where buf A region matches
  3371. `ediff-regexp-hide-A' and buf B region matches `ediff-regexp-hide-B'.
  3372. This function returns nil if the region number N (specified as
  3373. an argument) is not to be ignored and t if region N is to be ignored.
  3374.  
  3375. N is a region number used by Ediff internally. It is 1 less
  3376. the number seen by the user."
  3377.   (if (and (>= n 0) (< n ediff-number-of-differences))
  3378.       (let* ((ctl-buf ediff-control-buffer)
  3379.          (regex-A ediff-regexp-hide-A)
  3380.          (regex-B ediff-regexp-hide-B)
  3381.          (reg-A-match (ediff-eval-in-buffer
  3382.                ediff-A-buffer
  3383.                (goto-char (ediff-get-diff-posn 'A 'beg n ctl-buf))
  3384.                (re-search-forward
  3385.                 regex-A
  3386.                 (ediff-get-diff-posn 'A 'end n ctl-buf)
  3387.                 t)))
  3388.          (reg-B-match (ediff-eval-in-buffer
  3389.                ediff-B-buffer
  3390.                (goto-char (ediff-get-diff-posn 'B 'beg n ctl-buf))
  3391.                (re-search-forward
  3392.                 regex-B
  3393.                 (ediff-get-diff-posn 'B 'end n ctl-buf)
  3394.                 t))))
  3395.     (and reg-A-match reg-B-match))))
  3396.     
  3397.  
  3398. ;;; Quitting, suspending, etc.
  3399. (defun ediff-quit ()
  3400.   "Finish an Ediff session and exit Ediff.
  3401. Unselects the selected difference, if any, restores the read-only and modified
  3402. flags of the compared file buffers, kills Ediff buffers for this session
  3403. \(but not file-A and file-B\)."
  3404.   (interactive)
  3405.   (if (prog1
  3406.       (y-or-n-p "Do you really want to exit Ediff? ")
  3407.     (message ""))
  3408.       (ediff-really-quit)))
  3409.  
  3410.  
  3411. ;; Perform the quit operations.
  3412. (defun ediff-really-quit ()
  3413.   (setq ediff-help-message ediff-help-message-long)
  3414.   (ediff-restore-buffer-characteristics t) ;; restore as they were at setup
  3415.   (ediff-unhighlight-diffs-totally)
  3416.   (ediff-clear-diff-vector ediff-difference-vector 'fine-diffs-also)
  3417.   (if ediff-temp-file-A (delete-file ediff-temp-file-A))
  3418.   (if ediff-temp-file-B (delete-file ediff-temp-file-B))
  3419.   
  3420.   ;; restore buffer mode line id's in buffer-A/B
  3421.   (let ((control-buffer ediff-control-buffer))
  3422.     (condition-case nil
  3423.     (ediff-eval-in-buffer
  3424.      ediff-A-buffer
  3425.      (setq before-change-function nil)
  3426.      (setq ediff-this-buffer-control-sessions 
  3427.            (delq control-buffer ediff-this-buffer-control-sessions))
  3428.      (if (null ediff-this-buffer-control-sessions)
  3429.          (setq local-write-file-hooks 
  3430.            (delq 'ediff-block-write-file local-write-file-hooks)))
  3431.      (kill-local-variable 'mode-line-buffer-identification))
  3432.       (error))
  3433.       
  3434.     (condition-case nil
  3435.     (ediff-eval-in-buffer
  3436.      ediff-B-buffer
  3437.      (setq ediff-this-buffer-control-sessions 
  3438.            (delq control-buffer ediff-this-buffer-control-sessions))
  3439.      (if (null ediff-this-buffer-control-sessions)
  3440.          (setq local-write-file-hooks 
  3441.            (delq 'ediff-block-write-file local-write-file-hooks)))
  3442.      (setq before-change-function nil)
  3443.      (kill-local-variable 'mode-line-buffer-identification))
  3444.       (error)))
  3445.    
  3446.   (run-hooks 'ediff-quit-hooks))
  3447.   
  3448. (defun ediff-kill-buffer-carefully (buf)
  3449.   "Kill buffer BUF if it exists."
  3450.   (if (ediff-buffer-live-p buf)
  3451.       (kill-buffer (get-buffer buf))))
  3452.  
  3453. ;; The default way of quitting Ediff.
  3454. ;; Kills control buffers and leaves the
  3455. ;; frame split between the two diff'ed files.
  3456. (defun ediff-default-quit-hook ()
  3457.   (let ((buff-A ediff-A-buffer)
  3458.     (buff-B ediff-B-buffer))
  3459.     (ediff-kill-buffer-carefully ediff-diff-buffer)
  3460.     (ediff-kill-buffer-carefully ediff-fine-diff-buffer)
  3461.     (ediff-kill-buffer-carefully ediff-tmp-buffer)
  3462.     (ediff-kill-buffer-carefully ediff-error-buffer)
  3463.     (ediff-kill-buffer-carefully ediff-control-buffer)
  3464.     (ediff-kill-buffer-carefully ediff-patch-diagnostics)
  3465.     (delete-other-windows)
  3466.     ;; display only if not visible
  3467.     
  3468.     (condition-case nil
  3469.     (or (get-buffer-window buff-B t)
  3470.         (switch-to-buffer buff-B))
  3471.       (error))
  3472.     (condition-case nil
  3473.     (or (get-buffer-window buff-A t)
  3474.         (progn
  3475.           (if (get-buffer-window buff-B)
  3476.           (split-window-vertically))
  3477.           (switch-to-buffer buff-A)))
  3478.       (error))
  3479.     (message "")
  3480.     ))
  3481.     
  3482. ;; The default way of suspending Ediff.
  3483. ;; Buries Ediff buffers, kills all windows.
  3484. (defun ediff-default-suspend-hook ()
  3485.   (let ((buf-A ediff-A-buffer)
  3486.     (buf-B ediff-B-buffer)
  3487.     (buf-patch ediff-patch-buf)
  3488.     (buf-patch-diag ediff-patch-diagnostics)
  3489.     (buf-err  ediff-error-buffer)
  3490.     (buf-diff ediff-diff-buffer)
  3491.     (buf-fine-diff ediff-fine-diff-buffer))
  3492.     (bury-buffer) ;; ediff-control-buffer
  3493.     (delete-other-windows)
  3494.     (bury-buffer buf-err)
  3495.     (bury-buffer buf-diff)
  3496.     (bury-buffer buf-fine-diff)
  3497.     (bury-buffer buf-patch)
  3498.     (bury-buffer buf-patch-diag)
  3499.     (bury-buffer buf-A)
  3500.     (bury-buffer buf-B)))
  3501.      
  3502.      
  3503. (defun ediff-suspend ()
  3504.   "Suspend Ediff.
  3505. To resume, switch to the appropriate `Ediff Control Panel'
  3506. buffer and then type \\[ediff-recenter].  Ediff will automatically set
  3507. up an appropriate window config."
  3508.   (interactive)
  3509.   (let ((key (substitute-command-keys "\\[ediff-recenter]")))
  3510.   (run-hooks 'ediff-suspend-hooks)
  3511.   (message
  3512.    "To resume Ediff, switch to Ediff Control Panel and hit '%S'" key)))
  3513.  
  3514.  
  3515. (defun ediff-status-info ()
  3516.   "Show the names of the buffers or files being operated on by Ediff.
  3517. Hit \\[ediff-recenter] to reset the windows afterward."
  3518.   (interactive)
  3519.   (with-output-to-temp-buffer " *ediff-info*"
  3520.     (ediff-eval-in-buffer ediff-A-buffer
  3521.                (if buffer-file-name
  3522.                    (princ
  3523.                 (format "File A is: %s\n" buffer-file-name))
  3524.                  (princ 
  3525.                   (format "Buffer A is: %s\n" (buffer-name)))))
  3526.     (ediff-eval-in-buffer ediff-B-buffer
  3527.                (if buffer-file-name
  3528.                    (princ
  3529.                 (format "File B is: %s\n" buffer-file-name))
  3530.                  (princ 
  3531.                   (format "Buffer B is: %s\n" (buffer-name)))))
  3532.                   
  3533.     (let* ((A-line (ediff-eval-in-buffer ediff-A-buffer
  3534.                        (count-lines (point-min) (point))))
  3535.        (B-line (ediff-eval-in-buffer ediff-B-buffer
  3536.                        (count-lines (point-min) (point)))))
  3537.       (princ (format "\nPoint position in buffer A = %d\n" A-line))
  3538.       (princ (format "Point position in buffer B = %d\n" B-line)))
  3539.       
  3540.     (princ (format "\nCurrent difference number = %S\n"
  3541.            (cond ((< ediff-current-difference 0) 'start)
  3542.              ((>= ediff-current-difference
  3543.                   ediff-number-of-differences) 'end)
  3544.              (t (1+ ediff-current-difference)))))
  3545.  
  3546.     (cond (ediff-ignore-similar-regions
  3547.        (princ "\nSkipping over regions that differ only in white space & line breaks."))
  3548.       (t 
  3549.        (princ "\nNo skipping over regions that differ in white space & line breaks.")))
  3550.        
  3551.     (cond ((eq ediff-skip-diff-region-function 'ediff-show-all-diffs)
  3552.        (princ "\nSelective browsing by regexp is off.\n"))
  3553.       ((eq ediff-skip-diff-region-function 'ediff-hide-regexp-matches)
  3554.        (princ
  3555.         "\nIgnoring regions that match")
  3556.        (princ
  3557.         (format "\n\t regexp `%s' in buffer A  and\n\t regexp `%s' in buffer B\n"
  3558.             ediff-regexp-hide-A ediff-regexp-hide-B)))
  3559.       ((eq ediff-skip-diff-region-function 'ediff-focus-on-regexp-matches)
  3560.        (princ
  3561.         "\nFocusing on regions that match")
  3562.        (princ
  3563.         (format "\n\t regexp `%s' in buffer A  and\n\t regexp `%s' in buffer B\n"
  3564.             ediff-regexp-focus-A ediff-regexp-focus-B)))
  3565.       (t (princ "\nSelective browsing via a user-defined method.\n")))
  3566.     
  3567.     (princ "\nBug fixes to:  Michael Kifer <kifer@cs.sunysb.edu>\n")
  3568.     (princ   "Gripes to:     /dev/null <dev@null.gov>\n")
  3569.     ))
  3570.  
  3571.  
  3572.  
  3573. ;;; Support routines
  3574.  
  3575. ;; Select a difference by placing the ASCII flags around the appropriate
  3576. ;; group of lines in the A, B buffers
  3577. (defun ediff-select-difference (n)
  3578.   (if (and (ediff-buffer-live-p ediff-A-buffer)
  3579.        (ediff-buffer-live-p ediff-B-buffer)
  3580.        (>= n 0) (< n ediff-number-of-differences))
  3581.       (progn
  3582.     (ediff-remember-buffer-characteristics)
  3583.     (if (and window-system ediff-want-faces)
  3584.         (progn
  3585.           (ediff-highlight-diff n)
  3586.           (setq ediff-highlighting-style 'face))
  3587.       (setq ediff-highlighting-style 'ascii)
  3588.       (ediff-place-flags-in-buffer 'A ediff-A-buffer
  3589.                        ediff-control-buffer n)
  3590.       (ediff-place-flags-in-buffer 'B ediff-B-buffer
  3591.                        ediff-control-buffer n)) 
  3592.       
  3593.     (cond ((eq ediff-auto-refine 'on)
  3594.            (if (and
  3595.             (> ediff-auto-refine-limit
  3596.                (- (ediff-get-diff-posn 'A 'end n)
  3597.               (ediff-get-diff-posn 'A 'beg n)))
  3598.             (> ediff-auto-refine-limit
  3599.                (- (ediff-get-diff-posn 'B 'end n)
  3600.               (ediff-get-diff-posn 'B 'beg n))))
  3601.            (ediff-make-fine-diffs n 'noforce)
  3602.          (ediff-make-fine-diffs n 'skip)))
  3603.           
  3604.           ((eq ediff-auto-refine 'off)       ; highlight iff fine diffs
  3605.            (ediff-make-fine-diffs n 'skip))) ; already exist 
  3606.      
  3607.     (ediff-restore-buffer-characteristics)
  3608.     (run-hooks 'ediff-select-hooks))))
  3609.     
  3610.  
  3611. ;; Unselect a difference by removing the ASCII flags in the buffers.
  3612. (defun ediff-unselect-difference (n)
  3613.   (if (and (>= n 0) (< n ediff-number-of-differences))
  3614.       (progn 
  3615.     (ediff-remember-buffer-characteristics)
  3616.     
  3617.     (cond ((and window-system ediff-want-faces)
  3618.            (ediff-unhighlight-diff))
  3619.           ((eq ediff-highlighting-style 'ascii)
  3620.            (ediff-remove-flags-from-buffer
  3621.         ediff-A-buffer
  3622.         (ediff-get-diff-posn 'A 'beg n)
  3623.         (ediff-get-diff-posn 'A 'end n)
  3624.         ediff-before-flag-A ediff-after-flag-A)
  3625.            (ediff-remove-flags-from-buffer
  3626.         ediff-B-buffer
  3627.         (ediff-get-diff-posn 'B 'beg n)
  3628.         (ediff-get-diff-posn 'B 'end n)
  3629.         ediff-before-flag-B ediff-after-flag-B)))
  3630.         
  3631.     (ediff-restore-buffer-characteristics)
  3632.     (setq ediff-highlighting-style nil)
  3633.     
  3634.     ;; unhighlight fine diffs
  3635.     (ediff-set-fine-diff-properties ediff-current-difference 'default)
  3636.     
  3637.     (run-hooks 'ediff-unselect-hooks))))
  3638.   
  3639.  
  3640. ;; Unselects prev diff and selects a new one, if FLAG has value other than
  3641. ;; 'select-only or 'unselect-only.  If FLAG is 'select-only, the
  3642. ;; next difference is selected, but the current selection is not
  3643. ;; unselected.  If FLAG is 'unselect-only then the current selection is
  3644. ;; unselected, but the next one is not selected.  If NO-RECENTER is non-nil,
  3645. ;; don't recenter buffers after selecting/unselecting.
  3646. ;; 
  3647. ;; Don't use `ediff-select-difference' and `ediff-unselect-difference'
  3648. ;; directly, since this will screw up the undo info in the presence of
  3649. ;; ASCII flags. 
  3650. ;; Instead, use `ediff-unselect-and-select-difference' with appropriate
  3651. ;; flags.
  3652.  
  3653. (defun ediff-unselect-and-select-difference (n &optional flag no-recenter)
  3654.   (let (;; save buf modified info
  3655.     (control-buf ediff-control-buffer)
  3656.     (buf-A-modified (buffer-modified-p ediff-A-buffer))
  3657.     (buf-B-modified (buffer-modified-p ediff-B-buffer))
  3658.     ;; temporarily disable undo so highlighting won't confuse the user
  3659.     buf-A-undo buf-B-undo)
  3660.     
  3661.     (let ((ediff-current-difference n))
  3662.       (or no-recenter
  3663.       (ediff-recenter 'no-rehighlight)))
  3664.       
  3665.     (if (and (ediff-buffer-live-p ediff-A-buffer) 
  3666.          (ediff-buffer-live-p ediff-B-buffer))
  3667.     (progn
  3668.       (ediff-eval-in-buffer
  3669.        ediff-A-buffer
  3670.        (setq buf-A-undo buffer-undo-list))
  3671.       (ediff-eval-in-buffer
  3672.        ediff-B-buffer
  3673.        (setq buf-B-undo buffer-undo-list))
  3674.     
  3675.       (buffer-disable-undo ediff-A-buffer)
  3676.       (buffer-disable-undo ediff-B-buffer)))
  3677.     
  3678.     (unwind-protect    ;; we don't want to lose undo info due to error
  3679.     (progn
  3680.       (or (eq flag 'select-only)
  3681.           (ediff-unselect-difference ediff-current-difference))
  3682.       
  3683.       ;; Auto-save buffers while Ediff flags are temporarily removed.
  3684.       (ediff-eval-in-buffer
  3685.        ediff-A-buffer
  3686.        (if buf-A-modified
  3687.            (do-auto-save)))
  3688.       (ediff-eval-in-buffer
  3689.        ediff-B-buffer
  3690.        (if buf-B-modified
  3691.            (do-auto-save)))
  3692.       
  3693.       (or (eq flag 'unselect-only)
  3694.           (ediff-select-difference n))
  3695.       (setq ediff-current-difference n)
  3696.       ) ;; end protected section
  3697.       
  3698.       (ediff-eval-in-buffer
  3699.        control-buf
  3700.        (ediff-refresh-mode-line)
  3701.        ;; restore undo and buffer-modified info
  3702.        (ediff-eval-in-buffer
  3703.     ediff-A-buffer
  3704.     (set-buffer-modified-p buf-A-modified)
  3705.     (setq buffer-undo-list buf-A-undo)))
  3706.       
  3707.       (ediff-eval-in-buffer
  3708.        control-buf
  3709.        (ediff-eval-in-buffer
  3710.     ediff-B-buffer
  3711.     (set-buffer-modified-p buf-B-modified)
  3712.     (setq buffer-undo-list buf-B-undo)))
  3713.       )))
  3714.  
  3715. ;; Revise the mode line to display which difference we have selected
  3716.  
  3717. (defun ediff-refresh-mode-line ()
  3718.   (setq mode-line-buffer-identification
  3719.     (cond ((< ediff-current-difference 0)
  3720.            (list (format "%%b:  At start of %d diffs"
  3721.                  ediff-number-of-differences)))
  3722.           ((>= ediff-current-difference ediff-number-of-differences)
  3723.            (list (format "%%b:  At end of %d diffs"
  3724.                  ediff-number-of-differences)))
  3725.           (t
  3726.            (list (format "%%b:  diff %d of %d"
  3727.                  (1+ ediff-current-difference)
  3728.                  ediff-number-of-differences)))))
  3729.   ;; Force mode-line redisplay
  3730.   (force-mode-line-update))
  3731.  
  3732.  
  3733.  
  3734. ;; Verify that we have a difference selected.
  3735. (defun ediff-validate-difference ()
  3736.   (if (not (and (>= ediff-current-difference 0)
  3737.         (< ediff-current-difference ediff-number-of-differences)))
  3738.       (error "No difference selected")))
  3739.  
  3740. (defun ediff-read-file-name (prompt default-dir default-file)
  3741. ; This is a modified version of a similar function in `emerge.el'.
  3742. ; PROMPT should not have trailing ': ', so that it can be modified
  3743. ; according to context.
  3744. ; If default-file is set, it should be used as the default value.
  3745. ; If default-dir is non-nil, use it as the default directory.
  3746. ; Otherwise, use the value of Emacs' variable `default-directory.'
  3747.  
  3748.   ;; hack default-dir if it is not set
  3749.   (setq default-dir
  3750.     (file-name-as-directory
  3751.      (abbreviate-file-name
  3752.       (expand-file-name (or default-dir
  3753.                 (and default-file
  3754.                      (file-name-directory default-file))
  3755.                 default-directory)))))
  3756.  
  3757.   ;; strip the directory from default-file
  3758.   (if default-file
  3759.       (setq default-file (file-name-nondirectory default-file)))
  3760.  
  3761.   (let (f)
  3762.     (setq f (expand-file-name
  3763.          (read-file-name
  3764.           (format "%s%s: "
  3765.               prompt
  3766.               (if default-file
  3767.               (concat " (default " default-file ")")
  3768.             ""))
  3769.           default-dir
  3770.           default-file
  3771.           'confirm
  3772.           default-file
  3773.           )
  3774.          default-dir
  3775.          ))
  3776.     ;; If user enters a directory name, expand the default file in that
  3777.     ;; directory.  This allows the user to enter a directory name for the
  3778.     ;; B-file and diff against the default-file in that directory instead
  3779.     ;; of a DIRED listing!
  3780.     (if (and (file-directory-p f) default-file)
  3781.     (setq f (expand-file-name
  3782.          (file-name-nondirectory default-file) f)))
  3783.     f)) 
  3784.   
  3785. ;; If `prefix' is given, then it is used as a prefix for the temp file
  3786. ;; name. Otherwise, `.buffer-name' is used. If `file' is given, use this
  3787. ;; file and don't create a new one.
  3788. (defun ediff-make-temp-file (&optional prefix given-file)
  3789.   (let ((f (or given-file
  3790.            (make-temp-name (concat
  3791.                 ediff-temp-file-prefix
  3792.                 (or prefix
  3793.                     (format
  3794.                      ".%s"
  3795.                      "buf")))))))
  3796.     ;; create the file
  3797.     (write-region (point-min) (point-max) f nil 'no-message) 
  3798.     (set-file-modes f ediff-temp-file-mode)
  3799.     f))
  3800.  
  3801. ;; Quote metacharacters (using \) when executing diff in Unix, but not in
  3802. ;; EMX OS/2
  3803. (defun ediff-protect-metachars (str)
  3804.   (or (eq system-type 'emx)
  3805.       (let ((limit 0))
  3806.     (while (string-match emerge-metachars str limit)
  3807.       (setq str (concat (substring str 0 (match-beginning 0))
  3808.               "\\"
  3809.               (substring str (match-beginning 0))))
  3810.       (setq limit (1+ (match-end 0))))))
  3811.   str)
  3812.  
  3813. ;; Make sure the current buffer (for a file) has the same contents as the
  3814. ;; file on disk, and attempt to remedy the situation if not.
  3815. ;; Signal an error if we can't make them the same, or the user doesn't want
  3816. ;; to do what is necessary to make them the same.
  3817. ;; If file has file handlers (indicated by the optional arg), then we
  3818. ;; offer to instead of saving. This is one difference with Emerge. 
  3819. ;; Another is that we always offer to revert obsolete files, whether they
  3820. ;; are modified or not.
  3821. (defun ediff-verify-file-buffer (&optional file-magic)
  3822.   ;; First check if the file has been modified since the buffer visited it.
  3823.   (if (verify-visited-file-modtime (current-buffer))
  3824.       (if (buffer-modified-p)
  3825.       ;; If buffer is not obsolete and is modified, offer to save
  3826.       (if (yes-or-no-p 
  3827.            (format "Buffer out of sync with visited file. %s file %s? "
  3828.                (if file-magic "Revert" "Save")
  3829.                buffer-file-name))
  3830.           (if (not file-magic)
  3831.           (save-buffer)
  3832.         ;; for some reason, file-name-handlers append instead of
  3833.         ;; replacing, so we have to erase first.
  3834.         (erase-buffer)
  3835.         (revert-buffer t t))
  3836.         (error "Buffer out of sync for file %s" buffer-file-name))
  3837.     ;; If buffer is not obsolete and is not modified, do nothing
  3838.     nil)
  3839.     ;; If buffer is obsolete, offer to revert
  3840.     (if (yes-or-no-p
  3841.      (format "Buffer out of sync with visited file. Revert file %s? "
  3842.          buffer-file-name))
  3843.     (progn
  3844.       (if file-magic
  3845.           (erase-buffer))
  3846.       (revert-buffer t t))
  3847.       (error "Buffer out of sync for file %s" buffer-file-name))))
  3848.  
  3849.  
  3850. (defun ediff-block-write-file ()    
  3851.   "Prevent writing files A and B directly."
  3852.   (if (ediff-check-for-ascii-flags)
  3853.       (error "Type 'wa' and 'wb' in Ediff Control Panel to save buffs A/B.")))
  3854.  
  3855. (defun ediff-check-for-ascii-flags ()
  3856.   (eval
  3857.    (cons 'or
  3858.      (mapcar (function (lambda (buf)
  3859.                  (if (ediff-buffer-live-p buf)
  3860.                  (ediff-eval-in-buffer
  3861.                   buf
  3862.                   (eq ediff-highlighting-style 'ascii)))))
  3863.          ediff-this-buffer-control-sessions))))
  3864.  
  3865. (defun ediff-insert-in-front (overl beg end)
  3866.   "Capture overlays that had insertions in the front.
  3867. Called when overlay OVERL gets insertion in front."
  3868.   (if (ediff-overlay-get overl 'ediff-diff-num)
  3869.       (setq ediff-disturbed-overlays
  3870.         (cons overl ediff-disturbed-overlays)))
  3871.   )
  3872.   
  3873. (defun ediff-collect-extents-lucid (pos)
  3874.   "Collects all extents at POS having property `ediff-diff-num'.
  3875. Lucid Emacs causes headache by detaching empty extents, so I have to save
  3876. them before they disappear."
  3877.   (let (lis elt)
  3878.     (while (setq elt (extent-at pos nil 'ediff-diff-num elt))
  3879.       (setq lis (cons elt lis)))
  3880.     (setq ediff-disturbed-overlays lis)))
  3881.   
  3882. (defun ediff-move-disturbed-overlays (posn)  
  3883.   (mapcar (function (lambda (overl)
  3884.                (ediff-move-overlay overl
  3885.                        posn
  3886.                        (ediff-overlay-end overl))
  3887.                ))
  3888.       ediff-disturbed-overlays)
  3889.   (setq ediff-disturbed-overlays nil))
  3890.   
  3891. (defun ediff-adjust-disturbed-extents-lucid (posn &optional posn-type)
  3892. ;; POSN-TYPE tells if POSN should become a new start of the extents
  3893. ;; (if 'new-start) or a new end (if 'new-end). If POSN-TYPE is nil, then
  3894. ;; POSN is both the new start and the new end.
  3895.   (mapcar (function (lambda (overl)
  3896.                (cond ((and (null posn-type)
  3897.                    (equal (ediff-overlay-start overl)
  3898.                       (ediff-overlay-end overl)))
  3899.                   (ediff-move-overlay overl posn posn))
  3900.                
  3901.                  (posn-type
  3902.                   (ediff-move-overlay
  3903.                    overl
  3904.                    (if (eq posn-type 'new-start)
  3905.                    posn
  3906.                  (ediff-overlay-start overl))
  3907.                    (if (eq posn-type 'new-end)
  3908.                    posn
  3909.                  (ediff-overlay-end overl)))))))
  3910.       ediff-disturbed-overlays)
  3911.     (setq ediff-disturbed-overlays nil))
  3912.   
  3913. (defun ediff-save-buffer ()
  3914.   "Safe way of saving buffers A, B, and the diff output.
  3915. `wa' saves buffer A, `wb' saves buffer B, and `wf' saves the diff output."
  3916.   (interactive)
  3917.   (let ((hooks local-write-file-hooks))
  3918.     (ediff-unselect-and-select-difference ediff-current-difference
  3919.                       'unselect-only)
  3920.     (unwind-protect
  3921.     (ediff-eval-in-buffer
  3922.      (cond ((eq last-command-char ?a)
  3923.         ediff-A-buffer)
  3924.            ((eq last-command-char ?b)
  3925.         ediff-B-buffer)
  3926.            ((eq last-command-char ?f)
  3927.         (message "Saving diff output ...")(sit-for 1)
  3928.         ediff-diff-buffer))
  3929.      ;; temporarily remove writing block 
  3930.      (setq hooks (delq 'ediff-block-write-file hooks))
  3931.      (let ((local-write-file-hooks hooks))
  3932.        (save-buffer)))
  3933.       (ediff-unselect-and-select-difference ediff-current-difference
  3934.                         'select-only)
  3935.       )))
  3936.     
  3937.  
  3938.        
  3939. ;; Essentially `emerge-remove-flags-in-buffer', modified to allow deletion
  3940. ;; of read-only flags.
  3941. (defun ediff-remove-flags-from-buffer (buffer before-posn after-posn
  3942.                           before-flag after-flag)
  3943.   (ediff-eval-in-buffer
  3944.    buffer
  3945.    (let ((buffer-read-only nil)
  3946.      (before-change-function nil)
  3947.      (inhibit-read-only t)
  3948.      (before-flag-length (length before-flag))
  3949.      (after-flag-length (length after-flag))
  3950.      )
  3951.      (goto-char after-posn)
  3952.      (setq after-posn (point-marker)) ;; after-posn is now a marker
  3953.      ;; remove the flags, if they're there
  3954.      (goto-char (- before-posn before-flag-length))
  3955.      (if (ediff-if-lucid)
  3956.      (ediff-collect-extents-lucid (+ (point) before-flag-length)))
  3957.      (if (looking-at (regexp-quote before-flag))
  3958.      (delete-region (point) (+ (point) before-flag-length))
  3959.        ;; the flag isn't there
  3960.        (ding)
  3961.        (message "Trouble removing ASCII flag"))
  3962.      (if (ediff-if-lucid)
  3963.      (ediff-adjust-disturbed-extents-lucid (point)))
  3964.      
  3965.      (if (ediff-if-lucid)
  3966.      (ediff-collect-extents-lucid (point)))
  3967.      (goto-char after-posn)
  3968.      (if (looking-at (regexp-quote after-flag))
  3969.      (delete-region (point) (+ (point) after-flag-length))
  3970.        ;; the flag isn't there
  3971.        (ding)
  3972.        (message "Trouble removing ASCII flag"))
  3973.      (if (ediff-if-lucid)
  3974.      (ediff-adjust-disturbed-extents-lucid (point)))
  3975.      (setq after-posn nil) ;; after has become a marker--garbage-collect
  3976.      )))
  3977.  
  3978.  
  3979. ;; This is a modified `emerge-place-flags-in-buffer'.
  3980. (defun ediff-place-flags-in-buffer (buf-type buffer ctl-buffer difference)
  3981.   (ediff-eval-in-buffer
  3982.    buffer
  3983.    (ediff-place-flags-in-buffer1 buf-type ctl-buffer difference)))
  3984.  
  3985. ;; Modified `emerge-place-flags-in-buffer1'.
  3986. (defun ediff-place-flags-in-buffer1 (buf-type ctl-buffer difference)
  3987.   (let ((buffer-read-only nil)
  3988.     (inhibit-read-only t)
  3989.     (before-change-function nil)
  3990.     (before-flag-name (if (eq buf-type 'A)
  3991.                    'ediff-before-flag-A
  3992.                 'ediff-before-flag-B))
  3993.     (after-flag-name (if (eq buf-type 'A)
  3994.                    'ediff-after-flag-A
  3995.                  'ediff-after-flag-B))
  3996.     beg-of-line flag)
  3997.     
  3998.     ;; insert the flag before the difference
  3999.     (let ((before (ediff-get-diff-posn buf-type 'beg difference ctl-buffer)))
  4000.       (goto-char before)
  4001.       (setq beg-of-line (bolp))
  4002.       
  4003.       (setq flag (ediff-eval-in-buffer
  4004.           ctl-buffer
  4005.           (if beg-of-line
  4006.               (set before-flag-name ediff-before-flag-bol)
  4007.             (set before-flag-name ediff-before-flag-mol))))
  4008.      
  4009.       ;; insert the flag itself
  4010.       (if (ediff-if-lucid)
  4011.       (ediff-collect-extents-lucid (point)))
  4012.       (insert-before-markers flag)
  4013.       (if (ediff-if-lucid)
  4014.       ;; Lucid's extent end-points behave strangely; they won't
  4015.       ;; respect insert-before-markers
  4016.       (ediff-adjust-disturbed-extents-lucid (point) 'new-start))
  4017.       )
  4018.     ;; insert the flag after the difference
  4019.     (let* ((after (ediff-get-diff-posn buf-type 'end difference ctl-buffer)))
  4020.       (goto-char after)
  4021.       (setq beg-of-line (bolp))
  4022.  
  4023.       (setq flag (ediff-eval-in-buffer
  4024.           ctl-buffer
  4025.           (if beg-of-line
  4026.               (set after-flag-name ediff-after-flag-bol)
  4027.             (set after-flag-name ediff-after-flag-mol))))
  4028.  
  4029.       ;; insert the flag itself
  4030.       (if (ediff-if-lucid)
  4031.       (ediff-collect-extents-lucid (point)))
  4032.       (insert flag)
  4033.       (if (ediff-if-lucid)
  4034.       (ediff-adjust-disturbed-extents-lucid after 'new-end))
  4035.       )))
  4036.  
  4037.   
  4038. (defun ediff-get-diff-posn (buf-type pos &optional n control-buf)
  4039.   "Returns positions of difference sectors in the BUF-TYPE buffer.
  4040. BUF-TYPE should be a symbol--either `A' or `B'. 
  4041. POS is either `beg' or `end'--it specifies whether you want the position at the
  4042. beginning of a difference or at the end.
  4043.  
  4044. The optional argument N says which difference \(default:
  4045. `ediff-current-difference'\).  The optional argument CONTROL-BUF says
  4046. which control buffer is in effect in case it is not the current
  4047. buffer."
  4048.   (let (diff-overlay)
  4049.     (or control-buf
  4050.     (setq control-buf (current-buffer)))
  4051.  
  4052.     (ediff-eval-in-buffer
  4053.      control-buf
  4054.      (or n  (setq n ediff-current-difference))
  4055.      (if (or (< n 0) (>= n ediff-number-of-differences))
  4056.      (if (> ediff-number-of-differences 0)
  4057.          (error "There is no diff %d. Valid diffs are 1 to %d."
  4058.             (1+ n) ediff-number-of-differences)
  4059.        (error "No differences found.")))
  4060.      (setq diff-overlay (ediff-get-diff-overlay n buf-type)))
  4061.      
  4062.     (if (ediff-overlay-get diff-overlay 'detached)
  4063.     (ediff-move-overlay diff-overlay
  4064.                 (ediff-overlay-get diff-overlay 'ediff-marker)
  4065.                 (ediff-overlay-get diff-overlay 'ediff-marker)))
  4066.     (if (eq pos 'beg)
  4067.     (ediff-overlay-start diff-overlay)
  4068.       (ediff-overlay-end diff-overlay))
  4069.     ))
  4070.     
  4071.  
  4072.  
  4073. ;; These would highlight differences under X
  4074. (defun ediff-highlight-diff (n)
  4075.   "Put face on diff N.  Invoked for X displays only."
  4076.   (let* ((last-A (ediff-eval-in-buffer ediff-A-buffer (point-max)))
  4077.      (last-B (ediff-eval-in-buffer ediff-B-buffer (point-max)))
  4078.      (begin-A (ediff-get-diff-posn 'A 'beg n))
  4079.      (end-A (ediff-get-diff-posn 'A 'end n))
  4080.      (xtraA (if (equal begin-A end-A) 1 0))
  4081.      (end-A-hilit (min last-A (+ end-A xtraA)))
  4082.      
  4083.      (begin-B (ediff-get-diff-posn 'B 'beg n))
  4084.      (end-B (ediff-get-diff-posn 'B 'end n))
  4085.      (xtraB (if (equal begin-B end-B) 1 0))
  4086.      (end-B-hilit (min last-B (+ end-B xtraB))))
  4087.       
  4088.     (if (ediff-if-lucid)
  4089.     (progn 
  4090.       (ediff-move-overlay
  4091.        ediff-current-diff-overlay-A begin-A end-A-hilit)
  4092.       (ediff-move-overlay
  4093.        ediff-current-diff-overlay-B begin-B end-B-hilit))
  4094.       ;; Emacs 19.22 has a bug, which requires that ediff-move-overlay will
  4095.       ;; have the buffer as a parameter.  Believed fixed in 19.23.
  4096.       (ediff-move-overlay ediff-current-diff-overlay-A
  4097.               begin-A end-A-hilit ediff-A-buffer)
  4098.       (ediff-move-overlay ediff-current-diff-overlay-B
  4099.               begin-B end-B-hilit ediff-B-buffer))
  4100.     ;; giving priority of 0 and then changing it may look funny, but
  4101.     ;; this is intended to overcome an Emacs bug.
  4102.     (ediff-overlay-put ediff-current-diff-overlay-A 'priority  0)
  4103.     (ediff-overlay-put ediff-current-diff-overlay-B 'priority  0)
  4104.     (ediff-overlay-put ediff-current-diff-overlay-A 'priority  
  4105.          (ediff-highest-priority begin-A end-A-hilit ediff-A-buffer))
  4106.     (ediff-overlay-put ediff-current-diff-overlay-B 'priority 
  4107.          (ediff-highest-priority begin-B end-B-hilit ediff-B-buffer))
  4108.          
  4109.     (or (face-differs-from-default-p 'ediff-odd-diff-face-A-var)
  4110.     (not ediff-highlight-all-diffs)
  4111.     (progn
  4112.       (copy-face ediff-odd-diff-face-A 'ediff-odd-diff-face-A-var)
  4113.       (copy-face ediff-odd-diff-face-B 'ediff-odd-diff-face-B-var)
  4114.       (copy-face ediff-even-diff-face-A 'ediff-even-diff-face-A-var)
  4115.       (copy-face ediff-even-diff-face-B 'ediff-even-diff-face-B-var)))
  4116.       
  4117.     ;; unhighlight the background overlay for the diff n so they won't
  4118.     ;; interfere with the current diff overlay
  4119.     (ediff-overlay-put (ediff-get-diff-overlay n 'A) 'face nil)
  4120.     (ediff-overlay-put (ediff-get-diff-overlay n 'B) 'face nil)
  4121.     
  4122.     ;; (sit-for 0) ;; needed synch for some reason in v19.22
  4123.     ))
  4124.  
  4125.  
  4126. (defun ediff-unhighlight-diff ()
  4127.   "Remove overlays from buffers A and B."
  4128.     
  4129.   (ediff-move-overlay ediff-current-diff-overlay-A 1 1)
  4130.   (ediff-move-overlay ediff-current-diff-overlay-B 1 1)
  4131.   
  4132.   ;; rehighlight the overlay in the background of the
  4133.   ;; current difference region
  4134.   (ediff-overlay-put (ediff-get-diff-overlay ediff-current-difference 'A)
  4135.              'face (if (ediff-odd-p ediff-current-difference)
  4136.                    'ediff-odd-diff-face-A-var
  4137.                  'ediff-even-diff-face-A-var))
  4138.   (ediff-overlay-put (ediff-get-diff-overlay ediff-current-difference 'B)
  4139.              'face (if (ediff-odd-p ediff-current-difference)
  4140.                    'ediff-odd-diff-face-B-var
  4141.                  'ediff-even-diff-face-B-var))
  4142.   )
  4143.  
  4144.  
  4145. ;; delete highlighting overlays, restore faces to their original form
  4146. (defun ediff-unhighlight-diffs-totally ()
  4147.     (setq buffer-read-only nil)
  4148.     (ediff-unselect-and-select-difference -1)
  4149.     
  4150.     (if (and window-system ediff-want-faces)
  4151.     (let ((inhibit-quit t))
  4152.       (if (face-differs-from-default-p 'ediff-odd-diff-face-A-var)
  4153.           (progn
  4154.         (copy-face 'default 'ediff-odd-diff-face-A-var)
  4155.         (copy-face 'default 'ediff-odd-diff-face-B-var)
  4156.         (copy-face 'default 'ediff-even-diff-face-A-var)
  4157.         (copy-face 'default 'ediff-even-diff-face-B-var)))
  4158.       (if (ediff-overlayp ediff-current-diff-overlay-A)
  4159.           (ediff-delete-overlay ediff-current-diff-overlay-A))
  4160.       (setq ediff-current-diff-overlay-A nil)
  4161.       (if (ediff-overlayp ediff-current-diff-overlay-B)
  4162.           (ediff-delete-overlay ediff-current-diff-overlay-B))
  4163.       (setq ediff-current-diff-overlay-B nil)))
  4164.     )
  4165.     
  4166. (defun ediff-clear-diff-vector (vec &optional fin-diffs-also)
  4167.   ;; null out the difference overlays so they won't slow down future
  4168.   ;; editing operations
  4169.   (mapcar (function
  4170.        (lambda (elt)
  4171.          (ediff-delete-overlay (ediff-get-diff-overlay-from-vector elt 'A))
  4172.          (ediff-delete-overlay (ediff-get-diff-overlay-from-vector elt 'B))
  4173.          (if fin-diffs-also
  4174.          (ediff-clear-diff-vector 
  4175.           (ediff-get-fine-diff-vector-from-vec elt)))
  4176.          ))
  4177.       vec)
  4178.   ;; allow them to be garbage collected
  4179.   (setq vec nil))
  4180.       
  4181. (defun ediff-operate-on-flags (action)
  4182.   "Re/unhighlights buffers A and B with all flags from all Ediff sessions.
  4183. This is usually needed only when a
  4184. buffer is involved in multiple Ediff sessions."
  4185.   (let* ((A-sessions (ediff-eval-in-buffer
  4186.               ediff-A-buffer
  4187.               ediff-this-buffer-control-sessions))
  4188.      (B-sessions (ediff-eval-in-buffer
  4189.               ediff-B-buffer
  4190.               ediff-this-buffer-control-sessions))
  4191.      (sessions (ediff-union A-sessions B-sessions))
  4192.      (flag (if (eq action 'remove) 'unselect-only 'select-only)))
  4193.      
  4194.     (mapcar (function (lambda (buf)
  4195.             (ediff-eval-in-buffer
  4196.              buf
  4197.              (or (if (eq action 'insert)
  4198.                  (memq ediff-highlighting-style '(ascii off))
  4199.                    (not (eq ediff-highlighting-style 'ascii)))
  4200.                  (ediff-unselect-and-select-difference
  4201.                   ediff-current-difference 
  4202.                   flag 'no-recenter))
  4203.              )))
  4204.         sessions)))
  4205.  
  4206.  
  4207.  
  4208. ;;; Refinement of current diff        
  4209. ;; Split region along word boundaries. Each word will be on its own line.
  4210. ;; Output to buffer out-buffer.
  4211. (defun ediff-forward-word ()
  4212.   "Move point one word forward. Used for splitting diff regions into words.
  4213. This is the default for `ediff-forward-word-function'."
  4214.   (or (> (skip-chars-forward ediff-word-1) 0)
  4215.       (> (skip-chars-forward ediff-word-2) 0)))
  4216.  
  4217. (defun ediff-wordify (beg end in-buffer out-buffer)
  4218.   (let (sv-point string)
  4219.     (save-excursion
  4220.      (set-buffer in-buffer)
  4221.      (setq string (buffer-substring beg end))
  4222.  
  4223.      (set-buffer out-buffer)
  4224.      (erase-buffer)
  4225.      (insert string)
  4226.      (goto-char (point-min))
  4227.      (skip-chars-forward ediff-whitespace)
  4228.      (delete-region (point-min) (point))
  4229.      
  4230.      (while (not (eobp))
  4231.        (funcall ediff-forward-word-function)
  4232.        (setq sv-point (point))
  4233.        (skip-chars-forward ediff-whitespace)
  4234.        (delete-region sv-point (point))
  4235.        (insert "\n")))))
  4236.  
  4237. ;; `n' is the diff region to work on. 
  4238. ;; if `flag' is 'noforce then make fine-diffs only if this region's fine
  4239. ;; diffs have not been computed before.
  4240. ;; if `flag' is 'skip then don't compute fine diffs for this region.
  4241. (defun ediff-make-fine-diffs (&optional n flag)       
  4242.   (interactive)
  4243.   (or n  (setq n ediff-current-difference))
  4244.   
  4245.   (if (< ediff-number-of-differences 1)
  4246.       (error "No differences found."))
  4247.   
  4248.   (or (< n 0)
  4249.       (>= n ediff-number-of-differences)
  4250.       ;; n is within the range
  4251.       (let ((file-A ediff-temp-file-A)
  4252.     (file-B ediff-temp-file-B))
  4253.     
  4254.     (cond ((and (eq flag 'noforce) (ediff-get-fine-diff-vector n))
  4255.            nil)
  4256.           ((eq flag 'skip)
  4257.            (or (ediff-get-fine-diff-vector n)
  4258.            (eq ediff-auto-refine 'off)
  4259.            (message "Region %d is larger than auto-refine limit. Hit %S to force-refine."
  4260.                 (1+ n)
  4261.                 (substitute-command-keys
  4262.                  "\\[ediff-make-fine-diffs]")
  4263.                 )))
  4264.           (t
  4265.            ;; delete old fine diffs
  4266.            (ediff-clear-diff-vector (ediff-get-fine-diff-vector n))
  4267.            ;; recompute fine diffs
  4268.            (setq ediff-tmp-buffer (get-buffer-create "*ediff-tmp*"))
  4269.            
  4270.            (ediff-wordify
  4271.         (ediff-get-diff-posn 'A 'beg n)
  4272.         (ediff-get-diff-posn 'A 'end n)
  4273.         ediff-A-buffer
  4274.         ediff-tmp-buffer)
  4275.            (ediff-eval-in-buffer
  4276.         ediff-tmp-buffer
  4277.         (setq file-A (ediff-make-temp-file ".fine-diffs-A" file-A)))
  4278.            
  4279.            (ediff-wordify
  4280.         (ediff-get-diff-posn 'B 'beg n)
  4281.         (ediff-get-diff-posn 'B 'end n)
  4282.         ediff-B-buffer
  4283.         ediff-tmp-buffer)
  4284.            (ediff-eval-in-buffer
  4285.         ediff-tmp-buffer
  4286.         (setq file-B (ediff-make-temp-file ".fine-diffs-B" file-B)))
  4287.            
  4288.            ;; save temp file names.
  4289.            (setq ediff-temp-file-A file-A
  4290.              ediff-temp-file-B file-B)
  4291.            
  4292.            ;; set the new vector of fine diffs, if none exists
  4293.            (ediff-set-fine-diff-vector
  4294.         n
  4295.         (ediff-setup-diff-regions file-A file-B 'use-old-diff-buf n
  4296.                       ediff-fine-diff-program
  4297.                       ediff-fine-diff-options
  4298.                       ediff-fine-diff-ok-lines-regexp))
  4299.            (if (eq (length (ediff-get-fine-diff-vector n)) 0)
  4300.            (progn
  4301.              (message "No diffs found in region %d, except for white space and line breaks."
  4302.                   (1+ n))
  4303.              (ediff-mark-diff-as-space-only n t))
  4304.          (ediff-mark-diff-as-space-only n nil)))
  4305.           ) ;; end cond
  4306.     (ediff-set-fine-diff-properties n)
  4307.     )))
  4308.     
  4309.     
  4310. (defun ediff-set-fine-diff-properties (n &optional default)
  4311.   (or (not window-system)
  4312.       (< n 0)
  4313.       (>= n ediff-number-of-differences)
  4314.       ;; in a window system, set faces and priorities of fine overlays
  4315.       (let ((fine-diff-vector  (ediff-get-fine-diff-vector n))
  4316.         (face-A (if default 'default (face-name ediff-fine-diff-face-A)))
  4317.         (face-B (if default 'default (face-name ediff-fine-diff-face-B)))
  4318.         (priority-A (if default
  4319.                 0
  4320.               (1+ (ediff-overlay-get ediff-current-diff-overlay-A
  4321.                          'priority))))
  4322.         (priority-B (if default
  4323.                 0
  4324.               (1+ (ediff-overlay-get ediff-current-diff-overlay-B
  4325.                          'priority)))))
  4326.     (mapcar
  4327.      (function (lambda (vec)
  4328.              (ediff-overlay-put 
  4329.               (ediff-get-diff-overlay-from-vector vec 'A)
  4330.               'face face-A)
  4331.              (ediff-overlay-put
  4332.               (ediff-get-diff-overlay-from-vector vec 'A)
  4333.               'priority priority-A)
  4334.              
  4335.              (ediff-overlay-put
  4336.               (ediff-get-diff-overlay-from-vector vec 'B)
  4337.               'face face-B)
  4338.              (ediff-overlay-put
  4339.               (ediff-get-diff-overlay-from-vector vec 'B)
  4340.               'priority priority-B)
  4341.              ))
  4342.      fine-diff-vector)
  4343.     )))
  4344.     
  4345. (defun ediff-convert-diffs-to-overlays-refine (A-buffer B-buffer
  4346.                         diff-list refine-region)
  4347.   (let* ((current-diff -1)
  4348.      (reg-A-start (ediff-get-diff-posn 'A 'beg refine-region))
  4349.      (reg-B-start (ediff-get-diff-posn 'B 'beg refine-region))
  4350.      diff-overlay-list list-element
  4351.      a-begin a-end b-begin b-end
  4352.      a-overlay b-overlay)
  4353.  
  4354.     (ediff-eval-in-buffer A-buffer (goto-char reg-A-start))
  4355.     (ediff-eval-in-buffer B-buffer (goto-char reg-B-start))
  4356.      
  4357.     (while diff-list
  4358.       (setq current-diff (1+ current-diff)
  4359.         list-element (car diff-list)
  4360.         a-begin      (aref list-element 0)
  4361.         a-end      (aref list-element 1)
  4362.         b-begin      (aref list-element 2)
  4363.         b-end      (aref list-element 3))
  4364.         
  4365.       ;; put overlays at appropriate places in buffers
  4366.       (setq a-overlay (ediff-make-overlay 
  4367.                (ediff-goto-word (1+ a-begin) A-buffer)
  4368.                (ediff-goto-word a-end A-buffer 'end)
  4369.                A-buffer))
  4370.              
  4371.     (setq b-overlay (ediff-make-overlay 
  4372.              (ediff-goto-word (1+ b-begin) B-buffer)
  4373.              (ediff-goto-word b-end B-buffer 'end)
  4374.              B-buffer))
  4375.              
  4376.       ;; record all overlays for this difference
  4377.       (setq diff-overlay-list (nconc diff-overlay-list
  4378.                      (list (vector a-overlay b-overlay)))
  4379.         diff-list (cdr diff-list))
  4380.       ) ;; while
  4381.     ;; convert the list of difference information into a vector for
  4382.     ;; fast access
  4383.     (apply 'vector diff-overlay-list)))
  4384.  
  4385. ;; goto word #n starting at current position in buffer `buf'
  4386. ;; For ediff, a word is either a string of a-z,A-Z, incl `-' and `_';
  4387. ;; or a string of other non-blanks. A blank is a \n\t\C-j
  4388. ;; If `flag' is non-nil, goto the end of the n-th word.
  4389. (defun ediff-goto-word (n buf &optional flag)
  4390.   (ediff-eval-in-buffer
  4391.    buf
  4392.    (skip-chars-forward ediff-whitespace)
  4393.    (while (> n 1)
  4394.      (funcall ediff-forward-word-function)
  4395.      (skip-chars-forward ediff-whitespace)
  4396.      (setq n (1- n)))
  4397.    ;(if flag
  4398.    (if (and flag (> n 0))
  4399.        (funcall ediff-forward-word-function))
  4400.    (point)))
  4401.  
  4402.   
  4403.        
  4404.        
  4405.  
  4406. ;;; Misc
  4407.        
  4408. (defun ediff-union (list1 list2)
  4409.   "Combine LIST1 and LIST2 using a set-union operation.
  4410. The result list contains all items that appear in either LIST1 or LIST2.
  4411. This is a non-destructive function; it makes a copy of the data if necessary
  4412. to avoid corrupting the original LIST1 and LIST2.
  4413. This is a slightly simplified version from `cl-seq.el'.  Added here to
  4414. avoid loading cl-*."
  4415.   (cond ((null list1) list2) ((null list2) list1)
  4416.     ((equal list1 list2) list1)
  4417.     (t
  4418.      (or (>= (length list1) (length list2))
  4419.          (setq list1 (prog1 list2 (setq list2 list1))))
  4420.      (while list2
  4421.        (or (memq (car list2) list1)
  4422.            (setq list1 (cons (car list2) list1)))
  4423.        (setq list2 (cdr list2)))
  4424.      list1)))
  4425.  
  4426. ;(defun ediff-debug ()
  4427. ;  (interactive)
  4428. ;  (with-output-to-temp-buffer "*ediff-debug*"
  4429. ;    (princ 
  4430. ;     (format "Ctl buffer: %S\n\nediff-difference-vector:\n"
  4431. ;         ediff-control-buffer))
  4432. ;    (mapcar (function
  4433. ;         (lambda (overl-vec)
  4434. ;           (princ (format "Diff %d:  %S %S %S\n\t %S %S %S\n" 
  4435. ;                  (1+ (ediff-overlay-get (aref overl-vec 0)
  4436. ;                             'ediff-diff-num))
  4437. ;                  (ediff-overlay-get (aref overl-vec 0)
  4438. ;                         'ediff-control-buffer) 
  4439. ;                  (ediff-overlay-get (aref overl-vec 0)
  4440. ;                         'insert-in-front-hooks)
  4441. ;                  (aref overl-vec 0)
  4442. ;                  (ediff-overlay-get (aref overl-vec 1)
  4443. ;                         'ediff-control-buffer) 
  4444. ;                  (ediff-overlay-get (aref overl-vec 0)
  4445. ;                         'insert-in-front-hooks)
  4446. ;                  (aref overl-vec 1)
  4447. ;                  ))))
  4448. ;        ediff-difference-vector)
  4449. ;    (princ "\nediff-disturbed-overlays:\n")
  4450. ;    (mapcar (function
  4451. ;         (lambda (overl)
  4452. ;           (princ (format "%S  %S\n"
  4453. ;                  (ediff-overlay-get overl 'ediff-control-buffer)
  4454. ;                  overl
  4455. ;                  ))))
  4456. ;        ediff-disturbed-overlays)))
  4457.  
  4458.   
  4459. (run-hooks 'ediff-load-hooks)
  4460.   
  4461.  
  4462. (provide 'ediff)
  4463.  
  4464. ;;; ediff.el ends here
  4465.