From: Tino Calancha <tino.calancha@gmail.com>
To: Juri Linkov <juri@linkov.net>
Cc: 26338@debbugs.gnu.org, tino.calancha@gmail.com,
Marcin Borkowski <mbork@mbork.pl>,
npostavs@users.sourceforge.net, Dmitry Gutov <dgutov@yandex.ru>
Subject: bug#26338: 26.0.50; Collect all matches for REGEXP in current buffer
Date: Fri, 07 Apr 2017 23:47:16 +0900 [thread overview]
Message-ID: <87a87sqnpn.fsf@calancha-pc> (raw)
In-Reply-To: <871st6342v.fsf@localhost> (Juri Linkov's message of "Thu, 06 Apr 2017 01:03:04 +0300")
Juri Linkov <juri@linkov.net> writes:
>>> Sorry if this was said already, but why a macro and not a map-like
>>> function?
>> No special reason. It's the second idea which came to my mind after
>> my initial proposal was declined. Maybe because is shorter to do:
>> (with-collect-matches regexp)
>> than
>> (foo-collect-matches regexp nil #'identity)
>>
>> if you are just interested in the list of matches. Implementing it as
>> a map function might be also nice. Don't see a big enthusiasm on
>> the proposal, though :-(
>>
>> So far people think that it's easy to write a while loop. I wonder if they
>> think the same about the existence of `dolist': the should
>> never use it and always write a `while' loop instead. Don't think they
>> do that anyway.
>>
>> I will repeat it once more. I find nice, having an operator returning
>> a list with matches for REGEXP. If such operator, in addition, accepts
>> a body of code or a function, then i find this operator very nice
>> and elegant.
>
> A mapcar-like function presumes a lambda where you can process every
> match as you need, but going this way you'd have a temptation to
> implement an analogous API from other programming languages like e.g.
> https://apidock.com/ruby/String/scan
I am not crazy with the mapcar-like implemention either.
Actually, I have changed my mind after nice Noah suggestion. He
mentioned the possibility of extend `cl-loop' with a new clause to iterate on
matches for a regexp.
I think this clause fits well in cl-loop; this way we don't need to
introduce a new function/macro name.
--8<-----------------------------cut here---------------start------------->8---
commit 59e66771d13fce73ff5220ce3df677b9247c9c52
Author: Tino Calancha <tino.calancha@gmail.com>
Date: Fri Apr 7 23:31:08 2017 +0900
New clause in cl-loop to iterate in the matches of a regexp
Add new clause in cl-loop facility to loop over the matches for
REGEXP in the current buffer (Bug#26338).
* lisp/emacs-lisp/cl-macs.el (cl--parse-loop-clause): Add new clause.
(cl-loop): update docstring.
* doc/misc/cl.texi (For Clauses): Document the new clause.
* etc/NEWS: Mention this change.
diff --git a/doc/misc/cl.texi b/doc/misc/cl.texi
index 2339d57631..6c5c43ad09 100644
--- a/doc/misc/cl.texi
+++ b/doc/misc/cl.texi
@@ -2030,6 +2030,21 @@ For Clauses
This clause iterates over a sequence, with @var{var} a @code{setf}-able
reference onto the elements; see @code{in-ref} above.
+@item for @var{var} being the matches of @var{regexp}
+This clause iterates over the matches for @var{regexp} in the current buffer.
+By default, @var{var} is bound to the full match. Optionally, @var{var}
+might be bound to a subpart of the match. It's also possible to restrict
+the loop to a given number of matches. For example,
+
+@example
+(cl-loop for x being the matches of "^(defun \\(\\S +\\)"
+ using '(group 1 limit 10)
+ collect x)
+@end example
+
+@noindent
+collects the next 10 function names after point.
+
@item for @var{var} being the symbols [of @var{obarray}]
This clause iterates over symbols, either over all interned symbols
or over all symbols in @var{obarray}. The loop is executed with
diff --git a/etc/NEWS b/etc/NEWS
index aaca229d5c..03f6ecb88b 100644
--- a/etc/NEWS
+++ b/etc/NEWS
@@ -862,6 +862,10 @@ instead of its first.
* Lisp Changes in Emacs 26.1
+++
+** New clause in cl-loop to iterate in the matches for a regexp
+in the current buffer.
+
++++
** Emacs now supports records for user-defined types, via the new
functions 'copy-record', 'make-record', 'record', and 'recordp'.
Records are now used internally to represent cl-defstruct and defclass
diff --git a/lisp/emacs-lisp/cl-macs.el b/lisp/emacs-lisp/cl-macs.el
index 25c9f99992..50596c066e 100644
--- a/lisp/emacs-lisp/cl-macs.el
+++ b/lisp/emacs-lisp/cl-macs.el
@@ -892,6 +892,7 @@ cl-loop
the overlays/intervals [of BUFFER] [from POS1] [to POS2]
the frames/buffers
the windows [of FRAME]
+ the matches of/for REGEXP [using (group GROUP [limit LIMIT])]
Iteration clauses:
repeat INTEGER
while/until/always/never/thereis CONDITION
@@ -1339,6 +1340,33 @@ cl--parse-loop-clause
(push (list temp-idx `(1+ ,temp-idx))
loop-for-steps)))
+ ((memq word '(match matches))
+ (let* ((_ (or (and (not (memq (car cl--loop-args) '(of for)))
+ (error "Expected `of'"))))
+ (regexp (cl--pop2 cl--loop-args))
+ (group-limit
+ (and (eq (car cl--loop-args) 'using)
+ (consp (cadr cl--loop-args))
+ (>= (length (cadr cl--loop-args)) 2)
+ (cadr (cl--pop2 cl--loop-args))))
+ (group
+ (or (and group-limit
+ (cl-find 'group group-limit)
+ (nth (1+ (cl-position 'group group-limit)) group-limit))
+ 0))
+ (limit
+ (and group-limit
+ (cl-find 'limit group-limit)
+ (nth (1+ (cl-position 'limit group-limit)) group-limit)))
+ (count (make-symbol "--cl-count")))
+ (push (list count 0) loop-for-bindings)
+ (push (list var nil) loop-for-bindings)
+ (push `(re-search-forward ,regexp nil t) cl--loop-body)
+ (push `(or (null ,limit) (and (natnump ,limit) (< ,count ,limit))) cl--loop-body)
+ (push (list count `(1+ ,count)) loop-for-sets)
+ (push (list var `(match-string-no-properties ,group))
+ loop-for-sets)))
+
((memq word hash-types)
(or (memq (car cl--loop-args) '(in of))
(error "Expected `of'"))
--8<-----------------------------cut here---------------end--------------->8---
In GNU Emacs 26.0.50 (build 7, x86_64-pc-linux-gnu, GTK+ Version 3.22.11)
of 2017-04-07
Repository revision: 67aeaa74af8504f950f653136d749c6dd03a60de
next prev parent reply other threads:[~2017-04-07 14:47 UTC|newest]
Thread overview: 31+ messages / expand[flat|nested] mbox.gz Atom feed top
2017-04-02 12:41 bug#26338: 26.0.50; Collect all matches for REGEXP in current buffer Tino Calancha
2017-04-02 15:57 ` Dmitry Gutov
2017-04-03 3:58 ` Tino Calancha
2017-04-02 22:10 ` Juri Linkov
2017-04-03 4:01 ` Tino Calancha
2017-04-03 6:13 ` Tino Calancha
2017-04-03 23:35 ` Juri Linkov
2017-04-04 1:37 ` Tino Calancha
2017-04-04 2:20 ` Tino Calancha
2017-04-04 14:32 ` Marcin Borkowski
2017-04-05 11:58 ` Tino Calancha
2017-04-05 13:11 ` npostavs
2017-04-07 10:06 ` Tino Calancha
2017-04-07 14:40 ` Drew Adams
2017-04-08 4:45 ` Tino Calancha
2017-04-08 5:49 ` Drew Adams
2017-04-08 15:29 ` Tino Calancha
2017-04-08 15:42 ` Drew Adams
2017-04-08 11:46 ` Philipp Stephani
2017-04-08 13:42 ` Tino Calancha
2017-04-08 14:41 ` Philipp Stephani
2017-04-08 15:20 ` Tino Calancha
2017-04-22 19:42 ` Philipp Stephani
2017-04-08 15:38 ` npostavs
2017-04-22 19:36 ` Philipp Stephani
2017-04-05 22:03 ` Juri Linkov
2017-04-07 14:47 ` Tino Calancha [this message]
2017-04-07 15:28 ` Noam Postavsky
2017-04-07 15:54 ` Drew Adams
2017-04-08 13:49 ` Tino Calancha
2020-09-15 15:41 ` Lars Ingebrigtsen
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=87a87sqnpn.fsf@calancha-pc \
--to=tino.calancha@gmail.com \
--cc=26338@debbugs.gnu.org \
--cc=dgutov@yandex.ru \
--cc=juri@linkov.net \
--cc=mbork@mbork.pl \
--cc=npostavs@users.sourceforge.net \
/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).