unofficial mirror of emacs-devel@gnu.org 
 help / color / mirror / code / Atom feed
From: dalanicolai <dalanicolai@gmail.com>
To: Eli Zaretskii <eliz@gnu.org>
Cc: emacs-devel@gnu.org
Subject: Re: Two questions about overlays
Date: Tue, 21 Feb 2023 15:52:06 +0100	[thread overview]
Message-ID: <CACJP=3mo3DSG1x=bvCOfQjG5t5toAE1UbW132GViiOCn4c=5JQ@mail.gmail.com> (raw)
In-Reply-To: <CACJP=3mygFJq_+NP5bDV4dLDYTiBiP45F85iq_zYCYECyGsWyg@mail.gmail.com>


[-- Attachment #1.1: Type: text/plain, Size: 5965 bytes --]

Okay, finally, (I should have done this immediately), here is the best
and most brief explanation: So load the code of the attached UPDATED
scrap-mode.el. Then do `M-x scrap-expand-overlays`. It will print 1000, but
I expect it to print the number of overlays you are actually seeing on
screen, because the function calls

(print (length (overlays-in (window-start) (window-end)))))

after adding the 'space' display properties/expanding the overlays. So
although the display properties have been added, they seem to not have
taken effect at the time

(print (length (overlays-in (window-start) (window-end)))))

is evaluated. If you uncomment the `sit-for` in
`scrap-expand-overlays`, then you will see it prints a different
(i.e. the correct/expected) number.

Another way to get the expected number, is to now do

M-: (length (overlays-in (window-start) (window-end)))

On Tue, 21 Feb 2023 at 14:57, dalanicolai <dalanicolai@gmail.com> wrote:

