From mboxrd@z Thu Jan 1 00:00:00 1970 Path: news.gmane.org!.POSTED.blaine.gmane.org!not-for-mail From: Dmitrii Korobeinikov Newsgroups: gmane.emacs.bugs Subject: bug#35419: [Proposal] Buffer Lenses and the Case of Org-Mode (also, Jupyter) Date: Tue, 14 May 2019 23:42:53 +0600 Message-ID: References: <87muk1fn90.fsf@yantar92-laptop.i-did-not-set--mail-host-address--so-tickle-me> Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="000000000000b3501f0588dc90dc" Injection-Info: blaine.gmane.org; posting-host="blaine.gmane.org:195.159.176.226"; logging-data="245739"; mail-complaints-to="usenet@blaine.gmane.org" Cc: Philipp Stephani , 35419@debbugs.gnu.org, Noam Postavsky To: Ihor Radchenko Original-X-From: bug-gnu-emacs-bounces+geb-bug-gnu-emacs=m.gmane.org@gnu.org Tue May 14 19:45:32 2019 Return-path: Envelope-to: geb-bug-gnu-emacs@m.gmane.org Original-Received: from lists.gnu.org ([209.51.188.17]) by blaine.gmane.org with esmtps (TLS1.0:RSA_AES_256_CBC_SHA1:256) (Exim 4.89) (envelope-from ) id 1hQbUX-0011kL-Oz for geb-bug-gnu-emacs@m.gmane.org; Tue, 14 May 2019 19:45:30 +0200 Original-Received: from localhost ([127.0.0.1]:51822 helo=lists.gnu.org) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1hQbUW-0006IK-M6 for geb-bug-gnu-emacs@m.gmane.org; Tue, 14 May 2019 13:45:28 -0400 Original-Received: from eggs.gnu.org ([209.51.188.92]:58036) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1hQbTF-0005bf-FK for bug-gnu-emacs@gnu.org; Tue, 14 May 2019 13:44:11 -0400 Original-Received: from Debian-exim by eggs.gnu.org with spam-scanned (Exim 4.71) (envelope-from ) id 1hQbTD-0007GZ-25 for bug-gnu-emacs@gnu.org; Tue, 14 May 2019 13:44:09 -0400 Original-Received: from debbugs.gnu.org ([209.51.188.43]:36225) by eggs.gnu.org with esmtps (TLS1.0:RSA_AES_128_CBC_SHA1:16) (Exim 4.71) (envelope-from ) id 1hQbT8-000758-15; Tue, 14 May 2019 13:44:03 -0400 Original-Received: from Debian-debbugs by debbugs.gnu.org with local (Exim 4.84_2) (envelope-from ) id 1hQbT7-0000Yz-Tq; Tue, 14 May 2019 13:44:01 -0400 X-Loop: help-debbugs@gnu.org Resent-From: Dmitrii Korobeinikov Original-Sender: "Debbugs-submit" Resent-CC: bug-gnu-emacs@gnu.org, emacs-orgmode@gnu.org Resent-Date: Tue, 14 May 2019 17:44:01 +0000 Resent-Message-ID: Resent-Sender: help-debbugs@gnu.org X-GNU-PR-Message: followup 35419 X-GNU-PR-Package: emacs,org-mode Original-Received: via spool by 35419-submit@debbugs.gnu.org id=B35419.15578557942096 (code B ref 35419); Tue, 14 May 2019 17:44:01 +0000 Original-Received: (at 35419) by debbugs.gnu.org; 14 May 2019 17:43:14 +0000 Original-Received: from localhost ([127.0.0.1]:49767 helo=debbugs.gnu.org) by debbugs.gnu.org with esmtp (Exim 4.84_2) (envelope-from ) id 1hQbSL-0000Xk-BT for submit@debbugs.gnu.org; Tue, 14 May 2019 13:43:13 -0400 Original-Received: from mail-wm1-f46.google.com ([209.85.128.46]:54553) by debbugs.gnu.org with esmtp (Exim 4.84_2) (envelope-from ) id 1hQbSJ-0000XU-7f for 35419@debbugs.gnu.org; Tue, 14 May 2019 13:43:12 -0400 Original-Received: by mail-wm1-f46.google.com with SMTP id i3so3751979wml.4 for <35419@debbugs.gnu.org>; Tue, 14 May 2019 10:43:11 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=mime-version:references:in-reply-to:from:date:message-id:subject:to :cc; bh=wJknWIETo/Xfhl2j3U1a/M0fgAUVrl0BF6Km6D2pLik=; b=pmhZ+9yNQrih2ZdT2M5D4N9fX01aVKmQjzlXWhk4JZDa3fWzeFm22LJoA57w0VIDnx ZHjcliBZHeU0UtXYZwDBKIw+6eOZ9KU3XOSknIlN/UQcBF2VPSihNzRNSEOZYOJ+AxWN kpCQkZvVo+lep0BUQ6WTquTmmrqu1DJWWJGic/91LAdKGkBJmymYEQ3chOeL1xr2LpVg Grjp16f8azGP9A+MHBbgbQy4wZMaBJA+wIbPZJLa4+OYnXrBzxVcP+tXnNAQfW0QT4vC 4jYvY1imAGGt4Y6Yg/iFEzkYBAgjKkNSUfZdqvFbwqyIzt56S0ZNzRH8hZjpGczN+T1y AOXw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:mime-version:references:in-reply-to:from:date :message-id:subject:to:cc; bh=wJknWIETo/Xfhl2j3U1a/M0fgAUVrl0BF6Km6D2pLik=; b=ruWxvLkg9y6BhrjfhYrhM3i4IG5KW6JgIZrMYbFTWsnS9bvpS8VhrgHSZ02/R8ALaQ xHFf3OWsob9bL2BCenqwEeioBiSjOL47eOlIYUfzCXGrkVZxvkYzxxbyuvgoNZrjC7RI mtURArdgHWO9Fjuf59fwHLvep5X0fYpN1Ey/uCICPXvvNrmwlxBPMYb8IAau5dGlSKsO q7i/4oB80Oym6Log6JKCb+I379+6Ka2MCbgViu4/uNMoiz2WDRaIBNrB/+T2grM1lTuA lwDnBzbyjszF83yiZeCTMh3Z3TQ4JdvR/HPzsPJP8Yrfi4Ka9kXQyqMkm6ea17AftmaX l4aw== X-Gm-Message-State: APjAAAVIFwbsQob1v+FlnpofuyB5ict0KcRDerYHny76GuPPniO8JHBV fX5r/kGV4Kmr3qCsJO8hGCokCamLlSm5QNUODjA= X-Google-Smtp-Source: APXvYqxdORSwk3O9CbNN2xFicKDtZ0oXgc91wWgc5pVRS4jX8GQY9kWUaK4KjZKzFzDQ93MM+yQDbt2p1KeOsYCCRLY= X-Received: by 2002:a1c:9951:: with SMTP id b78mr10662014wme.96.1557855785404; Tue, 14 May 2019 10:43:05 -0700 (PDT) In-Reply-To: <87muk1fn90.fsf@yantar92-laptop.i-did-not-set--mail-host-address--so-tickle-me> X-BeenThere: debbugs-submit@debbugs.gnu.org X-Mailman-Version: 2.1.18 Precedence: list X-detected-operating-system: by eggs.gnu.org: GNU/Linux 2.2.x-3.x [generic] X-Received-From: 209.51.188.43 X-BeenThere: bug-gnu-emacs@gnu.org List-Id: "Bug reports for GNU Emacs, the Swiss army knife of text editors" List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: bug-gnu-emacs-bounces+geb-bug-gnu-emacs=m.gmane.org@gnu.org Original-Sender: "bug-gnu-emacs" Xref: news.gmane.org gmane.emacs.bugs:159283 Archived-At: --000000000000b3501f0588dc90dc Content-Type: text/plain; charset="UTF-8" Dear Ihor, > Note that indirect buffers always share *all* the contents with the master > buffer. As a result, it may not be easy to make things like flyspell > work on code blocks in org-mode, if these code blocks are treated as > lenses. I tried flyspell w/ different dictionaries on 2 buffers. The dictionary is switched every time I switch into one of the buffers. You are right, ispell and the like working w/ a file directly would have to learn to work w/ indirect buffers by managing multiple simultaneous processes. Fortunately, that doesn't seem like a big hurdle. >> (1) A question: when an indirect buffer is created and some region is >> narrowed to, is the rest of the buffer duplicated in memory somewhere? If >> this is so, there could be a useful efficiency-related modification to >> indirect buffers, which would allow "hard-narrowing": not duplicating the >> rest of the base buffer. > > There is no duplication of the buffer content in indirect buffers. > Internally, indirect buffer's content is a pointer to the main buffer > content. If you modify text in any of the indirect buffers or in the > main buffer, the text is modified in all of them and in the main buffer. > Only the buffer-local variables are duplicated. > You can refer to "27.11 Indirect Buffers" in the elisp manual for > details. Bad choice of wording on my side, I didn't mean duplication, but rather keeping unnecessary info, like text properties in the newly created indirect buffer, in the regions which were "permanently" chosen to be narrowed-out. Anyway, this is a premature optimization for now. > > The next immediately outstanding question is: > > (2) how can "embedding" (of a buffer as a part of another buffer as an > > area) be done efficiently? This could possibly be approached as two > > problems: (i) displaying the area and (ii) interacting with it. > > Any ideas? > > These issues have been discussed in > https://lists.gnu.org/archive/html/emacs-devel/2018-07/msg00863.html. > As I remember, the discussion stopped without a clear conclusion. It was > not clear how to separate the main buffer contents from the nested > buffer (I treat them as analogue of the buffer lenses). Another issue > was how the keymaps and buffer-local variables would interact when the > point is within a lense. It was not clear what should be the priority. The short answer is probably that lens-mode looks at the changes to the buffer and decides what's what. Here is my vision for this. Say, you have an indirect buffer, call it A, it's base has contents: > line 1 > line 2 > line 3 Also, there is a buffer, call it B, where we want to embed A, with contents: > word 1 > instruction: lens that displays A > word 2 Lens-mode decides to identify the second line as a lens and constructs layout of the file. > [text] > [lens#A] > [text] Now, construct and display the final buffer as: > word 1 > line 1 > line 2 > line 3 > word 2 The core question: how is this "displaying" done. In part, somewhat like how indirect buffers function. A displayed piece of text is a pointer/reference to the text in the indirect buffer. Of course, this should be done in a way so that the modes running in B don't change the properties of the text (following the layout constructed by lens-mode as in the example above). Though, this might better&easier be done at the display unit level. What about interaction? Well, when the cursor is inside the lens, the controller decides what to do w/ each keybinding, whether to redirect it to the indirect buffer or not. And what about the case when borders are crossed? As I see it, any code that executes - does so as is. For instance, consider a buffer with a lens (contents: "lens"), which looks like this: "word1 lens word2". Place the cursor on the first word, run a command to remove 2 words. Now there are to possibilities here, depending on the implementation of the function which does the removal: 1. Implementation identifies the boundaries of deletion and removes the words as contents of the main buffer. Lens-mode identifies the removal and deletes the lens altogether. 2. Delete "word1" -> the cursor is on "lens" -> next deletion command is redirected to the indirect buffer, which does the removal. Lens-mode identifies the lens as empty and removes it. The second way might not be always desirable, so whenever a function is called with the cursor outside a lens, as an option, decide to never redirect the input to a lens while the function runs. For another case, consider the first example with lines and, cursor being in the beginning of the file, remove three lines. To make this interesting, assume the first kind of implementation runs, identifying the bounds, never redirecting a command to the lens. Well, if the implementation of the lens is inspired by how indirect buffers work, I imagine, the necessary changes in the embedded buffer take place directly, in which case everything works as desirable. Afterwards, lens-mode processes the changes and decides if some lens was removed or added, updating the file layout. Best regards, Dmitrii. --000000000000b3501f0588dc90dc Content-Type: text/html; charset="UTF-8" Content-Transfer-Encoding: quoted-printable
Dear Ihor,

