1
0
mirror of https://github.com/m00natic/vlfi.git synced 2024-10-05 18:30:51 +01:00
vlfi/vlfi.el

817 lines
33 KiB
EmacsLisp
Raw Normal View History

;;; vlfi.el --- View Large Files Improved -*- lexical-binding: t -*-
;; Copyright (C) 2006, 2012, 2013 Free Software Foundation, Inc.
2013-04-14 17:55:39 +01:00
;; Version: 0.7
;; Keywords: large files, utilities
;; Authors: 2006 Mathias Dahl <mathias.dahl@gmail.com>
;; 2012 Sam Steingold <sds@gnu.org>
;; 2013 Andrey Kotlarski <m00naticus@gmail.com>
;; This file is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 2, or (at your option)
;; any later version.
;; This file is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs; see the file COPYING. If not, write to
;; the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
;; Boston, MA 02111-1307, USA.
;;; Commentary:
2013-02-02 14:12:30 +00:00
;; This package provides the M-x vlfi command, which visits part of a
;; large file in a read-only buffer without visiting the entire file.
;; The buffer uses VLFI mode, which defines several commands for
;; moving around, searching and editing selected chunk of file.
2013-02-02 14:12:30 +00:00
;; This package is an improved fork of the vlf.el package.
;;; Code:
2013-02-02 14:12:30 +00:00
(defgroup vlfi nil
"View Large Files in Emacs."
2013-02-02 14:12:30 +00:00
:prefix "vlfi-"
:group 'files)
2013-02-02 14:12:30 +00:00
(defcustom vlfi-batch-size 1024
"Defines how large each batch of file data is (in bytes)."
:type 'integer
2013-02-02 14:12:30 +00:00
:group 'vlfi)
;;; Keep track of file position.
(defvar vlfi-start-pos 0
"Absolute position of the visible chunk start.")
(defvar vlfi-end-pos vlfi-batch-size
"Absolute position of the visible chunk end.")
(defvar vlfi-file-size 0 "Total size of presented file.")
2013-02-02 14:12:30 +00:00
(defvar vlfi-mode-map
(let ((map (make-sparse-keymap)))
2013-02-02 14:12:30 +00:00
(define-key map [M-next] 'vlfi-next-batch)
(define-key map [M-prior] 'vlfi-prev-batch)
2013-04-01 01:07:20 +01:00
(define-key map "+" 'vlfi-change-batch-size)
(define-key map "-"
2013-02-02 14:12:30 +00:00
(lambda () "Decrease vlfi batch size by factor of 2."
2013-04-13 20:36:33 +01:00
(interactive)
(vlfi-change-batch-size t)))
2013-03-31 00:35:41 +00:00
(define-key map "s" 'vlfi-re-search-forward)
(define-key map "r" 'vlfi-re-search-backward)
(define-key map "o" 'vlfi-occur)
(define-key map "[" 'vlfi-beginning-of-file)
(define-key map "]" 'vlfi-end-of-file)
2013-04-02 00:55:53 +01:00
(define-key map "e" 'vlfi-edit-mode)
2013-04-13 20:57:09 +01:00
(define-key map "j" 'vlfi-jump-to-chunk)
2013-04-14 00:18:06 +01:00
(define-key map "l" 'vlfi-goto-line)
map)
2013-02-02 14:12:30 +00:00
"Keymap for `vlfi-mode'.")
2013-02-02 14:12:30 +00:00
(define-derived-mode vlfi-mode special-mode "VLFI"
"Mode to browse large files in."
(setq buffer-read-only t)
(set-buffer-modified-p nil)
(buffer-disable-undo)
2013-04-11 14:53:38 +01:00
(add-hook 'write-contents-functions 'vlfi-write)
(make-local-variable 'revert-buffer-function)
2013-04-11 20:57:49 +01:00
(setq revert-buffer-function 'vlfi-revert)
2013-02-02 14:12:30 +00:00
(make-local-variable 'vlfi-batch-size)
2013-04-07 22:56:42 +01:00
(put 'vlfi-batch-size 'permanent-local t)
2013-02-02 14:12:30 +00:00
(make-local-variable 'vlfi-start-pos)
2013-04-07 22:56:42 +01:00
(put 'vlfi-start-pos 'permanent-local t)
(make-local-variable 'vlfi-end-pos)
2013-04-07 22:56:42 +01:00
(put 'vlfi-end-pos 'permanent-local t)
2013-04-02 00:55:53 +01:00
(make-local-variable 'vlfi-file-size)
(put 'vlfi-file-size 'permanent-local t))
2013-04-13 20:49:08 +01:00
;;;###autoload
(defun vlfi (file)
"View Large FILE.
2013-04-13 20:49:08 +01:00
Batches of the file data from FILE will be displayed in a read-only
buffer. You can customize number of bytes displayed by customizing
`vlfi-batch-size'."
(interactive "fFile to open: ")
2013-04-13 20:49:08 +01:00
(with-current-buffer (generate-new-buffer "*vlfi*")
(setq buffer-file-name file
2013-04-13 22:52:07 +01:00
vlfi-file-size (vlfi-get-file-size file))
(vlfi-insert-file)
2013-04-13 20:49:08 +01:00
(vlfi-mode)
(switch-to-buffer (current-buffer))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; integration with other packages
;;;###autoload
(defun dired-vlfi ()
"In Dired, visit the file on this line in VLFI mode."
(interactive)
(vlfi (dired-get-file-for-visit)))
2013-04-13 20:49:08 +01:00
;;;###autoload
(eval-after-load "dired"
'(define-key dired-mode-map "V" 'dired-vlfi))
;;;###autoload
(defun vlfi-if-file-too-large (size op-type &optional filename)
"If file SIZE larger than `large-file-warning-threshold', \
allow user to view file with `vlfi', open it normally or abort.
OP-TYPE specifies the file operation being performed over FILENAME."
(and large-file-warning-threshold size
(> size large-file-warning-threshold)
(let ((char nil))
(while (not (memq (setq char
(read-event
(propertize
(format
"File %s is large (%s): \
%s normally (o), %s with vlfi (v) or abort (a)"
(if filename
(file-name-nondirectory filename)
"")
(file-size-human-readable size)
op-type op-type)
'face 'minibuffer-prompt)))
'(?o ?O ?v ?V ?a ?A))))
(cond ((memq char '(?o ?O)))
((memq char '(?v ?V))
(vlfi filename)
2013-04-13 20:49:08 +01:00
(error ""))
((memq char '(?a ?A))
(error "Aborted"))))))
;; hijack `abort-if-file-too-large'
;;;###autoload
(fset 'abort-if-file-too-large 'vlfi-if-file-too-large)
2013-04-14 00:50:31 +01:00
;; scroll auto batching
(defadvice scroll-up (around vlfi-scroll-up
activate compile)
"Slide to next batch if at end of buffer in `vlfi-mode'."
(if (and (eq major-mode 'vlfi-mode)
(eobp))
(progn (vlfi-next-batch 1)
(goto-char (point-min)))
ad-do-it))
(defadvice scroll-down (around vlfi-scroll-down
activate compile)
"Slide to previous batch if at beginning of buffer in `vlfi-mode'."
(if (and (eq major-mode 'vlfi-mode)
(bobp))
(progn (vlfi-prev-batch 1)
(goto-char (point-max)))
ad-do-it))
2013-04-13 20:49:08 +01:00
;; non recent Emacs
(unless (fboundp 'file-size-human-readable)
(defun file-size-human-readable (file-size)
"Print FILE-SIZE in MB."
(format "%.1fMB" (/ file-size 1024.0))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; utilities
2013-02-02 14:12:30 +00:00
(defun vlfi-change-batch-size (decrease)
"Change the buffer-local value of `vlfi-batch-size'.
Normally, the value is doubled;
with the prefix argument DECREASE it is halved."
(interactive "P")
2013-02-02 14:12:30 +00:00
(or (assq 'vlfi-batch-size (buffer-local-variables))
(error "%s is not local in this buffer" 'vlfi-batch-size))
(setq vlfi-batch-size (if decrease
2013-04-13 20:36:33 +01:00
(/ vlfi-batch-size 2)
(* vlfi-batch-size 2)))
(vlfi-move-to-batch vlfi-start-pos))
2013-02-02 14:12:30 +00:00
(defun vlfi-format-buffer-name ()
"Return format for vlfi buffer name."
(format "%s(%s)[%d/%d](%d)"
2013-04-13 20:36:33 +01:00
(file-name-nondirectory buffer-file-name)
(file-size-human-readable vlfi-file-size)
(/ vlfi-end-pos vlfi-batch-size)
(/ vlfi-file-size vlfi-batch-size)
vlfi-batch-size))
2013-02-02 14:12:30 +00:00
(defun vlfi-update-buffer-name ()
"Update the current buffer name."
2013-02-02 14:12:30 +00:00
(rename-buffer (vlfi-format-buffer-name) t))
2013-04-13 22:52:07 +01:00
(defmacro vlfi-get-file-size (file)
"Get size in bytes of FILE."
`(nth 7 (file-attributes ,file)))
2013-04-13 20:49:08 +01:00
(defun vlfi-insert-file (&optional from-end)
"Insert first chunk of current file contents in current buffer.
With FROM-END prefix, start from the back."
(if from-end
(setq vlfi-start-pos (max 0 (- vlfi-file-size vlfi-batch-size))
vlfi-end-pos vlfi-file-size)
(setq vlfi-start-pos 0
vlfi-end-pos (min vlfi-batch-size vlfi-file-size)))
(vlfi-move-to-chunk vlfi-start-pos vlfi-end-pos))
(defun vlfi-beginning-of-file ()
"Jump to beginning of file content."
(interactive)
(vlfi-insert-file))
(defun vlfi-end-of-file ()
"Jump to end of file content."
(interactive)
(vlfi-insert-file t))
(defun vlfi-revert (&optional ignore-auto noconfirm)
"Revert current chunk. Ignore IGNORE-AUTO.
Ask for confirmation if NOCONFIRM is nil."
(ignore ignore-auto)
(or noconfirm
(yes-or-no-p (format "Revert buffer from file %s? "
buffer-file-name))
(vlfi-move-to-chunk vlfi-start-pos vlfi-end-pos)))
2013-04-13 20:49:08 +01:00
2013-04-13 20:57:09 +01:00
(defun vlfi-jump-to-chunk (n)
"Go to to chunk N."
(interactive "nGoto to chunk: ")
(vlfi-move-to-batch (* (1- n) vlfi-batch-size)))
2013-04-13 20:49:08 +01:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; batch movement
2013-02-02 14:12:30 +00:00
(defun vlfi-next-batch (append)
"Display the next batch of file data.
When prefix argument is supplied and positive
jump over APPEND number of batches.
When prefix argument is negative
append next APPEND number of batches to the existing buffer."
(interactive "p")
2013-04-15 15:27:23 +01:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-14 23:26:58 +01:00
(setq vlfi-file-size (vlfi-get-file-size buffer-file-name)))
(let ((end (min (+ vlfi-end-pos (* vlfi-batch-size
(abs append)))
vlfi-file-size)))
(let ((inhibit-read-only t)
2013-04-13 20:36:33 +01:00
(do-append (< append 0))
(pos (position-bytes (point))))
(if do-append
2013-04-13 20:36:33 +01:00
(goto-char (point-max))
(setq vlfi-start-pos (- end vlfi-batch-size))
(erase-buffer))
(insert-file-contents buffer-file-name nil (if do-append
2013-04-13 20:36:33 +01:00
vlfi-end-pos
vlfi-start-pos)
end)
(setq vlfi-end-pos end)
(goto-char (or (byte-to-position (+ pos (vlfi-adjust-chunk)))
(point-max)))))
(set-visited-file-modtime)
(set-buffer-modified-p nil)
2013-02-02 14:12:30 +00:00
(vlfi-update-buffer-name))
2013-02-02 14:12:30 +00:00
(defun vlfi-prev-batch (prepend)
"Display the previous batch of file data.
When prefix argument is supplied and positive
jump over PREPEND number of batches.
When prefix argument is negative
append previous PREPEND number of batches to the existing buffer."
(interactive "p")
2013-02-02 14:12:30 +00:00
(if (zerop vlfi-start-pos)
(error "Already at BOF"))
2013-04-15 15:27:23 +01:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-14 23:26:58 +01:00
(setq vlfi-file-size (vlfi-get-file-size buffer-file-name)))
(let ((inhibit-read-only t)
2013-04-13 20:36:33 +01:00
(start (max 0 (- vlfi-start-pos (* vlfi-batch-size
(abs prepend)))))
(do-prepend (< prepend 0))
(pos (- (position-bytes (point-max))
(position-bytes (point)))))
(if do-prepend
2013-04-13 20:36:33 +01:00
(goto-char (point-min))
2013-04-14 23:26:58 +01:00
(setq vlfi-end-pos (min (+ start vlfi-batch-size)
vlfi-file-size))
(erase-buffer))
(insert-file-contents buffer-file-name nil start
2013-04-13 20:36:33 +01:00
(if do-prepend
vlfi-start-pos
vlfi-end-pos))
(setq vlfi-start-pos start)
(setq pos (+ pos (vlfi-adjust-chunk)))
(goto-char (or (byte-to-position (- (position-bytes (point-max))
pos))
(point-max))))
(set-visited-file-modtime)
(set-buffer-modified-p nil)
2013-02-02 14:12:30 +00:00
(vlfi-update-buffer-name))
(defun vlfi-move-to-batch (start &optional minimal)
"Move to batch determined by START.
Adjust according to file start/end and show `vlfi-batch-size' bytes.
When given MINIMAL flag, skip non important operations."
2013-04-15 15:27:23 +01:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-14 23:26:58 +01:00
(setq vlfi-file-size (vlfi-get-file-size buffer-file-name)))
(setq vlfi-start-pos (max 0 start)
2013-04-14 23:26:58 +01:00
vlfi-end-pos (min (+ vlfi-start-pos vlfi-batch-size)
vlfi-file-size))
(if (= vlfi-file-size vlfi-end-pos) ; re-check file size
(setq vlfi-start-pos (max 0 (- vlfi-end-pos vlfi-batch-size))))
(let ((inhibit-read-only t)
(pos (position-bytes (point))))
(erase-buffer)
(insert-file-contents buffer-file-name nil
2013-04-13 20:36:33 +01:00
vlfi-start-pos vlfi-end-pos)
(goto-char (or (byte-to-position (+ pos (vlfi-adjust-chunk)))
(point-max))))
(set-buffer-modified-p nil)
2013-04-14 23:26:58 +01:00
(set-visited-file-modtime)
(or minimal(vlfi-update-buffer-name)))
(defun vlfi-move-to-chunk (start end &optional minimal)
"Move to chunk determined by START END.
When given MINIMAL flag, skip non important operations."
2013-04-15 15:27:23 +01:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-13 22:52:07 +01:00
(setq vlfi-file-size (vlfi-get-file-size buffer-file-name)))
2013-03-31 18:44:41 +01:00
(setq vlfi-start-pos (max 0 start)
2013-04-13 20:36:33 +01:00
vlfi-end-pos (min end vlfi-file-size))
(let ((inhibit-read-only t)
(pos (position-bytes (point))))
2013-03-31 00:36:54 +00:00
(erase-buffer)
(insert-file-contents buffer-file-name nil
2013-04-13 20:36:33 +01:00
vlfi-start-pos vlfi-end-pos)
(goto-char (or (byte-to-position (+ pos (vlfi-adjust-chunk)))
(point-max))))
2013-03-31 00:36:54 +00:00
(set-buffer-modified-p nil)
2013-04-14 23:26:58 +01:00
(set-visited-file-modtime)
(or minimal (vlfi-update-buffer-name)))
2013-03-31 00:36:54 +00:00
(defun vlfi-adjust-chunk ()
"Adjust chunk beginning until content can be properly decoded.
Return number of bytes moved back for this to happen."
(let ((shift 0))
(while (and (not (zerop vlfi-start-pos))
(< shift 3)
(/= (- vlfi-end-pos vlfi-start-pos)
(length (encode-coding-region
(point-min) (point-max)
buffer-file-coding-system t))))
(setq shift (1+ shift)
vlfi-start-pos (1- vlfi-start-pos))
(let ((inhibit-read-only t))
(erase-buffer)
(insert-file-contents buffer-file-name nil
vlfi-start-pos vlfi-end-pos)))
(set-buffer-modified-p nil)
shift))
2013-04-13 20:49:08 +01:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; search
2013-04-13 20:49:08 +01:00
(defun vlfi-re-search (regexp count backward)
"Search for REGEXP COUNT number of times forward or BACKWARD."
(let* ((match-chunk-start vlfi-start-pos)
(match-chunk-end vlfi-end-pos)
(match-start-pos (+ vlfi-start-pos (position-bytes (point))))
2013-04-13 20:36:33 +01:00
(match-end-pos match-start-pos)
(to-find count)
(reporter (make-progress-reporter
(concat "Searching for " regexp "...")
(if backward
(- vlfi-file-size vlfi-end-pos)
vlfi-start-pos)
vlfi-file-size))
2013-04-13 20:36:33 +01:00
(batch-step (/ vlfi-batch-size 8))) ; amount of chunk overlap
(unwind-protect
2013-04-13 20:36:33 +01:00
(catch 'end-of-file
(if backward
(while (not (zerop to-find))
(cond ((re-search-backward regexp nil t)
(setq to-find (1- to-find)
match-chunk-start vlfi-start-pos
match-chunk-end vlfi-end-pos
2013-04-13 20:36:33 +01:00
match-start-pos (+ vlfi-start-pos
(position-bytes
(match-beginning 0)))
2013-04-13 20:36:33 +01:00
match-end-pos (+ vlfi-start-pos
(position-bytes
(match-end 0)))))
2013-04-13 20:36:33 +01:00
((zerop vlfi-start-pos)
(throw 'end-of-file nil))
(t (let ((batch-move (- vlfi-start-pos
(- vlfi-batch-size
batch-step))))
(vlfi-move-to-batch
(if (< match-start-pos batch-move)
(- match-start-pos vlfi-batch-size)
batch-move) t))
2013-04-13 20:36:33 +01:00
(goto-char (if (< match-start-pos
vlfi-end-pos)
(or (byte-to-position
(- match-start-pos
vlfi-start-pos))
(point-max))
2013-04-13 20:36:33 +01:00
(point-max)))
2013-04-15 15:27:23 +01:00
(progress-reporter-update
reporter (- vlfi-file-size
vlfi-start-pos)))))
2013-04-13 20:36:33 +01:00
(while (not (zerop to-find))
(cond ((re-search-forward regexp nil t)
(setq to-find (1- to-find)
match-chunk-start vlfi-start-pos
match-chunk-end vlfi-end-pos
2013-04-13 20:36:33 +01:00
match-start-pos (+ vlfi-start-pos
(position-bytes
(match-beginning 0)))
2013-04-13 20:36:33 +01:00
match-end-pos (+ vlfi-start-pos
(position-bytes
(match-end 0)))))
2013-04-13 20:36:33 +01:00
((= vlfi-end-pos vlfi-file-size)
(throw 'end-of-file nil))
(t (let ((batch-move (- vlfi-end-pos batch-step)))
(vlfi-move-to-batch
(if (< batch-move match-end-pos)
match-end-pos
batch-move) t))
2013-04-13 20:36:33 +01:00
(goto-char (if (< vlfi-start-pos match-end-pos)
(or (byte-to-position
(- match-end-pos
vlfi-start-pos))
(point-max))
2013-04-13 20:36:33 +01:00
(point-min)))
(progress-reporter-update reporter
2013-04-13 20:36:33 +01:00
vlfi-end-pos)))))
(progress-reporter-done reporter))
2013-04-01 10:57:58 +01:00
(if backward
(vlfi-goto-match match-chunk-start match-chunk-end
match-end-pos match-start-pos
2013-04-13 20:36:33 +01:00
count to-find)
(vlfi-goto-match match-chunk-start match-chunk-end
match-start-pos match-end-pos
2013-04-13 20:36:33 +01:00
count to-find)))))
2013-04-01 10:57:58 +01:00
(defun vlfi-goto-match (match-chunk-start match-chunk-end
match-pos-start
match-pos-end
count to-find)
"Move to MATCH-CHUNK-START MATCH-CHUNK-END surrounding \
MATCH-POS-START and MATCH-POS-END.
2013-04-01 12:38:50 +01:00
According to COUNT and left TO-FIND, show if search has been
successful. Return nil if nothing found."
(if (= count to-find)
(progn (vlfi-move-to-chunk match-chunk-start match-chunk-end)
(goto-char (or (byte-to-position (- match-pos-start
vlfi-start-pos))
(point-max)))
(message "Not found")
nil)
(let ((success (zerop to-find)))
(if success
(vlfi-update-buffer-name)
(vlfi-move-to-chunk match-chunk-start match-chunk-end))
(let* ((match-end (or (byte-to-position (- match-pos-end
vlfi-start-pos))
(point-max)))
(overlay (make-overlay (byte-to-position
(- match-pos-start
vlfi-start-pos))
match-end)))
2013-04-16 10:25:06 +01:00
(overlay-put overlay 'face 'match)
(unless success
(goto-char match-end)
(message "Moved to the %d match which is last"
(- count to-find)))
2013-04-13 20:36:33 +01:00
(sit-for 0.1)
(delete-overlay overlay)
t))))
2013-03-29 16:12:20 +00:00
(defun vlfi-re-search-forward (regexp count)
2013-04-10 00:10:00 +01:00
"Search forward for REGEXP prefix COUNT number of times.
Search is performed chunk by chunk in `vlfi-batch-size' memory."
2013-03-31 18:44:41 +01:00
(interactive (list (read-regexp "Search whole file"
2013-04-13 20:36:33 +01:00
(if regexp-history
2013-04-15 15:27:23 +01:00
(car regexp-history)))
2013-04-13 20:36:33 +01:00
(or current-prefix-arg 1)))
(vlfi-re-search regexp count nil))
2013-03-29 16:12:20 +00:00
(defun vlfi-re-search-backward (regexp count)
2013-04-10 00:10:00 +01:00
"Search backward for REGEXP prefix COUNT number of times.
Search is performed chunk by chunk in `vlfi-batch-size' memory."
2013-04-01 10:06:48 +01:00
(interactive (list (read-regexp "Search whole file backward"
2013-04-13 20:36:33 +01:00
(if regexp-history
2013-04-15 15:27:23 +01:00
(car regexp-history)))
2013-04-13 20:36:33 +01:00
(or current-prefix-arg 1)))
(vlfi-re-search regexp count t))
2013-03-29 16:12:20 +00:00
2013-04-14 00:18:06 +01:00
(defun vlfi-goto-line (n)
"Go to line N."
(interactive "nGo to line: ")
(let ((start-pos vlfi-start-pos)
(end-pos vlfi-end-pos)
(pos (point))
(success nil))
(unwind-protect
(progn (vlfi-beginning-of-file)
(goto-char (point-min))
(setq success (vlfi-re-search-forward "[\n\C-m]"
(1- n))))
(unless success
(vlfi-move-to-chunk start-pos end-pos)
(goto-char pos)))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; occur
2013-04-16 10:25:06 +01:00
(defvar vlfi-occur-mode-map
(let ((map (make-sparse-keymap)))
(define-key map "n" 'vlfi-occur-next-match)
(define-key map "p" 'vlfi-occur-prev-match)
(define-key map "\C-m" 'vlfi-occur-visit)
(define-key map [mouse-1] 'vlfi-occur-visit)
2013-04-16 10:25:06 +01:00
map)
"Keymap for command `vlfi-occur-mode'.")
(define-derived-mode vlfi-occur-mode special-mode "VLFI[occur]"
"Major mode for showing occur matches of VLFI opened files.")
(defun vlfi-occur-next-match ()
"Move cursor to next match."
(interactive)
(if (eq (get-char-property (point) 'face) 'match)
(goto-char (next-single-property-change (point) 'face)))
(goto-char (or (text-property-any (point) (point-max) 'face 'match)
(text-property-any (point-min) (point)
'face 'match))))
(defun vlfi-occur-prev-match ()
"Move cursor to previous match."
(interactive)
(if (eq (get-char-property (point) 'face) 'match)
(goto-char (previous-single-property-change (point) 'face)))
(while (not (eq (get-char-property (point) 'face) 'match))
(goto-char (or (previous-single-property-change (point) 'face)
(point-max)))))
(defun vlfi-occur-visit (&optional event)
"Visit current `vlfi-occur' link in a vlfi buffer.
The same for mouse EVENT."
(interactive (list last-nonmenu-event))
(when event
(switch-to-buffer (window-buffer (posn-window (event-end event))))
(goto-char (posn-point (event-end event))))
(let* ((pos (point))
(pos-relative (- pos (line-beginning-position)))
(file (get-char-property pos 'file)))
(if file
(let ((chunk-start (get-char-property pos 'chunk-start))
(chunk-end (get-char-property pos 'chunk-end))
(buffer (get-char-property pos 'buffer))
(match-pos (or (get-char-property pos 'match-pos)
(+ (get-char-property pos 'line-pos)
(if (< 8 pos-relative)
(- pos-relative 8)
0)))))
(unless (buffer-live-p buffer)
(let ((occur-buffer (current-buffer)))
(setq buffer (vlfi file))
(switch-to-buffer occur-buffer)))
(pop-to-buffer buffer)
(vlfi-move-to-chunk chunk-start chunk-end)
(set-buffer buffer)
(goto-char match-pos)))))
(defun vlfi-occur (regexp)
"Make occur style index for REGEXP."
(interactive (list (read-regexp "List lines matching regexp"
(if regexp-history
(car regexp-history)))))
(let ((start-pos vlfi-start-pos)
(end-pos vlfi-end-pos)
(pos (point)))
2013-04-16 10:25:06 +01:00
(vlfi-beginning-of-file)
(goto-char (point-min))
(vlfi-build-occur regexp)
(vlfi-move-to-chunk start-pos end-pos)
(goto-char pos)))
(defun vlfi-build-occur (regexp)
"Build occur style index for REGEXP."
(let ((line 1)
(last-match-line 0)
(last-line-pos (point-min))
(file buffer-file-name)
(match-end-pos (+ vlfi-start-pos (position-bytes (point))))
(occur-buffer (generate-new-buffer
(concat "*VLFI-occur " (file-name-nondirectory
buffer-file-name)
"*")))
(line-regexp (concat "\\(?5:[\n\C-m]\\)\\|\\(?10:"
regexp "\\)"))
(batch-step (/ vlfi-batch-size 8))
(reporter (make-progress-reporter
(concat "Building index for " regexp "...")
vlfi-start-pos vlfi-file-size)))
(unwind-protect
(progn
(while (/= vlfi-end-pos vlfi-file-size)
(if (re-search-forward line-regexp nil t)
(progn
(setq match-end-pos (+ vlfi-start-pos
(position-bytes
(match-end 0))))
(if (match-string 5)
(setq line (1+ line)
last-line-pos (point))
(let* ((chunk-start vlfi-start-pos)
(chunk-end vlfi-end-pos)
(vlfi-buffer (current-buffer))
(line-pos (line-beginning-position))
(line-text (buffer-substring
line-pos (line-end-position))))
(with-current-buffer occur-buffer
(unless (= line last-match-line)
(insert (propertize
(format "%7d:" line)
'file file
'buffer vlfi-buffer
'chunk-start chunk-start
'chunk-end chunk-end
'face 'shadow
'mouse-face '(highlight)
'line-pos line-pos))
(insert (propertize
(format "%s\n" line-text)
'file file
'buffer vlfi-buffer
'chunk-start chunk-start
'chunk-end chunk-end
'mouse-face '(highlight)
'line-pos line-pos)))
(forward-line -1)
(let ((line-start (+ (line-beginning-position)
8))
(match-pos (match-beginning 10)))
(add-text-properties
(+ line-start match-pos (- last-line-pos))
(+ line-start (match-end 10)
(- last-line-pos))
(list 'face 'match 'match-pos match-pos)))
(forward-line)
(setq last-match-line line)))))
(let ((batch-move (- vlfi-end-pos batch-step)))
(vlfi-move-to-batch (if (< batch-move match-end-pos)
match-end-pos
batch-move) t))
(goto-char (if (< vlfi-start-pos match-end-pos)
(or (byte-to-position (- match-end-pos
vlfi-start-pos))
(point-min))
(point-min)))
(setq last-match-line 0
last-line-pos (point-min))
(progress-reporter-update reporter vlfi-end-pos)))
(progress-reporter-done reporter))
(with-current-buffer occur-buffer
(goto-char (point-min))
(let ((match-count (count-lines (point-min) (point-max))))
(insert (propertize
(format "%d matches for \"%s\" in file \
\(from %d lines\): %s\n" match-count regexp line file)
2013-04-16 10:25:06 +01:00
'face 'underline)))
(vlfi-occur-mode))
(display-buffer occur-buffer))))
2013-04-13 20:49:08 +01:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
2013-04-02 00:55:53 +01:00
;;; editing
2013-04-13 20:49:08 +01:00
2013-04-02 00:55:53 +01:00
(defvar vlfi-edit-mode-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map text-mode-map)
(define-key map "\C-c\C-c" 'vlfi-write)
(define-key map "\C-c\C-q" 'vlfi-discard-edit)
map)
2013-04-07 22:56:42 +01:00
"Keymap for command `vlfi-edit-mode'.")
(define-derived-mode vlfi-edit-mode vlfi-mode "VLFI[edit]"
"Major mode for editing large file chunks."
(setq buffer-read-only nil)
(buffer-enable-undo)
(message (substitute-command-keys
2013-04-13 20:36:33 +01:00
"Editing: Type \\[vlfi-write] to write chunk \
2013-04-07 22:56:42 +01:00
or \\[vlfi-discard-edit] to discard changes.")))
2013-04-02 00:55:53 +01:00
2013-04-13 20:49:08 +01:00
(defun vlfi-discard-edit ()
"Discard edit and refresh chunk from file."
(interactive)
(vlfi-move-to-chunk vlfi-start-pos vlfi-end-pos)
(vlfi-mode)
(message "Switched to VLFI mode."))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; saving
(defun vlfi-write ()
"Write current chunk to file. Always return true to disable save.
If changing size of chunk shift remaining file content."
(interactive)
(when (and (buffer-modified-p)
2013-04-15 15:27:23 +01:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-13 20:49:08 +01:00
(y-or-n-p "File has changed since visited or saved. \
Save anyway? ")))
2013-04-13 22:53:24 +01:00
(let ((pos (point))
(size-change (- vlfi-end-pos vlfi-start-pos
2013-04-13 20:49:08 +01:00
(length (encode-coding-region
(point-min) (point-max)
2013-04-13 22:53:24 +01:00
buffer-file-coding-system t)))))
2013-04-13 20:49:08 +01:00
(cond ((zerop size-change)
(write-region nil nil buffer-file-name vlfi-start-pos t))
((< 0 size-change)
(vlfi-file-shift-back size-change))
(t (vlfi-file-shift-forward (- size-change))))
2013-04-13 22:53:24 +01:00
(vlfi-move-to-chunk vlfi-start-pos vlfi-end-pos)
2013-04-13 20:49:08 +01:00
(goto-char pos))
(vlfi-mode)
t))
(defun vlfi-file-shift-back (size-change)
"Shift file contents SIZE-CHANGE bytes back."
(write-region nil nil buffer-file-name vlfi-start-pos t)
(buffer-disable-undo)
(let ((read-start-pos vlfi-end-pos)
(coding-system-for-write 'no-conversion)
2013-04-15 15:27:23 +01:00
(reporter (make-progress-reporter "Adjusting file content..."
vlfi-end-pos
vlfi-file-size)))
(while (vlfi-shift-batch read-start-pos (- read-start-pos
size-change))
(setq read-start-pos (+ read-start-pos vlfi-batch-size))
(progress-reporter-update reporter read-start-pos))
;; pad end with space
(erase-buffer)
(insert-char 32 size-change)
(write-region nil nil buffer-file-name (- vlfi-file-size
size-change) t)
(progress-reporter-done reporter)))
(defun vlfi-shift-batch (read-pos write-pos)
"Read `vlfi-batch-size' bytes from READ-POS and write them \
back at WRITE-POS. Return nil if EOF is reached, t otherwise."
(erase-buffer)
2013-04-15 15:27:23 +01:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-14 23:26:58 +01:00
(setq vlfi-file-size (vlfi-get-file-size buffer-file-name)))
(let ((read-end (+ read-pos vlfi-batch-size)))
(insert-file-contents-literally buffer-file-name nil
read-pos
(min vlfi-file-size read-end))
(write-region nil nil buffer-file-name write-pos 0)
(< read-end vlfi-file-size)))
(defun vlfi-file-shift-forward (size-change)
"Shift file contents SIZE-CHANGE bytes forward.
Done by saving content up front and then writing previous batch."
(buffer-disable-undo)
(let ((size (+ vlfi-batch-size size-change))
(read-pos vlfi-end-pos)
(write-pos vlfi-start-pos)
(reporter (make-progress-reporter "Adjusting file content..."
vlfi-start-pos
vlfi-file-size)))
(when (vlfi-shift-batches size read-pos write-pos t)
(setq write-pos (+ read-pos size-change)
read-pos (+ read-pos size))
(progress-reporter-update reporter write-pos)
(let ((coding-system-for-write 'no-conversion))
(while (vlfi-shift-batches size read-pos write-pos nil)
(setq write-pos (+ read-pos size-change)
read-pos (+ read-pos size))
(progress-reporter-update reporter write-pos))))
(progress-reporter-done reporter)))
(defun vlfi-shift-batches (size read-pos write-pos hide-read)
"Append SIZE bytes of file starting at READ-POS.
Then write initial buffer content to file at WRITE-POS.
If HIDE-READ is non nil, temporarily hide literal read content.
Return nil if EOF is reached, t otherwise."
(or (verify-visited-file-modtime (current-buffer))
(setq vlfi-file-size (vlfi-get-file-size buffer-file-name)))
(let ((read-more (< read-pos vlfi-file-size))
(start-write-pos (point-min))
(end-write-pos (point-max)))
(when read-more
(goto-char end-write-pos)
(insert-file-contents-literally buffer-file-name nil read-pos
(min vlfi-file-size (+ read-pos
size))))
;; write
(if hide-read ; hide literal region if user has to choose encoding
(narrow-to-region start-write-pos end-write-pos))
(write-region start-write-pos end-write-pos
buffer-file-name write-pos 0)
(delete-region start-write-pos end-write-pos)
(if hide-read (widen))
read-more))
2013-04-11 15:44:32 +01:00
2013-02-02 14:12:30 +00:00
(provide 'vlfi)
2013-02-02 14:12:30 +00:00
;;; vlfi.el ends here