> B.t.w. I am using Emacs 29 (and 30), and the overlay-recenter is still
> there.
> Maybe it would be handy to remove it then?
>
> On Tue, 21 Feb 2023 at 14:50, dalanicolai <dalanicolai@gmail.com> wrote:
>
>> Let me, additionally, very briefly explain what the code is doing, and
>> what I am expecting (as an attempt to make it immediately clear what I
>> mean).
>>
>> So the code is first simply creating (for my dir) 108 overlays.  Then
>> in 'scrap-dir-images' are made to display the images from the
>> directory. Immediately, after the
>> '(seq-do-indexed (lambda (im n) (overlay-put ... 'display ...)'
>> the function print the value of '(overlays-in (window-start)
>> (window-end))',
>> which return 108. However, I expect it to print 2, as the overlays should
>> have the size of the images now (and I see only two images simultaneously
>> on
>> screen). Indeed when doing 'M-: (overlays-in (window-start) (window-end))'
>> immediately after creating the 'image gallery', it returns 2.
>>
>> On Tue, 21 Feb 2023 at 14:39, dalanicolai <dalanicolai@gmail.com> wrote:
>>
>>> In this example code, I am simply displaying all images at once,
>>> because I am assuming that the images in the directory are not too
>>> many and that they are small (which is not a very reasonable
>>> assumption, but this is just my personal 'test' function).
>>>
>>> Indeed, the example does not call sit-for, because it should show that
>>> the printed number of overlays, is the number of all images in the
>>> directory (instead of only the number of images currently on the
>>> screen, i.e. within '(overlays-in (window-start) (window-end))' ).
>>>
>>> For example, I have a directory with 108 images, when I run 'M-x
>>> scrap-dir-images' it print 108, although it should print the number of
>>> images on screen i.e. by '(overlays-in (window-start) (window-end))',
>>> which value is what 'scrap-dir-images' prints.  However, the images
>>> are normal 'foto' size, so that I only see two images on screen, and
>>> indeed doing 'M-: (overlays-in (window-start) (window-end))' manually
>>> now prints 2.  If I had added the '(sit-for)' in the example
>>> code, then I would have 'fixed' the problem, and 'scrap-dir-images'
>>> would have printed 2 immediately because the display property
>>> ('expansion of the overlays') got enough time to take effect.
>>>
>>> I hope this clears things up, but of course I would be happy to try
>>> another explanation (e.g. sending by adding an animated gif).
>>>
>>> But this function is a no-op in Emacs 29 and later, since the overlays
>>>> were reimplemented in a way that makes it unnecessary to "center" the
>>>> list of overlays.  So you can forget about that and ignore this
>>>> function.
>>>>
>>>
>>> Thanks, that is indeed handy to know.
>>>
>>> On Tue, 21 Feb 2023 at 14:21, Eli Zaretskii <eliz@gnu.org> wrote:
>>>
>>>> > From: dalanicolai <dalanicolai@gmail.com>
>>>> > Date: Tue, 21 Feb 2023 13:46:56 +0100
>>>> >
>>>> > So now here are the questions:
>>>> >
>>>> > - when displaying multiple pages in columns, I would like to use the
>>>> > overlays-in function to determine which overlays should display
>>>> > images.  So I am creating a full 'book roll' by giving the overlays a
>>>> > size via the 'space' display property, after which I use overlays-in
>>>> > to determine which overlays are actually visible. However, after it
>>>> > takes some time for the 'space' display property to take effect, so I
>>>> > am manuall adding a 'sit-for' with some reasonable delay
>>>> > time. However, I would like to ask if there is someone has an idea for
>>>> > a 'better' mechanism to wait until/detect if the 'overlay expansion'
>>>> > has finished.
>>>> >
>>>> > If the explanation is not clear then please load the following file
>>>> > and do 'M-x scrap-dir-images' on a directory that contains enough
>>>> > images to not fit all on a single screen. It will print the number of
>>>> > overlays found via 'overlays-in' directly after 'displaying the
>>>> > images' (here by assigning the image as display property instead of
>>>> > space). You will find it prints all overlays in the buffer (instead of
>>>> > only the ones on screen). To find what I expect it to print now
>>>> > (again) do 'M-: (overlays-in (window-start) (window-end))'.
>>>>
>>>> I did all that, and I still don't understand the question.  In
>>>> particular, your code doesn't call sit-for, so I'm unsure what exactly
>>>> is the problem you are asking about here.
>>>>
>>>> > My second question is about the function 'overlay-recenter' I don't
>>>> > really understand its docstring. What kind of 'overlay lookup' would
>>>> > go faster? What is 'overlay-lookup' anyway?
>>>>
>>>> Looking up overlays that are relevant to a particular buffer position.
>>>>
>>>> But this function is a no-op in Emacs 29 and later, since the overlays
>>>> were reimplemented in a way that makes it unnecessary to "center" the
>>>> list of overlays.  So you can forget about that and ignore this
>>>> function.
>>>>
>>>