> Note that indirect buffers always share *all* the contents with= the master
> buffer. As a result, it may not be easy to make = things like flysp= ell
> work on code blocks in org-mode, if these code bl= ocks are treated as
> lenses.=C2=A0

I= tried flyspell w/ different dictionaries on 2 buffers.
The dictionary is s= witched every time I switch into one of the buffers.
You are righ= t, ispell = and the like working w/ a file directly would have to learn to work w/ indi= rect buffers by managing multiple simultaneous processes.
Fortuna= tely, that doesn't seem like a big hurdle.

>= ;> (1) A question: when an indirect buffer is created and some region is=
>> narrowed to, is the rest of the buffer duplicated in me= mory somewhere? If
>> this is so, there could be a useful e= fficiency-related modification to
>> indirect buffers, whic= h would allow "hard-narrowing": not duplicating the
>= ;> rest of the base buffer.
>=C2=A0
> There is= no duplication of the buffer content in indirect buffers.
> I= nternally, indirect buffer's content is a pointer to the main buffer
> content. If you modify text in any of the indirect buffers or = in the
> main buffer, the text is modified in all of them and = in the main buffer.
> Only the buffer-local variables are dupl= icated.
> You can refer to "27.11 Indirect Buffers" = in the elisp manual for
> details.=C2=A0

