1
0
mirror of https://github.com/m00natic/vlfi.git synced 2025-03-13 22:28:05 +00:00
vlfi/vlf.el

875 lines
35 KiB
EmacsLisp
Raw Normal View History

2013-08-25 17:57:43 +03:00
;;; vlf.el --- View Large Files -*- lexical-binding: t -*-
;; Copyright (C) 2006, 2012, 2013 Free Software Foundation, Inc.
;; Version: 0.9.1
;; Keywords: large files, utilities
;; Maintainer: Andrey Kotlarski <m00naticus@gmail.com>
;; Authors: 2006 Mathias Dahl <mathias.dahl@gmail.com>
;; 2012 Sam Steingold <sds@gnu.org>
;; 2013 Andrey Kotlarski <m00naticus@gmail.com>
;; URL: https://github.com/m00natic/vlfi
;; 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-08-25 17:57:43 +03:00
;; This package provides the M-x vlf command, which visits part of a
2013-05-01 02:02:28 +03:00
;; large file without loading the entire file.
2013-08-25 17:57:43 +03:00
;; The buffer uses VLF mode, which defines several commands for
2013-05-01 02:02:28 +03:00
;; moving around, searching and editing selected part of file.
;; This package was inspired by a snippet posted by Kevin Rodgers,
;; showing how to use `insert-file-contents' to extract part of a
;; file.
;;; Code:
2013-08-25 17:57:43 +03:00
(defgroup vlf nil
"View Large Files in Emacs."
2013-08-25 17:57:43 +03:00
:prefix "vlf-"
:group 'files)
2013-08-25 17:57:43 +03:00
(defcustom vlf-batch-size 1024
"Defines how large each batch of file data is (in bytes)."
:type 'integer
2013-08-25 17:57:43 +03:00
:group 'vlf)
(put 'vlf-batch-size 'permanent-local t)
;;; Keep track of file position.
2013-08-25 17:57:43 +03:00
(defvar vlf-start-pos 0
"Absolute position of the visible chunk start.")
2013-08-25 17:57:43 +03:00
(put 'vlf-start-pos 'permanent-local t)
2013-08-25 17:57:43 +03:00
(defvar vlf-end-pos 0 "Absolute position of the visible chunk end.")
(put 'vlf-end-pos 'permanent-local t)
(defvar vlf-file-size 0 "Total size of presented file.")
(put 'vlf-file-size 'permanent-local t)
(defvar vlf-encode-size 0 "Size in bytes of current batch decoded.")
(put 'vlf-encode-size 'permanent-local t)
(defvar vlf-mode-map
(let ((map (make-sparse-keymap)))
2013-08-25 17:57:43 +03:00
(define-key map [M-next] 'vlf-next-batch)
(define-key map [M-prior] 'vlf-prev-batch)
(define-key map "+" 'vlf-change-batch-size)
2013-04-01 03:07:20 +03:00
(define-key map "-"
2013-08-25 17:57:43 +03:00
(lambda () "Decrease vlf batch size by factor of 2."
2013-04-13 22:36:33 +03:00
(interactive)
2013-08-25 17:57:43 +03:00
(vlf-change-batch-size t)))
(define-key map "s" 'vlf-re-search-forward)
(define-key map "r" 'vlf-re-search-backward)
(define-key map "o" 'vlf-occur)
(define-key map "[" 'vlf-beginning-of-file)
(define-key map "]" 'vlf-end-of-file)
(define-key map "e" 'vlf-edit-mode)
(define-key map "j" 'vlf-jump-to-chunk)
(define-key map "l" 'vlf-goto-line)
map)
2013-08-25 17:57:43 +03:00
"Keymap for `vlf-mode'.")
2013-08-25 17:57:43 +03:00
(define-derived-mode vlf-mode special-mode "VLF"
"Mode to browse large files in."
(setq buffer-read-only t)
(set-buffer-modified-p nil)
(buffer-disable-undo)
2013-08-25 17:57:43 +03:00
(add-hook 'write-file-functions 'vlf-write nil t)
(make-local-variable 'revert-buffer-function)
2013-08-25 17:57:43 +03:00
(setq revert-buffer-function 'vlf-revert)
(make-local-variable 'vlf-batch-size)
(make-local-variable 'vlf-start-pos)
(make-local-variable 'vlf-end-pos)
(make-local-variable 'vlf-file-size)
(make-local-variable 'vlf-encode-size))
2013-04-13 22:49:08 +03:00
;;;###autoload
2013-08-25 17:57:43 +03:00
(defun vlf (file)
"View Large FILE.
2013-04-13 22:49:08 +03: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
2013-08-25 17:57:43 +03:00
`vlf-batch-size'."
(interactive "fFile to open: ")
2013-08-25 17:57:43 +03:00
(with-current-buffer (generate-new-buffer "*vlf*")
(set-visited-file-name file)
2013-08-25 17:57:43 +03:00
(vlf-mode)
(setq vlf-file-size (vlf-get-file-size buffer-file-name))
(vlf-insert-file)
2013-04-13 22:49:08 +03:00
(switch-to-buffer (current-buffer))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; integration with other packages
;;;###autoload
2013-08-25 17:57:43 +03:00
(defun dired-vlf ()
"In Dired, visit the file on this line in VLF mode."
(interactive)
2013-08-25 17:57:43 +03:00
(vlf (dired-get-file-for-visit)))
2013-04-13 22:49:08 +03:00
;;;###autoload
(eval-after-load "dired"
2013-08-25 17:57:43 +03:00
'(define-key dired-mode-map "V" 'dired-vlf))
2013-04-13 22:49:08 +03:00
;;;###autoload
2013-08-25 17:57:43 +03:00
(defadvice abort-if-file-too-large (around vlf-if-file-too-large
(size op-type
&optional filename)
compile activate)
2013-04-13 22:49:08 +03:00
"If file SIZE larger than `large-file-warning-threshold', \
2013-08-25 17:57:43 +03:00
allow user to view file with `vlf', open it normally, or abort.
2013-04-13 22:49:08 +03:00
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): \
2013-08-25 17:57:43 +03:00
%s normally (o), %s with vlf (v) or abort (a)"
2013-04-13 22:49:08 +03:00
(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))
2013-08-25 17:57:43 +03:00
(vlf filename)
2013-04-13 22:49:08 +03:00
(error ""))
((memq char '(?a ?A))
(error "Aborted"))))))
2013-04-14 02:50:31 +03:00
;; scroll auto batching
2013-08-25 17:57:43 +03:00
(defadvice scroll-up (around vlf-scroll-up
2013-04-14 02:50:31 +03:00
activate compile)
2013-08-25 17:57:43 +03:00
"Slide to next batch if at end of buffer in `vlf-mode'."
(if (and (derived-mode-p 'vlf-mode)
2013-04-14 02:50:31 +03:00
(eobp))
2013-08-25 17:57:43 +03:00
(progn (vlf-next-batch 1)
2013-04-14 02:50:31 +03:00
(goto-char (point-min)))
ad-do-it))
2013-08-25 17:57:43 +03:00
(defadvice scroll-down (around vlf-scroll-down
2013-04-14 02:50:31 +03:00
activate compile)
2013-08-25 17:57:43 +03:00
"Slide to previous batch if at beginning of buffer in `vlf-mode'."
(if (and (derived-mode-p 'vlf-mode)
2013-04-14 02:50:31 +03:00
(bobp))
2013-08-25 17:57:43 +03:00
(progn (vlf-prev-batch 1)
2013-04-14 02:50:31 +03:00
(goto-char (point-max)))
ad-do-it))
;; non-recent Emacs
;;;###autoload
2013-04-13 22:49:08 +03:00
(unless (fboundp 'file-size-human-readable)
(defun file-size-human-readable (file-size)
"Print FILE-SIZE in MB."
(format "%.1fMB" (/ file-size 1048576.0))))
2013-04-13 22:49:08 +03:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; utilities
2013-08-25 17:57:43 +03:00
(defun vlf-change-batch-size (decrease)
"Change the buffer-local value of `vlf-batch-size'.
Normally, the value is doubled;
with the prefix argument DECREASE it is halved."
(interactive "P")
2013-08-25 17:57:43 +03:00
(setq vlf-batch-size (if decrease
(/ vlf-batch-size 2)
(* vlf-batch-size 2)))
(vlf-move-to-batch vlf-start-pos))
2013-02-02 16:12:30 +02:00
2013-08-25 17:57:43 +03:00
(defun vlf-format-buffer-name ()
"Return format for vlf buffer name."
(format "%s(%s)[%d/%d](%d)"
2013-04-13 22:36:33 +03:00
(file-name-nondirectory buffer-file-name)
2013-08-25 17:57:43 +03:00
(file-size-human-readable vlf-file-size)
(/ vlf-end-pos vlf-batch-size)
(/ vlf-file-size vlf-batch-size)
vlf-batch-size))
2013-08-25 17:57:43 +03:00
(defun vlf-update-buffer-name ()
"Update the current buffer name."
2013-08-25 17:57:43 +03:00
(rename-buffer (vlf-format-buffer-name) t))
2013-08-25 17:57:43 +03:00
(defun vlf-get-file-size (file)
2013-04-14 00:52:07 +03:00
"Get size in bytes of FILE."
2013-05-02 12:04:12 +03:00
(nth 7 (file-attributes file)))
2013-04-14 00:52:07 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-verify-size ()
"Update file size information if necessary and visited file time."
(unless (verify-visited-file-modtime (current-buffer))
2013-08-25 17:57:43 +03:00
(setq vlf-file-size (vlf-get-file-size buffer-file-name))
(set-visited-file-modtime)))
2013-08-25 17:57:43 +03:00
(defun vlf-insert-file (&optional from-end)
2013-04-13 22:49:08 +03:00
"Insert first chunk of current file contents in current buffer.
With FROM-END prefix, start from the back."
(if from-end
2013-08-25 17:57:43 +03:00
(setq vlf-start-pos (max 0 (- vlf-file-size vlf-batch-size))
vlf-end-pos vlf-file-size)
(setq vlf-start-pos 0
vlf-end-pos (min vlf-batch-size vlf-file-size)))
(vlf-move-to-chunk vlf-start-pos vlf-end-pos))
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-beginning-of-file ()
2013-04-13 22:49:08 +03:00
"Jump to beginning of file content."
(interactive)
2013-08-25 17:57:43 +03:00
(vlf-insert-file))
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-end-of-file ()
2013-04-13 22:49:08 +03:00
"Jump to end of file content."
(interactive)
2013-08-25 17:57:43 +03:00
(vlf-insert-file t))
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-revert (&optional _ignore-auto noconfirm)
"Revert current chunk. Ignore _IGNORE-AUTO.
Ask for confirmation if NOCONFIRM is nil."
(if (or noconfirm
(yes-or-no-p (format "Revert buffer from file %s? "
buffer-file-name)))
2013-08-25 17:57:43 +03:00
(vlf-move-to-chunk vlf-start-pos vlf-end-pos)))
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-jump-to-chunk (n)
2013-04-13 22:57:09 +03:00
"Go to to chunk N."
(interactive "nGoto to chunk: ")
2013-08-25 17:57:43 +03:00
(vlf-move-to-batch (* (1- n) vlf-batch-size)))
2013-04-13 22:57:09 +03:00
2013-04-13 22:49:08 +03:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; batch movement
2013-08-25 17:57:43 +03:00
(defun vlf-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-08-25 17:57:43 +03:00
(vlf-verify-size)
(let ((end (min (+ vlf-end-pos (* vlf-batch-size
2013-04-15 01:26:58 +03:00
(abs append)))
2013-08-25 17:57:43 +03:00
vlf-file-size)))
(let ((inhibit-read-only t)
2013-04-13 22:36:33 +03:00
(do-append (< append 0))
(pos (position-bytes (point))))
(if do-append
2013-04-13 22:36:33 +03:00
(goto-char (point-max))
2013-08-25 17:57:43 +03:00
(setq vlf-start-pos (- end vlf-batch-size))
2013-04-13 22:36:33 +03:00
(erase-buffer))
(insert-file-contents buffer-file-name nil (if do-append
2013-08-25 17:57:43 +03:00
vlf-end-pos
vlf-start-pos)
2013-04-13 22:36:33 +03:00
end)
2013-08-25 17:57:43 +03:00
(setq vlf-end-pos end)
(goto-char (or (byte-to-position (+ pos (vlf-adjust-chunk)))
(point-max)))))
(set-visited-file-modtime)
(set-buffer-modified-p nil)
2013-08-25 17:57:43 +03:00
(vlf-update-buffer-name))
2013-08-25 17:57:43 +03:00
(defun vlf-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-08-25 17:57:43 +03:00
(if (zerop vlf-start-pos)
(error "Already at BOF"))
2013-08-25 17:57:43 +03:00
(vlf-verify-size)
(let ((inhibit-read-only t)
2013-08-25 17:57:43 +03:00
(start (max 0 (- vlf-start-pos (* vlf-batch-size
2013-04-13 22:36:33 +03:00
(abs prepend)))))
(do-prepend (< prepend 0))
(pos (- (position-bytes (point-max))
(position-bytes (point)))))
(if do-prepend
2013-04-13 22:36:33 +03:00
(goto-char (point-min))
2013-08-25 17:57:43 +03:00
(setq vlf-end-pos (min (+ start vlf-batch-size)
vlf-file-size))
(erase-buffer))
(insert-file-contents buffer-file-name nil start
2013-04-13 22:36:33 +03:00
(if do-prepend
2013-08-25 17:57:43 +03:00
vlf-start-pos
vlf-end-pos))
(setq vlf-start-pos start
pos (+ pos (vlf-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-08-25 17:57:43 +03:00
(vlf-update-buffer-name))
2013-08-25 17:57:43 +03:00
(defun vlf-move-to-batch (start &optional minimal)
"Move to batch determined by START.
2013-08-25 17:57:43 +03:00
Adjust according to file start/end and show `vlf-batch-size' bytes.
When given MINIMAL flag, skip non important operations."
2013-08-25 17:57:43 +03:00
(vlf-verify-size)
(setq vlf-start-pos (max 0 start)
vlf-end-pos (min (+ vlf-start-pos vlf-batch-size)
vlf-file-size))
(if (= vlf-file-size vlf-end-pos) ; re-check file size
(setq vlf-start-pos (max 0 (- vlf-end-pos vlf-batch-size))))
(let ((inhibit-read-only t)
(pos (position-bytes (point))))
(erase-buffer)
(insert-file-contents buffer-file-name nil
2013-08-25 17:57:43 +03:00
vlf-start-pos vlf-end-pos)
(goto-char (or (byte-to-position (+ pos (vlf-adjust-chunk)))
(point-max))))
(set-buffer-modified-p nil)
2013-04-15 01:26:58 +03:00
(set-visited-file-modtime)
2013-08-25 17:57:43 +03:00
(or minimal(vlf-update-buffer-name)))
2013-08-25 17:57:43 +03:00
(defun vlf-move-to-chunk (start end &optional minimal)
"Move to chunk determined by START END.
When given MINIMAL flag, skip non important operations."
2013-08-25 17:57:43 +03:00
(vlf-verify-size)
(setq vlf-start-pos (max 0 start)
vlf-end-pos (min end vlf-file-size))
(let ((inhibit-read-only t)
(pos (position-bytes (point))))
2013-03-31 02:36:54 +02:00
(erase-buffer)
(insert-file-contents buffer-file-name nil
2013-08-25 17:57:43 +03:00
vlf-start-pos vlf-end-pos)
(goto-char (or (byte-to-position (+ pos (vlf-adjust-chunk)))
(point-max))))
2013-03-31 02:36:54 +02:00
(set-buffer-modified-p nil)
2013-04-15 01:26:58 +03:00
(set-visited-file-modtime)
2013-08-25 17:57:43 +03:00
(or minimal (vlf-update-buffer-name)))
2013-03-31 02:36:54 +02:00
2013-08-25 17:57:43 +03:00
(defun vlf-adjust-chunk ()
"Adjust chunk beginning until content can be properly decoded.
2013-08-25 17:57:43 +03:00
Set `vlf-encode-size' to size of buffer when encoded.
Return number of bytes moved back for this to happen."
(let ((shift 0)
2013-08-25 17:57:43 +03:00
(chunk-size (- vlf-end-pos vlf-start-pos)))
(while (and (< shift 4)
(< 4 (abs (- chunk-size
2013-08-25 17:57:43 +03:00
(setq vlf-encode-size
(length (encode-coding-region
(point-min) (point-max)
buffer-file-coding-system
t))))))
2013-08-25 17:57:43 +03:00
(not (zerop vlf-start-pos)))
(setq shift (1+ shift)
2013-08-25 17:57:43 +03:00
vlf-start-pos (1- vlf-start-pos)
chunk-size (1+ chunk-size))
(let ((inhibit-read-only t))
(erase-buffer)
(insert-file-contents buffer-file-name nil
2013-08-25 17:57:43 +03:00
vlf-start-pos vlf-end-pos)))
(set-buffer-modified-p nil)
shift))
2013-04-13 22:49:08 +03:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; search
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-re-search (regexp count backward batch-step)
2013-05-01 02:18:37 +03:00
"Search for REGEXP COUNT number of times forward or BACKWARD.
BATCH-STEP is amount of overlap between successive chunks."
(assert (< 0 count))
2013-08-25 17:57:43 +03:00
(let* ((match-chunk-start vlf-start-pos)
(match-chunk-end vlf-end-pos)
(match-start-pos (+ vlf-start-pos (position-bytes (point))))
2013-04-13 22:36:33 +03:00
(match-end-pos match-start-pos)
(to-find count)
(reporter (make-progress-reporter
(concat "Searching for " regexp "...")
(if backward
2013-08-25 17:57:43 +03:00
(- vlf-file-size vlf-end-pos)
vlf-start-pos)
vlf-file-size)))
(unwind-protect
2013-04-13 22:36:33 +03: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)
2013-08-25 17:57:43 +03:00
match-chunk-start vlf-start-pos
match-chunk-end vlf-end-pos
match-start-pos (+ vlf-start-pos
(position-bytes
(match-beginning 0)))
2013-08-25 17:57:43 +03:00
match-end-pos (+ vlf-start-pos
(position-bytes
(match-end 0)))))
2013-08-25 17:57:43 +03:00
((zerop vlf-start-pos)
2013-04-13 22:36:33 +03:00
(throw 'end-of-file nil))
2013-08-25 17:57:43 +03:00
(t (let ((batch-move (- vlf-start-pos
(- vlf-batch-size
2013-04-13 22:36:33 +03:00
batch-step))))
2013-08-25 17:57:43 +03:00
(vlf-move-to-batch
2013-04-13 22:36:33 +03:00
(if (< match-start-pos batch-move)
2013-08-25 17:57:43 +03:00
(- match-start-pos vlf-batch-size)
batch-move) t))
2013-04-13 22:36:33 +03:00
(goto-char (if (< match-start-pos
2013-08-25 17:57:43 +03:00
vlf-end-pos)
(or (byte-to-position
(- match-start-pos
2013-08-25 17:57:43 +03:00
vlf-start-pos))
(point-max))
2013-04-13 22:36:33 +03:00
(point-max)))
2013-04-15 17:27:23 +03:00
(progress-reporter-update
2013-08-25 17:57:43 +03:00
reporter (- vlf-file-size
vlf-start-pos)))))
2013-04-13 22:36:33 +03:00
(while (not (zerop to-find))
(cond ((re-search-forward regexp nil t)
(setq to-find (1- to-find)
2013-08-25 17:57:43 +03:00
match-chunk-start vlf-start-pos
match-chunk-end vlf-end-pos
match-start-pos (+ vlf-start-pos
(position-bytes
(match-beginning 0)))
2013-08-25 17:57:43 +03:00
match-end-pos (+ vlf-start-pos
(position-bytes
(match-end 0)))))
2013-08-25 17:57:43 +03:00
((= vlf-end-pos vlf-file-size)
2013-04-13 22:36:33 +03:00
(throw 'end-of-file nil))
2013-08-25 17:57:43 +03:00
(t (let ((batch-move (- vlf-end-pos batch-step)))
(vlf-move-to-batch
2013-04-13 22:36:33 +03:00
(if (< batch-move match-end-pos)
match-end-pos
batch-move) t))
2013-08-25 17:57:43 +03:00
(goto-char (if (< vlf-start-pos match-end-pos)
(or (byte-to-position
(- match-end-pos
2013-08-25 17:57:43 +03:00
vlf-start-pos))
(point-min))
2013-04-13 22:36:33 +03:00
(point-min)))
(progress-reporter-update reporter
2013-08-25 17:57:43 +03:00
vlf-end-pos)))))
(progress-reporter-done reporter))
2013-04-01 12:57:58 +03:00
(if backward
2013-08-25 17:57:43 +03:00
(vlf-goto-match match-chunk-start match-chunk-end
match-end-pos match-start-pos
2013-04-13 22:36:33 +03:00
count to-find)
2013-08-25 17:57:43 +03:00
(vlf-goto-match match-chunk-start match-chunk-end
match-start-pos match-end-pos
2013-04-13 22:36:33 +03:00
count to-find)))))
2013-04-01 12:57:58 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-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 14:38:50 +03:00
According to COUNT and left TO-FIND, show if search has been
successful. Return nil if nothing found."
(if (= count to-find)
2013-08-25 17:57:43 +03:00
(progn (vlf-move-to-chunk match-chunk-start match-chunk-end)
(goto-char (or (byte-to-position (- match-pos-start
2013-08-25 17:57:43 +03:00
vlf-start-pos))
(point-max)))
(message "Not found")
nil)
(let ((success (zerop to-find)))
(if success
2013-08-25 17:57:43 +03:00
(vlf-update-buffer-name)
(vlf-move-to-chunk match-chunk-start match-chunk-end))
(let* ((match-end (or (byte-to-position (- match-pos-end
2013-08-25 17:57:43 +03:00
vlf-start-pos))
(point-max)))
(overlay (make-overlay (byte-to-position
(- match-pos-start
2013-08-25 17:57:43 +03:00
vlf-start-pos))
match-end)))
2013-04-16 12:25:06 +03: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 22:36:33 +03:00
(sit-for 0.1)
(delete-overlay overlay)
t))))
2013-08-25 17:57:43 +03:00
(defun vlf-re-search-forward (regexp count)
2013-04-10 02:10:00 +03:00
"Search forward for REGEXP prefix COUNT number of times.
2013-08-25 17:57:43 +03:00
Search is performed chunk by chunk in `vlf-batch-size' memory."
2013-03-31 20:44:41 +03:00
(interactive (list (read-regexp "Search whole file"
2013-04-13 22:36:33 +03:00
(if regexp-history
2013-04-15 17:27:23 +03:00
(car regexp-history)))
2013-04-13 22:36:33 +03:00
(or current-prefix-arg 1)))
2013-08-25 17:57:43 +03:00
(vlf-re-search regexp count nil (/ vlf-batch-size 8)))
2013-03-29 18:12:20 +02:00
2013-08-25 17:57:43 +03:00
(defun vlf-re-search-backward (regexp count)
2013-04-10 02:10:00 +03:00
"Search backward for REGEXP prefix COUNT number of times.
2013-08-25 17:57:43 +03:00
Search is performed chunk by chunk in `vlf-batch-size' memory."
2013-04-01 12:06:48 +03:00
(interactive (list (read-regexp "Search whole file backward"
2013-04-13 22:36:33 +03:00
(if regexp-history
2013-04-15 17:27:23 +03:00
(car regexp-history)))
2013-04-13 22:36:33 +03:00
(or current-prefix-arg 1)))
2013-08-25 17:57:43 +03:00
(vlf-re-search regexp count t (/ vlf-batch-size 8)))
2013-03-29 18:12:20 +02:00
2013-08-25 17:57:43 +03:00
(defun vlf-goto-line (n)
"Go to line N. If N is negative, count from the end of file."
2013-04-14 02:18:06 +03:00
(interactive "nGo to line: ")
2013-08-25 17:57:43 +03:00
(let ((start-pos vlf-start-pos)
(end-pos vlf-end-pos)
2013-04-14 02:18:06 +03:00
(pos (point))
(success nil))
(unwind-protect
(if (< 0 n)
2013-08-25 17:57:43 +03:00
(progn (vlf-beginning-of-file)
(goto-char (point-min))
2013-08-25 17:57:43 +03:00
(setq success (vlf-re-search "[\n\C-m]" (1- n)
nil 0)))
2013-08-25 17:57:43 +03:00
(vlf-end-of-file)
(goto-char (point-max))
2013-08-25 17:57:43 +03:00
(setq success (vlf-re-search "[\n\C-m]" (- n) t 0)))
(if success
(message "Onto line %s" n)
2013-08-25 17:57:43 +03:00
(vlf-move-to-chunk start-pos end-pos)
2013-04-14 02:18:06 +03:00
(goto-char pos)))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; occur
2013-08-25 17:57:43 +03:00
(defvar vlf-occur-mode-map
2013-04-16 12:25:06 +03:00
(let ((map (make-sparse-keymap)))
2013-08-25 17:57:43 +03:00
(define-key map "n" 'vlf-occur-next-match)
(define-key map "p" 'vlf-occur-prev-match)
(define-key map "\C-m" 'vlf-occur-visit)
(define-key map [mouse-1] 'vlf-occur-visit)
(define-key map "o" 'vlf-occur-show)
2013-04-16 12:25:06 +03:00
map)
2013-08-25 17:57:43 +03:00
"Keymap for command `vlf-occur-mode'.")
2013-04-16 12:25:06 +03:00
2013-08-25 17:57:43 +03:00
(define-derived-mode vlf-occur-mode special-mode "VLF[occur]"
"Major mode for showing occur matches of VLF opened files.")
2013-04-16 12:25:06 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-occur-next-match ()
2013-04-16 12:25:06 +03:00
"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))))
2013-08-25 17:57:43 +03:00
(defun vlf-occur-prev-match ()
2013-04-16 12:25:06 +03:00
"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)))))
2013-08-25 17:57:43 +03:00
(defun vlf-occur-show (&optional event)
"Visit current `vlf-occur' link in a vlf buffer but stay in the \
occur buffer. If original VLF buffer has been killed,
open new VLF session each time.
EVENT may hold details of the invocation."
(interactive (list last-nonmenu-event))
(let ((occur-buffer (if event
(window-buffer (posn-window
(event-end event)))
(current-buffer))))
2013-08-25 17:57:43 +03:00
(vlf-occur-visit event)
(pop-to-buffer occur-buffer)))
2013-08-25 17:57:43 +03:00
(defun vlf-occur-visit (&optional event)
"Visit current `vlf-occur' link in a vlf buffer.
If original VLF buffer has been killed,
open new VLF session each time.
EVENT may hold details of the invocation."
(interactive (list last-nonmenu-event))
(when event
(set-buffer (window-buffer (posn-window (event-end event))))
(goto-char (posn-point (event-end event))))
(let* ((pos (point))
2013-04-16 15:38:36 +03:00
(pos-relative (- pos (line-beginning-position) 1))
(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 (+ (get-char-property pos 'line-pos)
pos-relative)))
(or (buffer-live-p buffer)
(let ((occur-buffer (current-buffer)))
2013-08-25 17:57:43 +03:00
(setq buffer (vlf file))
(switch-to-buffer occur-buffer)))
(pop-to-buffer buffer)
(if (buffer-modified-p)
2013-08-25 17:57:43 +03:00
(cond ((and (= vlf-start-pos chunk-start)
(= vlf-end-pos chunk-end))
(goto-char match-pos))
2013-08-25 17:57:43 +03:00
((y-or-n-p "VLF buffer has been modified. \
Really jump to new chunk? ")
2013-08-25 17:57:43 +03:00
(vlf-move-to-chunk chunk-start chunk-end)
(goto-char match-pos)))
2013-08-25 17:57:43 +03:00
(vlf-move-to-chunk chunk-start chunk-end)
(goto-char match-pos))))))
2013-08-25 17:57:43 +03:00
(defun vlf-occur (regexp)
"Make whole file occur style index for REGEXP.
Prematurely ending indexing will still show what's found so far."
(interactive (list (read-regexp "List lines matching regexp"
(if regexp-history
(car regexp-history)))))
2013-08-25 17:57:43 +03:00
(let ((start-pos vlf-start-pos)
(end-pos vlf-end-pos)
(pos (point)))
2013-08-25 17:57:43 +03:00
(vlf-beginning-of-file)
2013-04-16 12:25:06 +03:00
(goto-char (point-min))
2013-08-25 17:57:43 +03:00
(unwind-protect (vlf-build-occur regexp)
(vlf-move-to-chunk start-pos end-pos)
(goto-char pos))))
2013-08-25 17:57:43 +03:00
(defun vlf-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)
2013-04-16 15:51:30 +03:00
(total-matches 0)
2013-08-25 17:57:43 +03:00
(match-end-pos (+ vlf-start-pos (position-bytes (point))))
(occur-buffer (generate-new-buffer
2013-08-25 17:57:43 +03:00
(concat "*VLF-occur " (file-name-nondirectory
buffer-file-name)
"*")))
(line-regexp (concat "\\(?5:[\n\C-m]\\)\\|\\(?10:"
regexp "\\)"))
2013-08-25 17:57:43 +03:00
(batch-step (/ vlf-batch-size 8))
(end-of-file nil)
(reporter (make-progress-reporter
(concat "Building index for " regexp "...")
2013-08-25 17:57:43 +03:00
vlf-start-pos vlf-file-size)))
(unwind-protect
(progn
(while (not end-of-file)
(if (re-search-forward line-regexp nil t)
(progn
2013-08-25 17:57:43 +03:00
(setq match-end-pos (+ vlf-start-pos
(position-bytes
(match-end 0))))
(if (match-string 5)
(setq line (1+ line) ; line detected
last-line-pos (point))
2013-08-25 17:57:43 +03:00
(let* ((chunk-start vlf-start-pos)
(chunk-end vlf-end-pos)
(vlf-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) ;new match line
(insert "\n:") ; insert line number
2013-04-16 15:38:36 +03:00
(let* ((overlay-pos (1- (point)))
(overlay (make-overlay
overlay-pos
(1+ overlay-pos))))
(overlay-put overlay 'before-string
(propertize
(number-to-string line)
'face 'shadow)))
(insert (propertize line-text ; insert line
2013-04-16 15:38:36 +03:00
'file file
2013-08-25 17:57:43 +03:00
'buffer vlf-buffer
2013-04-16 15:38:36 +03:00
'chunk-start chunk-start
'chunk-end chunk-end
'mouse-face '(highlight)
2013-04-16 15:51:30 +03:00
'line-pos line-pos
'help-echo
(format "Move to line %d"
line))))
(setq last-match-line line
total-matches (1+ total-matches))
(let ((line-start (1+
(line-beginning-position)))
(match-pos (match-beginning 10)))
(add-text-properties ; mark match
(+ line-start match-pos (- last-line-pos))
(+ line-start (match-end 10)
(- last-line-pos))
(list 'face 'match
2013-04-16 15:51:30 +03:00
'help-echo
(format "Move to match %d"
total-matches))))))))
2013-08-25 17:57:43 +03:00
(setq end-of-file (= vlf-end-pos vlf-file-size))
(unless end-of-file
2013-08-25 17:57:43 +03:00
(let ((batch-move (- vlf-end-pos batch-step)))
(vlf-move-to-batch (if (< batch-move match-end-pos)
match-end-pos
batch-move) t))
2013-08-25 17:57:43 +03:00
(goto-char (if (< vlf-start-pos match-end-pos)
(or (byte-to-position (- match-end-pos
2013-08-25 17:57:43 +03:00
vlf-start-pos))
(point-min))
(point-min)))
(setq last-match-line 0
last-line-pos (line-beginning-position))
2013-08-25 17:57:43 +03:00
(progress-reporter-update reporter vlf-end-pos))))
(progress-reporter-done reporter))
2013-04-16 15:51:30 +03:00
(if (zerop total-matches)
(progn (with-current-buffer occur-buffer
(set-buffer-modified-p nil))
(kill-buffer occur-buffer)
(message "No matches for \"%s\"" regexp))
(with-current-buffer occur-buffer
(goto-char (point-min))
(insert (propertize
2013-04-16 15:51:30 +03:00
(format "%d matches from %d lines for \"%s\" \
in file: %s" total-matches line regexp file)
'face 'underline))
(set-buffer-modified-p nil)
(forward-char 2)
2013-08-25 17:57:43 +03:00
(vlf-occur-mode))
2013-04-16 15:51:30 +03:00
(display-buffer occur-buffer)))))
2013-04-13 22:49:08 +03:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
2013-04-02 02:55:53 +03:00
;;; editing
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defvar vlf-edit-mode-map
2013-04-02 02:55:53 +03:00
(let ((map (make-sparse-keymap)))
(set-keymap-parent map text-mode-map)
2013-08-25 17:57:43 +03:00
(define-key map "\C-c\C-c" 'vlf-write)
(define-key map "\C-c\C-q" 'vlf-discard-edit)
(define-key map "\C-v" vlf-mode-map)
2013-04-02 02:55:53 +03:00
map)
2013-08-25 17:57:43 +03:00
"Keymap for command `vlf-edit-mode'.")
2013-04-08 00:56:42 +03:00
2013-08-25 17:57:43 +03:00
(define-derived-mode vlf-edit-mode vlf-mode "VLF[edit]"
2013-04-08 00:56:42 +03:00
"Major mode for editing large file chunks."
(setq buffer-read-only nil)
(buffer-enable-undo)
(message (substitute-command-keys
2013-08-25 17:57:43 +03:00
"Editing: Type \\[vlf-write] to write chunk \
or \\[vlf-discard-edit] to discard changes.")))
2013-04-02 02:55:53 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-discard-edit ()
2013-04-13 22:49:08 +03:00
"Discard edit and refresh chunk from file."
(interactive)
(set-buffer-modified-p nil)
2013-08-25 17:57:43 +03:00
(vlf-move-to-chunk vlf-start-pos vlf-end-pos)
(vlf-mode)
(message "Switched to VLF mode."))
2013-04-13 22:49:08 +03:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; saving
2013-08-25 17:57:43 +03:00
(defun vlf-write ()
2013-04-13 22:49:08 +03:00
"Write current chunk to file. Always return true to disable save.
If changing size of chunk, shift remaining file content."
2013-04-13 22:49:08 +03:00
(interactive)
(when (and (buffer-modified-p)
2013-04-15 17:27:23 +03:00
(or (verify-visited-file-modtime (current-buffer))
2013-04-13 22:49:08 +03:00
(y-or-n-p "File has changed since visited or saved. \
Save anyway? ")))
2013-04-14 00:53:24 +03:00
(let ((pos (point))
2013-08-25 17:57:43 +03:00
(size-change (- vlf-encode-size
(setq vlf-encode-size
(length (encode-coding-region
(point-min) (point-max)
buffer-file-coding-system
t))))))
2013-04-13 22:49:08 +03:00
(cond ((zerop size-change)
2013-08-25 17:57:43 +03:00
(write-region nil nil buffer-file-name vlf-start-pos t))
2013-04-13 22:49:08 +03:00
((< 0 size-change)
2013-08-25 17:57:43 +03:00
(vlf-file-shift-back size-change))
(t (vlf-file-shift-forward (- size-change))))
(vlf-move-to-chunk vlf-start-pos vlf-end-pos)
2013-04-13 22:49:08 +03:00
(goto-char pos))
2013-08-25 17:57:43 +03:00
(vlf-mode))
t)
2013-04-13 22:49:08 +03:00
2013-08-25 17:57:43 +03:00
(defun vlf-file-shift-back (size-change)
"Shift file contents SIZE-CHANGE bytes back."
2013-08-25 17:57:43 +03:00
(write-region nil nil buffer-file-name vlf-start-pos t)
(buffer-disable-undo)
2013-08-25 17:57:43 +03:00
(let ((read-start-pos vlf-end-pos)
(coding-system-for-write 'no-conversion)
2013-04-15 17:27:23 +03:00
(reporter (make-progress-reporter "Adjusting file content..."
2013-08-25 17:57:43 +03:00
vlf-end-pos
vlf-file-size)))
(while (vlf-shift-batch read-start-pos (- read-start-pos
size-change))
2013-08-25 17:57:43 +03:00
(setq read-start-pos (+ read-start-pos vlf-batch-size))
(progress-reporter-update reporter read-start-pos))
;; pad end with space
(erase-buffer)
2013-08-25 17:57:43 +03:00
(vlf-verify-size)
(insert-char 32 size-change)
2013-08-25 17:57:43 +03:00
(write-region nil nil buffer-file-name (- vlf-file-size
size-change) t)
(progress-reporter-done reporter)))
2013-08-25 17:57:43 +03:00
(defun vlf-shift-batch (read-pos write-pos)
"Read `vlf-batch-size' bytes from READ-POS and write them \
back at WRITE-POS. Return nil if EOF is reached, t otherwise."
(erase-buffer)
2013-08-25 17:57:43 +03:00
(vlf-verify-size)
(let ((read-end (+ read-pos vlf-batch-size)))
(insert-file-contents-literally buffer-file-name nil
read-pos
2013-08-25 17:57:43 +03:00
(min vlf-file-size read-end))
(write-region nil nil buffer-file-name write-pos 0)
2013-08-25 17:57:43 +03:00
(< read-end vlf-file-size)))
2013-08-25 17:57:43 +03:00
(defun vlf-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)
2013-08-25 17:57:43 +03:00
(let ((size (+ vlf-batch-size size-change))
(read-pos vlf-end-pos)
(write-pos vlf-start-pos)
(reporter (make-progress-reporter "Adjusting file content..."
2013-08-25 17:57:43 +03:00
vlf-start-pos
vlf-file-size)))
(when (vlf-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))
2013-08-25 17:57:43 +03:00
(while (vlf-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)))
2013-08-25 17:57:43 +03:00
(defun vlf-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."
2013-08-25 17:57:43 +03:00
(vlf-verify-size)
(let ((read-more (< read-pos vlf-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
2013-08-25 17:57:43 +03:00
(min vlf-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 17:44:32 +03:00
2013-08-25 17:57:43 +03:00
(provide 'vlf)
2013-08-25 17:57:43 +03:00
;;; vlf.el ends here