组织模式:在议程视图中过滤标签?

时间:2012-04-09 13:34:39

标签: emacs org-mode

当议程构建其视图时,是否可以过滤标签?我尝试过以下内容仅显示与工作相关的约会:

("j" "Jobb"
   ((agenda ""
       ((org-agenda-skip-function '(org-agenda-skip-entry-if 'notregexp":jobb:"))))
    (tags-todo "jobb"))
    ((org-agenda-compact-blocks nil)))

仅当实际约会被直接标记时才有效,但如果约会从父标题继承其标记,则不会这样:

 * Tider                                                              :jobb:                                                                                                                                                         
 ** Millas arbetstider                                                                                                                                                                                                               
   <2012-04-11 ons 05:00-09:00>                                                                                                                                                                                                     
   <2012-04-12 tor 04:15-08:30>                                                                                                                                                                                                     
   <2012-04-13 fre 14:30-18:30>                           

是否有另一种方法可以执行此操作,以便显示继承其标记的约会?

3 个答案:

答案 0 :(得分:12)

问题在于org-agenda-skip-entries-if'notregexp的互动方式。它将跳过与:jobb:不匹配的任何条目。即使后面的条目继承了标记,它也没有明确列出,因此它们被跳过。似乎没有任何内置方法可以使用org-agenda-skip-entries-if匹配(或不匹配)标签。如果有这样的功能,它可能是寻找标签的更有效方法,但我不知道这样的功能。

您必须创建一个自定义函数,以提供所需的搜索格式。

如果您将议程命令更改为:

("j" "Jobb"
         ((agenda ""
                  ((org-agenda-skip-function '(zin/org-agenda-skip-tag "jobb" 't))))
          (tags-todo "jobb"))
         ((org-agenda-compact-blocks nil)))

并将zin/org-agenda-skip-tag定义为:

(defun zin/org-agenda-skip-tag (tag &optional others)
  "Skip all entries that correspond to TAG.

If OTHERS is true, skip all entries that do not correspond to TAG."
  (let ((next-headline (save-excursion (or (outline-next-heading) (point-max))))
        (current-headline (or (and (org-at-heading-p)
                                   (point))
                              (save-excursion (org-back-to-heading)))))
    (if others
        (if (not (member tag (org-get-tags-at current-headline)))
            next-headline
          nil)
      (if (member tag (org-get-tags-at current-headline))
          next-headline
        nil))))

您将得到我理解的您想要的议程视图。如果我向后退,并且未来3天的条目不应出现,您只需将功能更改为(zin/org-agenda-skip-tag "jobb")(zin/org-agenda-skip-tag "jobb" 'nil),在这种情况下它们就相同。

日程视图

在这种情况下,test-new是我正在使用的组织文件的名称,可以忽略它。我还设置了两个标题TODO,以便在测试函数时可以看到它们,因为我将议程限制为只有一个文件。

Week-agenda (W15):
Monday      9 April 2012 W15
Tuesday    10 April 2012
Wednesday  11 April 2012
  test-new:    5:00- 9:00 TODO Millas arbetstider                        :jobb::
Thursday   12 April 2012
  test-new:    4:15- 8:30 TODO Millas arbetstider                        :jobb::
Friday     13 April 2012
  test-new:   14:30-18:30 TODO Millas arbetstider                        :jobb::
Saturday   14 April 2012
Sunday     15 April 2012

================================================================================
Headlines with TAGS match: jobb
  test-new:   TODO Tider                                                  :jobb:
  test-new:   TODO Millas arbetstider                                    :jobb::

答案 1 :(得分:3)

在使用Jonathan的答案中的函数几个月来进行这种逐块过滤之后,我发现自己想要能够处理更复杂查询的东西(比如其他块类型使用的匹配字符串),而我想想我将在这里发布给任何在将来偶然发现这个问题的人。

编辑:my/org-match-at-point-p的原始实现现在有点过时了。组织模式源现在具有词法范围,这会更改org-make-tags-matcher的合同。过去,todotags-list变量需要围绕对org-make-tags-matcher的调用进行动态调整,而现在它们似乎被传递给调用返回的函数。 (是的!这好多了!)我已经调整了下面的代码以匹配新版本,但我不再使用Org模式了,所以它只是经过了轻微的测试。

(defun my/org-match-at-point-p (match)
  "Return non-nil if headline at point matches MATCH.
Here MATCH is a match string of the same format used by
`org-tags-view'."
  (funcall (cdr (org-make-tags-matcher match))
           (org-get-todo-state)
           (org-get-tags-at)
           (org-reduced-level (org-current-level))))

(defun my/org-agenda-skip-without-match (match)
  "Skip current headline unless it matches MATCH.

Return nil if headline containing point matches MATCH (which
should be a match string of the same format used by
`org-tags-view').  If headline does not match, return the
position of the next headline in current buffer.

Intended for use with `org-agenda-skip-function', where this will
skip exactly those headlines that do not match." 
  (save-excursion
    (unless (org-at-heading-p) (org-back-to-heading)) 
    (let ((next-headline (save-excursion
                           (or (outline-next-heading) (point-max)))))
      (if (my/org-match-at-point-p match) nil next-headline))))

如果您仍在使用较早版本的Org模式,则这是原始代码。请注意,此版本假定定义my/org-match-at-point-p的文件是词法范围的;如果没有,您可以使用简单的my/defun-dyn安全地替换defun宏。

(defmacro my/defun-dyn (&rest def)
  "As `defun', but always in dynamic scope.
A function defined in this way ignores the value of
`lexical-binding' and treats it as if it were nil.

\(fn NAME ARGLIST &optional DOCSTRING DECL &rest BODY)"
  (declare (debug defun)
           (indent defun)
           (doc-string 3))
  `(eval '(defun ,@def) nil))

(my/defun-dyn my/org-match-at-point-p (match &optional todo-only)
  "Return non-nil if headline at point matches MATCH.
Here MATCH is a match string of the same format used by
`org-tags-view'.

If the optional argument TODO-ONLY is non-nil, do not declare a
match unless headline at point is a todo item."
  (let ((todo      (org-get-todo-state))
        (tags-list (org-get-tags-at)))
    (eval (cdr (org-make-tags-matcher match)))))

答案 2 :(得分:-1)

可以通过在议程视图中单击“ /”来过滤Agenda Dispatcher

“ /” = 在所有议程文件中搜索正则表达式 在org-agenda-text-search-extra-files中列出的文件中。这使用 Emacs命令多次出现。前缀参数可用于 指定每个匹配项的上下文行数...