Skip to content

d12frosted/vulpea-journal

Repository files navigation

vulpea-journal

version

A daily journaling interface for vulpea that integrates seamlessly with vulpea-ui sidebar.

What is vulpea-journal?

vulpea-journal brings the power of daily journaling to your vulpea-based note system. Think of it as org-journal rebuilt from the ground up for vulpea, with a modern reactive UI.

Key features:

  • Daily notes - One file per day, automatically created with customizable templates
  • Sidebar widgets - Calendar, navigation, related notes - all in the vulpea-ui sidebar
  • Calendar integration - See which days have entries, jump to any date
  • Previous years - “On this day” view showing what you wrote in past years
  • Zero window management - Uses vulpea-ui sidebar, no custom layouts to break

Quick Start

1. Install

vulpea-journal requires:

;; With use-package and straight.el
(use-package vulpea-journal
  :straight (:host github :repo "d12frosted/vulpea-journal")
  :after (vulpea vulpea-ui)
  :config
  (vulpea-journal-setup))

2. Widgets are auto-registered

Journal widgets automatically register with vulpea-ui when you load vulpea-journal-ui. They only appear when viewing journal notes.

Default widget order interleaves with vulpea-ui widgets:

WidgetOrderAppears…
journal-nav50Before stats
stats100(vulpea-ui)
journal-calendar150After stats
outline200(vulpea-ui)
backlinks300(vulpea-ui)
created-today350After backlinks
previous-years360After created-today
links400(vulpea-ui)

To customize the order, see Widget Order Configuration below.

3. Add a keybinding

(global-set-key (kbd "C-c j") #'vulpea-journal)

4. Start journaling!

Press C-c j to open today’s journal. The sidebar will show journal-specific widgets.

How It Works

Journal Notes

Journal notes are regular vulpea notes identified by a tag (default: "journal"). Each note has:

  • A file path based on the date (e.g., journal/2025-12-08.org)
  • A title based on the date (e.g., 2025-12-08 Sunday)
  • A CREATED property storing the date for querying

When you call vulpea-journal, it:

  1. Finds or creates the note for that date
  2. Opens it in your main window
  3. Shows the vulpea-ui sidebar with journal widgets

Sidebar Integration

vulpea-journal doesn’t create its own windows. Instead, it provides widgets that plug into vulpea-ui’s sidebar system. This means:

  • No window management bugs
  • Consistent UI with the rest of vulpea-ui
  • Widgets automatically appear/disappear based on what you’re viewing

Journal widgets check if the current note is a journal entry. When you view a non-journal note, they simply hide themselves.

Configuration

Template

Customize how journal notes are created:

(setq vulpea-journal-default-template
      '(:file-name "journal/%Y-%m-%d.org"    ; File path (strftime format)
        :title "%Y-%m-%d %A"                  ; Note title (strftime format)
        :tags ("journal")                     ; Tags (first one identifies journals)
        :head "#+created: %<[%Y-%m-%d]>"      ; Header content
        :body "* Morning\n\n* Evening\n"))    ; Initial body

Important: The :file-name and :title use strftime format because they’re expanded for the target date, not the current time. When you open the journal for December 25th, the file will be journal/2025-12-25.org regardless of today’s date.

Other keys (:head, :body) use vulpea’s %<format> syntax and are expanded at creation time.

Dynamic Templates

For more control, use a function:

(setq vulpea-journal-default-template
      (lambda (date)
        (let ((weekday (format-time-string "%u" date)))
          (list
           :file-name "journal/%Y-%m-%d.org"
           :title (format-time-string "%Y-%m-%d %A" date)
           :tags '("journal")
           :head "#+created: %<[%Y-%m-%d]>"
           :body (if (member weekday '("6" "7"))
                     "* Weekend\n\n"
                   "* Work\n\n* Personal\n")))))

Calendar Widget

;; Start week on Sunday (0) or Monday (1, default)
(setq vulpea-journal-ui-calendar-week-start 1)

Created Today Widget

;; Include journal notes in the "created today" list
(setq vulpea-journal-ui-created-today-exclude-journal nil)  ; default: t

Previous Years Widget

;; How many years to look back
(setq vulpea-journal-ui-previous-years-count 5)  ; default: 5

;; Characters to show in preview
(setq vulpea-journal-ui-previous-years-preview-chars 256)

;; Hide org drawers in preview
(setq vulpea-journal-ui-previous-years-hide-drawers t)  ; default: t

;; Start with previews expanded
(setq vulpea-journal-ui-previous-years-expanded t)  ; default: t

Widget Order Configuration

Customize where journal widgets appear relative to vulpea-ui widgets:

(setq vulpea-journal-ui-widget-orders
      '((nav . 50)              ; before stats (100)
        (calendar . 150)        ; after stats, before outline (200)
        (created-today . 350)   ; after backlinks (300)
        (previous-years . 360)))

