Skip to content

Richardsl/heatmap-calendar-obsidian

Repository files navigation

Heatmap Calendar plugin for Obsidian

Visualize your data in a heatmap calendar similar to the github activity calendar using this Obsidian plugin.

Useful for tracking progress for exercise, finances, social time, project progression, passions, vices etc.

To be used with Obsidian Dataview, but could be used standalone or with other plugins aswell (if you know some javascript).

Shows a black logo in light color mode and a white one in dark color mode.

Β 

Howto

  1. Annotate the data you want to track in your daily notes (see Dataview annotation documentation)

  2. Create a DataviewJS block where you want the Heatmap Calendar to display.

  3. Collect the data you want to display using DataviewJS

  4. Pass the data into Heatmap Calendar using renderHeatmapCalendar()

Β 

Visualized Concept: heatmap calendar example

Full Example Code:

\```dataviewjs // PS. remove backslash \ at the very beginning!

dv.span("** 😊 Title  πŸ˜₯**") /* optional βΉοΈπŸ’€βš‘βš πŸ§©β†‘β†“β³πŸ“”πŸ’ΎπŸ“πŸ“πŸ”„πŸ“πŸ”€βŒ¨οΈπŸ•ΈοΈπŸ“…πŸ”βœ¨ */
const calendarData = {
	year: 2022,  // (optional) defaults to current year
	colors: {    // (optional) defaults to green
		blue:        ["#8cb9ff", "#69a3ff", "#428bff", "#1872ff", "#0058e2"], // first entry is considered default if supplied
		green:       ["#c6e48b", "#7bc96f", "#49af5d", "#2e8840", "#196127"],
		red:         ["#ff9e82", "#ff7b55", "#ff4d1a", "#e73400", "#bd2a00"],
		orange:      ["#ffa244", "#fd7f00", "#dd6f00", "#bf6000", "#9b4e00"],
		pink:        ["#ff96cb", "#ff70b8", "#ff3a9d", "#ee0077", "#c30062"],
		orangeToRed: ["#ffdf04", "#ffbe04", "#ff9a03", "#ff6d02", "#ff2c01"]
	},
	showCurrentDayBorder: true, // (optional) defaults to true
	defaultEntryIntensity: 4,   // (optional) defaults to 4
	intensityScaleStart: 10,    // (optional) defaults to lowest value passed to entries.intensity
	intensityScaleEnd: 100,     // (optional) defaults to highest value passed to entries.intensity
	entries: [],                // (required) populated in the DataviewJS loop below
}

//DataviewJS loop
for (let page of dv.pages('"daily notes"').where(p => p.exercise)) {
	//dv.span("<br>" + page.file.name) // uncomment for troubleshooting
	calendarData.entries.push({
		date: page.file.name,     // (required) Format YYYY-MM-DD
		intensity: page.exercise, // (required) the data you want to track, will map color intensities automatically
		content: "πŸ‹οΈ",           // (optional) Add text to the date cell
		color: "orange",          // (optional) Reference from *calendarData.colors*. If no color is supplied; colors[0] is used
	})
}

renderHeatmapCalendar(this.container, calendarData)

```

Β 

Colors:

The heatmap uses a green color scheme by default, just like Github.

Default Color: green (no color specified)

heatmap calendar custom colors example

Β 

Custom Color

You can customize the colors of the heatmap by supplying a color array to calendarData.colors:

heatmap calendar custom colors example

Β 

More color options

Β 

Multi-Color:

You can use multiple colors to display different data-entries in the same heatmap. Specifying the name you gave the color in calendarData.colors (eg. "blue", "pink" etc).

heatmap calendar custom colors example

Styling Background (empty days):

Use Obsidian's built in "CSS snippets" for custom styling including styling the empty days (aka the background cells).

But remember this will affect all of you heatmaps in all of your notes. heatmap calendar custom colors example

heatmap calendar custom colors example

Global color schemes via settings:

You can also add a color scheme via the Settings panel. This scheme which will be available everywhere.

In order to do so go to Obsidian Settings > Heatmap Calendar, you will see a list of available colors, and you can add your own. You must specify a β€œColor name” by which you will reference it in your render call, and provide a valid array of colors.

When you do so, you can now reference your scheme everywhere by passing your name to the colors option. For example, let's say you have defined a new color called githubGreen. Now, in your code, you can reference it like so:

```dataviewjs
const calendarData = {
	colors: "githubGreen",
	entries: [],
}

renderHeatmapCalendar(this.container, calendarData)
```