Bad ch= oice of wording on my side, I didn't mean duplication, but rather keepi= ng unnecessary info, like text properties in the newly created indirect buf= fer, in the regions which were "permanently" chosen to be narrowe= d-out.
Anyway, this is a premature optimization for now.

> > The next immediately outstanding question is:
> > (2) how can "embedding" (of a buffer as a part = of another buffer as an
> > area) be done efficiently? This= could possibly be approached as two
> > problems: (i) disp= laying the area and (ii) interacting with it.
> > Any ideas= ?
>
> These issues have been discussed in
> https://lists.gnu.o= rg/archive/html/emacs-d= evel/2018-07/msg00863.html.
> As I remember, the discus= sion stopped without a clear conclusion. It was
> not clear ho= w to separate the main buffer contents from the nested
> buffe= r (I treat them as analogue of the buffer lenses). Another issue
= > was how the = keymaps and buffer-local variables would interact when the
> point is within a lense. It was not clear what should be the priority.

The short answer is probably that lens-mode looks at the c= hanges to the buffer and decides what's what.
Here is my visi= on for this.

Say, you have an indirect buffer, cal= l it A, it's base has contents:

> line 1
> line 2
> line 3

Also, th= ere is a buffer, call it B, where we want to embed A, with contents:
<= div>
> word 1
> instruction: lens that displa= ys A
> word 2

