🕓 Event Helpers
Commands to extend NotePlan's events handling, including calendar lists, time blocks, and date offset templating. See link for more details, and configuration.
🕓 Event Helpers plugin
This plugin provides commands to help you do useful things with Events and Calendars that can't (yet) be done by NotePlan itself:
- insert day's events as list: insert a list of this day's calendar events into the current note -- including doing so automatically in day/week note templates
- insert matching events: insert a list of this day's calendar events that match certain patterns into the current note
- time blocks to calendar: takes NotePlan-defined time blocks and converts to them to full Calendar events in your current default calendar, as set by iCal. (See also Display of Time Blocks below.)
- process date offsets: finds date offset patterns and turns them into due dates, based on date at start of section. (See Date Offsets below for full details.)
- shift dates: takes dates in the selected lines and shifts them forwards or backwards by a given date interval. It works on both
>YYYY-MM-DD
and>YYYY-Wnn
style dates. (User George Crump (@george65) has created a video showing how this command works.)
Most of these commands require configuration, described in the sections below. On macOS, click the gear button on the 'Event Helpers' line in the Plugin Preferences panel to access the settings. Or on on iOS/iPadOS devices, use the Events: update plugin settings command instead.
This plugin is a labour of love designed to strengthen the NotePlan product and grow its community. To support my late-night coding you might like to buy me a coffee!
"insert day's events as list" and "insert matching events" commands
Settings:
-
Events heading: in
/insert today's events as list
command (orevents()
template call), the heading to put before the list of the day's events. Optional. -
Events List display format: the format string to use to customise how events are displayed (when run as a /command, not through a Template). The available placeholders are
CAL
,TITLE
,LOCATION
,EVENTLINK
,DATE
,START
,END
,NOTES
,ATTENDEES
,NOTES
,URL
,MEETINGNOTE
. Most of these are self-explanatory for events in most types of calendars, other than:ATTENDEES
gives the full details of event attendees provided by the operating system, and can be a mix of names, email addresses, and other details;ATTENDEENAMES
just gives the name of event attendees, or if that's missing, just the email address;DATE
is formatted using the locale settings from your operating system, unless you override that with the 'Shared Settings > Locale' setting;EVENTLINK
is specific to NotePlan: it will make a nicely-formatted link to the actual calendar event, and clicking on it will show a pop with all the event's details.MEETINGNOTE
will insert a link that when clicked, will help you create a meeting note for this particular event. For this there's a setting 'Meeting Note Template title' which you can use to set which template to pick if you have several; if it isn't set then a list will be presented.START
andEND
are the start and end times of the event. (The format of these can be controlled -- see 'Time options' below.)
Note: each placeholder has to be surrounded by
*|...|*
to distinguish them as placeholders and not other markdown text to include.Default:
- (*|CAL, |**|START|*) *|TITLE|**|\nEVENTLINK|**|\nwith ATTENDEES|**|\nNOTES|*
-
Events List display format for all-day events: the format string to use to customise how all-day events are displayed (when run as a /command, not through a Template). The available placeholders are as above, with the exception of
START
andEND
.Default:
- (*|CAL, |**|START|*) *|TITLE|**|\nEVENTLINK|**|\nwith ATTENDEES|**|\nNOTES|*
-
Sort order of events list: by 'time' (= increasing start time of events) or by 'calendar' first, which then secondarily sorts by start time.
-
Calendars to include: optional ["array","of calendar","names"] to filter by when showing list of events. If empty or missing, no filtering will be done.
-
Calendar name mappings: optional - add mappings for your calendar names to appear as in the output - e.g. from "Jonathan (iCloud)" to "Me" (and from "Us (iCloud)" to "Us") with
Jonathan (iCloud);Me, Us (iCloud);Us
. Note: separate mapping from main name by;
a character, and separate mapping pairs with the,
character. -
Meeting Note Template title: use to set which template to pick if you have several; if it isn't set then a list of meeting note templates will be offered.
-
Matching Events heading: in
/insert matching events
command (orlistMatchingEvents()
template call), the heading to put before the list of matching events -
Events match list: for
/add matching events
is an array of pairs of strings. The first string is what is matched for in an event's title. If it does match, the second string is used as the format for how to insert the event details at the cursor. This uses the same placeholders as above. For example:{ "#meeting" : "### *|TITLE|* (*|START|*)\nWith *|ATTENDEES|**|\n NOTES|**|\nEVENTLINK|*", "holiday" : "*|TITLE|*\nHoliday:: *|NOTES|*" }
You can also add the special
*|STOPMATCHING|*
placeholder which will mean only the first match in this list is applied for a given event. -
Stop after first match in the list above?: If true, only the first match in the list above is used for a given event. (Note: this doesn't stop matching the rest of the events in the Calendars.) This is the equivalent of setting 'STOPMATCHING' on every item in the above list.
Unfortunately, the NP plugin settings can change the order of the items in this list without warning. So to use this successfully, you may need to manually edit the settings file, which is found at
<NotePlan root>/Plugins/jgclark.EventHelpers/settings.json
. The default isfalse
. -
Include time blocks from completed tasks?: whether to include time blocks from lines with completed tasks.
-
Name of Calendar to write to: the name of the calendar for
/time blocks to calendar
to write events to. Must be a writable calendar. If empty, then the default system calendar will be used. (Note: you have to specifically set a default calendar in the settings of the macOS Calendar app or in iOS Settings app > Calendar > Default Calendar.) -
Default event duration: Event duration (in minutes) to use when making an event from a time block, if no end time is given.
-
Confirm Event Creation?: optional boolean tag to indicate whether to ask user to confirm each event to be created
-
Remove time blocks when processed?: in
time blocks...
whether to remove time block after making an event from it -
Add event link?: whether to add a nicely-formatted event link when creating an event from a time block. (This can return rather long strings (e.g.
⏰event:287B39C1-4D0A-46DC-BD72-84D79167EFDF
) and so you might want to use a theme option to shorten them until needed (details below).) -
Processed tag name: if this is set, then this tag will get added on the end of the line with the time block, to show that it has been processed. Otherwise, next time this command is run, it will create another event. This can be used with or without addEventID.
-
Locale: optional Locale to use for times in events. If not given, will default to what the OS reports, or failing that, 'en-US'.
-
Time options: Optional Time format settings.
Default:
{\n\t\"hour\": \"2-digit\", \n\t\"minute\": \"2-digit\", \n\t\"hour12\": false\n}
Using Event Lists from a Template
You can use these commands from Templates, when they are applied to a note, or are used to create one:
<%- events(...) %>
produces a list of all events for the period wherever you wish it to appear in the Template. By default it gives a simple markdown list of event title and start time for today, though this can all be customised.<%- matchingEvents(...) %>
produces a list of all matching events. This is more powerful, with each configured match able to have a different format of output. The matches and format strings are entered in a JSON-formatted array, which can only be specified in the Plugin's settings.
These work particularly well in Daily Note templates (or in Weekly Note templates with daysToCover: 7
-- see below).
You can customise the output by adding parameters to the commands. Note: these need to be comma separated, the values enclosed in double quotes, and all surrounded by curly quotes. (Sorry: I didn't design this syntax ...) For example:
<%- events( {format:"### *|CAL|*: *|TITLE|* (*|START|*-*|END|*)*|\nEVENTLINK|**|with ATTENDEES|**|\nLOCATION|**|\nNOTES|*", allday_format:"### *|TITLE|**|\nEVENTLINK|**|\nNOTES|*", includeHeadings:false, calendars:"home,children"} ) %>
The following Parameters are available:
name | type | description | example |
---|---|---|---|
includeHeadings | boolean | adds heading "Events" | includeHeadings: false |
includeAllDayEvents | boolean | include/exclude all day events | includeAllDayEvents: false |
calendarSet | string | limit which calendars are included | calendarSet:"list,of,calendar,names" |
calendarNameMappings | string | customize the name of the calendars | calendarNameMappings:"Jonathan (iCloud);Me, Us (iCloud);Us" |
daysToCover | number | include more than the current day | daysToCover: 3 includes today + 2 days |
format | string | customize the format | 'format:"..."' |
allday_format | string | customize format for all day events | 'allday_format:"..." |
format
and allday_format
use the same placeholders (surrounded by *|...|*
) as above, and can be mixed with any markdown characters or other text and they will get replaced accordingly with the fields from each matching event found.
You can include other text (including line breaks indicated by \n
) within the placeholder. For example in *|\nwith ATTENDEENAMES|*
, if the ATTENDEENAMES is not empty, then it will output the attendees list on a newline and after the text 'with '.
NB: the Sort order
setting above also controls how the output of this list is sorted.
/shift dates
This command takes plain or scheduled day or week dates (i.e. YYYY-MM-DD
, >YYYY-MM-DD
, YYYY-Wnn
or ``>YYYY-Wnn`) in the selected lines and shifts them forwards or backwards by a given date interval. This allows you to copy a set of tasks to use again, and have the dates moved forward by a month or year etc.
Its settings are:
- Remove @done dates? Whether to remove
@done(...)
dates; by default it will. (If you don't remove such dates, then they will also get shifted.) - Set any finished (i.e. completed or cancelled) tasks to open? Default: true.
- Remove any 'processed tag name' on tasks or checklists? Whether to remove any 'processed tag name' (from the settings for "/time blocks to calendar" command above) from tasks or checklists. Default: true.
Note: if the line contains a blockID (link to another line), this will be removed before the date is shifted.
/process date offsets
User George Crump (@george65) has created a video showing how this command works.
The command is best understood with some examples. Here's some Christmas planning:
This ... | ... becomes this |
---|---|
### Christmas Cards 2021-12-25 * Write cards {-20d} * Post overseas cards {-15d} * Post cards to this country {-10d} * Store spare cards for next year {+3d} | ### Christmas Cards 2021-12-25 * Write cards >2021-12-05 * Post overseas cards >2021-12-10 * Post cards to this country >2021-12-15 * Store spare cards for next year >2021-12-28 |
* Bob's birthday on 2021-09-14 * Find present {-6d} * Wrap & post present {-3d} * Call Bob {0d} | * Bob's birthday on 2021-09-14 * Find present >2021-09-08 * Wrap & post present >2021-09-11 * Call Bob >2021-09-14 |
You can use this within a line to have both a deadline and a calculated start date:
This ... | ... becomes this |
---|---|
* Post cards deadline 2021-12-18 {-10d} | * Post cards deadline 2021-12-18 >2021-12-08 |
The /process date offsets
command looks for a valid base date in the previous heading, previous main task if it has sub-tasks, or in the line itself. If it does, then it changes any date offset patterns (such as {-10d}
, {+2w}
, {-3m}
) into dates (e.g. 2022-05-02
). This is particularly useful when set up in templates, that when applied to a note, sets the due date at the start, and calculates the other dates for you.
In more detail:
- The base date is by default of the form
YYYY-MM-DD
, not preceded by characters0-9(<
, all of which could confuse. - Valid date offsets are specified as
[^][+/-][0-9][bdwmqyBDWMQY]
. This allows forb
usiness days,d
ays,w
eeks,m
onths,q
uarters ory
ears. (Business days skip weekends. If the existing date happens to be on a weekend, it's treated as being the next working day. Public holidays aren't accounted for.) You can use upper- or lower-case letters.-{+3d}
means three days after the 'base' date {-3d}
means three days before the 'base' date- You can use
{0d}
to mean no offset -- i.e. on the day itself. - An offset that starts with
{^...}
calculates before or after the last calculated date (not the base date). An example of this is:
For example ... | ... becomes |
---|---|
* Bob's birthday on 2021-09-05 * Find present {^+6d} * Wrap & post present {^+3d} * Call Bob {0d} | * Bob's birthday on 2021-09-05 to 2021-09-14 * Find present >2021-09-08 * Wrap & post present >2021-09-11 * Call Bob >2021-09-14 |
### Easter Preparations >2022-03-01 * Use up sweet treats (Shrove Tuesday) {0d} * Start Lent (Ash Wednesday) {^+1d} * End of Lent {^+-6w} * Remember Last Supper {^+1d} * Good Friday {^+1d} * Easter Sunday {^+2d} | ### Easter Preparations >2022-03-01 to >2022-04-17 * Use up sweet treats (Shrove Tuesday) >2022-03-01 * Start Lent (Ash Wednesday) >2022-03-02 * End of Lent >2022-04-13 * Remember Last Supper >2022-04-14 * Good Friday >2022-04-15 * Easter Sunday >2022-04-17 |
If a base date can't be found, the command will ask you to supply a date.
Note: any blockIDs (links to another line) are removed before the offset is calculated.
Display of Time Blocks
If you're using the time blocks to calendar command with a format that includes the START and END times, then it's likely that the NotePlan will still see a time block for the text of the event, and so in the calendar area show the event and a time block for it. To avoid this you can either
- put brackets around the START and END times
- or use set something in the 'text must contain' string in NotePlan's 'Todo' preferences pane.
NotePlan allows extensive customisation of fonts and colours through its Themes. It also supports doing more advanced highlighting using regex. To add colour highlighting for time blocks, add the following to your favourite theme's .json file:
"timeblocks": {
"regex": "(?:^\\s*(?:\\*(?!\\s+\\[[\\-\\>]\\])\\s+|\\-(?!\\h+\\[[\\-\\>]\\]\\s+)|[\\d+]\\.|\\#{1,5}\\s+))(?:\\[\\s\\]\\s+)?.*?\\s(((at|from)\\s+([0-2]?\\d|noon|midnight)(:[0-5]\\d)?(\\s?(AM|am|PM|pm)?)(\\s*(\\-|\\–|\\~|\\〜|to)\\s*([0-2]?\\d)(:[0-5]\\d)?(\\s*(AM|am|PM|pm)?))?|([0-2]?\\d|noon|midnight)(:[0-5]\\d)\\s*(AM|am|PM|pm)?(\\s*(\\-|\\–|\\~|\\〜|to)\\s*([0-2]?\\d|noon|midnight)(:[0-5]\\d)?(\\s*(AM|am|PM|pm)?)?)?))(?=\\s|$)",
"matchPosition": 1,
"color": "#CC4999"
}
If you're adding event IDs through the /time blocks to calendar
command, then you might want to hide the long event:ID
string until you move the cursor to the ⏰ marker. To do this add the following:
"eventID": {
"regex": "(event:[A-F0-9-]{36,37})",
"matchPosition": 1,
"color": "#444444",
"isHiddenWithoutCursor": true,
"isRevealOnCursorRange": true
}
Support
If you find an issue with this plugin, or would like to suggest new features for it, please raise a Bug or Feature 'Issue'.
History
See CHANGELOG for the plugin's history.