Example: Move calendar before stats:

(use-package vulpea-journal
  :custom
  (vulpea-journal-ui-widget-orders
   '((nav . 50)
     (calendar . 90)            ; now before stats
     (created-today . 350)
     (previous-years . 360))))

Reference orders for vulpea-ui widgets: stats=100, outline=200, backlinks=300, links=400.

Commands

CommandDescription
vulpea-journalOpen today’s journal (or specify date programmatically)
vulpea-journal-todayOpen today’s journal
vulpea-journal-datePrompt for a date and open its journal
vulpea-journal-nextGo to next journal entry
vulpea-journal-previousGo to previous journal entry
vulpea-journal-setupEnable calendar/sidebar integration and register widgets

Sidebar Keybindings

When viewing a journal note, the sidebar provides quick navigation:

KeyAction
[Go to previous journal entry
]Go to next journal entry
tGo to today’s journal
dPick a date

Calendar Integration

After calling vulpea-journal-setup, the Emacs calendar gains journal superpowers via vulpea-journal-calendar-mode:

KeyAction
jOpen journal for date at point
]Jump to next journal entry
[Jump to previous journal entry

Days with journal entries are highlighted in the calendar.

The minor mode is automatically enabled in calendar buffers. You can toggle it with M-x vulpea-journal-calendar-mode if needed.

Widgets Reference

vulpea-journal-widget-nav

Shows the current journal date and navigation buttons:

images/vulpea-journal-ui-navigation.png

Clicking Prev/Next navigates by one day (creating the entry if needed). Clicking Today jumps to today’s journal.

vulpea-journal-widget-calendar

Interactive month calendar:

images/vulpea-journal-ui-calendar.png

  • Bold = today
  • Highlighted = selected date
  • Dot (·) = has journal entry
  • Click any date to open its journal

Use < and > to navigate months without changing the selected date.

vulpea-journal-widget-created-today

Lists all notes created on the journal’s date (from the CREATED property):

images/vulpea-journal-ui-created-today.png

Click a note to visit it. Times come from the CREATED property timestamp.

vulpea-journal-widget-previous-years

Shows journal entries from the same date in previous years:

images/vulpea-journal-ui-previous-years.png

Click the date to visit that journal. Click ▸=/=▾ to expand/collapse the preview.

Tips & Tricks

Open journal on Emacs startup

(add-hook 'emacs-startup-hook #'vulpea-journal)

Weekly review workflow

Use vulpea-journal-dates-in-range to query entries:

(defun my/journal-this-week ()
  "Get all journal dates from this week."
  (let* ((today (current-time))
         (dow (string-to-number (format-time-string "%u" today)))
         (start (time-subtract today (days-to-time (1- dow))))
         (end (time-add start (days-to-time 7))))
    (vulpea-journal-dates-in-range start end)))

Custom journal tag

If you want a different tag than "journal":

(setq vulpea-journal-default-template
      '(:file-name "daily/%Y-%m-%d.org"
        :title "%Y-%m-%d %A"
        :tags ("daily-note")  ; First tag is used for identification
        :head "#+created: %<[%Y-%m-%d]>"))

Troubleshooting

Widgets don’t appear in sidebar

  1. Ensure vulpea-journal-ui is loaded (widgets auto-register on load)
  2. Check that you’re viewing a journal note (has the journal tag)
  3. Try M-x vulpea-ui-sidebar-refresh

Date extraction fails

vulpea-journal extracts dates from the CREATED property. Ensure your template includes:

:head "#+created: %<[%Y-%m-%d]>"

Supported formats:

  • [2025-12-08]
  • [2025-12-08 08:54]
  • 2025-12-08

Calendar marks don’t appear

Ensure vulpea-journal-setup is called in your config. This adds the necessary hooks:

(add-hook 'calendar-today-visible-hook #'vulpea-journal-calendar-mark-entries)
(add-hook 'calendar-today-invisible-hook #'vulpea-journal-calendar-mark-entries)

Calendar keybindings don’t work

vulpea-journal-setup enables vulpea-journal-calendar-mode in calendar buffers. If keybindings aren’t working:

  1. Verify the minor mode is active by running M-x vulpea-journal-calendar-mode to toggle it
  2. Ensure vulpea-journal-setup was called before opening the calendar

Existing journal files not in database

If you have existing journal files that are not properly indexed in the vulpea database (e.g., manually created files, or files without an :ID: property), vulpea-journal will show an error when you try to navigate to that date. This is a safety measure to prevent accidentally overwriting your files.