Lens-mode decides to id= entify the second line as a lens and constructs layout of the file.

> [text]
> [lens#A]
> [text]=

Now, construct and display the final buffer as:

> word 1
> line 1
> li= ne 2
> line 3
> word 2

T= he core question: how is this "displaying" done.
In par= t, somewhat like how indirect buffers function.
A displayed piece= of text is a pointer/reference to the text in the indirect buffer.
Of course, this should be done in a way so that the modes running in B d= on't change the properties of the text (following the layout constructe= d by lens-mode as in the example above). Though, this might better&easi= er be done at the display unit level.

What about i= nteraction?
Well, when the cursor is inside the lens, the control= ler decides what to do w/ each keybinding, whether to redirect it to the indirect bu= ffer or not.
And what about the case when borders are crossed?
As I see it, any code that executes - does so as is.
For = instance, consider a buffer with a lens (contents: "lens"), which= looks like this: "word1 lens word2".
Place the cursor = on the first word, run a command to remove 2 words.
Now there are= to possibilities here, depending on the implementation of the function whi= ch does the removal:
1. Implementation identifies the boundaries = of deletion and removes the words as contents of the main buffer. Lens-mode= identifies the removal and deletes the lens altogether.
2. Delet= e "word1" -> the cursor is on "lens" -> next dele= tion command is redirected to the indirect buffer, which does the removal. = Lens-mode identifies the lens as empty and removes it.

=
The second way might not be always desirable, so whenever a function i= s called with the cursor outside a lens, as an option, decide to never redi= rect the input to a lens while the function runs.

= For another case, consider the first example with lines and, cursor being i= n the beginning of the file, remove three lines.
To make this int= eresting, assume the first kind of implementation runs, identifying the bou= nds, never redirecting a command to the lens.
Well, if the implem= entation of the lens is inspired by how indirect buffers work, I imagine, t= he necessary changes in the embedded buffer take place directly, in which c= ase everything works as desirable.

Afterwards, lens-mode pr= ocesses the changes and decides if some lens was removed or added, updating= the file layout.

Best regards,
Dmitrii.
--000000000000b3501f0588dc90dc--