diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..e93f73b
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,3 @@
+
+.repository.json
+swiftbar-plugins.code-workspace
diff --git a/NotePlan3.15m.rb b/NotePlan3.15m.rb
new file mode 100644
index 0000000..31bfd8f
--- /dev/null
+++ b/NotePlan3.15m.rb
@@ -0,0 +1,346 @@
+#!/usr/bin/env ruby
+# coding: utf-8
+
+# Todo Today for NotePlan v3
+# v1.0
+# Jonathan Clark
+# jgclark
+# A todo list taken from NotePlan v3 and displayed with customizable color-code. Mark tasks "done" simply by clicking on them in the menubar drop-down list. This was based on "Todo.NotePlan" by Richard Guay which in turn was based on "Todo Colour" plugin by Srdgh.
+# ruby
+#
+# https://noteplan.co/
+
+# true
+
+# bitbar documentation: https://github.com/matryer/xbar-plugins/blob/main/CONTRIBUTING.md
+# brief swiftbar documentation: https://github.com/swiftbar/SwiftBar
+# Main detail is:
+# Script output for both header and body is split by line (\n). Each line must follow this format: - | [param = ...]
+# Where:
+# - "Item Title" can be any string, this will be used as a menu item title.
+# - [param = ...] is an optional set of parameters\modificators. Each parameter is a key-value separated by =. Use | to separate parameters from the title.
+
+# Modifications by Jonathan Clark
+# 2022/09/15:
+# - add support for weekly notes as well
+# - remove logic that stops reading notes at first blank line
+# - now only print H1 and H2 headers
+# - now refreshes plugin after clicking on a task in the list
+# - cleanup code
+# 2020/10/30:
+# - Update NP data storage filepaths for NotePlan 3 beta
+# (including CloudKit change at v3.0.15 beta)
+# - Make CloudKit location the default
+# - tweak colours and flags to suit my needs
+# - ignore tasks with dates scheduled into the future
+# - improve some non-tasks it was including
+# - code clean up
+# 2020/11/29:
+# - auto-detect storage type (CloudKit > iCloud Drive > Drobpox if there are multiple)
+# - add option to specify the file extension in use (default to md, but can be txt)
+#
+# Modifications by Guillaume Barrette
+# 2017/07/01:
+# - Added option to show subtasks
+# 2017/06/15:
+# - Changed TRUE/FALSE constant to true/false since uppercase are deprecated in ruby 2.4
+# - Changed labels to start with '#' to follow NotePlan way of tagging
+# - Allow to change Fonts by the user
+# - Added a new parameter for users to specify if want the task to be archived
+# at the end of the file or not
+# - Added alternate action to mark as cancelled instead of done (using the
+# Option modifier key)
+# - Allow indentation at beginning of task
+# 2017/06/03:
+# - Added 'divide_with_header' to allow to show sections separated by headers
+# - Updated the algorithm to skip all items that are not a task (Skip anything that
+# doesn't starts with '- ' or '* ' and if followed by [x], [>], [-])
+# 2017/05/28:
+# - Fixed the line number of item to mark as done by getting the id before stripping
+# the lines that are not a task
+# - Scheduled task (to another day - [>]) are now skipped also
+# 2017/05/20:
+# - Added Black and White NotePlan menubar icon
+# - Repaired a bug when there was no newline on the last line the done task would
+# get appended to the last line instead of a new line at the end
+# - Added the time in the @done(YYYY-MM-DD HH:MM) so it's like NotePlan preference
+# - Added User Parameters so it's easy to determine if we want to append the
+# @done(...) string at the end of the done task and if we want the black or white
+# menubar icon
+# - Changed the menubar icon to a templateImage so the color changes automatically
+# when using a dark menubar (removed the white icon)
+# - Removed 'use_black_icon' parameters since now it's automatic
+# - Changed encoding method and removed the use of 'force_encoding("utf-8")'
+# - Repaired a bug if there was no file already created for that day in NotePlan
+#
+# Modifications by Richard Guay
+# 2017/05/20:
+# - Added using emoji option
+# - fixed character encoding on removing an item
+# - Proper parsing of [ ] in the todo.
+# - cleanup
+require 'date'
+
+#################################
+# User Parameters:
+insert_date_on_done_task = true # If true, the date will be inserted with the @done tag
+use_emoji_as_icon = false # If true, will show emoji, otherwise it will use the black or white icon.
+use_star = true # if true, will look for and use '*' instead of '-'
+show_alt_task = false # If true, tasks marked with the alternate character ('* ' if use_star is FALSE or '- ' if use_star is TRUE) will be shown in the task list. For example, this could be useful to use them as bullet list.
+show_subtasks = true # If true, subtasks are shown
+divide_with_header = true # If true, headers are shown
+archive_task_at_end = false # If true, the task will get archived to the end of the note
+file_extension = '.md' # Defaults to file extension type 'md' -- can change to '.txt'
+
+standard_font = '' # Font used for tasks
+header_font = 'SFPro-Bold' # Font used for headers if listed with 'divide_with_header'
+#################################
+
+Encoding.default_internal = Encoding::UTF_8
+Encoding.default_external = Encoding::UTF_8
+
+USERNAME = ENV['LOGNAME'] # pull username from environment
+USER_DIR = ENV['HOME'] # pull home directory from environment
+DROPBOX_DIR = "#{USER_DIR}/Dropbox/Apps/NotePlan/Documents".freeze
+ICLOUDDRIVE_DIR = "#{USER_DIR}/Library/Mobile Documents/iCloud~co~noteplan~NotePlan/Documents".freeze
+CLOUDKIT_DIR = "#{USER_DIR}/Library/Containers/co.noteplan.NotePlan3/Data/Library/Application Support/co.noteplan.NotePlan3".freeze
+data_root_filepath = DROPBOX_DIR if Dir.exist?(DROPBOX_DIR) && Dir[File.join(DROPBOX_DIR, '**', '*')].count { |file| File.file?(file) } > 1
+data_root_filepath = ICLOUDDRIVE_DIR if Dir.exist?(ICLOUDDRIVE_DIR) && Dir[File.join(ICLOUDDRIVE_DIR, '**', '*')].count { |file| File.file?(file) } > 1
+data_root_filepath = CLOUDKIT_DIR if Dir.exist?(CLOUDKIT_DIR) && Dir[File.join(CLOUDKIT_DIR, '**', '*')].count { |file| File.file?(file) } > 1
+
+daily_file_loc = File.expand_path(data_root_filepath + '/Calendar/' + Date.today.strftime('%Y%m%d') + file_extension)
+weekly_file_loc = File.expand_path(data_root_filepath + '/Calendar/' + Date.today.strftime('%Y-W%W') + file_extension)
+
+if ARGV.empty?
+ # Add further priority labels here
+ priority_labels = ['@urgent', '#high']
+
+ # Change priority color here
+ priority_marker = '🔴'
+
+ # Customise label color-code here:
+ labels = {
+ '@admin' => 'orange',
+ '@liz' => 'yellow',
+ '@home' => 'green',
+ '@martha' => 'purple', # pink is too light
+ '@Health' => 'cadetblue',
+ '@church' => 'blue', # lightblue is too light
+ '@tutorials' => 'violet',
+ '@Envato' => 'darkorange',
+ '@workflow' => 'purple',
+ '@tutorial' => 'cobaltblue'
+ }
+
+ lines_in_daily_file = File.exist?(daily_file_loc.to_s) ? IO.readlines(daily_file_loc.to_s) : []
+ lines_in_weekly_file = File.exist?(weekly_file_loc.to_s) ? IO.readlines(weekly_file_loc.to_s) : []
+ lines = []
+
+ # Go through daily file, removing all lines that are not a todo.
+ line_numbers = []
+ line_count = 0
+ task_style_to_search = show_alt_task ? ['- ', '* '] : use_star ? ['* '] : ['- ']
+ lines_in_daily_file.each_index do |key|
+ # Clean out leading and trailing whitespace
+ line = lines_in_daily_file[key].gsub(/\s+$/, '')
+ task_line = show_subtasks ? line.gsub(/^\s+/, '') : line
+ if task_line.start_with?(*task_style_to_search) && !task_line[2..4].start_with?('[x]', '[>]', '[-]') # Get only active Task items
+ # Now check if doesn't have a >YYYY-MM-DD that schedules it into the future
+ break if task_line =~ /\s>\d{4}\-\d{2}\-\d{2}/
+
+ # It's a todo line to display. Remove the leading task marker and add to the list.
+ if use_star
+ lines.push(line.gsub(/^(\s*)\*\s*(\[ \]\s*)*/, '\1'))
+ else
+ lines.push(line.gsub(/^(\s*)\-\s*(\[ \]\s*)*/, '\1'))
+ end
+ line_numbers.push(line_count)
+ elsif divide_with_header && line =~ /^(#\s|##\s)/ # i.e. this is a header line
+ lines.push(line)
+ line_numbers.push(line_count)
+ else
+ # ignore the line
+ end
+ line_count += 1
+ end
+ daily_task_count = lines.size
+
+ # repeat for weekly note, but to distinguish them, make the line_numbers negative instead
+ line_count = 0
+ lines_in_weekly_file.each_index do |key|
+ # Clean out leading and trailing whitespace
+ line = lines_in_weekly_file[key].gsub(/\s+$/, '')
+ task_line = show_subtasks ? line.gsub(/^\s+/, '') : line
+ if task_line.start_with?(*task_style_to_search) && !task_line[2..4].start_with?('[x]', '[>]', '[-]') # Get only active Task items
+ # Now check if doesn't have a >YYYY-MM-DD that schedules it into the future
+ break if task_line =~ /\s>\d{4}\-\d{2}\-\d{2}/
+
+ # It's a todo line to display. Remove the leading task marker and add to the list.
+ if use_star
+ lines.push(line.gsub(/^(\s*)\*\s*(\[ \]\s*)*/, '\1'))
+ else
+ lines.push(line.gsub(/^(\s*)\-\s*(\[ \]\s*)*/, '\1'))
+ end
+ line_numbers.push(line_count)
+ elsif divide_with_header && line =~ /^(#\s|##\s)/ # i.e. this is a H1 or H2 line
+ lines.push(line)
+ line_numbers.push(line_count)
+ else
+ # ignore the line
+ end
+ line_count += 1
+ end
+ weekly_task_count = lines.size - daily_task_count
+
+ # Give the header. It's the NotePlan icon or an emoji briefcase with the number of items todo
+ icon_base64 = 'iVBORw0KGgoAAAANSUhEUgAAADgAAAA4CAQAAAACj/OVAAAABGdBTUEAALGPC/xhBQAAACBjSFJNAAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAAAmJLR0QA/4ePzL8AAAAJcEhZcwAAFiUAABYlAUlSJPAAAAAHdElNRQfkChwAHRNqrC5wAAABSElEQVRYw2NgGAWjYBRQCXAy8AIxnYAxw384NKKHhf9R4KiFoxYOpIXGDPxkWsjFYEKqZQ5Q476QYSFMRpfcYJtDgoU3yQvuZ2iG/mfwIsLCBgxdl8hLFhD4h0EMj4VKWPX8p8RCEHzPwILVwv84IZGgHY8RG9H4t/GodSU+FmPwGEMsVCI1axygwLIF5Gb+V2RY9oGy8oaV4R9J1rFQo5BLIdIyR2qWrBcJWLaL2nXELwIWnqCmZROJDNIsalgmT1KS+cMgTIllzCSmUAh8A0zZZIEXFGT8Y6Ra1kCFoq2IeOt24DEG3Skb8Khtprx6wlYfMjJ8pbR6wq8Zm6gwZRZextCoRUQTwx1D113yGlGrSWhEbSG3zapOQTPxO1RGhdSsIUd2y5uNQWa0bzFq4ciyMAjJOnF6jdXYMrgwmI6Oj42CUUAVAABntNYrW391eQAAACV0RVh0ZGF0ZTpjcmVhdGUAMjAyMC0xMC0yOFQwMDoyOToxOSswMDowMDOfhXoAAAAldEVYdGRhdGU6bW9kaWZ5ADIwMjAtMTAtMjhUMDA6Mjg6MjMrMDA6MDCH/w5VAAAAAElFTkSuQmCC'
+
+ line_count = 0
+ lines.each { |line| line_count += 1 unless line.start_with?('#') }
+ if use_emoji_as_icon
+ puts "💼#{line_count}"
+ else
+ puts "#{line_count} |templateImage=#{icon_base64}"
+ end
+
+ puts '---' # end of header marker. (Each --- after the first one will be interpreted as a menu separator.)
+
+ cfn = File.expand_path(__FILE__)
+
+ # Create the list of items to do in the menu.
+ item_number = 0
+ puts "# Today's note (#{daily_task_count} open tasks) | color=blue,lightblue font=#{header_font} href=noteplan://x-callback-url/openNote?noteDate=#{Date.today.strftime('%Y%m%d')}" if !daily_task_count.zero?
+ now_in_weekly_section = false
+ lines.each do |item|
+ # first check whether we're about to move to weekly items
+ if (item_number == daily_task_count)
+ puts "---" if !daily_task_count.zero?
+ puts "# This Week's note (#{weekly_task_count} open tasks for #{Date.today.strftime('W%W')}) | color=purple,violet font=#{header_font} href=noteplan://x-callback-url/openNote?noteDate=#{Date.today.strftime('%Y-W%W')}"
+ now_in_weekly_section = true
+ end
+ line_color = ''
+ line = item.chomp
+ if priority_labels.any? { |s| line.include? s }
+ # If line contains priority label, display in priority color
+ # line_color = priority_color
+ # If line contains priority label, prefix item with priority_marker
+ line = priority_marker + ' ' + line
+ else
+ # If line contains no priority label, cycle through labels hash,
+ # and if line contains a label display in corresponding color
+ labels.each { |label, label_color| line_color = label_color if line.include?(label) }
+ end
+ # If the line contains no label, display in default color. Otherwise, in
+ # chosen color. Clicking line launches this script with line number as
+ # the parameter.
+ line_font = standard_font
+ # If this is a H1 or H2 line, then print as a title, and put a separator before it
+ if line.start_with?('# ', '## ')
+ puts('---') unless item_number == 0
+ line_font = header_font
+ end
+ if !now_in_weekly_section
+ line_params = "#{line_color.empty? ? '' : 'color=' + line_color} #{line_font.empty? ? '' : 'font=' + line_font} bash='#{cfn}' param1=#{line_numbers[item_number]}D"
+ else
+ line_params = "#{line_color.empty? ? '' : 'color=' + line_color} #{line_font.empty? ? '' : 'font=' + line_font} bash='#{cfn}' param1=#{line_numbers[item_number]}W"
+ end
+ puts("#{line} | " + line_params + " param2=x terminal=false trim=false refresh=true\n")
+ puts("#{line} | alternate=true " + line_params + " param2=- terminal=false trim=false refresh=true\n") # alternative 'cancel' item used when 'option' key is pressed
+ item_number += 1
+ end
+ puts '---'
+ puts "Click an item to mark as 'done'"
+ puts "Click an item to mark as 'cancelled' | alternate=true" # alternative 'cancel' item used when 'option' key is pressed
+ puts 'Refresh now (normally every 15m) | refresh=true'
+
+else
+ # This is what to do when clicking on an item:
+ # - set it as done
+ # - (if wanted) move the item to the Archive section
+ # (and create it first if needed).
+
+ # Get the task number to complete/cancel (starting from 0)
+ item = ARGV[0]
+ do_num = item.to_i # keep just numeric portion, dropping terminal characters
+ # Get value of param2, which is either 'x' or '-'
+ mark = ARGV[1]
+ puts "Checking off for item #{item} line #{do_num} and mark '#{mark}'"
+
+ # If the item finishes 'D' then we're updating the daily note
+ if item.end_with?('D')
+ # Get the list of todos and setup variables
+ daily_todo_file = File.open(daily_file_loc.to_s)
+ lines_in_daily_file = IO.readlines(daily_todo_file)
+ unless lines_in_daily_file[do_num].start_with?('#') # Do nothing if the item is a header
+ task = ''
+ lines = []
+ line_number = 0
+
+ lines_in_daily_file[-1] = lines_in_daily_file[-1] + "\n" unless lines_in_daily_file[-1].include? "\n"
+
+ # Process the todo list lines
+ lines_in_daily_file.each do |line|
+ if line_number != do_num
+ # It is one of the other lines. Just push it into the stack
+ lines.push(line)
+ else
+ # Get the line to be moved to the archive area
+ task = if insert_date_on_done_task
+ line.chomp + (mark == 'x' ? " @done(#{Time.new.strftime('%Y-%m-%d %H:%M')})\n" : "\n")
+ else
+ task = line.chomp + "\n"
+ end
+ task = task.gsub(/^(\s*)([\-\*]+)\s*(\[ \]\s*)*/, '\1\2 [' + mark + '] ') # Works with both task style, useful if mix with 'show_alt_task', also it keeps the indentation at beginning of the line
+ lines.push(task) if !archive_task_at_end
+ end
+ line_number += 1
+ end
+
+ # Add the task to the bottom
+ lines.push(task) if archive_task_at_end
+
+ # Save the file
+ IO.write(daily_todo_file, lines.join)
+ end
+
+ # ... otherwise update the weekly note
+ else
+ # Get the list of todos and setup variables
+ weekly_todo_file = File.open(weekly_file_loc.to_s)
+ lines_in_weekly_file = IO.readlines(weekly_todo_file)
+
+ unless lines_in_weekly_file[do_num].start_with?('#') # Do nothing if the item is a header
+ task = ''
+ lines = []
+ line_number = 0
+ lines_in_weekly_file[-1] = lines_in_weekly_file[-1] + "\n" unless lines_in_weekly_file[-1].include? "\n"
+
+ # Process the todo list lines
+ lines_in_weekly_file.each do |line|
+ if line_number != do_num
+ # It is one of the other lines. Just push it into the stack
+ lines.push(line)
+ else
+ # Get the line to be moved to the archive area
+ task = if insert_date_on_done_task
+ line.chomp + (mark == 'x' ? " @done(#{Time.new.strftime('%Y-%m-%d %H:%M')})\n" : "\n")
+ else
+ task = line.chomp + "\n"
+ end
+ task = task.gsub(/^(\s*)([\-\*]+)\s*(\[ \]\s*)*/, '\1\2 [' + mark + '] ') # Works with both task style, useful if mix with 'show_alt_task', also it keeps the indentation at beginning of the line
+ if !archive_task_at_end
+ lines.push(task)
+ end
+ end
+ line_number += 1
+ end
+
+ # Add the task to the bottom
+ lines.push(task) if archive_task_at_end
+
+ # Save the file
+ IO.write(weekly_todo_file, lines.join)
+ end
+
+ end
+end
diff --git a/repository.json b/repository.json
old mode 100644
new mode 100755
index cd2f3a8..947c890
--- a/repository.json
+++ b/repository.json
@@ -2071,13 +2071,29 @@
"author": "z0mbix",
"author.github": "z0mbix",
"version": "1.0"
+ }
+ ]
+ },
+ {
+ "category": "Task Managers",
+ "plugins": [
+ {
+ "source": "./Lifestyle/ToDo/NotePlan3.15m.rb",
+ "title": "To do Today for NotePlan 3",
+ "author": "Jonathan Clark",
+ "author.github": "jgclark",
+ "desc": "Displays today's todo list taken from NotePlan v3 and displayed with customizable color-code. Mark tasks 'done' simply by clicking on them in the menubar drop-down list. This was based on 'Todo.NotePlan' by Richard Guay which in turn was based on 'Todo Colour' plugin by Srdgh.",
+ "image": "https://noteplan.co/static/icon-aef6fdb335c829b1363315ef21c3146d.png",
+ "dependencies": "ruby",
+ "abouturl": "https://noteplan.co/",
+ "version": "v2.2"
},
{
"source": "./Lifestyle/ToDo/todoNotePlan.15m.rb",
"title": "NotePlan Todo in Colour",
"author": "Richard Guay",
"author.github": "raguay",
- "desc": "A todo list taken from NotePlan and displayed with customizable color-code. Mark tasks \"done\" simply by clicking on them in the menubar drop-down list. This was based on \"Todo Colour\" plugin by Srdgh.",
+ "desc": "A todo list taken from NotePlan v1 and displayed with customizable color-code. Mark tasks \"done\" simply by clicking on them in the menubar drop-down list. This was based on \"Todo Colour\" plugin by Srdgh.",
"image": "http://customct.com/images/NotePlanPlugin-01.png",
"dependencies": "ruby",
"abouturl": "http://customct.com/bitbar",