From 2fa8bd84adf6e1e677bd01a66c43756db9261100 Mon Sep 17 00:00:00 2001 From: Valerie Burzynski Date: Mon, 18 Mar 2024 17:36:01 -0500 Subject: [PATCH] feat: regex fixes --- TASKS.md | 37 +- main.js | 681 ------------------------------------ src/swap-checkbox-status.ts | 20 +- 3 files changed, 17 insertions(+), 721 deletions(-) delete mode 100644 main.js diff --git a/TASKS.md b/TASKS.md index 031ef38..9346792 100644 --- a/TASKS.md +++ b/TASKS.md @@ -1,27 +1,6 @@ -## Future Roadmap +# Tasks -### customization and configuration - -- [x] make the accents customizable through the settings -- [x] customize the checkbox options using the settings menu -- [x] ability to customize all of the options -- [x] ability to add and remove options -- [x] ability to rename -- [x] ability to reorder the options -- [x] ~~ability to have aliases~~ - - Cancelled for now, config allows mutliple works and fuzzy search fill match any of them -- [ ] the settings should have presets for common themes like Things, Minimal, etc. - -## Brainstorming... - -### Feature: Workflow Customization - -- [ ] enable/disable transforming non-task lines into tasks -- [ ] enable an optional second step which prompts whether to transform tasks, non-tasks, or both. -- [ ] alternatively -- split into three commands: - - [ ] apply the task status transformation only to task lines - - [ ] apply the task status transformation only to non-task lines - - [ ] apply the task status transformation to both types of lines +## Brainstorming ### Feature: Quickly apply previously selected status @@ -36,7 +15,7 @@ ### Commands and Hotkeys - [ ] implement individual command for each type of custom checkbox accent - - this is allow users to directly apply a custom accent through either a hotkey or command palette. + - this would allow users to directly apply a custom accent through either a hotkey or command palette. ### Multiple Line Selection Features @@ -68,16 +47,6 @@ Scenario: start of selection intersects with a callout block Scenario: end of selection intersects with a callout block ``` -## Bug Fixes - -- [x] making a multi-line selection in reverse (down to up), doesn't work -- [x] when a selection range doesn't start at the beginning of a line, the checkbox marker gets added at the start of the selection, not the start of the line. - -## Other Ideas - -- [ ] ability to quickly make a checkbox into a non-checkbox. - - would this become a list item or regular text? (probably an unordered list item) - ## Release Workflow ```shell diff --git a/main.js b/main.js deleted file mode 100644 index 3ded9c9..0000000 --- a/main.js +++ /dev/null @@ -1,681 +0,0 @@ -/* -THIS IS A GENERATED/BUNDLED FILE BY ESBUILD -if you want to view the source, please visit the github repository of this plugin -*/ - -var __defProp = Object.defineProperty; -var __getOwnPropDesc = Object.getOwnPropertyDescriptor; -var __getOwnPropNames = Object.getOwnPropertyNames; -var __hasOwnProp = Object.prototype.hasOwnProperty; -var __export = (target, all) => { - for (var name in all) - __defProp(target, name, { get: all[name], enumerable: true }); -}; -var __copyProps = (to, from, except, desc) => { - if (from && typeof from === "object" || typeof from === "function") { - for (let key of __getOwnPropNames(from)) - if (!__hasOwnProp.call(to, key) && key !== except) - __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); - } - return to; -}; -var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); - -// main.ts -var main_exports = {}; -__export(main_exports, { - default: () => TaskStatusPlugin -}); -module.exports = __toCommonJS(main_exports); -var import_obsidian4 = require("obsidian"); - -// src/settings.ts -var import_obsidian = require("obsidian"); - -// src/themes/AnuPpuccinTheme.ts -var AnuPpuccinTheme = { - name: "AnuPpuccin Theme", - statuses: [ - { character: " ", title: "Unchecked" }, - { character: "x", title: "Checked" }, - { character: ">", title: "Rescheduled" }, - { character: "<", title: "Scheduled" }, - { character: "!", title: "Important" }, - { character: "-", title: "Cancelled" }, - { character: "/", title: "In Progress" }, - { character: "?", title: "Question" }, - { character: "*", title: "Star" }, - { character: "n", title: "Note" }, - { character: "l", title: "Location" }, - { character: "i", title: "Information" }, - { character: "I", title: "Idea" }, - { character: "S", title: "Amount" }, - { character: "p", title: "Pro" }, - { character: "c", title: "Con" }, - { character: "b", title: "Bookmark" }, - { character: '"', title: "Quote" }, - { character: "0", title: "Speech bubble 0" }, - { character: "1", title: "Speech bubble 1" }, - { character: "2", title: "Speech bubble 2" }, - { character: "3", title: "Speech bubble 3" }, - { character: "4", title: "Speech bubble 4" }, - { character: "5", title: "Speech bubble 5" }, - { character: "6", title: "Speech bubble 6" }, - { character: "7", title: "Speech bubble 7" }, - { character: "8", title: "Speech bubble 8" }, - { character: "9", title: "Speech bubble 9" } - ] -}; - -// src/themes/AuraThemeTheme.ts -var AuraTheme = { - name: "Aura Theme", - statuses: [ - { character: " ", title: "incomplete" }, - { character: "x", title: "complete / done" }, - { character: "-", title: "cancelled" }, - { character: ">", title: "deferred" }, - { character: "/", title: "in progress, or half-done" }, - { character: "!", title: "Important" }, - { character: "?", title: "question" }, - { character: "R", title: "review" }, - { character: "+", title: "Inbox / task that should be processed later" }, - { character: "b", title: "bookmark" }, - { character: "B", title: "brainstorm" }, - { character: "D", title: "deferred or scheduled" }, - { character: "I", title: "Info" }, - { character: "i", title: "idea" }, - { character: "N", title: "note" }, - { character: "Q", title: "quote" }, - { character: "W", title: "win / success / reward" }, - { character: "P", title: "pro" }, - { character: "C", title: "con" } - ] -}; - -// src/themes/EbullientworksTheme.ts -var EbullientworksTheme = { - name: "Ebullientworks Theme", - statuses: [ - { character: " ", title: "Unchecked" }, - { character: "x", title: "Checked" }, - { character: "-", title: "Cancelled" }, - { character: "/", title: "In Progress" }, - { character: ">", title: "Deferred" }, - { character: "!", title: "Important" }, - { character: "?", title: "Question" }, - { character: "r", title: "Review" } - ] -}; - -// src/themes/ITSTheme.ts -var ITSTheme = { - name: "ITS Theme with SlRvb Checkboxes", - statuses: [ - { character: " ", title: "Unchecked" }, - { character: "x", title: "Regular" }, - { character: "X", title: "Checked" }, - { character: "-", title: "Dropped" }, - { character: ">", title: "Forward" }, - { character: "D", title: "Date" }, - { character: "?", title: "Question" }, - { character: "/", title: "Half Done" }, - { character: "+", title: "Add" }, - { character: "R", title: "Research" }, - { character: "!", title: "Important" }, - { character: "i", title: "Idea" }, - { character: "B", title: "Brainstorm" }, - { character: "P", title: "Pro" }, - { character: "C", title: "Con" }, - { character: "Q", title: "Quote" }, - { character: "N", title: "Note" }, - { character: "b", title: "Bookmark" }, - { character: "I", title: "Information" }, - { character: "p", title: "Paraphrase" }, - { character: "L", title: "Location" }, - { character: "E", title: "Example" }, - { character: "A", title: "Answer" }, - { character: "r", title: "Reward" }, - { character: "c", title: "Choice" }, - { character: "d", title: "Doing" }, - { character: "T", title: "Time" }, - { character: "@", title: "Character / Person" }, - { character: "t", title: "Talk" }, - { character: "O", title: "Outline / Plot" }, - { character: "~", title: "Conflict" }, - { character: "W", title: "World" }, - { character: "f", title: "Clue / Find" }, - { character: "F", title: "Foreshadow" }, - { character: "H", title: "Favorite / Health" }, - { character: "&", title: "Symbolism" }, - { character: "s", title: "Secret" } - ] -}; - -// src/themes/LYTModeTheme.ts -var LYTModeTheme = { - name: "LYT Mode Theme", - statuses: [ - { character: " ", title: "Unchecked" }, - { character: "x", title: "Checked" }, - { character: ">", title: "Rescheduled" }, - { character: "<", title: "Scheduled" }, - { character: "!", title: "Important" }, - { character: "-", title: "Cancelled" }, - { character: "/", title: "In Progress" }, - { character: "?", title: "Question" }, - { character: "*", title: "Star" }, - { character: "n", title: "Note" }, - { character: "l", title: "Location" }, - { character: "i", title: "Information" }, - { character: "I", title: "Idea" }, - { character: "S", title: "Amount" }, - { character: "p", title: "Pro" }, - { character: "c", title: "Con" }, - { character: "b", title: "Bookmark" }, - { character: "f", title: "Fire" }, - { character: "k", title: "Key" }, - { character: "w", title: "Win" }, - { character: "u", title: "Up" }, - { character: "d", title: "Down" } - ] -}; - -// src/themes/MinimalTheme.ts -var MinimalTheme = { - name: "Minimal Theme", - statuses: [ - { character: " ", title: "to-do" }, - { character: "/", title: "incomplete" }, - { character: "x", title: "done" }, - { character: "-", title: "canceled" }, - { character: ">", title: "forwarded" }, - { character: "<", title: "scheduling" }, - { character: "?", title: "question" }, - { character: "!", title: "important" }, - { character: "*", title: "star" }, - { character: '"', title: "quote" }, - { character: "l", title: "location" }, - { character: "b", title: "bookmark" }, - { character: "i", title: "information" }, - { character: "S", title: "savings" }, - { character: "I", title: "idea" }, - { character: "p", title: "pros" }, - { character: "c", title: "cons" }, - { character: "f", title: "fire" }, - { character: "k", title: "key" }, - { character: "w", title: "win" }, - { character: "u", title: "up" }, - { character: "d", title: "down" } - ] -}; - -// src/themes/ThingsTheme.ts -var ThingsTheme = { - name: "Things Theme", - statuses: [ - // Basic - { character: " ", title: "to-do" }, - { character: "/", title: "incomplete" }, - { character: "x", title: "done" }, - { character: "-", title: "canceled" }, - { character: ">", title: "forwarded" }, - { character: "<", title: "scheduling" }, - // Extras - { character: "?", title: "question" }, - { character: "!", title: "important" }, - { character: "*", title: "star" }, - { character: '"', title: "quote" }, - { character: "l", title: "location" }, - { character: "b", title: "bookmark" }, - { character: "i", title: "information" }, - { character: "S", title: "savings" }, - { character: "I", title: "idea" }, - { character: "p", title: "pros" }, - { character: "c", title: "cons" }, - { character: "f", title: "fire" }, - { character: "k", title: "key" }, - { character: "w", title: "win" }, - { character: "u", title: "up" }, - { character: "d", title: "down" } - ] -}; - -// src/themes/index.ts -var themes = [ - AnuPpuccinTheme, - AuraTheme, - EbullientworksTheme, - ITSTheme, - LYTModeTheme, - MinimalTheme, - ThingsTheme -]; - -// src/default-settings.ts -var DEFAULT_SETTINGS = { - // prettier-ignore - checkboxOptions: [ - { title: "to-do", character: " " }, - { title: "incomplete", character: "/" }, - { title: "done", character: "x" }, - { title: "canceled", character: "-" }, - { title: "forwarded", character: ">" }, - { title: "scheduling", character: "<" }, - { title: "question", character: "?" }, - { title: "important", character: "!" }, - { title: "star", character: "*" }, - { title: "quote", character: '"' }, - { title: "location", character: "l" }, - { title: "bookmark", character: "b" }, - { title: "information", character: "i" }, - { title: "savings", character: "S" }, - { title: "idea", character: "I" }, - { title: "pros", character: "p" }, - { title: "cons", character: "c" }, - { title: "fire", character: "f" }, - { title: "key", character: "k" }, - { title: "win", character: "w" }, - { title: "up", character: "u" }, - { title: "down", character: "d" }, - { title: "draft pull request", character: "D" }, - { title: "open pull request", character: "P" }, - { title: "merged pull request", character: "M" } - ] -}; -var default_settings_default = DEFAULT_SETTINGS; - -// src/settings.ts -function swap(arr, indexA, indexB) { - if (indexB < 0 || indexB === arr.length) - return; - const temp = arr[indexA]; - arr[indexA] = arr[indexB]; - arr[indexB] = temp; -} -function moveToTop(arr, index) { - const item = arr.splice(index, 1); - arr.unshift(item[0]); -} -var Settings = class extends import_obsidian.PluginSettingTab { - /** - * constructs the settings - * @param app obsidian application instance - * @param plugin plugin instance - */ - constructor(app, plugin) { - super(app, plugin); - this.plugin = plugin; - } - /** - * Renders the settings view - */ - display() { - const { containerEl } = this; - containerEl.empty(); - this.displayTaskStatuses(); - this.displayListActions(); - this.displayImportOptions(); - } - /** - * Render the custom task statuses editing section - */ - displayTaskStatuses() { - this.plugin.settings.checkboxOptions.forEach((checkboxOption, index) => { - new import_obsidian.Setting(this.containerEl).setName(`${(index + 1).toString().padStart(2, "0")}.`).addText(async (text) => { - text.setPlaceholder("name").setValue(checkboxOption.title).onChange(async (value) => { - this.plugin.settings.checkboxOptions[index].title = value; - this.plugin.saveSettings(); - }); - }).addText(async (text) => { - text.setPlaceholder("character").setValue(checkboxOption.character).onChange(async (value) => { - this.plugin.settings.checkboxOptions[index].character = value; - this.plugin.saveSettings(); - }); - }).addExtraButton((button) => { - button.setIcon("double-up-arrow-glyph").setTooltip("Move to top").onClick(() => { - moveToTop(this.plugin.settings.checkboxOptions, index); - this.plugin.saveSettings(); - this.display(); - }); - }).addExtraButton((button) => { - button.setIcon("up-chevron-glyph").setTooltip("Move up").onClick(() => { - swap(this.plugin.settings.checkboxOptions, index, index - 1); - this.plugin.saveSettings(); - this.display(); - }); - }).addExtraButton((button) => { - button.setIcon("down-chevron-glyph").setTooltip("Move down").onClick(() => { - swap(this.plugin.settings.checkboxOptions, index, index + 1); - this.plugin.saveSettings(); - this.display(); - }); - }).addExtraButton((button) => { - button.setIcon("cross").setTooltip("Delete").onClick(() => { - this.plugin.settings.checkboxOptions.splice(index, 1); - this.plugin.saveSettings(); - this.display(); - }); - }); - }); - } - displayListActions() { - new import_obsidian.Setting(this.containerEl).setName("Status List Actions").addButton((button) => { - button.setButtonText("Clear list").setWarning().onClick(() => { - this.plugin.settings.checkboxOptions = []; - this.plugin.saveSettings(); - this.display(); - }); - }).addButton((button) => { - button.setButtonText("Reset to default").onClick(() => { - this.plugin.settings = structuredClone(default_settings_default); - this.plugin.saveSettings(); - this.display(); - }); - }).addButton((button) => { - button.setButtonText("Add new status").setCta().onClick(() => { - this.plugin.settings.checkboxOptions.push({ title: "undefined", character: "x" }); - this.plugin.saveSettings(); - this.display(); - }); - }); - } - displayImportOptions() { - themes.forEach(({ name, statuses }) => { - new import_obsidian.Setting(this.containerEl).setName(name).setDesc(`Add any missing custom checkbox statuses supported by the ${name}.`).addButton((button) => { - button.setButtonText(name).onClick(async () => { - statuses.forEach((option) => { - const found = this.plugin.settings.checkboxOptions.some((o) => o.character == option.character); - if (!found) { - this.plugin.settings.checkboxOptions.push({ ...option }); - } - }); - this.plugin.saveSettings(); - this.display(); - }); - }); - }); - } -}; - -// src/register-ribbon.ts -var import_obsidian3 = require("obsidian"); - -// src/modals/quick-action-modal.ts -var import_obsidian2 = require("obsidian"); - -// src/swap-checkbox-status.ts -var _SwapCheckboxStatus = class _SwapCheckboxStatus { - constructor(editor) { - /** - * When true, the class will output debug console messages - */ - this.DEBUG = true; - this.editor = editor; - } - /** - * Initiates the transformation of editor selected lines - * @param marker the status marker string to put inside the checkbox - */ - swap(marker) { - const selections = this.editor.listSelections(); - selections.forEach((selection) => { - this.transformSelectionOrLine(selection, marker); - }); - } - /** - * execute the appropriate transform depending on whether a single line or multiple are selected - * @param selection an editor selection - * @param marker the status marker to put inside the checkbox - */ - transformSelectionOrLine(selection, marker) { - this.log(`selection: -${selection}`); - if (selection.anchor.line === selection.head.line && selection.anchor.ch === selection.head.ch) { - this.transformLine(selection.anchor, marker); - } else { - this.transformSelection(selection, marker); - } - } - /** - * transform a single line by it's line number - * @param line the line number - * @param target the status marker to put inside the checkbox - */ - transformLine(anchor, target) { - const line = anchor.line; - this.log("start transformLine"); - const original = this.editor.getLine(line); - const replacement = this.getLineReplacement(original, target); - this.log(`original: -${original}`); - this.log(`replacement: -${replacement}`); - this.editor.setLine(line, replacement); - let ch = replacement.length; - if (original.length === replacement.length) { - ch = anchor.ch; - } else { - ch = anchor.ch + replacement.length - original.length; - } - this.editor.setSelection({ line, ch }); - } - getLineReplacement(original, target) { - switch (true) { - case _SwapCheckboxStatus.taskRegex.test(original): - return this.transformTasks(original, target); - case _SwapCheckboxStatus.nonTaskRegex.test(original): - return this.transformNonTasks(original, target); - case _SwapCheckboxStatus.blankLineRegex.test(original): - return this.transformBlankLines(original, target); - default: - this.log("detected other"); - return original; - } - } - transformTasks(original, target) { - this.log("transforming task(s)"); - return original.replace(_SwapCheckboxStatus.taskRegex, `$1[${target}]$2`); - } - transformNonTasks(original, target) { - this.log("transforming non-task(s)"); - const regex = new RegExp(_SwapCheckboxStatus.nonTaskRegex.source, "gm"); - const parts = regex.exec(original); - if (parts === null) - return original; - const bullet = parts[3] || "-"; - return original.replace(regex, `$2${bullet} [${target}] $4`); - } - transformBlankLines(original, target) { - this.log("transforming blank line(s)"); - return original.replace(_SwapCheckboxStatus.blankLineRegex, `$1- [${target}] `); - } - /** - * transform a selection of text - * @param selection - * @param target - */ - transformSelection(selection, target) { - this.log("start transformSelection"); - this.log("selection", selection); - const { anchor, head } = selection; - const anchorIsStart = anchor.ch <= head.ch && anchor.line <= head.line; - const cursorStart = anchorIsStart ? anchor : head; - const cursorEnd = anchorIsStart ? head : anchor; - cursorStart.ch = 0; - cursorEnd.ch = 0; - cursorEnd.line += 1; - let replacement = this.editor.getRange(cursorStart, cursorEnd); - if (_SwapCheckboxStatus.taskRegex.test(replacement)) { - replacement = this.transformTasks(replacement, target); - } - if (_SwapCheckboxStatus.nonTaskRegex.test(replacement)) { - replacement = this.transformNonTasks(replacement, target); - } - this.log("replacement", replacement); - this.editor.replaceRange(replacement, cursorStart, cursorEnd); - } - /** - * conditional log output - * @param args arguments to pass on to console.log - */ - log(...args) { - this.DEBUG && console.log(...args); - } -}; -// matches a checklist item: -// start of a line; any amount of whitespace or `>` characters (for callouts); -// then a dash or asterix for a bullet point, a single whitespace, then square bracket syntax -// for the checkbox; followed by any remaining characters til the end of the line. -// Capturing Groups: -// $1 - everything before the checkbox -// $2 - everything after the checkbox -_SwapCheckboxStatus.taskRegex = /^([\s>]*[-*+]\s)\[[^\]]\](?!\()(.*)$/m; -/* -(?!^#+) Ignore Heading (negative lookahead) -(?!^\s*$) Ignore blank lines -(?!^\s{0,3}([-_*]) *(?:\1 *){2,}$) Ignore horizontal rules and thematic breaks -(?!^[\s>]* \[![\w-]+\]) Ignore first line of Obsidian Callouts -^ start of a line -(?!(?:\s*>?)*\s*[-*+]\s+\[[^\]]\](?!\()) Ignore lines that are tasks (they may or may not be inside block quotes) -([>\s]*)? capture any amount of indentation and nested blockquote markers -(?:(?:[-*+]|[0-9]+\.)\s)? don't capture any ordered or unordered list markings -(.*) capture the rest of the text -*/ -// matches a line that is not a checklist -// Capturing Groups: -// $1 - used internally to detect horizontal rules -// $2 - indentation -// $3 - bullet character -// $4 - content -_SwapCheckboxStatus.nonTaskRegex = /(?!^#+)(?!^\s*$)(?!^\s{0,3}([-_*])\s*(?:\1 *){2,}$)(?!^[\s>]* \[![\w-]+\])^(?!(?:\s*>?)*\s*[-*+]\s+\[[^\]]\](?!\())([>\s]*)?(?:(?:([-*+])|[0-9]+\.)\s)?(.*)/m; -// matches a blank line -_SwapCheckboxStatus.blankLineRegex = /^(\s*)$/m; -var SwapCheckboxStatus = _SwapCheckboxStatus; -var swap_checkbox_status_default = SwapCheckboxStatus; - -// src/modals/quick-action-modal.ts -var QuickActionModal = class extends import_obsidian2.SuggestModal { - /** - * - * @param app Obsidian instance - * @param plugin plugin instance - * @param editor editor instance - */ - constructor(app, plugin, editor) { - super(app); - this.plugin = plugin; - this.editor = editor; - } - /** - * filters the checkbox options; the results are used as suggestions - * @param query the search string - * @returns collection of options - */ - getSuggestions(query) { - return this.plugin.settings.checkboxOptions.filter( - (option) => option.title.toLowerCase().includes(query.toLowerCase()) - ); - } - /** - * renders each suggestion - * @param option the checkbox option to display - * @param el the suggestion HTML element - */ - renderSuggestion(option, el) { - el.setCssStyles({ - display: "flex", - flexDirection: "row", - alignItems: "center", - textAlign: "center" - }); - const input = el.createEl("input", { - attr: { - "type": "checkbox", - "data-task": option.character - } - }); - input.classList.add("task-list-item"); - input.checked = option.character !== " "; - const span = el.createEl("span", { text: option.title }); - span.classList.add("cm-list-1"); - } - /** - * Handler for when the user chooses an option - * @param option the option selected by the user - * @param evt the triggering mouse or keyboard event - */ - onChooseSuggestion(option, evt) { - new import_obsidian2.Notice(`Selected ${option.title}`); - new swap_checkbox_status_default(this.editor).swap(option.character); - } -}; - -// src/register-ribbon.ts -var register_ribbon_default = (plugin) => { - (0, import_obsidian3.addIcon)( - "search-check", - ` - - - - - - ` - ); - plugin.addRibbonIcon( - "search-check", - "Change Checkbox Status", - (_event) => { - const activeView = plugin.app.workspace.getActiveViewOfType(import_obsidian3.MarkdownView); - if (!activeView) - return; - const editor = activeView.editor; - new QuickActionModal(plugin.app, plugin, editor).open(); - } - ); -}; - -// src/commands/open-task-quick-menu.ts -var changeTaskStatus = (plugin) => ({ - id: "change-task-status", - name: "change task status", - // hotkeys: [{ modifiers: ["Mod", "Shift"], key: "l" }], - editorCallback: (editor, view) => { - new QuickActionModal(plugin.app, plugin, editor).open(); - } -}); -var open_task_quick_menu_default = changeTaskStatus; - -// main.ts -var TaskStatusPlugin = class extends import_obsidian4.Plugin { - /** - * Setup the plugin when it loads in obsidian - */ - async onload() { - console.log("loading Obsidian Task Status"); - await this.loadSettings(); - register_ribbon_default(this); - this.addCommand(open_task_quick_menu_default(this)); - this.addSettingTab(new Settings(this.app, this)); - } - /** - * Teardown the plugin when it gets unloaded - */ - onunload() { - console.log("unloading Obsidian Task Status"); - } - /** - * Trigger the rendering of the settings view - */ - async loadSettings() { - this.settings = Object.assign({}, default_settings_default, await this.loadData()); - } - /** - * persist/save the plugin settings - */ - async saveSettings() { - await this.saveData(this.settings); - } -}; -//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["main.ts", "src/settings.ts", "src/themes/AnuPpuccinTheme.ts", "src/themes/AuraThemeTheme.ts", "src/themes/EbullientworksTheme.ts", "src/themes/ITSTheme.ts", "src/themes/LYTModeTheme.ts", "src/themes/MinimalTheme.ts", "src/themes/ThingsTheme.ts", "src/themes/index.ts", "src/default-settings.ts", "src/register-ribbon.ts", "src/modals/quick-action-modal.ts", "src/swap-checkbox-status.ts", "src/commands/open-task-quick-menu.ts"],
  "sourcesContent": ["import { Plugin } from 'obsidian';\nimport { TaskStatusPluginInterface, TaskStatusPluginSettings } from 'src/types';\nimport Settings from 'src/settings';\nimport registerRibbon from 'src/register-ribbon';\nimport changeTaskStatus from 'src/commands/open-task-quick-menu';\nimport DEFAULT_SETTINGS from 'src/default-settings';\n\nexport default class TaskStatusPlugin extends Plugin implements TaskStatusPluginInterface {\n  settings: TaskStatusPluginSettings;\n\n  /**\n   * Setup the plugin when it loads in obsidian\n   */\n  async onload() {\n    console.log('loading Obsidian Task Status');\n    await this.loadSettings();\n    registerRibbon(this);\n    this.addCommand(changeTaskStatus(this));\n    this.addSettingTab(new Settings(this.app, this));\n  }\n\n  /**\n   * Teardown the plugin when it gets unloaded\n   */\n  onunload() {\n    console.log('unloading Obsidian Task Status');\n  }\n\n  /**\n   * Trigger the rendering of the settings view\n   */\n  async loadSettings() {\n    this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());\n  }\n\n  /**\n   * persist/save the plugin settings\n   */\n  async saveSettings() {\n    await this.saveData(this.settings);\n  }\n}\n", "import { App, PluginSettingTab, Setting } from 'obsidian';\nimport { CheckboxOption, TaskStatusPluginInterface } from './types';\nimport { themes } from './themes';\nimport DEFAULT_SETTINGS from './default-settings';\n\n/**\n * Swap two indexes in an array\n * @param arr\n * @param indexA\n * @param indexB\n * @returns\n */\nfunction swap<T>(arr: T[], indexA: number, indexB: number): void {\n  if (indexB < 0 || indexB === arr.length) return;\n\n  const temp = arr[indexA];\n  arr[indexA] = arr[indexB];\n  arr[indexB] = temp;\n}\n\n/**\n * move an array item to the top of the list\n * @param arr the list to modify\n * @param index the index of the item to move to the top\n */\nfunction moveToTop<T>(arr: T[], index: number): void {\n  const item = arr.splice(index, 1);\n  arr.unshift(item[0]);\n}\n\nexport default class Settings extends PluginSettingTab {\n  plugin: TaskStatusPluginInterface;\n\n  /**\n   * constructs the settings\n   * @param app obsidian application instance\n   * @param plugin plugin instance\n   */\n  constructor(app: App, plugin: TaskStatusPluginInterface) {\n    super(app, plugin);\n    this.plugin = plugin;\n  }\n\n  /**\n   * Renders the settings view\n   */\n  display(): void {\n    const { containerEl } = this;\n    containerEl.empty();\n    this.displayTaskStatuses();\n    this.displayListActions();\n    this.displayImportOptions();\n  }\n\n  /**\n   * Render the custom task statuses editing section\n   */\n  displayTaskStatuses(): void {\n    // create a series of settings to edit the list of custom task statuses\n    // the setting will have a name, two inputs (status name and marker), and buttons to move the\n    // item or remove it\n    this.plugin.settings.checkboxOptions.forEach((checkboxOption, index) => {\n      new Setting(this.containerEl)\n        .setName(`${(index + 1).toString().padStart(2, '0')}.`)\n        .addText(async (text) => {\n          text\n            .setPlaceholder('name')\n            .setValue(checkboxOption.title)\n            .onChange(async (value) => {\n              this.plugin.settings.checkboxOptions[index].title = value;\n              this.plugin.saveSettings();\n            });\n        })\n        .addText(async (text) => {\n          text\n            .setPlaceholder('character')\n            .setValue(checkboxOption.character)\n            .onChange(async (value) => {\n              this.plugin.settings.checkboxOptions[index].character = value;\n              this.plugin.saveSettings();\n            });\n        })\n        .addExtraButton((button) => {\n          button\n            .setIcon('double-up-arrow-glyph')\n            .setTooltip('Move to top')\n            .onClick(() => {\n              moveToTop(this.plugin.settings.checkboxOptions, index);\n              this.plugin.saveSettings();\n              this.display();\n            });\n        })\n        .addExtraButton((button) => {\n          button\n            .setIcon('up-chevron-glyph')\n            .setTooltip('Move up')\n            .onClick(() => {\n              swap(this.plugin.settings.checkboxOptions, index, index - 1);\n              this.plugin.saveSettings();\n              this.display();\n            });\n        })\n        .addExtraButton((button) => {\n          button\n            .setIcon('down-chevron-glyph')\n            .setTooltip('Move down')\n            .onClick(() => {\n              swap(this.plugin.settings.checkboxOptions, index, index + 1);\n              this.plugin.saveSettings();\n              this.display();\n            });\n        })\n        .addExtraButton((button) => {\n          button\n            .setIcon('cross')\n            .setTooltip('Delete')\n            .onClick(() => {\n              this.plugin.settings.checkboxOptions.splice(index, 1);\n              this.plugin.saveSettings();\n              this.display();\n            });\n        });\n    });\n  }\n\n  displayListActions() {\n    new Setting(this.containerEl)\n      .setName('Status List Actions')\n      .addButton((button) => {\n        button\n          .setButtonText('Clear list')\n          .setWarning()\n          .onClick(() => {\n            this.plugin.settings.checkboxOptions = [];\n            this.plugin.saveSettings();\n            this.display();\n          });\n      })\n      .addButton((button) => {\n        button\n          .setButtonText('Reset to default')\n          .onClick(() => {\n            this.plugin.settings = structuredClone(DEFAULT_SETTINGS);\n            this.plugin.saveSettings();\n            this.display();\n          });\n      })\n      .addButton((button) => {\n        button\n          .setButtonText('Add new status')\n          .setCta()\n          .onClick(() => {\n            this.plugin.settings.checkboxOptions.push({ title: 'undefined', character: 'x' });\n            this.plugin.saveSettings();\n            this.display();\n          });\n      });\n  }\n\n  displayImportOptions() {\n    themes.forEach(({name, statuses}) => {\n      new Setting(this.containerEl)\n        .setName(name)\n        .setDesc(`Add any missing custom checkbox statuses supported by the ${name}.`)\n        .addButton((button) => {\n          button.setButtonText(name).onClick(async () => {\n            statuses.forEach((option: CheckboxOption) => {\n              const found = this.plugin.settings.checkboxOptions.some((o) => o.character == option.character)\n              if(!found) {\n                this.plugin.settings.checkboxOptions.push({ ...option });\n              }\n            });\n            this.plugin.saveSettings();\n            this.display();\n          });\n        });\n    });\n  }\n}\n", "import { Theme } from \"./types\";\n\n/**\n * Statuses supported by the AnuPpuccin theme.\n * {@link https://github.com/AnubisNekhet/AnuPpuccin}\n */\nexport const AnuPpuccinTheme: Theme = {\n  name: 'AnuPpuccin Theme',\n  statuses: [\n    { character: ' ', title: 'Unchecked' },\n    { character: 'x', title: 'Checked' },\n    { character: '>', title: 'Rescheduled' },\n    { character: '<', title: 'Scheduled' },\n    { character: '!', title: 'Important' },\n    { character: '-', title: 'Cancelled' },\n    { character: '/', title: 'In Progress' },\n    { character: '?', title: 'Question' },\n    { character: '*', title: 'Star' },\n    { character: 'n', title: 'Note' },\n    { character: 'l', title: 'Location' },\n    { character: 'i', title: 'Information' },\n    { character: 'I', title: 'Idea' },\n    { character: 'S', title: 'Amount' },\n    { character: 'p', title: 'Pro' },\n    { character: 'c', title: 'Con' },\n    { character: 'b', title: 'Bookmark' },\n    { character: '\"', title: 'Quote' },\n    { character: '0', title: 'Speech bubble 0' },\n    { character: '1', title: 'Speech bubble 1' },\n    { character: '2', title: 'Speech bubble 2' },\n    { character: '3', title: 'Speech bubble 3' },\n    { character: '4', title: 'Speech bubble 4' },\n    { character: '5', title: 'Speech bubble 5' },\n    { character: '6', title: 'Speech bubble 6' },\n    { character: '7', title: 'Speech bubble 7' },\n    { character: '8', title: 'Speech bubble 8' },\n    { character: '9', title: 'Speech bubble 9' },\n  ],\n};\n", "import { Theme } from './types';\n\n/**\n * Status supported by the Aura theme.\n * {@link https://github.com/ashwinjadhav818/obsidian-aura}\n */\nexport const AuraTheme: Theme = {\n  name: 'Aura Theme',\n  statuses: [\n    { character: ' ', title: 'incomplete' },\n    { character: 'x', title: 'complete / done' },\n    { character: '-', title: 'cancelled' },\n    { character: '>', title: 'deferred' },\n    { character: '/', title: 'in progress, or half-done' },\n    { character: '!', title: 'Important' },\n    { character: '?', title: 'question' },\n    { character: 'R', title: 'review' },\n    { character: '+', title: 'Inbox / task that should be processed later' },\n    { character: 'b', title: 'bookmark' },\n    { character: 'B', title: 'brainstorm' },\n    { character: 'D', title: 'deferred or scheduled' },\n    { character: 'I', title: 'Info' },\n    { character: 'i', title: 'idea' },\n    { character: 'N', title: 'note' },\n    { character: 'Q', title: 'quote' },\n    { character: 'W', title: 'win / success / reward' },\n    { character: 'P', title: 'pro' },\n    { character: 'C', title: 'con' },\n  ],\n};\n", "import { Theme } from \"./types\";\n\n/**\n * Status supported by the Ebullientworks theme.\n * {@link https://github.com/ebullient/obsidian-theme-ebullientworks}\n */\nexport const EbullientworksTheme: Theme = {\n  name: 'Ebullientworks Theme',\n  statuses: [\n    { character: ' ', title: 'Unchecked' },\n    { character: 'x', title: 'Checked' },\n    { character: '-', title: 'Cancelled' },\n    { character: '/', title: 'In Progress' },\n    { character: '>', title: 'Deferred' },\n    { character: '!', title: 'Important' },\n    { character: '?', title: 'Question' },\n    { character: 'r', title: 'Review' },\n  ],\n};\n", "import { Theme } from \"./types\";\n\n/**\n * Statuses supported by the ITS theme.\n * {@link https://github.com/SlRvb/Obsidian--ITS-Theme}\n */\nexport const ITSTheme: Theme = {\n  name: 'ITS Theme with SlRvb Checkboxes',\n  statuses: [\n    { character: ' ', title: 'Unchecked' },\n    { character: 'x', title: 'Regular' },\n    { character: 'X', title: 'Checked' },\n    { character: '-', title: 'Dropped' },\n    { character: '>', title: 'Forward' },\n    { character: 'D', title: 'Date' },\n    { character: '?', title: 'Question' },\n    { character: '/', title: 'Half Done' },\n    { character: '+', title: 'Add' },\n    { character: 'R', title: 'Research' },\n    { character: '!', title: 'Important' },\n    { character: 'i', title: 'Idea' },\n    { character: 'B', title: 'Brainstorm' },\n    { character: 'P', title: 'Pro' },\n    { character: 'C', title: 'Con' },\n    { character: 'Q', title: 'Quote' },\n    { character: 'N', title: 'Note' },\n    { character: 'b', title: 'Bookmark' },\n    { character: 'I', title: 'Information' },\n    { character: 'p', title: 'Paraphrase' },\n    { character: 'L', title: 'Location' },\n    { character: 'E', title: 'Example' },\n    { character: 'A', title: 'Answer' },\n    { character: 'r', title: 'Reward' },\n    { character: 'c', title: 'Choice' },\n    { character: 'd', title: 'Doing' },\n    { character: 'T', title: 'Time' },\n    { character: '@', title: 'Character / Person' },\n    { character: 't', title: 'Talk' },\n    { character: 'O', title: 'Outline / Plot' },\n    { character: '~', title: 'Conflict' },\n    { character: 'W', title: 'World' },\n    { character: 'f', title: 'Clue / Find' },\n    { character: 'F', title: 'Foreshadow' },\n    { character: 'H', title: 'Favorite / Health' },\n    { character: '&', title: 'Symbolism' },\n    { character: 's', title: 'Secret' },\n  ]\n};\n", "import { Theme } from \"./types\";\n\n/**\n * Status supported by the LYT Mode theme.\n * {@link https://github.com/nickmilo/LYT-Mode}\n */\nexport const LYTModeTheme: Theme = {\n  name: 'LYT Mode Theme',\n  statuses: [\n    { character: ' ', title: 'Unchecked' },\n    { character: 'x', title: 'Checked' },\n    { character: '>', title: 'Rescheduled' },\n    { character: '<', title: 'Scheduled' },\n    { character: '!', title: 'Important' },\n    { character: '-', title: 'Cancelled' },\n    { character: '/', title: 'In Progress' },\n    { character: '?', title: 'Question' },\n    { character: '*', title: 'Star' },\n    { character: 'n', title: 'Note' },\n    { character: 'l', title: 'Location' },\n    { character: 'i', title: 'Information' },\n    { character: 'I', title: 'Idea' },\n    { character: 'S', title: 'Amount' },\n    { character: 'p', title: 'Pro' },\n    { character: 'c', title: 'Con' },\n    { character: 'b', title: 'Bookmark' },\n    { character: 'f', title: 'Fire' },\n    { character: 'k', title: 'Key' },\n    { character: 'w', title: 'Win' },\n    { character: 'u', title: 'Up' },\n    { character: 'd', title: 'Down' },\n  ],\n};\n", "import { Theme } from \"./types\";\n\n/**\n * Statuses supported by the Minimal theme.\n * {@link https://github.com/kepano/obsidian-minimal}\n */\nexport const MinimalTheme: Theme = {\n  name: 'Minimal Theme',\n  statuses: [\n    { character: ' ', title: 'to-do' },\n    { character: '/', title: 'incomplete' },\n    { character: 'x', title: 'done' },\n    { character: '-', title: 'canceled' },\n    { character: '>', title: 'forwarded' },\n    { character: '<', title: 'scheduling' },\n    { character: '?', title: 'question' },\n    { character: '!', title: 'important' },\n    { character: '*', title: 'star' },\n    { character: '\"', title: 'quote' },\n    { character: 'l', title: 'location' },\n    { character: 'b', title: 'bookmark' },\n    { character: 'i', title: 'information' },\n    { character: 'S', title: 'savings' },\n    { character: 'I', title: 'idea' },\n    { character: 'p', title: 'pros' },\n    { character: 'c', title: 'cons' },\n    { character: 'f', title: 'fire' },\n    { character: 'k', title: 'key' },\n    { character: 'w', title: 'win' },\n    { character: 'u', title: 'up' },\n    { character: 'd', title: 'down' },\n  ]\n};\n", "import { Theme } from \"./types\";\n\n/**\n * Statuses supported by the Things theme.\n * {@link https://github.com/colineckert/obsidian-things}\n */\nexport const ThingsTheme: Theme = {\n  name: 'Things Theme',\n  statuses: [\n    // Basic\n    { character: ' ', title: 'to-do' },\n    { character: '/', title: 'incomplete' },\n    { character: 'x', title: 'done' },\n    { character: '-', title: 'canceled' },\n    { character: '>', title: 'forwarded' },\n    { character: '<', title: 'scheduling' },\n    // Extras\n    { character: '?', title: 'question' },\n    { character: '!', title: 'important' },\n    { character: '*', title: 'star' },\n    { character: '\"', title: 'quote' },\n    { character: 'l', title: 'location' },\n    { character: 'b', title: 'bookmark' },\n    { character: 'i', title: 'information' },\n    { character: 'S', title: 'savings' },\n    { character: 'I', title: 'idea' },\n    { character: 'p', title: 'pros' },\n    { character: 'c', title: 'cons' },\n    { character: 'f', title: 'fire' },\n    { character: 'k', title: 'key' },\n    { character: 'w', title: 'win' },\n    { character: 'u', title: 'up' },\n    { character: 'd', title: 'down' },\n  ]\n};\n\n", "import { AnuPpuccinTheme } from './AnuPpuccinTheme';\nimport { AuraTheme } from './AuraThemeTheme';\nimport { EbullientworksTheme } from './EbullientworksTheme';\nimport { ITSTheme } from './ITSTheme';\nimport { LYTModeTheme } from './LYTModeTheme';\nimport { MinimalTheme } from './MinimalTheme';\nimport { ThingsTheme } from './ThingsTheme';\n\nexport const themes = [\n  AnuPpuccinTheme,\n  AuraTheme,\n  EbullientworksTheme,\n  ITSTheme,\n  LYTModeTheme,\n  MinimalTheme,\n  ThingsTheme,\n];\n", "import { TaskStatusPluginSettings } from './types';\n\nconst DEFAULT_SETTINGS: TaskStatusPluginSettings = {\n  // prettier-ignore\n  checkboxOptions: [\n    { title: 'to-do',               character: ' ' },\n    { title: 'incomplete',          character: '/' },\n    { title: 'done',                character: 'x' },\n    { title: 'canceled',            character: '-' },\n    { title: 'forwarded',           character: '>' },\n    { title: 'scheduling',          character: '<' },\n    { title: 'question',            character: '?' },\n    { title: 'important',           character: '!' },\n    { title: 'star',                character: '*' },\n    { title: 'quote',               character: '\"' },\n    { title: 'location',            character: 'l' },\n    { title: 'bookmark',            character: 'b' },\n    { title: 'information',         character: 'i' },\n    { title: 'savings',             character: 'S' },\n    { title: 'idea',                character: 'I' },\n    { title: 'pros',                character: 'p' },\n    { title: 'cons',                character: 'c' },\n    { title: 'fire',                character: 'f' },\n    { title: 'key',                 character: 'k' },\n    { title: 'win',                 character: 'w' },\n    { title: 'up',                  character: 'u' },\n    { title: 'down',                character: 'd' },\n    { title: 'draft pull request',  character: 'D' },\n    { title: 'open pull request',   character: 'P' },\n    { title: 'merged pull request', character: 'M' },\n  ],\n};\n\nexport default DEFAULT_SETTINGS;\n", "import { MarkdownView, addIcon } from 'obsidian';\nimport { TaskStatusPluginInterface } from \"./types\";\nimport QuickActionModal from './modals/quick-action-modal';\n\n/**\n * Registers a Ribbon Icon that can open the action modal\n */\nexport default (plugin: TaskStatusPluginInterface) => {\n  // Add an icon (modified version of https://lucide.dev/icons/search-check)\n  addIcon(\n    'search-check',\n    `<g stroke=\"none\" stroke-width=\"1\" fill=\"none\" fill-rule=\"evenodd\" stroke-linecap=\"round\" stroke-linejoin=\"round\">\n      <g transform=\"translate(3, 3)\" stroke=\"currentcolor\" stroke-width=\"6\">\n        <polyline points=\"26.1111111 41.7777778 36.5555556 52.2222222 57.4444444 31.3333333\"></polyline>\n        <circle cx=\"41.7777778\" cy=\"41.7777778\" r=\"41.7777778\"></circle>\n        <line x1=\"94\" y1=\"94\" x2=\"71.5444444\" y2=\"71.5444444\"></line>\n      </g>\n    </g>`\n  );\n\n  // create an icon in the left ribbon.\n  plugin.addRibbonIcon(\n    'search-check',\n    'Change Checkbox Status',\n    (_event: MouseEvent) => {\n      const activeView = plugin.app.workspace.getActiveViewOfType(MarkdownView);\n      if (!activeView) return;\n\n      const editor = activeView.editor;\n      new QuickActionModal(plugin.app, plugin, editor).open();\n    }\n  );\n}\n", "import { Editor, Notice, SuggestModal, App } from \"obsidian\";\nimport SwapCheckboxStatus from \"src/swap-checkbox-status\";\nimport { CheckboxOption, TaskStatusPluginInterface } from \"src/types\";\n\n/**\n * A serchable modal that allows the user to select a checkbox status symbol\n */\nexport default class QuickActionModal extends SuggestModal<CheckboxOption> {\n  editor: Editor;\n  plugin: TaskStatusPluginInterface;\n\n  /**\n   *\n   * @param app Obsidian instance\n   * @param plugin plugin instance\n   * @param editor editor instance\n   */\n  constructor(app: App, plugin: TaskStatusPluginInterface, editor: Editor) {\n    super(app);\n    this.plugin = plugin;\n    this.editor = editor;\n  }\n\n  /**\n   * filters the checkbox options; the results are used as suggestions\n   * @param query the search string\n   * @returns collection of options\n   */\n  getSuggestions(query: string): CheckboxOption[] {\n    return this.plugin.settings.checkboxOptions.filter((option) =>\n      option.title.toLowerCase().includes(query.toLowerCase())\n    );\n  }\n\n  /**\n   * renders each suggestion\n   * @param option the checkbox option to display\n   * @param el the suggestion HTML element\n   */\n  renderSuggestion(option: CheckboxOption, el: HTMLElement) {\n    el.setCssStyles({\n      display: 'flex',\n      flexDirection: 'row',\n      alignItems: 'center',\n      textAlign: 'center',\n    });\n\n    // show a preview of the checkbox\n    const input = el.createEl('input', {\n      attr: {\n        'type': 'checkbox',\n        'data-task': option.character,\n      },\n    });\n    input.classList.add('task-list-item');\n    input.checked = option.character !== ' ';\n\n    // show the name of the checkbox option\n    const span = el.createEl(\"span\", { text: option.title });\n    span.classList.add('cm-list-1')\n  }\n\n  /**\n   * Handler for when the user chooses an option\n   * @param option the option selected by the user\n   * @param evt the triggering mouse or keyboard event\n   */\n  onChooseSuggestion(option: CheckboxOption, evt: MouseEvent | KeyboardEvent) {\n    new Notice(`Selected ${option.title}`);\n    new SwapCheckboxStatus(this.editor).swap(option.character);\n  }\n}\n", "import { Editor, EditorPosition, EditorSelection } from 'obsidian';\n\n// TODO: ignore certain types of blocks -- like comment blocks and code blocks\n\nclass SwapCheckboxStatus {\n  editor: Editor;\n\n  /**\n   * When true, the class will output debug console messages\n   */\n  DEBUG = true;\n\n  // matches a checklist item:\n  // start of a line; any amount of whitespace or `>` characters (for callouts);\n  // then a dash or asterix for a bullet point, a single whitespace, then square bracket syntax\n  // for the checkbox; followed by any remaining characters til the end of the line.\n  // Capturing Groups:\n  //   $1 - everything before the checkbox\n  //   $2 - everything after the checkbox\n  public static readonly taskRegex = /^([\\s>]*[-*+]\\s)\\[[^\\]]\\](?!\\()(.*)$/m;\n\n  /*\n  (?!^#+)                                   Ignore Heading (negative lookahead)\n  (?!^\\s*$)                                 Ignore blank lines\n  (?!^\\s{0,3}([-_*]) *(?:\\1 *){2,}$)        Ignore horizontal rules and thematic breaks\n  (?!^[\\s>]* \\[![\\w-]+\\])                   Ignore first line of Obsidian Callouts\n  ^                                         start of a line\n  (?!(?:\\s*>?)*\\s*[-*+]\\s+\\[[^\\]]\\](?!\\())  Ignore lines that are tasks (they may or may not be inside block quotes)\n  ([>\\s]*)?                                 capture any amount of indentation and nested blockquote markers\n  (?:(?:[-*+]|[0-9]+\\.)\\s)?                 don't capture any ordered or unordered list markings\n  (.*)                                      capture the rest of the text\n  */\n\n  // matches a line that is not a checklist\n  // Capturing Groups:\n  //  $1 - used internally to detect horizontal rules\n  //  $2 - indentation\n  //  $3 - bullet character\n  //  $4 - content\n  public static readonly nonTaskRegex =\n    /(?!^#+)(?!^\\s*$)(?!^\\s{0,3}([-_*])\\s*(?:\\1 *){2,}$)(?!^[\\s>]* \\[![\\w-]+\\])^(?!(?:\\s*>?)*\\s*[-*+]\\s+\\[[^\\]]\\](?!\\())([>\\s]*)?(?:(?:([-*+])|[0-9]+\\.)\\s)?(.*)/m;\n\n  // matches a blank line\n  public static readonly blankLineRegex = /^(\\s*)$/m;\n\n  constructor(editor: Editor) {\n    this.editor = editor;\n  }\n\n  /**\n   * Initiates the transformation of editor selected lines\n   * @param marker the status marker string to put inside the checkbox\n   */\n  swap(marker: string) {\n    const selections = this.editor.listSelections();\n    selections.forEach((selection) => {\n      this.transformSelectionOrLine(selection, marker);\n    });\n  }\n\n  /**\n   * execute the appropriate transform depending on whether a single line or multiple are selected\n   * @param selection an editor selection\n   * @param marker the status marker to put inside the checkbox\n   */\n  transformSelectionOrLine(selection: EditorSelection, marker: string) {\n    this.log(`selection:\\n${selection}`);\n    if (selection.anchor.line === selection.head.line && selection.anchor.ch === selection.head.ch) {\n      this.transformLine(selection.anchor, marker);\n    } else {\n      this.transformSelection(selection, marker);\n    }\n  }\n\n  /**\n   * transform a single line by it's line number\n   * @param line the line number\n   * @param target the status marker to put inside the checkbox\n   */\n  transformLine(anchor: EditorPosition, target: string) {\n    const line: number = anchor.line;\n    this.log('start transformLine');\n    const original = this.editor.getLine(line);\n    const replacement = this.getLineReplacement(original, target);\n\n    this.log(`original:\\n${original}`);\n    this.log(`replacement:\\n${replacement}`);\n\n    this.editor.setLine(line, replacement);\n\n    let ch = replacement.length;\n    if (original.length === replacement.length) {\n      ch = anchor.ch;\n    } else {\n      ch = anchor.ch + replacement.length - original.length;\n    }\n    this.editor.setSelection({ line, ch });\n  }\n\n  getLineReplacement(original: string, target: string): string {\n    switch (true) {\n      // when the line is a task, replace the task marker\n      case SwapCheckboxStatus.taskRegex.test(original):\n        return this.transformTasks(original, target);\n      // when the line is not a task, transform it into a task\n      case SwapCheckboxStatus.nonTaskRegex.test(original):\n        return this.transformNonTasks(original, target);\n      // when the line is blank, start a blank task\n      case SwapCheckboxStatus.blankLineRegex.test(original):\n        return this.transformBlankLines(original, target);\n      // otherwise don't apply any transformation\n      default:\n        this.log('detected other');\n        return original;\n    }\n  }\n\n  transformTasks(original: string, target: string) {\n    this.log('transforming task(s)');\n    return original.replace(SwapCheckboxStatus.taskRegex, `$1[${target}]$2`);\n  }\n\n  transformNonTasks(original: string, target: string) {\n    this.log('transforming non-task(s)');\n    const regex = new RegExp(SwapCheckboxStatus.nonTaskRegex.source, 'gm');\n    const parts = regex.exec(original);\n    if (parts === null) return original;\n\n    // match 1 is a used inside the regex to detect horizontal rules\n    // use matches 2 (indentation) and 3 (the content);\n    const bullet = parts[3] || '-';\n    return original.replace(regex, `$2${bullet} [${target}] $4`);\n  }\n\n  transformBlankLines(original: string, target: string) {\n    this.log('transforming blank line(s)');\n    return original.replace(SwapCheckboxStatus.blankLineRegex, `$1- [${target}] `);\n  }\n\n  /**\n   * transform a selection of text\n   * @param selection\n   * @param target\n   */\n  transformSelection(selection: EditorSelection, target: string) {\n    this.log('start transformSelection');\n    this.log('selection', selection);\n\n    const { anchor, head } = selection;\n\n    // determine the start and end\n    const anchorIsStart = anchor.ch <= head.ch && anchor.line <= head.line;\n    const cursorStart = anchorIsStart ? anchor : head;\n    const cursorEnd = anchorIsStart ? head : anchor;\n\n    // expand the selection to cover entire lines\n    cursorStart.ch = 0;\n    cursorEnd.ch = 0;\n    cursorEnd.line += 1;\n\n    // get the existing original text\n    let replacement = this.editor.getRange(cursorStart, cursorEnd);\n\n    // transform selected lines that contain tasks\n    if (SwapCheckboxStatus.taskRegex.test(replacement)) {\n      replacement = this.transformTasks(replacement, target);\n    }\n\n    // all selected non-blank lines are transformed into tasks as well\n    if (SwapCheckboxStatus.nonTaskRegex.test(replacement)) {\n      replacement = this.transformNonTasks(replacement, target);\n    }\n\n    // NOTE: non-blank lines are left blank when muli-line selection is used\n\n    this.log('replacement', replacement);\n    this.editor.replaceRange(replacement, cursorStart, cursorEnd);\n  }\n\n  /**\n   * conditional log output\n   * @param args arguments to pass on to console.log\n   */\n  log(...args: unknown[]) {\n    this.DEBUG && console.log(...args);\n  }\n}\n\nexport default SwapCheckboxStatus;\n", "import { Command, MarkdownView, Editor } from \"obsidian\";\nimport { TaskStatusPluginInterface } from \"src/types\";\nimport QuickActionModal from '../modals/quick-action-modal';\n\n/**\n * command creator which constructs a command object configured to open the quick action modal\n * @param plugin the plugin instance\n * @returns a command object\n */\nconst changeTaskStatus = (plugin: TaskStatusPluginInterface): Command => ({\n  id: 'change-task-status',\n  name: 'change task status',\n  // hotkeys: [{ modifiers: [\"Mod\", \"Shift\"], key: \"l\" }],\n  editorCallback: (editor: Editor, view: MarkdownView) => {\n    new QuickActionModal(plugin.app, plugin, editor).open();\n  }\n});\n\nexport default changeTaskStatus;\n\n"],
  "mappings": "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",
  "names": ["import_obsidian", "import_obsidian", "import_obsidian"]
}
 diff --git a/src/swap-checkbox-status.ts b/src/swap-checkbox-status.ts index b1110cf..b7c6e2c 100644 --- a/src/swap-checkbox-status.ts +++ b/src/swap-checkbox-status.ts @@ -17,7 +17,7 @@ class SwapCheckboxStatus { // Capturing Groups: // $1 - everything before the checkbox // $2 - everything after the checkbox - public static readonly taskRegex = /^([\s>]*[-*+]\s)\[[^\]]\](?!\()(.*)$/m; + public static readonly taskRegex = /^([\s>]*[-*+]\s)\[[^\]]\](?!\()(.*)$/gm; /* (?!^#+) Ignore Heading (negative lookahead) @@ -38,10 +38,10 @@ class SwapCheckboxStatus { // $3 - bullet character // $4 - content public static readonly nonTaskRegex = - /(?!^#+)(?!^\s*$)(?!^\s{0,3}([-_*])\s*(?:\1 *){2,}$)(?!^[\s>]* \[![\w-]+\])^(?!(?:\s*>?)*\s*[-*+]\s+\[[^\]]\](?!\())([>\s]*)?(?:(?:([-*+])|[0-9]+\.)\s)?(.*)/m; + /(?!^#+)(?!^\s*$)(?!^\s{0,3}([-_*])\s*(?:\1 *){2,}$)(?!^[\s>]* \[![\w-]+\])^(?!(?:\s*>?)*\s*[-*+]\s+\[[^\]]\](?!\())([>\s]*)?(?:(?:([-*+])|[0-9]+\.)\s)?(.*)/gm; // matches a blank line - public static readonly blankLineRegex = /^(\s*)$/m; + public static readonly blankLineRegex = /^(\s*)$/gm; constructor(editor: Editor) { this.editor = editor; @@ -98,6 +98,9 @@ class SwapCheckboxStatus { } getLineReplacement(original: string, target: string): string { + SwapCheckboxStatus.taskRegex.lastIndex = 0; + SwapCheckboxStatus.nonTaskRegex.lastIndex = 0; + SwapCheckboxStatus.blankLineRegex.lastIndex = 0; switch (true) { // when the line is a task, replace the task marker case SwapCheckboxStatus.taskRegex.test(original): @@ -117,23 +120,26 @@ class SwapCheckboxStatus { transformTasks(original: string, target: string) { this.log('transforming task(s)'); + SwapCheckboxStatus.taskRegex.lastIndex = 0; return original.replace(SwapCheckboxStatus.taskRegex, `$1[${target}]$2`); } transformNonTasks(original: string, target: string) { this.log('transforming non-task(s)'); - const regex = new RegExp(SwapCheckboxStatus.nonTaskRegex.source, 'gm'); - const parts = regex.exec(original); + + SwapCheckboxStatus.nonTaskRegex.lastIndex = 0; + const parts = SwapCheckboxStatus.nonTaskRegex.exec(original); if (parts === null) return original; // match 1 is a used inside the regex to detect horizontal rules // use matches 2 (indentation) and 3 (the content); const bullet = parts[3] || '-'; - return original.replace(regex, `$2${bullet} [${target}] $4`); + return original.replace(SwapCheckboxStatus.nonTaskRegex, `$2${bullet} [${target}] $4`); } transformBlankLines(original: string, target: string) { this.log('transforming blank line(s)'); + SwapCheckboxStatus.blankLineRegex.lastIndex = 0; return original.replace(SwapCheckboxStatus.blankLineRegex, `$1- [${target}] `); } @@ -162,11 +168,13 @@ class SwapCheckboxStatus { let replacement = this.editor.getRange(cursorStart, cursorEnd); // transform selected lines that contain tasks + SwapCheckboxStatus.taskRegex.lastIndex = 0; if (SwapCheckboxStatus.taskRegex.test(replacement)) { replacement = this.transformTasks(replacement, target); } // all selected non-blank lines are transformed into tasks as well + SwapCheckboxStatus.nonTaskRegex.lastIndex = 0; if (SwapCheckboxStatus.nonTaskRegex.test(replacement)) { replacement = this.transformNonTasks(replacement, target); }