* Help Understanding syntax-propertize-function
@ 2021-03-15 22:48 Reza Nikoopour
2021-03-15 23:30 ` Stefan Monnier
0 siblings, 1 reply; 10+ messages in thread
From: Reza Nikoopour @ 2021-03-15 22:48 UTC (permalink / raw)
To: emacs-devel
[-- Attachment #1: Type: text/plain, Size: 1720 bytes --]
Hello All,
I'm trying to implement here doc syntax highlighting. I've reviewed the
implementations in shell-script-mode and hcl-mode but I don't really
understand what is happening. I've read the documentation for
syntax-propertize-function (
https://www.gnu.org/software/emacs/manual/html_node/elisp/Syntax-Properties.html)
but that didn't help me understand what's going on.
Could someone help explain the following code:
(defun hcl--syntax-propertize-heredoc (end)
(let ((ppss (syntax-ppss)))
(when (eq t (nth 3 ppss))
(let ((key (get-text-property (nth 8 ppss) 'hcl-here-doc-marker))
(case-fold-search nil))
(when (re-search-forward
(concat "^\\(?:[ \t]*\\)" (regexp-quote key) "\\(\n\\)")
end 'move)
(let ((eol (match-beginning 1)))
(put-text-property eol (1+ eol)
'syntax-table (string-to-syntax "|"))))))))
(defun hcl--font-lock-open-heredoc (start string eol)
(unless (or (memq (char-before start) '(?< ?>))
(save-excursion
(goto-char start)
(hcl--in-string-or-comment-p)))
(let ((str (replace-regexp-in-string "['\"]" "" string)))
(put-text-property eol (1+ eol) 'hcl-here-doc-marker str)
(prog1 (string-to-syntax "|")
(goto-char (+ 2 start))))))
(defun hcl--syntax-propertize-function (start end)
(goto-char start)
(hcl--syntax-propertize-heredoc end)
(funcall
(syntax-propertize-rules
(hcl--here-doc-beg-re
(2 (hcl--font-lock-open-heredoc
(match-beginning 0) (match-string 1) (match-beginning 2))))
("\\s|" (0 (prog1 nil (hcl--syntax-propertize-heredoc end)))))
(point) end))
Cheers,
Reza
[-- Attachment #2: Type: text/html, Size: 3864 bytes --]
^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: Help Understanding syntax-propertize-function
2021-03-15 22:48 Help Understanding syntax-propertize-function Reza Nikoopour
@ 2021-03-15 23:30 ` Stefan Monnier
2021-03-15 23:58 ` Reza Nikoopour
2021-03-24 20:29 ` Filipp Gunbin
0 siblings, 2 replies; 10+ messages in thread
From: Stefan Monnier @ 2021-03-15 23:30 UTC (permalink / raw)
To: Reza Nikoopour; +Cc: emacs-devel
> Could someone help explain the following code:
I'm not sure which part you don't understand, so I'll stick to
generic ideas:
> (defun hcl--syntax-propertize-function (start end)
> (goto-char start)
> (hcl--syntax-propertize-heredoc end)
> (funcall
> (syntax-propertize-rules
> (hcl--here-doc-beg-re
> (2 (hcl--font-lock-open-heredoc
> (match-beginning 0) (match-string 1) (match-beginning 2))))
> ("\\s|" (0 (prog1 nil (hcl--syntax-propertize-heredoc end)))))
> (point) end))
Since we don't know where `start` will be, it can be within a heredoc.
So we first call `hcl--syntax-propertize-heredoc` which should detect
when we're inside a heredoc and if so process it until its end and if not
do nothing.
Once that's done, we know we're not in a heredoc, hence we're in
"normal code" and we then use `syntax-propertize-rules` to try and
detect a few interesting conditions. One of them is when we find
a match for `hcl--here-doc-beg-re`, in that case we call
`hcl--font-lock-open-heredoc` to place some syntactic marker at the
beginning of the heredoc. The other is when we see a char of
syntactic category `|`, which doesn't occur in the wild but should
presumably be the result of `hcl--font-lock-open-heredoc` having placed
it there, so it announces the beginning of a heredoc, in which case we
can call `hcl--syntax-propertize-heredoc` to process it.
Stefan
^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: Help Understanding syntax-propertize-function
2021-03-15 23:30 ` Stefan Monnier
@ 2021-03-15 23:58 ` Reza Nikoopour
2021-03-16 1:42 ` Stefan Monnier
2021-03-24 20:29 ` Filipp Gunbin
1 sibling, 1 reply; 10+ messages in thread
From: Reza Nikoopour @ 2021-03-15 23:58 UTC (permalink / raw)
To: Stefan Monnier; +Cc: emacs-devel
[-- Attachment #1: Type: text/plain, Size: 523 bytes --]
> I'm not sure which part you don't understand, so I'll stick to
> generic ideas:
Thanks that helped a lot and enabled me to figure out what to ask!
> So we first call `hcl--syntax-propertize-heredoc` which should detect
> when we're inside a heredoc and if so process it until its end and if not
> do nothing.
I'm struggling to understand what `hcl--syntax-propertize-heredoc` is
doing.
Could you provide a high level explanation like you did with
`hcl--syntax-propertize-function`.
That helped me a lot.
Cheers,
Reza
[-- Attachment #2: Type: text/html, Size: 653 bytes --]
^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: Help Understanding syntax-propertize-function
2021-03-15 23:58 ` Reza Nikoopour
@ 2021-03-16 1:42 ` Stefan Monnier
2021-03-16 15:37 ` Reza Nikoopour
0 siblings, 1 reply; 10+ messages in thread
From: Stefan Monnier @ 2021-03-16 1:42 UTC (permalink / raw)
To: Reza Nikoopour; +Cc: emacs-devel
> I'm struggling to understand what `hcl--syntax-propertize-heredoc` is
> doing.
> Could you provide a high level explanation like you did with
> `hcl--syntax-propertize-function`.
> That helped me a lot.
> (defun hcl--syntax-propertize-heredoc (end)
The purpose of the function is to do the "syntax-propertize" of the
inside of a heredoc. It presumes that point *may* be in a heredoc but
not necessarily.
> (let ((ppss (syntax-ppss)))
> (when (eq t (nth 3 ppss))
Here we fetched the syntax state at point and then we checked that we
are indeed inside a heredoc (or at least a "string like thing" that was
opened using the `|` syntax category; in this mode should should only
ever happen if we have placed this syntax because we found a heredoc
marker).
> (let ((key (get-text-property (nth 8 ppss) 'hcl-here-doc-marker))
This fetches the string that was used in the heredoc opener and which
has to be used as heredoc closer. It's been placed on the opening char
at position (nth 8 ppss) by the same code that placed the `|` syntax
property on that same character.
> (case-fold-search nil))
I guess here that the code does that because HCL defines the heredoc
marker to be case-significant.
> (when (re-search-forward
> (concat "^\\(?:[ \t]*\\)" (regexp-quote key) "\\(\n\\)")
> end 'move)
Here we look for the heredoc end marker. If we can't find one before
`end`, it just means that the heredoc extends further and hence we have
nothing to do (we could actually remove/override any `|` syntax that
might appear before `end` in case such a thing is possible, but
apparently the rest of code is arranged so that this is not needed).
> (let ((eol (match-beginning 1)))
> (put-text-property eol (1+ eol)
> 'syntax-table (string-to-syntax "|"))))))))
If we did find the heredoc end marker before `end`, then we mark it as
being the end by adding the `|` syntax to the ending character.
Stefan
^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: Help Understanding syntax-propertize-function
2021-03-15 23:30 ` Stefan Monnier
2021-03-15 23:58 ` Reza Nikoopour
@ 2021-03-24 20:29 ` Filipp Gunbin
2021-03-24 22:10 ` Stefan Monnier
1 sibling, 1 reply; 10+ messages in thread
From: Filipp Gunbin @ 2021-03-24 20:29 UTC (permalink / raw)
To: Stefan Monnier; +Cc: emacs-devel, Reza Nikoopour
There's the prog1 form at the end of hcl--font-lock-open-heredoc:
...
(prog1 (string-to-syntax "|")
(goto-char (+ 2 start))))))
It looks like it's meant to be a special "(prog1 EXP . EXPS)" form for
syntax-propertize-rules. But it's inside of a defun used as SYNTAX:
...
(syntax-propertize-rules
(hcl--here-doc-beg-re
(2 (hcl--font-lock-open-heredoc
(match-beginning 0) (match-string 1) (match-beginning 2))))
I wonder - how could that work? Does syntax-propertize-rules inspect
the SYNTAX expression to check for suchlike forms?
Filipp
^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: Help Understanding syntax-propertize-function
2021-03-24 20:29 ` Filipp Gunbin
@ 2021-03-24 22:10 ` Stefan Monnier
2021-03-24 22:54 ` Filipp Gunbin
0 siblings, 1 reply; 10+ messages in thread
From: Stefan Monnier @ 2021-03-24 22:10 UTC (permalink / raw)
To: Reza Nikoopour; +Cc: emacs-devel
> There's the prog1 form at the end of hcl--font-lock-open-heredoc:
>
> ...
> (prog1 (string-to-syntax "|")
> (goto-char (+ 2 start))))))
Indeed, there is, but I don't know what worries you bout it.
> It looks like it's meant to be a special "(prog1 EXP . EXPS)" form for
> syntax-propertize-rules.
I don't think so.
> I wonder - how could that work? Does syntax-propertize-rules inspect
> the SYNTAX expression to check for suchlike forms?
No it doesn't.
Stefan
^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: Help Understanding syntax-propertize-function
2021-03-24 22:10 ` Stefan Monnier
@ 2021-03-24 22:54 ` Filipp Gunbin
0 siblings, 0 replies; 10+ messages in thread
From: Filipp Gunbin @ 2021-03-24 22:54 UTC (permalink / raw)
To: Stefan Monnier; +Cc: emacs-devel, Reza Nikoopour
On 24/03/2021 18:10 -0400, Stefan Monnier wrote:
>> There's the prog1 form at the end of hcl--font-lock-open-heredoc:
>>
>> ...
>> (prog1 (string-to-syntax "|")
>> (goto-char (+ 2 start))))))
>
> Indeed, there is, but I don't know what worries you bout it.
>
>> It looks like it's meant to be a special "(prog1 EXP . EXPS)" form for
>> syntax-propertize-rules.
>
> I don't think so.
Then prog1 is confusing there. It's not special, as you confirm, and
it's not needed otherwise. Just
(goto-char (+ 2 start))
(string-to-syntax "|")
will do - without raising suspicions :-)
Filipp
^ permalink raw reply [flat|nested] 10+ messages in thread
end of thread, other threads:[~2021-03-24 22:54 UTC | newest]
Thread overview: 10+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2021-03-15 22:48 Help Understanding syntax-propertize-function Reza Nikoopour
2021-03-15 23:30 ` Stefan Monnier
2021-03-15 23:58 ` Reza Nikoopour
2021-03-16 1:42 ` Stefan Monnier
2021-03-16 15:37 ` Reza Nikoopour
2021-03-16 17:29 ` Stefan Monnier
2021-03-16 17:50 ` Reza Nikoopour
2021-03-24 20:29 ` Filipp Gunbin
2021-03-24 22:10 ` Stefan Monnier
2021-03-24 22:54 ` Filipp Gunbin
Code repositories for project(s) associated with this external index
https://git.savannah.gnu.org/cgit/emacs.git
https://git.savannah.gnu.org/cgit/emacs/org-mode.git
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.