[-- Attachment #1.2: Type: text/html, Size: 7783 bytes --]

[-- Attachment #2: scrap-mode.el --]
[-- Type: text/x-emacs-lisp, Size: 4337 bytes --]

;;; image-scroll-mode.el --- Image and document display engine  -*- lexical-binding: t; -*-

;; Copyright (C) 2023  Daniel Nicolai

;; Author: Daniel Nicolai <dalanicolai@gmail.com>
;; Keywords: tools

;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.

;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
;; GNU General Public License for more details.

;; You should have received a copy of the GNU General Public License
;; along with this program.  If not, see <https://www.gnu.org/licenses/>.

;;; Commentary:

;;

;;; Code:
(defvar scrap-incompatible-modes '(visual-line-mode
                                   global-hl-line-mode))

(defvar-local scrap-overlays nil)
(defvar-local scrap-columns nil)
(defvar-local scrap-next-function #'scrap-next)

(defsubst scrap-current-overlay ()
  (car (overlays-at (point))))

(defsubst scrap-get-overlay-n ()
  (overlay-get (scrap-current-overlay) 'n))

(defun scrap-dir-images (dir)
  (interactive
   (list (read-file-name "Select document: " nil nil t nil #'file-directory-p)))
  (pop-to-buffer (get-buffer-create dir))
  (setq buffer-file-name dir)
  (let ((images (seq-filter #'image-supported-file-p (directory-files dir)))
	(max-w (window-text-width nil t))
	;; (max-h 200)
	)
    (scrap-create-overlays (length images) 1)
    (seq-do-indexed (lambda (im n)
		      (overlay-put (nth n scrap-overlays)
				   'display (create-image (concat (file-name-as-directory dir) im) nil nil
							  :max-width max-w)))
		    images))
  (print (length (overlays-in (window-start) (window-end)))))

(defun scrap-expand-overlays ()
  (interactive)
  (let ((number-of-overlays 1000))
    (pop-to-buffer "*scrap-example*")
    (scrap-create-overlays number-of-overlays 1)
    (dolist (o scrap-overlays)
      (overlay-put o 'display '(space . (:width (100) :height (140))))
      (overlay-put o 'face '(:background "gray"))))
  ;; (sit-for 0.05)
  (print (length (overlays-in (window-start) (window-end)))))

(defun scrap-create-overlays (number
                              &optional columns hspace vspace
                              &rest overlay-props)
  (dolist (m scrap-incompatible-modes)
    (funcall m -1))
  (toggle-truncate-lines 1) ; also disables visual-mode
  (setq scrap-columns columns)
  (let (overlays)
    (dotimes (i number)
      (let* ((n (1+ i))
             (o (make-overlay
                 (point)
                 (progn (insert " ") (point))
                 (unless (= (% columns 3) 0)
		   (insert (make-string (or hspace 1) (string-to-char " ")))))))
        (when (and (= (% n (or columns 1)) 0)
                   (not (= n number)))
          (insert "\n")
          (insert (make-string (or vspace 1) (string-to-char "\n"))))
        (overlay-put o 'n n)
        (dotimes (j (/ (length overlay-props) 2))
          (let ((m (* j 2)))
            (overlay-put o (nth m overlay-props) (nth (+ m 1) overlay-props))))
        (push o overlays)))
    (goto-char (point-min))
    (setq scrap-overlays (nreverse overlays))
    (scrap-mode)))

(defun scrap-next (n &optional previous)
  (let* ((current (1- (scrap-get-overlay-n)))
         (next (nth (+ current (if previous (- n) n)) scrap-overlays)))
    (goto-char (overlay-start next))
    (overlay-recenter (point))))

(defun scrap-scroll-next (n)
  (interactive "p")
  (funcall scrap-next-function n))

(defun scrap-scroll-previous (n)
  (interactive "p")
  (funcall scrap-next-function n t))

(defun scrap-scroll-next-line (n)
  (interactive "p")
  (funcall scrap-next-function (* n scrap-columns)))

(defun scrap-scroll-previous-line (n)
  (interactive "p")
  (funcall scrap-next-function (* n scrap-columns) t))

(define-minor-mode scrap-mode "Display images in a grid."
  :lighter "Scrap"
  :keymap `((,(kbd "C-n") . scrap-scroll-next)
            (,(kbd "C-p") . scrap-scroll-previous)
            (,(kbd "C-f") . scrap-scroll-next-line)
            (,(kbd "C-b") . scrap-scroll-previous-line)))

(provide 'image-scroll-mode)
;;; image-scroll-mode.el ends here

  parent reply	other threads:[~2023-02-21 14:52 UTC|newest]

Thread overview: 9+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2023-02-21 12:46 Two questions about overlays dalanicolai
2023-02-21 13:21 ` Eli Zaretskii
2023-02-21 13:39   ` dalanicolai
2023-02-21 13:50     ` dalanicolai
2023-02-21 13:57       ` dalanicolai
2023-02-21 14:16         ` dalanicolai
2023-02-21 14:52         ` dalanicolai [this message]
2023-02-21 15:03     ` Eli Zaretskii
2023-02-21 15:17       ` dalanicolai

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

  List information: https://www.gnu.org/software/emacs/

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to='CACJP=3mo3DSG1x=bvCOfQjG5t5toAE1UbW132GViiOCn4c=5JQ@mail.gmail.com' \
    --to=dalanicolai@gmail.com \
    --cc=eliz@gnu.org \
    --cc=emacs-devel@gnu.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).