Β 

Β 

The color schemes used in the examples were created at leonardocolor.io.


Β 

Data Intensity:

Set which intensity of color to use (eg. from light-green to dark-green etc).

heatmap calendar custom colors example

More
They color-range will be distributed between the highest and lowest number you pass to "intensity".

If the number range 0-100 is used, numbers between 1-20 would map to the lightest color, 40-60 would map to mid intensity color, and 80-100 would map to max intensity. You can add more intensities in order to increase color resolution; simply supply more colors to calendarData.colors.yourcolor

Dataview's time variables are supported without any conversion, as they return milliseconds by default.
[time:: 1 hours, 35 minutes] => intensity: page.time

Β 


Other Notes:

Β 

Development (Windows):

npm run dev - will start an automatic TS to JS transpiler and automatically copy the generated JS/CSS/manifest files to the example vault when modified (Remember to run npm install first).

After the files have been transpiled, the hot-reload plugin (https://github.com/pjeby/hot-reload) then reloads Obsidian automatically. Hot-reload is installed in the example vault by default. its used to avoid restarting obsidian after every change to code.
(remember to add an empty .hotreload file to "EXAMPLE_VAULT/.obsidian/plugins/heatmap-calendar/" if not already present, as this tells hot-reload to watch for changes)

npm run build generates the files ready for distribution.

Β 

Tip: ctrl-shift-i opens the devtools inside Obsidian.

Β 

Technical Explanation

All the plugin does, is add the function renderHeatmapCalendar() to the global namespace of you vault.

"this.container" is passed as the first argument because the plugin needs to know where to render the calendar. You don't have to worry about this.

"renderHeatmapCalendar()" then takes "calendarData" as the secondary argument. This is the javascript object you have to create yourself in order to give plugin instructions and data. Most of the properties are optional, but you have to supply an entries array as an absolute minimum.

See the beginning of the readme for the full code example.

absolute minimum code example:

\```dataviewjs

const calendarData = {
    entries: [],                
}

renderHeatmapCalendar(this.container, calendarData)

```

Β 

Β 

What's New:


Version [0.7.0] - 2024-06-04

New "First day of the week" setting, Thanks @antosha417
New setting for changing the first day of the week, ie. Sunday/Monday.

For the American users :-)

heatmap calendar custom colors example

Thanks @antosha417



Alternating Month Styling, Thanks @lksrpp
The months now how their own classes so that they can be styled individually.

The example CSS Snippet below can be found in the EXAMPLE_VAULT in the "./obsidian/snippets" folder:

heatmap calendar custom colors example

Thanks @lksrpp



Tighter Heatmap Margins/paddings, Thanks @Chailotl
New setting for

heatmap calendar custom colors example

heatmap calendar custom colors example

Thanks @Chailotl




Version [0.6.0] - 2023-04-12

  • Feature: Add ability to define global colors via settings @sunyatasattva pull #74
  • Feature: Add more versatile custom styling of the "content" passed to date cell @sunyatasattva pull #73

Version [0.5.0] - 2022-06-30

  • Feature: Add darkmode support

Version [0.4.0] - 2022-06-25

  • Feature: Add hover preview feature courtesy of @arsenty from issue #12.
    to enable - add content: await dv.span([](${page.file.name})) to entries, and enable Settings -> Core Plugins -> Page Preview.
    Optionally install plugin Metatable to display metadata/frontmatter in the preview window aswell.
    See examples for more details. Note: if you enabled Use [[Wikilinks]] under Settings -> Files and links, you have to use the respective link structure: content: await dv.span([[${page.file.name}|]])

Version [0.3.0] - 2022-06-25

  • Feature: Can add more intensities in order to increase color resolution. simply supply more colors to calendarData.colors.yourcolor
  • Feature: Can set custom range on the intensity scaling using intensityScaleStart and intensityScaleEnd
  • Bugfix: Entries from other years would show up in the calendar

Version [0.2.0] - 2022-06-05

  • Feature: Add border around todays box to indicate what day it is. Can be removed by setting showCurrentDayBorder to false
  • Feature: Add better development solution/workflow by using automated file copying instead of symlinks

Version [0.1.1] - 2022-03-18

  • Bugfix: fix major date problem where year would render with incorrect number of days for different timezones issue#4.
  • Bugfix: fix problem with certain entries not showing up in the correct month
  • Bugfix: fix grid cells not scaling correctly with browser width, especially content in grid cells

Version [0.1.0] - 2022-02-23

  • initial release