To fix this, you have two options:

  1. Add proper properties and sync: Ensure each file has an :ID: property in its property drawer, then run M-x vulpea-db-sync to index them.
  2. Delete and recreate: If the files don’t contain important content, delete them and let vulpea-journal create new ones.

Additionally, if your notes lack the CREATED property, vulpea-journal won’t find them for date-based queries (like “notes created today”). The note itself will work fine if opened directly.

Migration script

If you have many existing journal files that need :ID: and CREATED properties, you can use this script to migrate them automatically.

WARNING: Back up your notes before running this script!

(defun vulpea-journal-migrate-files (directory)
  "Migrate journal files in DIRECTORY to vulpea-journal format.

Adds :ID: property if missing.
Adds CREATED property if missing (inferred from filename or title).

After running this, execute `vulpea-db-sync' to index the files."
  (interactive "DJournal directory: ")
  (let* ((files (directory-files directory t "\\.org$"))
         (count (length files))
         (modified 0)
         (skipped 0))
    (message "Processing %d files in %s..." count directory)
    (dolist (file files)
      (message "  Processing %s..." (file-name-nondirectory file))
      (with-current-buffer (find-file-noselect file)
        (let ((changed nil)
              (filename (file-name-base file)))
          ;; Check/add ID property
          (goto-char (point-min))
          (unless (org-entry-get (point) "ID")
            (org-id-get-create)
            (setq changed t)
            (message "    Added ID"))
          ;; Check/add CREATED property
          (goto-char (point-min))
          (unless (org-entry-get (point) "CREATED")
            (when-let* ((date (vulpea-journal-migrate--infer-date file)))
              (org-set-property "CREATED" date)
              (setq changed t)
              (message "    Added CREATED: %s" date)))
          ;; Save if modified
          (if changed
              (progn
                (save-buffer)
                (setq modified (1+ modified)))
            (setq skipped (1+ skipped)))
          (kill-buffer))))
    (message "Migration complete: %d modified, %d skipped" modified skipped)
    (message "Run M-x vulpea-db-sync to index the migrated files.")))

(defun vulpea-journal-migrate--infer-date (file)
  "Infer date from FILE path or title.
Returns date string in format [YYYY-MM-DD] or nil."
  (let ((filename (file-name-base file)))
    (cond
     ;; Try common filename patterns: YYYYMMDD, YYYY-MM-DD, YYYY_MM_DD
     ((string-match "\\([0-9]\\{4\\}\\)[_-]?\\([0-9]\\{2\\}\\)[_-]?\\([0-9]\\{2\\}\\)" filename)
      (format "[%s-%s-%s]"
              (match-string 1 filename)
              (match-string 2 filename)
              (match-string 3 filename)))
     ;; Try reading title from file
     (t
      (with-temp-buffer
        (insert-file-contents file)
        (goto-char (point-min))
        (when (re-search-forward "^#\\+title:[ \t]*\\(.+\\)" nil t)
          (let ((title (match-string 1)))
            (when (string-match "\\([0-9]\\{4\\}\\)[_-]?\\([0-9]\\{2\\}\\)[_-]?\\([0-9]\\{2\\}\\)" title)
              (format "[%s-%s-%s]"
                      (match-string 1 title)
                      (match-string 2 title)
                      (match-string 3 title))))))))))

Usage:

  1. Back up your journal directory
  2. Evaluate the code above
  3. Run M-x vulpea-journal-migrate-files and select your journal directory
  4. Run M-x vulpea-db-sync to index the migrated files

API Reference

Functions

;; Note identification
(vulpea-journal-note-p note)        ; Is NOTE a journal note?
(vulpea-journal-note-date note)     ; Extract date from journal NOTE

;; Note retrieval
(vulpea-journal-note date)          ; Get/create journal for DATE
(vulpea-journal-find-note date)     ; Find existing journal (no create)

;; Queries
(vulpea-journal-all-dates)          ; All dates with journals
(vulpea-journal-dates-in-month m y) ; Journals in month M of year Y
(vulpea-journal-dates-in-range a b) ; Journals between dates A and B
(vulpea-journal-notes-for-date-across-years date n)  ; Same date in past N years

Faces

  • vulpea-journal-calendar-entry-face - Calendar days with entries
  • vulpea-journal-ui-widget-title - Widget headers
  • vulpea-journal-ui-calendar-date - Regular calendar days
  • vulpea-journal-ui-calendar-today - Today in calendar
  • vulpea-journal-ui-calendar-entry - Days with entries in widget
  • vulpea-journal-ui-calendar-selected - Selected day in widget

Contributing

Contributions welcome! Please open issues and PRs on GitHub.

License

GPLv3. See LICENSE for details.

About

Modern daily journaling interface for Emacs with reactive sidebar widgets, built on vulpea

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published