timhor / obsidian-editor-shortcuts

Obsidian plugin to add keyboard shortcuts commonly found in code editors such as Visual Studio Code (vscode) or Sublime Text
MIT License
356 stars 25 forks source link

[Feature Request] Expand selection Up / Down / Left / Right #81

Open Mikestriken opened 3 months ago

Mikestriken commented 3 months ago

VSCode allows users to adjust the following:



Please add these key bindings as well!

I use alt + j / k /l / i as my arrow keys alt + o / u as word part left / right

and then just by pressing shift while using any of the above I move the cursor and select what's in between.

Mikestriken commented 3 months ago

I used generative AI (and referenced some of abrahamneben's code in his fork) to implement my feature request. The following was added to your code: Note: I used the source code of your last stable release to implement my code. in actions.ts:

export const select1LinesUp = (
  editor: Editor, 
  selection: EditorSelection
) => {
  let newHead: EditorPosition = {
    line: selection.head.line - 1,
    ch: selection.head.ch
  };

  // Ensure the new head position is within the bounds of the line
  if (newHead.line >= 0) {
    const lineAboveLength = editor.getLine(newHead.line).length;
    if (newHead.ch > lineAboveLength) {
      newHead.ch = lineAboveLength;
    }
  } else {
    // Prevent moving the head above the first line
    newHead.line = 0;
    newHead.ch = 0;
  }

  return { anchor: selection.anchor, head: newHead };
};

export const select1LinesDown = (
  editor: Editor, 
  selection: EditorSelection
) => {
  let newHead: EditorPosition = {
    line: selection.head.line + 1,
    ch: selection.head.ch
  };

  // Ensure the new head position is within the bounds of the line
  if (newHead.line < editor.lineCount()) {
    const lineBelowLength = editor.getLine(newHead.line).length;
    if (newHead.ch > lineBelowLength) {
      newHead.ch = 0;
    }
  } else {
    // Prevent moving the head below the last line
    newHead.line = editor.lineCount() - 1;
    newHead.ch = editor.getLine(newHead.line).length;
  }

  return { anchor: selection.anchor, head: newHead };
};

export const select1CharLeft = (
  _editor: Editor, 
  selection: EditorSelection
) => {
    const new_hd: EditorPosition = {
      line:selection.head.line, 
      ch:selection.head.ch-1
    };

    return { anchor: selection.anchor, head: new_hd };
};

export const select1CharRight = (
  _editor: Editor, 
  selection: EditorSelection
) => {
    const new_hd: EditorPosition = {
      line:selection.head.line, 
      ch:selection.head.ch+1
    };

    return { anchor: selection.anchor, head: new_hd };
};

// Define a reusable variable for punctuation characters
const punctuationChars = /[\s.,!?;:-_]/;

export const select1WordLeft = (
  editor: Editor,
  selection: EditorSelection
) => {
  const { anchor, head } = selection;  // Destructure the anchor and head positions from the selection
  const line = editor.getLine(head.line);  // Get the text of the line where the head is currently located
  let newHead = { ...head };  // Create a new position object for the head to track the updated cursor position

  // Handle the case where the cursor is at the beginning of the line and needs to move to the previous line
  if (head.ch === 0 && head.line > 0) {
    newHead.line--;  // Move up one line
    newHead.ch = editor.getLine(newHead.line).length;  // Set the cursor to the end of the previous line
  } else {
    // Move left to the end of the next word or punctuation group
    let ch = head.ch;
    if (ch > 0 && !/[\s]/.test(line[ch-1]) && !punctuationChars.test(line[ch-1])) {
      // If the character is part of a word, move left until the end of the word is found
      while (ch > 0 && !/[\s]/.test(line[ch-1]) && !punctuationChars.test(line[ch-1])) ch--;
    } else if (ch > 0 && punctuationChars.test(line[ch-1])) {
      // If the character is a punctuation mark, move left until the end of the punctuation group is found
      while (ch > 0 && punctuationChars.test(line[ch-1])) ch--;
    } else {
      ch--;  // If none of the above, simply move one character to the left
    }
    newHead.ch = ch;  // Update the head position with the new character position
  }

  // Adjust selection based on direction
  if (editor.posToOffset(newHead) < editor.posToOffset(anchor)) {
    // If the new head position is before the anchor, expand the selection to the left
    return { anchor, head: newHead };
  } else if (editor.posToOffset(newHead) > editor.posToOffset(anchor) && head.ch !== anchor.ch) {
    // If the new head position is after the anchor and the selection is non-collapsed, shrink the selection from the right
    return { anchor, head: newHead };
  } else {
    // If neither condition is met, maintain the current selection
    return { anchor: newHead, head: anchor };
  }
};

export const select1WordRight = (
  editor: Editor,
  selection: EditorSelection
) => {
  const { anchor, head } = selection;
  const line = editor.getLine(head.line);
  let newHead = { ...head };

  // Handle moving across lines
  if (head.ch === line.length && head.line < editor.lineCount() - 1) {
    newHead.line++;
    newHead.ch = 0;
  } else {
    // Move right to the end of the next word or punctuation group
    let ch = head.ch;
    if (ch < line.length && !/[\s]/.test(line[ch]) && !punctuationChars.test(line[ch])) {
      while (ch < line.length && !/[\s]/.test(line[ch]) && !punctuationChars.test(line[ch])) ch++;
    } else if (ch < line.length && punctuationChars.test(line[ch])) {
      while (ch < line.length && punctuationChars.test(line[ch])) ch++;
    } else {
      ch++;
    }
    newHead.ch = ch;
  }

  // Determine if we should shrink or expand the selection
  if (editor.posToOffset(newHead) > editor.posToOffset(head)) {
    // Shrink the selection from the left
    return { anchor, head: newHead };
  } else {
    // Expand the selection to the right
    return { anchor: newHead, head: anchor };
  }
};

in main.ts

import {
  // ...
  select1LinesUp,
  select1LinesDown,
  select1CharLeft,
  select1CharRight,
  select1WordLeft,
  select1WordRight,
  // ...
} from './actions';
 // ...
export default class CodeEditorShortcuts extends Plugin {
  settings: PluginSettings;
  async onload() {
    await this.loadSettings();
     // ... 
    this.addCommand({
      id: 'select1LinesUp',
      name: 'Select 1 lines up',
      editorCallback: (editor) => withMultipleSelections(editor, select1LinesUp),
    });

    this.addCommand({
      id: 'select1LinesDown',
      name: 'Select 1 lines Down',
      editorCallback: (editor) => withMultipleSelections(editor, select1LinesDown),
    });

    this.addCommand({
      id: 'select1CharLeft',
      name: 'Select 1 character left',
      editorCallback: (editor) => withMultipleSelections(editor, select1CharLeft),
    });

    this.addCommand({
      id: 'select1CharRight',
      name: 'Select 1 character right',
      editorCallback: (editor) => withMultipleSelections(editor, select1CharRight),
    });

    this.addCommand({
      id: 'select1WordLeft',
      name: 'Select 1 word left',
      editorCallback: (editor) => withMultipleSelections(editor, select1WordLeft),
    });

    this.addCommand({
      id: 'select1WordRight',
      name: 'Select 1 word right',
      editorCallback: (editor) => withMultipleSelections(editor, select1WordRight),
    });
    // ...
}

Known Bugs:

Additional Feature I would like implemented, word part detection;

IE: camelCase = 2 words not 1.

Mikestriken commented 3 months ago
Here is the compiled main.js code: ```js var __create = Object.create; var __defProp = Object.defineProperty; var __defProps = Object.defineProperties; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropDescs = Object.getOwnPropertyDescriptors; var __getOwnPropNames = Object.getOwnPropertyNames; var __getOwnPropSymbols = Object.getOwnPropertySymbols; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __propIsEnum = Object.prototype.propertyIsEnumerable; var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __spreadValues = (a, b) => { for (var prop in b || (b = {})) if (__hasOwnProp.call(b, prop)) __defNormalProp(a, prop, b[prop]); if (__getOwnPropSymbols) for (var prop of __getOwnPropSymbols(b)) { if (__propIsEnum.call(b, prop)) __defNormalProp(a, prop, b[prop]); } return a; }; var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b)); var __markAsModule = (target) => __defProp(target, "__esModule", { value: true }); var __export = (target, all) => { __markAsModule(target); for (var name in all) __defProp(target, name, { get: all[name], enumerable: true }); }; var __reExport = (target, module2, desc) => { if (module2 && typeof module2 === "object" || typeof module2 === "function") { for (let key of __getOwnPropNames(module2)) if (!__hasOwnProp.call(target, key) && key !== "default") __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable }); } return target; }; var __toModule = (module2) => { return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2); }; var __async = (__this, __arguments, generator) => { return new Promise((resolve, reject) => { var fulfilled = (value) => { try { step(generator.next(value)); } catch (e) { reject(e); } }; var rejected = (value) => { try { step(generator.throw(value)); } catch (e) { reject(e); } }; var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected); step((generator = generator.apply(__this, __arguments)).next()); }); }; // src/main.ts __export(exports, { default: () => CodeEditorShortcuts }); var import_obsidian3 = __toModule(require("obsidian")); // src/constants.ts var CASE; (function(CASE2) { CASE2["UPPER"] = "upper"; CASE2["LOWER"] = "lower"; CASE2["TITLE"] = "title"; CASE2["NEXT"] = "next"; })(CASE || (CASE = {})); var LOWERCASE_ARTICLES = ["the", "a", "an"]; var SEARCH_DIRECTION; (function(SEARCH_DIRECTION2) { SEARCH_DIRECTION2["FORWARD"] = "forward"; SEARCH_DIRECTION2["BACKWARD"] = "backward"; })(SEARCH_DIRECTION || (SEARCH_DIRECTION = {})); var MATCHING_BRACKETS = { "[": "]", "(": ")", "{": "}" }; var MATCHING_QUOTES = { "'": "'", '"': '"', "`": "`" }; var MATCHING_QUOTES_BRACKETS = __spreadValues(__spreadValues({}, MATCHING_QUOTES), MATCHING_BRACKETS); var CODE_EDITOR; (function(CODE_EDITOR2) { CODE_EDITOR2["SUBLIME"] = "sublime"; CODE_EDITOR2["VSCODE"] = "vscode"; })(CODE_EDITOR || (CODE_EDITOR = {})); var MODIFIER_KEYS = [ "Control", "Shift", "Alt", "Meta", "CapsLock", "Fn" ]; var LIST_CHARACTER_REGEX = /^\s*(-|\+|\*|\d+\.|>) (\[.\] )?/; // src/state.ts var SettingsState = { autoInsertListPrefix: true }; // src/utils.ts var defaultMultipleSelectionOptions = { repeatSameLineActions: true }; var withMultipleSelectionsNew = (editor, callback, options = defaultMultipleSelectionOptions) => { const selections = editor.listSelections(); let selectionIndexesToProcess; const newSelections = []; const changes = []; if (!options.repeatSameLineActions) { const seenLines = []; selectionIndexesToProcess = selections.reduce((indexes, currSelection, currIndex) => { const currentLine = currSelection.head.line; if (!seenLines.includes(currentLine)) { seenLines.push(currentLine); indexes.push(currIndex); } return indexes; }, []); } for (let i = 0; i < selections.length; i++) { if (selectionIndexesToProcess && !selectionIndexesToProcess.includes(i)) { continue; } const { changes: newChanges, newSelection } = callback(editor, selections[i], __spreadProps(__spreadValues({}, options.args), { iteration: i })); changes.push(...newChanges); if (options.combineSameLineSelections) { const existingSameLineSelection = newSelections.find((selection) => selection.from.line === newSelection.from.line); if (existingSameLineSelection) { existingSameLineSelection.from.ch = 0; continue; } } newSelections.push(newSelection); } editor.transaction({ changes, selections: newSelections }); }; var withMultipleSelections = (editor, callback, options = defaultMultipleSelectionOptions) => { const { cm } = editor; const selections = editor.listSelections(); let selectionIndexesToProcess; let newSelections = []; if (!options.repeatSameLineActions) { const seenLines = []; selectionIndexesToProcess = selections.reduce((indexes, currSelection, currIndex) => { const currentLine = currSelection.head.line; if (!seenLines.includes(currentLine)) { seenLines.push(currentLine); indexes.push(currIndex); } return indexes; }, []); } const applyCallbackOnSelections = () => { for (let i = 0; i < selections.length; i++) { if (selectionIndexesToProcess && !selectionIndexesToProcess.includes(i)) { continue; } const selection = editor.listSelections()[i]; if (selection) { const newSelection = callback(editor, selection, options.args); newSelections.push(newSelection); } } if (options.customSelectionHandler) { newSelections = options.customSelectionHandler(newSelections); } editor.setSelections(newSelections); }; if (cm && cm.operation) { cm.operation(applyCallbackOnSelections); } else { console.debug("cm object not found, operations will not be buffered"); applyCallbackOnSelections(); } }; var iterateCodeMirrorDivs = (callback) => { let codeMirrors; codeMirrors = document.querySelectorAll(".cm-content"); if (codeMirrors.length === 0) { codeMirrors = document.querySelectorAll(".CodeMirror"); } codeMirrors.forEach(callback); }; var getLineStartPos = (line) => ({ line, ch: 0 }); var getLineEndPos = (line, editor) => ({ line, ch: editor.getLine(line).length }); var getSelectionBoundaries = (selection) => { let { anchor: from, head: to } = selection; if (from.line > to.line) { [from, to] = [to, from]; } if (from.line === to.line && from.ch > to.ch) { [from, to] = [to, from]; } return { from, to, hasTrailingNewline: to.line > from.line && to.ch === 0 }; }; var getLeadingWhitespace = (lineContent) => { const indentation = lineContent.match(/^\s+/); return indentation ? indentation[0] : ""; }; var isLetterCharacter = (char) => /\p{L}\p{M}*/u.test(char); var isDigit = (char) => /\d/.test(char); var isLetterOrDigit = (char) => isLetterCharacter(char) || isDigit(char); var wordRangeAtPos = (pos, lineContent) => { let start = pos.ch; let end = pos.ch; while (start > 0 && isLetterOrDigit(lineContent.charAt(start - 1))) { start--; } while (end < lineContent.length && isLetterOrDigit(lineContent.charAt(end))) { end++; } return { anchor: { line: pos.line, ch: start }, head: { line: pos.line, ch: end } }; }; var findPosOfNextCharacter = ({ editor, startPos, checkCharacter, searchDirection }) => { let { line, ch } = startPos; let lineContent = editor.getLine(line); let matchFound = false; let matchedChar; if (searchDirection === SEARCH_DIRECTION.BACKWARD) { while (line >= 0) { const char = lineContent.charAt(Math.max(ch - 1, 0)); matchFound = checkCharacter(char); if (matchFound) { matchedChar = char; break; } ch--; if (ch <= 0) { line--; if (line >= 0) { lineContent = editor.getLine(line); ch = lineContent.length; } } } } else { while (line < editor.lineCount()) { const char = lineContent.charAt(ch); matchFound = checkCharacter(char); if (matchFound) { matchedChar = char; break; } ch++; if (ch >= lineContent.length) { line++; lineContent = editor.getLine(line); ch = 0; } } } return matchFound ? { match: matchedChar, pos: { line, ch } } : null; }; var hasSameSelectionContent = (editor, selections) => new Set(selections.map((selection) => { const { from, to } = getSelectionBoundaries(selection); return editor.getRange(from, to); })).size === 1; var getSearchText = ({ editor, allSelections, autoExpand }) => { const singleSearchText = hasSameSelectionContent(editor, allSelections); const firstSelection = allSelections[0]; const { from, to } = getSelectionBoundaries(firstSelection); let searchText = editor.getRange(from, to); if (searchText.length === 0 && autoExpand) { const wordRange = wordRangeAtPos(from, editor.getLine(from.line)); searchText = editor.getRange(wordRange.anchor, wordRange.head); } return { searchText, singleSearchText }; }; var escapeRegex = (input) => input.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); var withWordBoundaries = (input) => `(?<=\\W|^)${input}(?=\\W|$)`; var findAllMatches = ({ searchText, searchWithinWords, documentContent }) => { const escapedSearchText = escapeRegex(searchText); const searchExpression = new RegExp(searchWithinWords ? escapedSearchText : withWordBoundaries(escapedSearchText), "g"); return Array.from(documentContent.matchAll(searchExpression)); }; var findNextMatchPosition = ({ editor, latestMatchPos, searchText, searchWithinWords, documentContent }) => { const latestMatchOffset = editor.posToOffset(latestMatchPos); const matches = findAllMatches({ searchText, searchWithinWords, documentContent }); let nextMatch = null; for (const match of matches) { if (match.index > latestMatchOffset) { nextMatch = { anchor: editor.offsetToPos(match.index), head: editor.offsetToPos(match.index + searchText.length) }; break; } } if (!nextMatch) { const selectionIndexes = editor.listSelections().map((selection) => { const { from } = getSelectionBoundaries(selection); return editor.posToOffset(from); }); for (const match of matches) { if (!selectionIndexes.includes(match.index)) { nextMatch = { anchor: editor.offsetToPos(match.index), head: editor.offsetToPos(match.index + searchText.length) }; break; } } } return nextMatch; }; var findAllMatchPositions = ({ editor, searchText, searchWithinWords, documentContent }) => { const matches = findAllMatches({ searchText, searchWithinWords, documentContent }); const matchPositions = []; for (const match of matches) { matchPositions.push({ anchor: editor.offsetToPos(match.index), head: editor.offsetToPos(match.index + searchText.length) }); } return matchPositions; }; var toTitleCase = (selectedText) => { return selectedText.split(/(\s+)/).map((word, index, allWords) => { if (index > 0 && index < allWords.length - 1 && LOWERCASE_ARTICLES.includes(word.toLowerCase())) { return word.toLowerCase(); } return word.charAt(0).toUpperCase() + word.substring(1).toLowerCase(); }).join(""); }; var getNextCase = (selectedText) => { const textUpper = selectedText.toUpperCase(); const textLower = selectedText.toLowerCase(); const textTitle = toTitleCase(selectedText); switch (selectedText) { case textUpper: { return textLower; } case textLower: { return textTitle; } case textTitle: { return textUpper; } default: { return textUpper; } } }; var isNumeric = (input) => input.length > 0 && !isNaN(+input); var getNextListPrefix = (text, direction) => { const listChars = text.match(LIST_CHARACTER_REGEX); if (listChars && listChars.length > 0) { let prefix = listChars[0].trimStart(); const isEmptyListItem = prefix === listChars.input.trimStart(); if (isEmptyListItem) { return null; } if (isNumeric(prefix) && direction === "after") { prefix = +prefix + 1 + ". "; } if (prefix.startsWith("- [") && !prefix.includes("[ ]")) { prefix = "- [ ] "; } return prefix; } return ""; }; var formatRemainingListPrefixes = (editor, fromLine, indentation) => { const changes = []; for (let i = fromLine; i < editor.lineCount(); i++) { const contentsOfCurrentLine = editor.getLine(i); const listPrefixRegex = new RegExp(`^${indentation}\\d+\\.`); const lineStartsWithNumberPrefix = listPrefixRegex.test(contentsOfCurrentLine); if (!lineStartsWithNumberPrefix) { break; } const replacementContent = contentsOfCurrentLine.replace(/\d+\./, (match) => +match + 1 + "."); changes.push({ from: { line: i, ch: 0 }, to: { line: i, ch: contentsOfCurrentLine.length }, text: replacementContent }); } if (changes.length > 0) { editor.transaction({ changes }); } }; var toggleVaultConfig = (app, setting) => { const value = app.vault.getConfig(setting); setVaultConfig(app, setting, !value); }; var setVaultConfig = (app, setting, value) => { app.vault.setConfig(setting, value); }; // src/actions.ts var insertLineAbove = (editor, selection, args) => { const { line } = selection.head; const startOfCurrentLine = getLineStartPos(line); const contentsOfCurrentLine = editor.getLine(line); const indentation = getLeadingWhitespace(contentsOfCurrentLine); let listPrefix = ""; if (SettingsState.autoInsertListPrefix && line > 0 && editor.getLine(line - 1).trim().length > 0) { listPrefix = getNextListPrefix(contentsOfCurrentLine, "before"); if (isNumeric(listPrefix)) { formatRemainingListPrefixes(editor, line, indentation); } } const changes = [ { from: startOfCurrentLine, text: indentation + listPrefix + "\n" } ]; const newSelection = { from: __spreadProps(__spreadValues({}, startOfCurrentLine), { line: startOfCurrentLine.line + args.iteration, ch: indentation.length + listPrefix.length }) }; return { changes, newSelection }; }; var insertLineBelow = (editor, selection, args) => { const { line } = selection.head; const startOfCurrentLine = getLineStartPos(line); const endOfCurrentLine = getLineEndPos(line, editor); const contentsOfCurrentLine = editor.getLine(line); const indentation = getLeadingWhitespace(contentsOfCurrentLine); let listPrefix = ""; if (SettingsState.autoInsertListPrefix) { listPrefix = getNextListPrefix(contentsOfCurrentLine, "after"); if (listPrefix === null) { const changes2 = [ { from: startOfCurrentLine, to: endOfCurrentLine, text: "" } ]; const newSelection2 = { from: { line, ch: 0 } }; return { changes: changes2, newSelection: newSelection2 }; } if (isNumeric(listPrefix)) { formatRemainingListPrefixes(editor, line + 1, indentation); } } const changes = [ { from: endOfCurrentLine, text: "\n" + indentation + listPrefix } ]; const newSelection = { from: { line: line + 1 + args.iteration, ch: indentation.length + listPrefix.length } }; return { changes, newSelection }; }; var numLinesDeleted = 0; var deleteLine = (editor, selection, args) => { const { from, to, hasTrailingNewline } = getSelectionBoundaries(selection); if (to.line === editor.lastLine()) { const previousLine = Math.max(0, from.line - 1); const endOfPreviousLine = getLineEndPos(previousLine, editor); const changes2 = [ { from: from.line === 0 ? getLineStartPos(0) : endOfPreviousLine, to: to.ch === 0 ? getLineStartPos(to.line) : getLineEndPos(to.line, editor), text: "" } ]; const newSelection2 = { from: { line: previousLine, ch: Math.min(from.ch, endOfPreviousLine.ch) } }; return { changes: changes2, newSelection: newSelection2 }; } if (args.iteration === 0) { numLinesDeleted = 0; } const toLine = hasTrailingNewline ? to.line - 1 : to.line; const endOfNextLine = getLineEndPos(toLine + 1, editor); const changes = [ { from: getLineStartPos(from.line), to: getLineStartPos(toLine + 1), text: "" } ]; const newSelection = { from: { line: from.line - numLinesDeleted, ch: Math.min(to.ch, endOfNextLine.ch) } }; numLinesDeleted += toLine - from.line + 1; return { changes, newSelection }; }; var deleteToStartOfLine = (editor, selection) => { const pos = selection.head; let startPos = getLineStartPos(pos.line); if (pos.line === 0 && pos.ch === 0) { return selection; } if (pos.line === startPos.line && pos.ch === startPos.ch) { startPos = getLineEndPos(pos.line - 1, editor); } editor.replaceRange("", startPos, pos); return { anchor: startPos }; }; var deleteToEndOfLine = (editor, selection) => { const pos = selection.head; let endPos = getLineEndPos(pos.line, editor); if (pos.line === endPos.line && pos.ch === endPos.ch) { endPos = getLineStartPos(pos.line + 1); } editor.replaceRange("", pos, endPos); return { anchor: pos }; }; var joinLines = (editor, selection) => { var _a, _b; const { from, to } = getSelectionBoundaries(selection); const { line } = from; let endOfCurrentLine = getLineEndPos(line, editor); const joinRangeLimit = Math.max(to.line - line, 1); const selectionLength = editor.posToOffset(to) - editor.posToOffset(from); let trimmedChars = ""; for (let i = 0; i < joinRangeLimit; i++) { if (line === editor.lineCount() - 1) { break; } endOfCurrentLine = getLineEndPos(line, editor); const endOfNextLine = getLineEndPos(line + 1, editor); const contentsOfCurrentLine = editor.getLine(line); const contentsOfNextLine = editor.getLine(line + 1); const charsToTrim = (_a = contentsOfNextLine.match(LIST_CHARACTER_REGEX)) != null ? _a : []; trimmedChars += (_b = charsToTrim[0]) != null ? _b : ""; const newContentsOfNextLine = contentsOfNextLine.replace(LIST_CHARACTER_REGEX, ""); if (newContentsOfNextLine.length > 0 && contentsOfCurrentLine.charAt(endOfCurrentLine.ch - 1) !== " ") { editor.replaceRange(" " + newContentsOfNextLine, endOfCurrentLine, endOfNextLine); } else { editor.replaceRange(newContentsOfNextLine, endOfCurrentLine, endOfNextLine); } } if (selectionLength === 0) { return { anchor: endOfCurrentLine }; } return { anchor: from, head: { line: from.line, ch: from.ch + selectionLength - trimmedChars.length } }; }; var copyLine = (editor, selection, direction) => { const { from, to, hasTrailingNewline } = getSelectionBoundaries(selection); const fromLineStart = getLineStartPos(from.line); const toLine = hasTrailingNewline ? to.line - 1 : to.line; const toLineEnd = getLineEndPos(toLine, editor); const contentsOfSelectedLines = editor.getRange(fromLineStart, toLineEnd); if (direction === "up") { editor.replaceRange("\n" + contentsOfSelectedLines, toLineEnd); return selection; } else { editor.replaceRange(contentsOfSelectedLines + "\n", fromLineStart); const linesSelected = to.line - from.line + 1; return { anchor: { line: toLine + 1, ch: from.ch }, head: { line: toLine + linesSelected, ch: to.ch } }; } }; var isManualSelection = true; var setIsManualSelection = (value) => { isManualSelection = value; }; var isProgrammaticSelectionChange = false; var setIsProgrammaticSelectionChange = (value) => { isProgrammaticSelectionChange = value; }; var selectWordOrNextOccurrence = (editor) => { setIsProgrammaticSelectionChange(true); const allSelections = editor.listSelections(); const { searchText, singleSearchText } = getSearchText({ editor, allSelections, autoExpand: false }); if (searchText.length > 0 && singleSearchText) { const { from: latestMatchPos } = getSelectionBoundaries(allSelections[allSelections.length - 1]); const nextMatch = findNextMatchPosition({ editor, latestMatchPos, searchText, searchWithinWords: isManualSelection, documentContent: editor.getValue() }); const newSelections = nextMatch ? allSelections.concat(nextMatch) : allSelections; editor.setSelections(newSelections); const lastSelection = newSelections[newSelections.length - 1]; editor.scrollIntoView(getSelectionBoundaries(lastSelection)); } else { const newSelections = []; for (const selection of allSelections) { const { from, to } = getSelectionBoundaries(selection); if (from.line !== to.line || from.ch !== to.ch) { newSelections.push(selection); } else { newSelections.push(wordRangeAtPos(from, editor.getLine(from.line))); setIsManualSelection(false); } } editor.setSelections(newSelections); } }; var selectAllOccurrences = (editor) => { const allSelections = editor.listSelections(); const { searchText, singleSearchText } = getSearchText({ editor, allSelections, autoExpand: true }); if (!singleSearchText) { return; } const matches = findAllMatchPositions({ editor, searchText, searchWithinWords: true, documentContent: editor.getValue() }); editor.setSelections(matches); }; var selectLine = (_editor, selection) => { const { from, to } = getSelectionBoundaries(selection); const startOfCurrentLine = getLineStartPos(from.line); const startOfNextLine = getLineStartPos(to.line + 1); return { anchor: startOfCurrentLine, head: startOfNextLine }; }; var addCursorsToSelectionEnds = (editor, emulate = CODE_EDITOR.VSCODE) => { if (editor.listSelections().length !== 1) { return; } const selection = editor.listSelections()[0]; const { from, to, hasTrailingNewline } = getSelectionBoundaries(selection); const newSelections = []; const toLine = hasTrailingNewline ? to.line - 1 : to.line; for (let line = from.line; line <= toLine; line++) { const head = line === to.line ? to : getLineEndPos(line, editor); let anchor; if (emulate === CODE_EDITOR.VSCODE) { anchor = head; } else { anchor = line === from.line ? from : getLineStartPos(line); } newSelections.push({ anchor, head }); } editor.setSelections(newSelections); }; var goToLineBoundary = (editor, selection, boundary) => { const { from, to } = getSelectionBoundaries(selection); if (boundary === "start") { return { anchor: getLineStartPos(from.line) }; } else { return { anchor: getLineEndPos(to.line, editor) }; } }; var navigateLine = (editor, selection, position) => { const pos = selection.head; let line; let ch; if (position === "prev") { line = Math.max(pos.line - 1, 0); const endOfLine = getLineEndPos(line, editor); ch = Math.min(pos.ch, endOfLine.ch); } if (position === "next") { line = Math.min(pos.line + 1, editor.lineCount() - 1); const endOfLine = getLineEndPos(line, editor); ch = Math.min(pos.ch, endOfLine.ch); } if (position === "first") { line = 0; ch = 0; } if (position === "last") { line = editor.lineCount() - 1; const endOfLine = getLineEndPos(line, editor); ch = endOfLine.ch; } return { anchor: { line, ch } }; }; var moveCursor = (editor, direction) => { switch (direction) { case "up": editor.exec("goUp"); break; case "down": editor.exec("goDown"); break; case "left": editor.exec("goLeft"); break; case "right": editor.exec("goRight"); break; } }; var select1LinesUp = (editor, selection) => { let newHead = { line: selection.head.line - 1, ch: selection.head.ch }; if (newHead.line >= 0) { const lineAboveLength = editor.getLine(newHead.line).length; if (newHead.ch > lineAboveLength) { newHead.ch = lineAboveLength; } } else { newHead.line = 0; newHead.ch = 0; } return { anchor: selection.anchor, head: newHead }; }; var select1LinesDown = (editor, selection) => { let newHead = { line: selection.head.line + 1, ch: selection.head.ch }; if (newHead.line < editor.lineCount()) { const lineBelowLength = editor.getLine(newHead.line).length; if (newHead.ch > lineBelowLength) { newHead.ch = 0; } } else { newHead.line = editor.lineCount() - 1; newHead.ch = editor.getLine(newHead.line).length; } return { anchor: selection.anchor, head: newHead }; }; var select1CharLeft = (_editor, selection) => { const new_hd = { line: selection.head.line, ch: selection.head.ch - 1 }; return { anchor: selection.anchor, head: new_hd }; }; var select1CharRight = (_editor, selection) => { const new_hd = { line: selection.head.line, ch: selection.head.ch + 1 }; return { anchor: selection.anchor, head: new_hd }; }; var punctuationChars = /[\s.,!?;:-_]/; var select1WordLeft = (editor, selection) => { const { anchor, head } = selection; const line = editor.getLine(head.line); let newHead = __spreadValues({}, head); if (head.ch === 0 && head.line > 0) { newHead.line--; newHead.ch = editor.getLine(newHead.line).length; } else { let ch = head.ch; if (ch > 0 && !/[\s]/.test(line[ch - 1]) && !punctuationChars.test(line[ch - 1])) { while (ch > 0 && !/[\s]/.test(line[ch - 1]) && !punctuationChars.test(line[ch - 1])) ch--; } else if (ch > 0 && punctuationChars.test(line[ch - 1])) { while (ch > 0 && punctuationChars.test(line[ch - 1])) ch--; } else { ch--; } newHead.ch = ch; } if (editor.posToOffset(newHead) < editor.posToOffset(anchor)) { return { anchor, head: newHead }; } else if (editor.posToOffset(newHead) > editor.posToOffset(anchor) && head.ch !== anchor.ch) { return { anchor, head: newHead }; } else { return { anchor: newHead, head: anchor }; } }; var select1WordRight = (editor, selection) => { const { anchor, head } = selection; const line = editor.getLine(head.line); let newHead = __spreadValues({}, head); if (head.ch === line.length && head.line < editor.lineCount() - 1) { newHead.line++; newHead.ch = 0; } else { let ch = head.ch; if (ch < line.length && !/[\s]/.test(line[ch]) && !punctuationChars.test(line[ch])) { while (ch < line.length && !/[\s]/.test(line[ch]) && !punctuationChars.test(line[ch])) ch++; } else if (ch < line.length && punctuationChars.test(line[ch])) { while (ch < line.length && punctuationChars.test(line[ch])) ch++; } else { ch++; } newHead.ch = ch; } if (editor.posToOffset(newHead) > editor.posToOffset(head)) { return { anchor, head: newHead }; } else { return { anchor: newHead, head: anchor }; } }; var moveWord = (editor, direction) => { switch (direction) { case "left": editor.exec("goWordLeft"); break; case "right": editor.exec("goWordRight"); break; } }; var transformCase = (editor, selection, caseType) => { let { from, to } = getSelectionBoundaries(selection); let selectedText = editor.getRange(from, to); if (selectedText.length === 0) { const pos = selection.head; const { anchor, head } = wordRangeAtPos(pos, editor.getLine(pos.line)); [from, to] = [anchor, head]; selectedText = editor.getRange(anchor, head); } let replacementText = selectedText; switch (caseType) { case CASE.UPPER: { replacementText = selectedText.toUpperCase(); break; } case CASE.LOWER: { replacementText = selectedText.toLowerCase(); break; } case CASE.TITLE: { replacementText = toTitleCase(selectedText); break; } case CASE.NEXT: { replacementText = getNextCase(selectedText); break; } } editor.replaceRange(replacementText, from, to); return selection; }; var expandSelection = ({ editor, selection, openingCharacterCheck, matchingCharacterMap }) => { let { anchor, head } = selection; if (anchor.line >= head.line && anchor.ch > anchor.ch) { [anchor, head] = [head, anchor]; } const newAnchor = findPosOfNextCharacter({ editor, startPos: anchor, checkCharacter: openingCharacterCheck, searchDirection: SEARCH_DIRECTION.BACKWARD }); if (!newAnchor) { return selection; } const newHead = findPosOfNextCharacter({ editor, startPos: head, checkCharacter: (char) => char === matchingCharacterMap[newAnchor.match], searchDirection: SEARCH_DIRECTION.FORWARD }); if (!newHead) { return selection; } return { anchor: newAnchor.pos, head: newHead.pos }; }; var expandSelectionToBrackets = (editor, selection) => expandSelection({ editor, selection, openingCharacterCheck: (char) => /[([{]/.test(char), matchingCharacterMap: MATCHING_BRACKETS }); var expandSelectionToQuotes = (editor, selection) => expandSelection({ editor, selection, openingCharacterCheck: (char) => /['"`]/.test(char), matchingCharacterMap: MATCHING_QUOTES }); var expandSelectionToQuotesOrBrackets = (editor) => { const selections = editor.listSelections(); const newSelection = expandSelection({ editor, selection: selections[0], openingCharacterCheck: (char) => /['"`([{]/.test(char), matchingCharacterMap: MATCHING_QUOTES_BRACKETS }); editor.setSelections([...selections, newSelection]); }; var insertCursor = (editor, lineOffset) => { const selections = editor.listSelections(); const newSelections = []; for (const selection of selections) { const { line, ch } = selection.head; if (line === 0 && lineOffset < 0 || line === editor.lastLine() && lineOffset > 0) { break; } const targetLineLength = editor.getLine(line + lineOffset).length; newSelections.push({ anchor: { line: selection.anchor.line + lineOffset, ch: Math.min(selection.anchor.ch, targetLineLength) }, head: { line: line + lineOffset, ch: Math.min(ch, targetLineLength) } }); } editor.setSelections([...editor.listSelections(), ...newSelections]); }; var insertCursorAbove = (editor) => insertCursor(editor, -1); var insertCursorBelow = (editor) => insertCursor(editor, 1); var goToHeading = (app, editor, boundary) => { const file = app.metadataCache.getFileCache(app.workspace.getActiveFile()); if (!file.headings || file.headings.length === 0) { return; } const { line } = editor.getCursor("from"); let prevHeadingLine = 0; let nextHeadingLine = editor.lastLine(); file.headings.forEach(({ position }) => { const { end: headingPos } = position; if (line > headingPos.line && headingPos.line > prevHeadingLine) { prevHeadingLine = headingPos.line; } if (line < headingPos.line && headingPos.line < nextHeadingLine) { nextHeadingLine = headingPos.line; } }); editor.setSelection(boundary === "prev" ? getLineEndPos(prevHeadingLine, editor) : getLineEndPos(nextHeadingLine, editor)); }; // src/settings.ts var import_obsidian = __toModule(require("obsidian")); var DEFAULT_SETTINGS = { autoInsertListPrefix: true }; var SettingTab = class extends import_obsidian.PluginSettingTab { constructor(app, plugin) { super(app, plugin); this.plugin = plugin; } display() { const { containerEl } = this; containerEl.empty(); containerEl.createEl("h2", { text: "Code Editor Shortcuts" }); const listPrefixSetting = new import_obsidian.Setting(containerEl).setName("Auto insert list prefix").setDesc("Automatically insert list prefix when inserting a line above or below").addToggle((toggle) => toggle.setValue(this.plugin.settings.autoInsertListPrefix).onChange((value) => __async(this, null, function* () { this.plugin.settings.autoInsertListPrefix = value; yield this.plugin.saveSettings(); }))); new import_obsidian.Setting(containerEl).setName("Reset defaults").addButton((btn) => { btn.setButtonText("Reset").onClick(() => __async(this, null, function* () { this.plugin.settings = __spreadValues({}, DEFAULT_SETTINGS); listPrefixSetting.components[0].setValue(DEFAULT_SETTINGS.autoInsertListPrefix); yield this.plugin.saveSettings(); })); }); } }; // src/modals.ts var import_obsidian2 = __toModule(require("obsidian")); var GoToLineModal = class extends import_obsidian2.SuggestModal { constructor(app, lineCount, onSubmit) { super(app); this.lineCount = lineCount; this.onSubmit = onSubmit; const PROMPT_TEXT = `Enter a line number between 1 and ${lineCount}`; this.limit = 1; this.setPlaceholder(PROMPT_TEXT); this.emptyStateText = PROMPT_TEXT; } getSuggestions(line) { const lineNumber = parseInt(line); if (line.length > 0 && lineNumber > 0 && lineNumber <= this.lineCount) { return [line]; } return []; } renderSuggestion(line, el) { el.createEl("div", { text: line }); } onChooseSuggestion(line) { this.onSubmit(parseInt(line) - 1); } }; // src/main.ts var CodeEditorShortcuts = class extends import_obsidian3.Plugin { onload() { return __async(this, null, function* () { yield this.loadSettings(); this.addCommand({ id: "insertLineAbove", name: "Insert line above", hotkeys: [ { modifiers: ["Mod", "Shift"], key: "Enter" } ], editorCallback: (editor) => withMultipleSelectionsNew(editor, insertLineAbove) }); this.addCommand({ id: "insertLineBelow", name: "Insert line below", hotkeys: [ { modifiers: ["Mod"], key: "Enter" } ], editorCallback: (editor) => withMultipleSelectionsNew(editor, insertLineBelow) }); this.addCommand({ id: "deleteLine", name: "Delete line", hotkeys: [ { modifiers: ["Mod", "Shift"], key: "K" } ], editorCallback: (editor) => withMultipleSelectionsNew(editor, deleteLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { combineSameLineSelections: true })) }); this.addCommand({ id: "deleteToStartOfLine", name: "Delete to start of line", editorCallback: (editor) => withMultipleSelections(editor, deleteToStartOfLine) }); this.addCommand({ id: "deleteToEndOfLine", name: "Delete to end of line", editorCallback: (editor) => withMultipleSelections(editor, deleteToEndOfLine) }); this.addCommand({ id: "joinLines", name: "Join lines", hotkeys: [ { modifiers: ["Mod"], key: "J" } ], editorCallback: (editor) => withMultipleSelections(editor, joinLines, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { repeatSameLineActions: false })) }); this.addCommand({ id: "duplicateLine", name: "Duplicate line", hotkeys: [ { modifiers: ["Mod", "Shift"], key: "D" } ], editorCallback: (editor) => withMultipleSelections(editor, copyLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "down" })) }); this.addCommand({ id: "copyLineUp", name: "Copy line up", hotkeys: [ { modifiers: ["Alt", "Shift"], key: "ArrowUp" } ], editorCallback: (editor) => withMultipleSelections(editor, copyLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "up" })) }); this.addCommand({ id: "copyLineDown", name: "Copy line down", hotkeys: [ { modifiers: ["Alt", "Shift"], key: "ArrowDown" } ], editorCallback: (editor) => withMultipleSelections(editor, copyLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "down" })) }); this.addCommand({ id: "selectWordOrNextOccurrence", name: "Select word or next occurrence", hotkeys: [ { modifiers: ["Mod"], key: "D" } ], editorCallback: (editor) => selectWordOrNextOccurrence(editor) }); this.addCommand({ id: "selectAllOccurrences", name: "Select all occurrences", hotkeys: [ { modifiers: ["Mod", "Shift"], key: "L" } ], editorCallback: (editor) => selectAllOccurrences(editor) }); this.addCommand({ id: "selectLine", name: "Select line", hotkeys: [ { modifiers: ["Mod"], key: "L" } ], editorCallback: (editor) => withMultipleSelections(editor, selectLine) }); this.addCommand({ id: "addCursorsToSelectionEnds", name: "Add cursors to selection ends", hotkeys: [ { modifiers: ["Alt", "Shift"], key: "I" } ], editorCallback: (editor) => addCursorsToSelectionEnds(editor) }); this.addCommand({ id: "goToLineStart", name: "Go to start of line", editorCallback: (editor) => withMultipleSelections(editor, goToLineBoundary, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "start" })) }); this.addCommand({ id: "goToLineEnd", name: "Go to end of line", editorCallback: (editor) => withMultipleSelections(editor, goToLineBoundary, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "end" })) }); this.addCommand({ id: "goToNextLine", name: "Go to next line", editorCallback: (editor) => withMultipleSelections(editor, navigateLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "next" })) }); this.addCommand({ id: "goToPrevLine", name: "Go to previous line", editorCallback: (editor) => withMultipleSelections(editor, navigateLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "prev" })) }); this.addCommand({ id: "goToFirstLine", name: "Go to first line", editorCallback: (editor) => withMultipleSelections(editor, navigateLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "first" })) }); this.addCommand({ id: "goToLastLine", name: "Go to last line", editorCallback: (editor) => withMultipleSelections(editor, navigateLine, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: "last" })) }); this.addCommand({ id: "goToLineNumber", name: "Go to line number", editorCallback: (editor) => { const lineCount = editor.lineCount(); const onSubmit = (line) => editor.setCursor({ line, ch: 0 }); new GoToLineModal(this.app, lineCount, onSubmit).open(); } }); this.addCommand({ id: "goToNextChar", name: "Move cursor forward", editorCallback: (editor) => moveCursor(editor, "right") }); this.addCommand({ id: "goToPrevChar", name: "Move cursor backward", editorCallback: (editor) => moveCursor(editor, "left") }); this.addCommand({ id: "moveCursorUp", name: "Move cursor up", editorCallback: (editor) => moveCursor(editor, "up") }); this.addCommand({ id: "moveCursorDown", name: "Move cursor down", editorCallback: (editor) => moveCursor(editor, "down") }); this.addCommand({ id: "select1LinesUp", name: "Select 1 lines up", editorCallback: (editor) => withMultipleSelections(editor, select1LinesUp) }); this.addCommand({ id: "select1LinesDown", name: "Select 1 lines Down", editorCallback: (editor) => withMultipleSelections(editor, select1LinesDown) }); this.addCommand({ id: "select1CharLeft", name: "Select 1 character left", editorCallback: (editor) => withMultipleSelections(editor, select1CharLeft) }); this.addCommand({ id: "select1CharRight", name: "Select 1 character right", editorCallback: (editor) => withMultipleSelections(editor, select1CharRight) }); this.addCommand({ id: "select1WordLeft", name: "Select 1 word left", editorCallback: (editor) => withMultipleSelections(editor, select1WordLeft) }); this.addCommand({ id: "select1WordRight", name: "Select 1 word right", editorCallback: (editor) => withMultipleSelections(editor, select1WordRight) }); this.addCommand({ id: "goToPreviousWord", name: "Go to previous word", editorCallback: (editor) => moveWord(editor, "left") }); this.addCommand({ id: "goToNextWord", name: "Go to next word", editorCallback: (editor) => moveWord(editor, "right") }); this.addCommand({ id: "transformToUppercase", name: "Transform selection to uppercase", editorCallback: (editor) => withMultipleSelections(editor, transformCase, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: CASE.UPPER })) }); this.addCommand({ id: "transformToLowercase", name: "Transform selection to lowercase", editorCallback: (editor) => withMultipleSelections(editor, transformCase, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: CASE.LOWER })) }); this.addCommand({ id: "transformToTitlecase", name: "Transform selection to title case", editorCallback: (editor) => withMultipleSelections(editor, transformCase, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: CASE.TITLE })) }); this.addCommand({ id: "toggleCase", name: "Toggle case of selection", editorCallback: (editor) => withMultipleSelections(editor, transformCase, __spreadProps(__spreadValues({}, defaultMultipleSelectionOptions), { args: CASE.NEXT })) }); this.addCommand({ id: "expandSelectionToBrackets", name: "Expand selection to brackets", editorCallback: (editor) => withMultipleSelections(editor, expandSelectionToBrackets) }); this.addCommand({ id: "expandSelectionToQuotes", name: "Expand selection to quotes", editorCallback: (editor) => withMultipleSelections(editor, expandSelectionToQuotes) }); this.addCommand({ id: "expandSelectionToQuotesOrBrackets", name: "Expand selection to quotes or brackets", editorCallback: (editor) => expandSelectionToQuotesOrBrackets(editor) }); this.addCommand({ id: "insertCursorAbove", name: "Insert cursor above", editorCallback: (editor) => insertCursorAbove(editor) }); this.addCommand({ id: "insertCursorBelow", name: "Insert cursor below", editorCallback: (editor) => insertCursorBelow(editor) }); this.addCommand({ id: "goToNextHeading", name: "Go to next heading", editorCallback: (editor) => goToHeading(this.app, editor, "next") }); this.addCommand({ id: "goToPrevHeading", name: "Go to previous heading", editorCallback: (editor) => goToHeading(this.app, editor, "prev") }); this.addCommand({ id: "toggle-line-numbers", name: "Toggle line numbers", callback: () => toggleVaultConfig(this.app, "showLineNumber") }); this.addCommand({ id: "indent-using-tabs", name: "Indent using tabs", callback: () => setVaultConfig(this.app, "useTab", true) }); this.addCommand({ id: "indent-using-spaces", name: "Indent using spaces", callback: () => setVaultConfig(this.app, "useTab", false) }); this.addCommand({ id: "undo", name: "Undo", editorCallback: (editor) => editor.undo() }); this.addCommand({ id: "redo", name: "Redo", editorCallback: (editor) => editor.redo() }); this.registerSelectionChangeListeners(); this.addSettingTab(new SettingTab(this.app, this)); }); } registerSelectionChangeListeners() { this.app.workspace.onLayoutReady(() => { const handleSelectionChange = (evt) => { if (evt instanceof KeyboardEvent && MODIFIER_KEYS.includes(evt.key)) { return; } if (!isProgrammaticSelectionChange) { setIsManualSelection(true); } setIsProgrammaticSelectionChange(false); }; iterateCodeMirrorDivs((cm) => { this.registerDomEvent(cm, "keydown", handleSelectionChange); this.registerDomEvent(cm, "click", handleSelectionChange); this.registerDomEvent(cm, "dblclick", handleSelectionChange); }); }); } loadSettings() { return __async(this, null, function* () { const savedSettings = yield this.loadData(); this.settings = __spreadValues(__spreadValues({}, DEFAULT_SETTINGS), savedSettings); SettingsState.autoInsertListPrefix = this.settings.autoInsertListPrefix; }); } saveSettings() { return __async(this, null, function* () { yield this.saveData(this.settings); SettingsState.autoInsertListPrefix = this.settings.autoInsertListPrefix; }); } }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/constants.ts", "src/state.ts", "src/utils.ts", "src/actions.ts", "src/settings.ts", "src/modals.ts"],
  "sourcesContent": ["import { Plugin } from 'obsidian';\nimport {\n  addCursorsToSelectionEnds,\n  copyLine,\n  deleteLine,\n  deleteToStartOfLine,\n  deleteToEndOfLine,\n  expandSelectionToBrackets,\n  expandSelectionToQuotes,\n  expandSelectionToQuotesOrBrackets,\n  goToHeading,\n  goToLineBoundary,\n  insertLineAbove,\n  insertLineBelow,\n  joinLines,\n  moveCursor,\n  select1LinesUp,\n  select1LinesDown,\n  select1CharLeft,\n  select1CharRight,\n  select1WordLeft,\n  select1WordRight,\n  navigateLine,\n  isProgrammaticSelectionChange,\n  selectAllOccurrences,\n  selectLine,\n  selectWordOrNextOccurrence,\n  setIsManualSelection,\n  setIsProgrammaticSelectionChange,\n  transformCase,\n  insertCursorAbove,\n  insertCursorBelow,\n  moveWord,\n} from './actions';\nimport {\n  defaultMultipleSelectionOptions,\n  iterateCodeMirrorDivs,\n  setVaultConfig,\n  toggleVaultConfig,\n  withMultipleSelections,\n  withMultipleSelectionsNew,\n} from './utils';\nimport { CASE, MODIFIER_KEYS } from './constants';\nimport { SettingTab, DEFAULT_SETTINGS, PluginSettings } from './settings';\nimport { SettingsState } from './state';\nimport { GoToLineModal } from './modals';\n\nexport default class CodeEditorShortcuts extends Plugin {\n  settings: PluginSettings;\n\n  async onload() {\n    await this.loadSettings();\n\n    this.addCommand({\n      id: 'insertLineAbove',\n      name: 'Insert line above',\n      hotkeys: [\n        {\n          modifiers: ['Mod', 'Shift'],\n          key: 'Enter',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelectionsNew(editor, insertLineAbove),\n    });\n\n    this.addCommand({\n      id: 'insertLineBelow',\n      name: 'Insert line below',\n      hotkeys: [\n        {\n          modifiers: ['Mod'],\n          key: 'Enter',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelectionsNew(editor, insertLineBelow),\n    });\n\n    this.addCommand({\n      id: 'deleteLine',\n      name: 'Delete line',\n      hotkeys: [\n        {\n          modifiers: ['Mod', 'Shift'],\n          key: 'K',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelectionsNew(editor, deleteLine, {\n          ...defaultMultipleSelectionOptions,\n          combineSameLineSelections: true,\n        }),\n    });\n\n    this.addCommand({\n      id: 'deleteToStartOfLine',\n      name: 'Delete to start of line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, deleteToStartOfLine),\n    });\n\n    this.addCommand({\n      id: 'deleteToEndOfLine',\n      name: 'Delete to end of line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, deleteToEndOfLine),\n    });\n\n    this.addCommand({\n      id: 'joinLines',\n      name: 'Join lines',\n      hotkeys: [\n        {\n          modifiers: ['Mod'],\n          key: 'J',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, joinLines, {\n          ...defaultMultipleSelectionOptions,\n          repeatSameLineActions: false,\n        }),\n    });\n\n    this.addCommand({\n      id: 'duplicateLine',\n      name: 'Duplicate line',\n      hotkeys: [\n        {\n          modifiers: ['Mod', 'Shift'],\n          key: 'D',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, copyLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'down',\n        }),\n    });\n\n    this.addCommand({\n      id: 'copyLineUp',\n      name: 'Copy line up',\n      hotkeys: [\n        {\n          modifiers: ['Alt', 'Shift'],\n          key: 'ArrowUp',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, copyLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'up',\n        }),\n    });\n\n    this.addCommand({\n      id: 'copyLineDown',\n      name: 'Copy line down',\n      hotkeys: [\n        {\n          modifiers: ['Alt', 'Shift'],\n          key: 'ArrowDown',\n        },\n      ],\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, copyLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'down',\n        }),\n    });\n\n    this.addCommand({\n      id: 'selectWordOrNextOccurrence',\n      name: 'Select word or next occurrence',\n      hotkeys: [\n        {\n          modifiers: ['Mod'],\n          key: 'D',\n        },\n      ],\n      editorCallback: (editor) => selectWordOrNextOccurrence(editor),\n    });\n\n    this.addCommand({\n      id: 'selectAllOccurrences',\n      name: 'Select all occurrences',\n      hotkeys: [\n        {\n          modifiers: ['Mod', 'Shift'],\n          key: 'L',\n        },\n      ],\n      editorCallback: (editor) => selectAllOccurrences(editor),\n    });\n\n    this.addCommand({\n      id: 'selectLine',\n      name: 'Select line',\n      hotkeys: [\n        {\n          modifiers: ['Mod'],\n          key: 'L',\n        },\n      ],\n      editorCallback: (editor) => withMultipleSelections(editor, selectLine),\n    });\n\n    this.addCommand({\n      id: 'addCursorsToSelectionEnds',\n      name: 'Add cursors to selection ends',\n      hotkeys: [\n        {\n          modifiers: ['Alt', 'Shift'],\n          key: 'I',\n        },\n      ],\n      editorCallback: (editor) => addCursorsToSelectionEnds(editor),\n    });\n\n    this.addCommand({\n      id: 'goToLineStart',\n      name: 'Go to start of line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, goToLineBoundary, {\n          ...defaultMultipleSelectionOptions,\n          args: 'start',\n        }),\n    });\n\n    this.addCommand({\n      id: 'goToLineEnd',\n      name: 'Go to end of line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, goToLineBoundary, {\n          ...defaultMultipleSelectionOptions,\n          args: 'end',\n        }),\n    });\n\n    this.addCommand({\n      id: 'goToNextLine',\n      name: 'Go to next line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, navigateLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'next',\n        }),\n    });\n\n    this.addCommand({\n      id: 'goToPrevLine',\n      name: 'Go to previous line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, navigateLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'prev',\n        }),\n    });\n\n    this.addCommand({\n      id: 'goToFirstLine',\n      name: 'Go to first line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, navigateLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'first',\n        }),\n    });\n\n    this.addCommand({\n      id: 'goToLastLine',\n      name: 'Go to last line',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, navigateLine, {\n          ...defaultMultipleSelectionOptions,\n          args: 'last',\n        }),\n    });\n\n    this.addCommand({\n      id: 'goToLineNumber',\n      name: 'Go to line number',\n      editorCallback: (editor) => {\n        const lineCount = editor.lineCount();\n        const onSubmit = (line: number) => editor.setCursor({ line, ch: 0 });\n        new GoToLineModal(this.app, lineCount, onSubmit).open();\n      },\n    });\n\n    this.addCommand({\n      id: 'goToNextChar',\n      name: 'Move cursor forward',\n      editorCallback: (editor) => moveCursor(editor, 'right'),\n    });\n\n    this.addCommand({\n      id: 'goToPrevChar',\n      name: 'Move cursor backward',\n      editorCallback: (editor) => moveCursor(editor, 'left'),\n    });\n\n    this.addCommand({\n      id: 'moveCursorUp',\n      name: 'Move cursor up',\n      editorCallback: (editor) => moveCursor(editor, 'up'),\n    });\n\n    this.addCommand({\n      id: 'moveCursorDown',\n      name: 'Move cursor down',\n      editorCallback: (editor) => moveCursor(editor, 'down'),\n    });\n\n    // Test other code\n    this.addCommand({\n      id: 'select1LinesUp',\n      name: 'Select 1 lines up',\n      editorCallback: (editor) => withMultipleSelections(editor, select1LinesUp),\n    });\n    \n    this.addCommand({\n      id: 'select1LinesDown',\n      name: 'Select 1 lines Down',\n      editorCallback: (editor) => withMultipleSelections(editor, select1LinesDown),\n    });\n\n    this.addCommand({\n      id: 'select1CharLeft',\n      name: 'Select 1 character left',\n      editorCallback: (editor) => withMultipleSelections(editor, select1CharLeft),\n    });\n\n    this.addCommand({\n      id: 'select1CharRight',\n      name: 'Select 1 character right',\n      editorCallback: (editor) => withMultipleSelections(editor, select1CharRight),\n    });\n\n    this.addCommand({\n      id: 'select1WordLeft',\n      name: 'Select 1 word left',\n      editorCallback: (editor) => withMultipleSelections(editor, select1WordLeft),\n    });\n\n    this.addCommand({\n      id: 'select1WordRight',\n      name: 'Select 1 word right',\n      editorCallback: (editor) => withMultipleSelections(editor, select1WordRight),\n    });\n\n    this.addCommand({\n      id: 'goToPreviousWord',\n      name: 'Go to previous word',\n      editorCallback: (editor) => moveWord(editor, 'left'),\n    });\n\n    this.addCommand({\n      id: 'goToNextWord',\n      name: 'Go to next word',\n      editorCallback: (editor) => moveWord(editor, 'right'),\n    });\n\n    this.addCommand({\n      id: 'transformToUppercase',\n      name: 'Transform selection to uppercase',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, transformCase, {\n          ...defaultMultipleSelectionOptions,\n          args: CASE.UPPER,\n        }),\n    });\n\n    this.addCommand({\n      id: 'transformToLowercase',\n      name: 'Transform selection to lowercase',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, transformCase, {\n          ...defaultMultipleSelectionOptions,\n          args: CASE.LOWER,\n        }),\n    });\n\n    this.addCommand({\n      id: 'transformToTitlecase',\n      name: 'Transform selection to title case',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, transformCase, {\n          ...defaultMultipleSelectionOptions,\n          args: CASE.TITLE,\n        }),\n    });\n\n    this.addCommand({\n      id: 'toggleCase',\n      name: 'Toggle case of selection',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, transformCase, {\n          ...defaultMultipleSelectionOptions,\n          args: CASE.NEXT,\n        }),\n    });\n\n    this.addCommand({\n      id: 'expandSelectionToBrackets',\n      name: 'Expand selection to brackets',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, expandSelectionToBrackets),\n    });\n\n    this.addCommand({\n      id: 'expandSelectionToQuotes',\n      name: 'Expand selection to quotes',\n      editorCallback: (editor) =>\n        withMultipleSelections(editor, expandSelectionToQuotes),\n    });\n\n    this.addCommand({\n      id: 'expandSelectionToQuotesOrBrackets',\n      name: 'Expand selection to quotes or brackets',\n      editorCallback: (editor) => expandSelectionToQuotesOrBrackets(editor),\n    });\n\n    this.addCommand({\n      id: 'insertCursorAbove',\n      name: 'Insert cursor above',\n      editorCallback: (editor) => insertCursorAbove(editor),\n    });\n\n    this.addCommand({\n      id: 'insertCursorBelow',\n      name: 'Insert cursor below',\n      editorCallback: (editor) => insertCursorBelow(editor),\n    });\n\n    this.addCommand({\n      id: 'goToNextHeading',\n      name: 'Go to next heading',\n      editorCallback: (editor) => goToHeading(this.app, editor, 'next'),\n    });\n\n    this.addCommand({\n      id: 'goToPrevHeading',\n      name: 'Go to previous heading',\n      editorCallback: (editor) => goToHeading(this.app, editor, 'prev'),\n    });\n\n    this.addCommand({\n      id: 'toggle-line-numbers',\n      name: 'Toggle line numbers',\n      callback: () => toggleVaultConfig(this.app, 'showLineNumber'),\n    });\n\n    this.addCommand({\n      id: 'indent-using-tabs',\n      name: 'Indent using tabs',\n      callback: () => setVaultConfig(this.app, 'useTab', true),\n    });\n\n    this.addCommand({\n      id: 'indent-using-spaces',\n      name: 'Indent using spaces',\n      callback: () => setVaultConfig(this.app, 'useTab', false),\n    });\n\n    this.addCommand({\n      id: 'undo',\n      name: 'Undo',\n      editorCallback: (editor) => editor.undo(),\n    });\n\n    this.addCommand({\n      id: 'redo',\n      name: 'Redo',\n      editorCallback: (editor) => editor.redo(),\n    });\n\n    this.registerSelectionChangeListeners();\n\n    this.addSettingTab(new SettingTab(this.app, this));\n  }\n\n  private registerSelectionChangeListeners() {\n    this.app.workspace.onLayoutReady(() => {\n      // Change handler for selectWordOrNextOccurrence\n      const handleSelectionChange = (evt: Event) => {\n        if (evt instanceof KeyboardEvent && MODIFIER_KEYS.includes(evt.key)) {\n          return;\n        }\n        if (!isProgrammaticSelectionChange) {\n          setIsManualSelection(true);\n        }\n        setIsProgrammaticSelectionChange(false);\n      };\n      iterateCodeMirrorDivs((cm: HTMLElement) => {\n        this.registerDomEvent(cm, 'keydown', handleSelectionChange);\n        this.registerDomEvent(cm, 'click', handleSelectionChange);\n        this.registerDomEvent(cm, 'dblclick', handleSelectionChange);\n      });\n    });\n  }\n\n  async loadSettings() {\n    const savedSettings = await this.loadData();\n    this.settings = {\n      ...DEFAULT_SETTINGS,\n      ...savedSettings,\n    };\n    SettingsState.autoInsertListPrefix = this.settings.autoInsertListPrefix;\n  }\n\n  async saveSettings() {\n    await this.saveData(this.settings);\n    SettingsState.autoInsertListPrefix = this.settings.autoInsertListPrefix;\n  }\n}\n", "export enum CASE {\n  UPPER = 'upper',\n  LOWER = 'lower',\n  TITLE = 'title',\n  NEXT = 'next',\n}\n\nexport const LOWERCASE_ARTICLES = ['the', 'a', 'an'];\n\nexport enum SEARCH_DIRECTION {\n  FORWARD = 'forward',\n  BACKWARD = 'backward',\n}\n\nexport type MatchingCharacterMap = { [key: string]: string };\n\nexport const MATCHING_BRACKETS: MatchingCharacterMap = {\n  '[': ']',\n  '(': ')',\n  '{': '}',\n};\n\nexport const MATCHING_QUOTES: MatchingCharacterMap = {\n  \"'\": \"'\",\n  '\"': '\"',\n  '`': '`',\n};\n\nexport const MATCHING_QUOTES_BRACKETS: MatchingCharacterMap = {\n  ...MATCHING_QUOTES,\n  ...MATCHING_BRACKETS,\n};\n\nexport enum CODE_EDITOR {\n  SUBLIME = 'sublime',\n  VSCODE = 'vscode',\n}\n\nexport const MODIFIER_KEYS = [\n  'Control',\n  'Shift',\n  'Alt',\n  'Meta',\n  'CapsLock',\n  'Fn',\n];\n\n/**\n * Captures the prefix (including space) for bullet lists, numbered lists\n * and checklists\n */\nexport const LIST_CHARACTER_REGEX = /^\\s*(-|\\+|\\*|\\d+\\.|>) (\\[.\\] )?/;\n", "type CodeEditorShortcutsState = {\n  autoInsertListPrefix: boolean;\n};\n\n/**\n * Simple state object used to hold information from saved settings (accessible\n * anywhere it's imported without needing to thread it down to dependent\n * functions as an argument)\n */\nexport const SettingsState: CodeEditorShortcutsState = {\n  autoInsertListPrefix: true,\n};\n", "import {\n  App,\n  Editor,\n  EditorChange,\n  EditorRangeOrCaret,\n  EditorPosition,\n  EditorSelection,\n  EditorSelectionOrCaret,\n} from 'obsidian';\nimport {\n  SEARCH_DIRECTION,\n  LOWERCASE_ARTICLES,\n  LIST_CHARACTER_REGEX,\n} from './constants';\nimport { CustomSelectionHandler } from './custom-selection-handlers';\n\ntype EditorActionCallbackNew = (\n  editor: Editor,\n  selection: EditorSelection,\n  args: any,\n) => { changes: EditorChange[]; newSelection: EditorRangeOrCaret };\n\ntype EditorActionCallback = (\n  editor: Editor,\n  selection: EditorSelection,\n  args: string,\n) => EditorSelectionOrCaret;\n\ntype MultipleSelectionOptions = {\n  // Additional information to be passed to the EditorActionCallback\n  args?: string;\n\n  // Perform further processing of new selections before they are set\n  customSelectionHandler?: CustomSelectionHandler;\n\n  // Whether the action should be repeated for cursors on the same line\n  repeatSameLineActions?: boolean;\n};\n\nexport type EditorActionCallbackNewArgs = Record<string, any>;\n\ntype MultipleSelectionOptionsNew = {\n  // Additional information to be passed to the EditorActionCallback\n  args?: EditorActionCallbackNewArgs;\n\n  // Whether the action should be repeated for cursors on the same line\n  repeatSameLineActions?: boolean;\n\n  // Whether to combine cursors on the same line after the operation has\n  // finished (the cursor with a smaller line number takes precedence)\n  combineSameLineSelections?: boolean;\n};\n\nexport const defaultMultipleSelectionOptions = { repeatSameLineActions: true };\n\nexport const withMultipleSelectionsNew = (\n  editor: Editor,\n  callback: EditorActionCallbackNew,\n  options: MultipleSelectionOptionsNew = defaultMultipleSelectionOptions,\n) => {\n  const selections = editor.listSelections();\n  let selectionIndexesToProcess: number[];\n  const newSelections: EditorRangeOrCaret[] = [];\n  const changes: EditorChange[] = [];\n\n  if (!options.repeatSameLineActions) {\n    const seenLines: number[] = [];\n    selectionIndexesToProcess = selections.reduce(\n      (indexes, currSelection, currIndex) => {\n        const currentLine = currSelection.head.line;\n        if (!seenLines.includes(currentLine)) {\n          seenLines.push(currentLine);\n          indexes.push(currIndex);\n        }\n        return indexes;\n      },\n      [],\n    );\n  }\n\n  for (let i = 0; i < selections.length; i++) {\n    // Controlled by repeatSameLineActions\n    if (selectionIndexesToProcess && !selectionIndexesToProcess.includes(i)) {\n      continue;\n    }\n\n    const { changes: newChanges, newSelection } = callback(\n      editor,\n      selections[i],\n      {\n        ...options.args,\n        iteration: i,\n      },\n    );\n    changes.push(...newChanges);\n\n    if (options.combineSameLineSelections) {\n      const existingSameLineSelection = newSelections.find(\n        (selection) => selection.from.line === newSelection.from.line,\n      );\n      // Generally only happens when deleting consecutive lines using separate cursors\n      if (existingSameLineSelection) {\n        // Reset to 0 as `ch` will otherwise exceed the line length\n        existingSameLineSelection.from.ch = 0;\n        // Skip adding a new selection with the same line number\n        continue;\n      }\n    }\n\n    newSelections.push(newSelection);\n  }\n\n  editor.transaction({\n    changes,\n    selections: newSelections,\n  });\n};\n\nexport const withMultipleSelections = (\n  editor: Editor,\n  callback: EditorActionCallback,\n  options: MultipleSelectionOptions = defaultMultipleSelectionOptions,\n) => {\n  // @ts-expect-error: Obsidian's Editor interface does not explicitly\n  // include the CodeMirror cm object, but it is there when using the\n  // legacy editor\n  const { cm } = editor;\n\n  const selections = editor.listSelections();\n  let selectionIndexesToProcess: number[];\n  let newSelections: EditorSelectionOrCaret[] = [];\n\n  if (!options.repeatSameLineActions) {\n    const seenLines: number[] = [];\n    selectionIndexesToProcess = selections.reduce(\n      (indexes, currSelection, currIndex) => {\n        const currentLine = currSelection.head.line;\n        if (!seenLines.includes(currentLine)) {\n          seenLines.push(currentLine);\n          indexes.push(currIndex);\n        }\n        return indexes;\n      },\n      [],\n    );\n  }\n\n  const applyCallbackOnSelections = () => {\n    for (let i = 0; i < selections.length; i++) {\n      // Controlled by repeatSameLineActions\n      if (selectionIndexesToProcess && !selectionIndexesToProcess.includes(i)) {\n        continue;\n      }\n\n      // Can't reuse selections variable as positions may change on each iteration\n      const selection = editor.listSelections()[i];\n\n      // Selections may disappear (e.g. running delete line for two cursors on the same line)\n      if (selection) {\n        const newSelection = callback(editor, selection, options.args);\n        newSelections.push(newSelection);\n      }\n    }\n\n    if (options.customSelectionHandler) {\n      newSelections = options.customSelectionHandler(newSelections);\n    }\n    editor.setSelections(newSelections);\n  };\n\n  if (cm && cm.operation) {\n    // Group all the updates into one atomic operation (so undo/redo work as expected)\n    cm.operation(applyCallbackOnSelections);\n  } else {\n    // Safe fallback if cm doesn't exist (so undo/redo will step through each change)\n    console.debug('cm object not found, operations will not be buffered');\n    applyCallbackOnSelections();\n  }\n};\n\n/**\n * Executes the supplied callback for each top-level CodeMirror div element in the\n * DOM. This is an interim util made to work with both CM5 and CM6 as Obsidian's\n * `iterateCodeMirrors` method only works with CM5.\n */\nexport const iterateCodeMirrorDivs = (callback: (cm: HTMLElement) => any) => {\n  let codeMirrors: NodeListOf<HTMLElement>;\n  codeMirrors = document.querySelectorAll('.cm-content'); // CM6\n  if (codeMirrors.length === 0) {\n    codeMirrors = document.querySelectorAll('.CodeMirror'); // CM5\n  }\n  codeMirrors.forEach(callback);\n};\n\nexport const getLineStartPos = (line: number): EditorPosition => ({\n  line,\n  ch: 0,\n});\n\nexport const getLineEndPos = (\n  line: number,\n  editor: Editor,\n): EditorPosition => ({\n  line,\n  ch: editor.getLine(line).length,\n});\n\nexport const getSelectionBoundaries = (selection: EditorSelection) => {\n  let { anchor: from, head: to } = selection;\n\n  // In case user selects upwards\n  if (from.line > to.line) {\n    [from, to] = [to, from];\n  }\n\n  // In case user selects backwards on the same line\n  if (from.line === to.line && from.ch > to.ch) {\n    [from, to] = [to, from];\n  }\n\n  return { from, to, hasTrailingNewline: to.line > from.line && to.ch === 0 };\n};\n\nexport const getLeadingWhitespace = (lineContent: string) => {\n  const indentation = lineContent.match(/^\\s+/);\n  return indentation ? indentation[0] : '';\n};\n\n// Match any character from any language: https://www.regular-expressions.info/unicode.html\nconst isLetterCharacter = (char: string) => /\\p{L}\\p{M}*/u.test(char);\n\nconst isDigit = (char: string) => /\\d/.test(char);\n\nconst isLetterOrDigit = (char: string) =>\n  isLetterCharacter(char) || isDigit(char);\n\nexport const wordRangeAtPos = (\n  pos: EditorPosition,\n  lineContent: string,\n): { anchor: EditorPosition; head: EditorPosition } => {\n  let start = pos.ch;\n  let end = pos.ch;\n  while (start > 0 && isLetterOrDigit(lineContent.charAt(start - 1))) {\n    start--;\n  }\n  while (end < lineContent.length && isLetterOrDigit(lineContent.charAt(end))) {\n    end++;\n  }\n  return {\n    anchor: {\n      line: pos.line,\n      ch: start,\n    },\n    head: {\n      line: pos.line,\n      ch: end,\n    },\n  };\n};\n\nexport type CheckCharacter = (char: string) => boolean;\n\nexport const findPosOfNextCharacter = ({\n  editor,\n  startPos,\n  checkCharacter,\n  searchDirection,\n}: {\n  editor: Editor;\n  startPos: EditorPosition;\n  checkCharacter: CheckCharacter;\n  searchDirection: SEARCH_DIRECTION;\n}) => {\n  let { line, ch } = startPos;\n  let lineContent = editor.getLine(line);\n  let matchFound = false;\n  let matchedChar: string;\n\n  if (searchDirection === SEARCH_DIRECTION.BACKWARD) {\n    while (line >= 0) {\n      // ch will initially be 0 if searching from start of line\n      const char = lineContent.charAt(Math.max(ch - 1, 0));\n      matchFound = checkCharacter(char);\n      if (matchFound) {\n        matchedChar = char;\n        break;\n      }\n      ch--;\n      // inclusive because (ch - 1) means the first character will already\n      // have been checked\n      if (ch <= 0) {\n        line--;\n        if (line >= 0) {\n          lineContent = editor.getLine(line);\n          ch = lineContent.length;\n        }\n      }\n    }\n  } else {\n    while (line < editor.lineCount()) {\n      const char = lineContent.charAt(ch);\n      matchFound = checkCharacter(char);\n      if (matchFound) {\n        matchedChar = char;\n        break;\n      }\n      ch++;\n      if (ch >= lineContent.length) {\n        line++;\n        lineContent = editor.getLine(line);\n        ch = 0;\n      }\n    }\n  }\n\n  return matchFound\n    ? {\n        match: matchedChar,\n        pos: {\n          line,\n          ch,\n        },\n      }\n    : null;\n};\n\nexport const hasSameSelectionContent = (\n  editor: Editor,\n  selections: EditorSelection[],\n) =>\n  new Set(\n    selections.map((selection) => {\n      const { from, to } = getSelectionBoundaries(selection);\n      return editor.getRange(from, to);\n    }),\n  ).size === 1;\n\nexport const getSearchText = ({\n  editor,\n  allSelections,\n  autoExpand,\n}: {\n  editor: Editor;\n  allSelections: EditorSelection[];\n  autoExpand: boolean;\n}) => {\n  // Don't search if multiple selection contents are not identical\n  const singleSearchText = hasSameSelectionContent(editor, allSelections);\n  const firstSelection = allSelections[0];\n  const { from, to } = getSelectionBoundaries(firstSelection);\n  let searchText = editor.getRange(from, to);\n  if (searchText.length === 0 && autoExpand) {\n    const wordRange = wordRangeAtPos(from, editor.getLine(from.line));\n    searchText = editor.getRange(wordRange.anchor, wordRange.head);\n  }\n  return {\n    searchText,\n    singleSearchText,\n  };\n};\n\n/**\n * Escapes any special regex characters in the given string.\n *\n * Adapted from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions#escaping\n */\nconst escapeRegex = (input: string) =>\n  input.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&'); // $& means the whole matched string\n\n/**\n * Constructs a custom regex query with word boundaries because in `\\b` in JS doesn't\n * match word boundaries for unicode characters, even with the unicode flag on.\n *\n * Adapted from https://shiba1014.medium.com/regex-word-boundaries-with-unicode-207794f6e7ed.\n */\nconst withWordBoundaries = (input: string) => `(?<=\\\\W|^)${input}(?=\\\\W|$)`;\n\nexport const findAllMatches = ({\n  searchText,\n  searchWithinWords,\n  documentContent,\n}: {\n  searchText: string;\n  searchWithinWords: boolean;\n  documentContent: string;\n}) => {\n  const escapedSearchText = escapeRegex(searchText);\n  const searchExpression = new RegExp(\n    searchWithinWords\n      ? escapedSearchText\n      : withWordBoundaries(escapedSearchText),\n    'g',\n  );\n  return Array.from(documentContent.matchAll(searchExpression));\n};\n\nexport const findNextMatchPosition = ({\n  editor,\n  latestMatchPos,\n  searchText,\n  searchWithinWords,\n  documentContent,\n}: {\n  editor: Editor;\n  latestMatchPos: EditorPosition;\n  searchText: string;\n  searchWithinWords: boolean;\n  documentContent: string;\n}) => {\n  const latestMatchOffset = editor.posToOffset(latestMatchPos);\n  const matches = findAllMatches({\n    searchText,\n    searchWithinWords,\n    documentContent,\n  });\n  let nextMatch: EditorSelection | null = null;\n\n  for (const match of matches) {\n    if (match.index > latestMatchOffset) {\n      nextMatch = {\n        anchor: editor.offsetToPos(match.index),\n        head: editor.offsetToPos(match.index + searchText.length),\n      };\n      break;\n    }\n  }\n  // Circle back to search from the top\n  if (!nextMatch) {\n    const selectionIndexes = editor.listSelections().map((selection) => {\n      const { from } = getSelectionBoundaries(selection);\n      return editor.posToOffset(from);\n    });\n    for (const match of matches) {\n      if (!selectionIndexes.includes(match.index)) {\n        nextMatch = {\n          anchor: editor.offsetToPos(match.index),\n          head: editor.offsetToPos(match.index + searchText.length),\n        };\n        break;\n      }\n    }\n  }\n\n  return nextMatch;\n};\n\nexport const findAllMatchPositions = ({\n  editor,\n  searchText,\n  searchWithinWords,\n  documentContent,\n}: {\n  editor: Editor;\n  searchText: string;\n  searchWithinWords: boolean;\n  documentContent: string;\n}) => {\n  const matches = findAllMatches({\n    searchText,\n    searchWithinWords,\n    documentContent,\n  });\n  const matchPositions = [];\n  for (const match of matches) {\n    matchPositions.push({\n      anchor: editor.offsetToPos(match.index),\n      head: editor.offsetToPos(match.index + searchText.length),\n    });\n  }\n  return matchPositions;\n};\n\nexport const toTitleCase = (selectedText: string) => {\n  // use capture group to join with the same separator used to split\n  return selectedText\n    .split(/(\\s+)/)\n    .map((word, index, allWords) => {\n      if (\n        index > 0 &&\n        index < allWords.length - 1 &&\n        LOWERCASE_ARTICLES.includes(word.toLowerCase())\n      ) {\n        return word.toLowerCase();\n      }\n      return word.charAt(0).toUpperCase() + word.substring(1).toLowerCase();\n    })\n    .join('');\n};\n\nexport const getNextCase = (selectedText: string): string => {\n  const textUpper = selectedText.toUpperCase();\n  const textLower = selectedText.toLowerCase();\n  const textTitle = toTitleCase(selectedText);\n\n  switch (selectedText) {\n    case textUpper: {\n      return textLower;\n    }\n    case textLower: {\n      return textTitle;\n    }\n    case textTitle: {\n      return textUpper;\n    }\n    default: {\n      return textUpper;\n    }\n  }\n};\n\n/**\n * Checks if an input string is numeric.\n *\n * Adapted from https://stackoverflow.com/a/60548119\n */\nexport const isNumeric = (input: string) => input.length > 0 && !isNaN(+input);\n\n/**\n * Determines the next markdown list character prefix for a given line.\n *\n * If it's an ordered list and direction is `after`, the prefix will be\n * incremented by 1.\n *\n * If it's a checklist, the newly inserted checkbox will always be unticked.\n *\n * If the current list item is empty, this will be indicated by a `null` prefix.\n */\nexport const getNextListPrefix = (\n  text: string,\n  direction: 'before' | 'after',\n): string | null => {\n  const listChars = text.match(LIST_CHARACTER_REGEX);\n  if (listChars && listChars.length > 0) {\n    let prefix = listChars[0].trimStart();\n    const isEmptyListItem = prefix === listChars.input.trimStart();\n    if (isEmptyListItem) {\n      return null;\n    }\n    if (isNumeric(prefix) && direction === 'after') {\n      prefix = +prefix + 1 + '. ';\n    }\n    if (prefix.startsWith('- [') && !prefix.includes('[ ]')) {\n      prefix = '- [ ] ';\n    }\n    return prefix;\n  }\n  return '';\n};\n\nexport const formatRemainingListPrefixes = (\n  editor: Editor,\n  fromLine: number,\n  indentation: string,\n) => {\n  const changes: EditorChange[] = [];\n\n  for (let i = fromLine; i < editor.lineCount(); i++) {\n    const contentsOfCurrentLine = editor.getLine(i);\n    // Only prefixes at the same indentation level should be updated\n    const listPrefixRegex = new RegExp(`^${indentation}\\\\d+\\\\.`);\n    const lineStartsWithNumberPrefix = listPrefixRegex.test(\n      contentsOfCurrentLine,\n    );\n    if (!lineStartsWithNumberPrefix) {\n      break;\n    }\n    const replacementContent = contentsOfCurrentLine.replace(\n      /\\d+\\./,\n      (match) => +match + 1 + '.',\n    );\n    changes.push({\n      from: { line: i, ch: 0 },\n      to: { line: i, ch: contentsOfCurrentLine.length },\n      text: replacementContent,\n    });\n  }\n\n  if (changes.length > 0) {\n    editor.transaction({ changes });\n  }\n};\n\ntype VaultConfigSetting = 'showLineNumber' | 'useTab';\n\nexport const toggleVaultConfig = (app: App, setting: VaultConfigSetting) => {\n  // @ts-expect-error - getConfig is not in the public API\n  const value = app.vault.getConfig(setting);\n  setVaultConfig(app, setting, !value);\n};\n\nexport const setVaultConfig = (\n  app: App,\n  setting: VaultConfigSetting,\n  value: boolean,\n) => {\n  // @ts-expect-error - setConfig is not in the public API\n  app.vault.setConfig(setting, value);\n};\n", "import type {\n  App,\n  Editor,\n  EditorChange,\n  EditorPosition,\n  EditorSelection,\n} from 'obsidian';\nimport {\n  CASE,\n  SEARCH_DIRECTION,\n  MATCHING_BRACKETS,\n  MATCHING_QUOTES,\n  MATCHING_QUOTES_BRACKETS,\n  MatchingCharacterMap,\n  CODE_EDITOR,\n  LIST_CHARACTER_REGEX,\n} from './constants';\nimport { SettingsState } from './state';\nimport {\n  CheckCharacter,\n  EditorActionCallbackNewArgs,\n  findAllMatchPositions,\n  findNextMatchPosition,\n  findPosOfNextCharacter,\n  formatRemainingListPrefixes,\n  getLeadingWhitespace,\n  getLineEndPos,\n  getLineStartPos,\n  getNextCase,\n  toTitleCase,\n  getSelectionBoundaries,\n  wordRangeAtPos,\n  getSearchText,\n  getNextListPrefix,\n  isNumeric,\n} from './utils';\n\nexport const insertLineAbove = (\n  editor: Editor,\n  selection: EditorSelection,\n  args: EditorActionCallbackNewArgs,\n) => {\n  const { line } = selection.head;\n  const startOfCurrentLine = getLineStartPos(line);\n\n  const contentsOfCurrentLine = editor.getLine(line);\n  const indentation = getLeadingWhitespace(contentsOfCurrentLine);\n\n  let listPrefix = '';\n  if (\n    SettingsState.autoInsertListPrefix &&\n    line > 0 &&\n    // If inside a list, only insert prefix if within the same list\n    editor.getLine(line - 1).trim().length > 0\n  ) {\n    listPrefix = getNextListPrefix(contentsOfCurrentLine, 'before');\n    if (isNumeric(listPrefix)) {\n      formatRemainingListPrefixes(editor, line, indentation);\n    }\n  }\n\n  const changes: EditorChange[] = [\n    { from: startOfCurrentLine, text: indentation + listPrefix + '\\n' },\n  ];\n  const newSelection = {\n    from: {\n      ...startOfCurrentLine,\n      // Offset by iteration\n      line: startOfCurrentLine.line + args.iteration,\n      ch: indentation.length + listPrefix.length,\n    },\n  };\n  return {\n    changes,\n    newSelection,\n  };\n};\n\nexport const insertLineBelow = (\n  editor: Editor,\n  selection: EditorSelection,\n  args: EditorActionCallbackNewArgs,\n) => {\n  const { line } = selection.head;\n  const startOfCurrentLine = getLineStartPos(line);\n  const endOfCurrentLine = getLineEndPos(line, editor);\n\n  const contentsOfCurrentLine = editor.getLine(line);\n  const indentation = getLeadingWhitespace(contentsOfCurrentLine);\n\n  let listPrefix = '';\n  if (SettingsState.autoInsertListPrefix) {\n    listPrefix = getNextListPrefix(contentsOfCurrentLine, 'after');\n\n    // Performing this action on an empty list item should delete it\n    if (listPrefix === null) {\n      const changes: EditorChange[] = [\n        { from: startOfCurrentLine, to: endOfCurrentLine, text: '' },\n      ];\n      const newSelection = {\n        from: {\n          line,\n          ch: 0,\n        },\n      };\n      return {\n        changes,\n        newSelection,\n      };\n    }\n\n    if (isNumeric(listPrefix)) {\n      formatRemainingListPrefixes(editor, line + 1, indentation);\n    }\n  }\n\n  const changes: EditorChange[] = [\n    { from: endOfCurrentLine, text: '\\n' + indentation + listPrefix },\n  ];\n  const newSelection = {\n    from: {\n      // Offset by iteration\n      line: line + 1 + args.iteration,\n      ch: indentation.length + listPrefix.length,\n    },\n  };\n  return {\n    changes,\n    newSelection,\n  };\n};\n\n// Note: don't use the built-in exec method for 'deleteLine' as there is a bug\n// where running it on a line that is long enough to be wrapped will focus on\n// the previous line instead of the next line after deletion\nlet numLinesDeleted = 0;\nexport const deleteLine = (\n  editor: Editor,\n  selection: EditorSelection,\n  args: EditorActionCallbackNewArgs,\n) => {\n  const { from, to, hasTrailingNewline } = getSelectionBoundaries(selection);\n\n  if (to.line === editor.lastLine()) {\n    // There is no 'next line' when cursor is on the last line\n    const previousLine = Math.max(0, from.line - 1);\n    const endOfPreviousLine = getLineEndPos(previousLine, editor);\n    const changes: EditorChange[] = [\n      {\n        from: from.line === 0 ? getLineStartPos(0) : endOfPreviousLine,\n        to:\n          // Exclude line starting at trailing newline at end of document from being deleted\n          to.ch === 0\n            ? getLineStartPos(to.line)\n            : getLineEndPos(to.line, editor),\n        text: '',\n      },\n    ];\n    const newSelection = {\n      from: {\n        line: previousLine,\n        ch: Math.min(from.ch, endOfPreviousLine.ch),\n      },\n    };\n    return {\n      changes,\n      newSelection,\n    };\n  }\n\n  // Reset offset at the start of a new bulk delete operation\n  if (args.iteration === 0) {\n    numLinesDeleted = 0;\n  }\n  // Exclude line starting at trailing newline from being deleted\n  const toLine = hasTrailingNewline ? to.line - 1 : to.line;\n  const endOfNextLine = getLineEndPos(toLine + 1, editor);\n  const changes: EditorChange[] = [\n    {\n      from: getLineStartPos(from.line),\n      to: getLineStartPos(toLine + 1),\n      text: '',\n    },\n  ];\n  const newSelection = {\n    from: {\n      // Offset by the number of lines deleted in all previous iterations\n      line: from.line - numLinesDeleted,\n      ch: Math.min(to.ch, endOfNextLine.ch),\n    },\n  };\n  // This needs to be calculated after setting the new selection as it only\n  // applies for subsequent iterations\n  numLinesDeleted += toLine - from.line + 1;\n  return {\n    changes,\n    newSelection,\n  };\n};\n\nexport const deleteToStartOfLine = (\n  editor: Editor,\n  selection: EditorSelection,\n) => {\n  const pos = selection.head;\n  let startPos = getLineStartPos(pos.line);\n\n  if (pos.line === 0 && pos.ch === 0) {\n    // We're at the start of the document so do nothing\n    return selection;\n  }\n\n  if (pos.line === startPos.line && pos.ch === startPos.ch) {\n    // We're at the start of the line so delete the preceding newline\n    startPos = getLineEndPos(pos.line - 1, editor);\n  }\n\n  editor.replaceRange('', startPos, pos);\n  return {\n    anchor: startPos,\n  };\n};\n\nexport const deleteToEndOfLine = (\n  editor: Editor,\n  selection: EditorSelection,\n) => {\n  const pos = selection.head;\n  let endPos = getLineEndPos(pos.line, editor);\n\n  if (pos.line === endPos.line && pos.ch === endPos.ch) {\n    // We're at the end of the line so delete just the newline\n    endPos = getLineStartPos(pos.line + 1);\n  }\n\n  editor.replaceRange('', pos, endPos);\n  return {\n    anchor: pos,\n  };\n};\n\nexport const joinLines = (editor: Editor, selection: EditorSelection) => {\n  const { from, to } = getSelectionBoundaries(selection);\n  const { line } = from;\n\n  let endOfCurrentLine = getLineEndPos(line, editor);\n  const joinRangeLimit = Math.max(to.line - line, 1);\n  const selectionLength = editor.posToOffset(to) - editor.posToOffset(from);\n  let trimmedChars = '';\n\n  for (let i = 0; i < joinRangeLimit; i++) {\n    if (line === editor.lineCount() - 1) {\n      break;\n    }\n    endOfCurrentLine = getLineEndPos(line, editor);\n    const endOfNextLine = getLineEndPos(line + 1, editor);\n    const contentsOfCurrentLine = editor.getLine(line);\n    const contentsOfNextLine = editor.getLine(line + 1);\n\n    const charsToTrim = contentsOfNextLine.match(LIST_CHARACTER_REGEX) ?? [];\n    trimmedChars += charsToTrim[0] ?? '';\n\n    const newContentsOfNextLine = contentsOfNextLine.replace(\n      LIST_CHARACTER_REGEX,\n      '',\n    );\n    if (\n      newContentsOfNextLine.length > 0 &&\n      contentsOfCurrentLine.charAt(endOfCurrentLine.ch - 1) !== ' '\n    ) {\n      editor.replaceRange(\n        ' ' + newContentsOfNextLine,\n        endOfCurrentLine,\n        endOfNextLine,\n      );\n    } else {\n      editor.replaceRange(\n        newContentsOfNextLine,\n        endOfCurrentLine,\n        endOfNextLine,\n      );\n    }\n  }\n\n  if (selectionLength === 0) {\n    return {\n      anchor: endOfCurrentLine,\n    };\n  }\n  return {\n    anchor: from,\n    head: {\n      line: from.line,\n      ch: from.ch + selectionLength - trimmedChars.length,\n    },\n  };\n};\n\nexport const copyLine = (\n  editor: Editor,\n  selection: EditorSelection,\n  direction: 'up' | 'down',\n) => {\n  const { from, to, hasTrailingNewline } = getSelectionBoundaries(selection);\n  const fromLineStart = getLineStartPos(from.line);\n  // Exclude line starting at trailing newline from being duplicated\n  const toLine = hasTrailingNewline ? to.line - 1 : to.line;\n  const toLineEnd = getLineEndPos(toLine, editor);\n  const contentsOfSelectedLines = editor.getRange(fromLineStart, toLineEnd);\n  if (direction === 'up') {\n    editor.replaceRange('\\n' + contentsOfSelectedLines, toLineEnd);\n    return selection;\n  } else {\n    editor.replaceRange(contentsOfSelectedLines + '\\n', fromLineStart);\n    // This uses `to.line` instead of `toLine` to avoid a double adjustment\n    const linesSelected = to.line - from.line + 1;\n    return {\n      anchor: { line: toLine + 1, ch: from.ch },\n      head: { line: toLine + linesSelected, ch: to.ch },\n    };\n  }\n};\n\n/*\nProperties used to distinguish between selections that are programmatic\n(expanding from a cursor selection) vs. manual (using a mouse / Shift + arrow\nkeys). This controls the match behaviour for selectWordOrNextOccurrence.\n*/\nlet isManualSelection = true;\nexport const setIsManualSelection = (value: boolean) => {\n  isManualSelection = value;\n};\nexport let isProgrammaticSelectionChange = false;\nexport const setIsProgrammaticSelectionChange = (value: boolean) => {\n  isProgrammaticSelectionChange = value;\n};\n\nexport const selectWordOrNextOccurrence = (editor: Editor) => {\n  setIsProgrammaticSelectionChange(true);\n  const allSelections = editor.listSelections();\n  const { searchText, singleSearchText } = getSearchText({\n    editor,\n    allSelections,\n    autoExpand: false,\n  });\n\n  if (searchText.length > 0 && singleSearchText) {\n    const { from: latestMatchPos } = getSelectionBoundaries(\n      allSelections[allSelections.length - 1],\n    );\n    const nextMatch = findNextMatchPosition({\n      editor,\n      latestMatchPos,\n      searchText,\n      searchWithinWords: isManualSelection,\n      documentContent: editor.getValue(),\n    });\n    const newSelections = nextMatch\n      ? allSelections.concat(nextMatch)\n      : allSelections;\n    editor.setSelections(newSelections);\n    const lastSelection = newSelections[newSelections.length - 1];\n    editor.scrollIntoView(getSelectionBoundaries(lastSelection));\n  } else {\n    const newSelections = [];\n    for (const selection of allSelections) {\n      const { from, to } = getSelectionBoundaries(selection);\n      // Don't modify existing range selections\n      if (from.line !== to.line || from.ch !== to.ch) {\n        newSelections.push(selection);\n      } else {\n        newSelections.push(wordRangeAtPos(from, editor.getLine(from.line)));\n        setIsManualSelection(false);\n      }\n    }\n    editor.setSelections(newSelections);\n  }\n};\n\nexport const selectAllOccurrences = (editor: Editor) => {\n  const allSelections = editor.listSelections();\n  const { searchText, singleSearchText } = getSearchText({\n    editor,\n    allSelections,\n    autoExpand: true,\n  });\n  if (!singleSearchText) {\n    return;\n  }\n  const matches = findAllMatchPositions({\n    editor,\n    searchText,\n    searchWithinWords: true,\n    documentContent: editor.getValue(),\n  });\n  editor.setSelections(matches);\n};\n\nexport const selectLine = (_editor: Editor, selection: EditorSelection) => {\n  const { from, to } = getSelectionBoundaries(selection);\n  const startOfCurrentLine = getLineStartPos(from.line);\n  // if a line is already selected, expand the selection to the next line\n  const startOfNextLine = getLineStartPos(to.line + 1);\n  return { anchor: startOfCurrentLine, head: startOfNextLine };\n};\n\nexport const addCursorsToSelectionEnds = (\n  editor: Editor,\n  emulate: CODE_EDITOR = CODE_EDITOR.VSCODE,\n) => {\n  // Only apply the action if there is exactly one selection\n  if (editor.listSelections().length !== 1) {\n    return;\n  }\n  const selection = editor.listSelections()[0];\n  const { from, to, hasTrailingNewline } = getSelectionBoundaries(selection);\n  const newSelections = [];\n  // Exclude line starting at trailing newline from having cursor added\n  const toLine = hasTrailingNewline ? to.line - 1 : to.line;\n  for (let line = from.line; line <= toLine; line++) {\n    const head = line === to.line ? to : getLineEndPos(line, editor);\n    let anchor: EditorPosition;\n    if (emulate === CODE_EDITOR.VSCODE) {\n      anchor = head;\n    } else {\n      anchor = line === from.line ? from : getLineStartPos(line);\n    }\n    newSelections.push({\n      anchor,\n      head,\n    });\n  }\n  editor.setSelections(newSelections);\n};\n\nexport const goToLineBoundary = (\n  editor: Editor,\n  selection: EditorSelection,\n  boundary: 'start' | 'end',\n) => {\n  const { from, to } = getSelectionBoundaries(selection);\n  if (boundary === 'start') {\n    return { anchor: getLineStartPos(from.line) };\n  } else {\n    return { anchor: getLineEndPos(to.line, editor) };\n  }\n};\n\nexport const navigateLine = (\n  editor: Editor,\n  selection: EditorSelection,\n  position: 'next' | 'prev' | 'first' | 'last',\n) => {\n  const pos = selection.head;\n  let line: number;\n  let ch: number;\n\n  if (position === 'prev') {\n    line = Math.max(pos.line - 1, 0);\n    const endOfLine = getLineEndPos(line, editor);\n    ch = Math.min(pos.ch, endOfLine.ch);\n  }\n  if (position === 'next') {\n    line = Math.min(pos.line + 1, editor.lineCount() - 1);\n    const endOfLine = getLineEndPos(line, editor);\n    ch = Math.min(pos.ch, endOfLine.ch);\n  }\n  if (position === 'first') {\n    line = 0;\n    ch = 0;\n  }\n  if (position === 'last') {\n    line = editor.lineCount() - 1;\n    const endOfLine = getLineEndPos(line, editor);\n    ch = endOfLine.ch;\n  }\n\n  return { anchor: { line, ch } };\n};\n\nexport const moveCursor = (\n  editor: Editor,\n  direction: 'up' | 'down' | 'left' | 'right',\n) => {\n  switch (direction) {\n    case 'up':\n      editor.exec('goUp');\n      break;\n    case 'down':\n      editor.exec('goDown');\n      break;\n    case 'left':\n      editor.exec('goLeft');\n      break;\n    case 'right':\n      editor.exec('goRight');\n      break;\n  }\n};\n\nexport const select1LinesUp = (\n  editor: Editor, \n  selection: EditorSelection\n) => {\n  let newHead: EditorPosition = {\n    line: selection.head.line - 1,\n    ch: selection.head.ch\n  };\n\n  // Ensure the new head position is within the bounds of the line\n  if (newHead.line >= 0) {\n    const lineAboveLength = editor.getLine(newHead.line).length;\n    if (newHead.ch > lineAboveLength) {\n      newHead.ch = lineAboveLength;\n    }\n  } else {\n    // Prevent moving the head above the first line\n    newHead.line = 0;\n    newHead.ch = 0;\n  }\n\n  return { anchor: selection.anchor, head: newHead };\n};\n\nexport const select1LinesDown = (\n  editor: Editor, \n  selection: EditorSelection\n) => {\n  let newHead: EditorPosition = {\n    line: selection.head.line + 1,\n    ch: selection.head.ch\n  };\n\n  // Ensure the new head position is within the bounds of the line\n  if (newHead.line < editor.lineCount()) {\n    const lineBelowLength = editor.getLine(newHead.line).length;\n    if (newHead.ch > lineBelowLength) {\n      newHead.ch = 0;\n    }\n  } else {\n    // Prevent moving the head below the last line\n    newHead.line = editor.lineCount() - 1;\n    newHead.ch = editor.getLine(newHead.line).length;\n  }\n\n  return { anchor: selection.anchor, head: newHead };\n};\n\nexport const select1CharLeft = (\n  _editor: Editor, \n  selection: EditorSelection\n) => {\n    const new_hd: EditorPosition = {\n      line:selection.head.line, \n      ch:selection.head.ch-1\n    };\n\n    return { anchor: selection.anchor, head: new_hd };\n};\n\nexport const select1CharRight = (\n  _editor: Editor, \n  selection: EditorSelection\n) => {\n    const new_hd: EditorPosition = {\n      line:selection.head.line, \n      ch:selection.head.ch+1\n    };\n\n    return { anchor: selection.anchor, head: new_hd };\n};\n\n// Define a reusable variable for punctuation characters\nconst punctuationChars = /[\\s.,!?;:-_]/;\n\nexport const select1WordLeft = (\n  editor: Editor,\n  selection: EditorSelection\n) => {\n  const { anchor, head } = selection;  // Destructure the anchor and head positions from the selection\n  const line = editor.getLine(head.line);  // Get the text of the line where the head is currently located\n  let newHead = { ...head };  // Create a new position object for the head to track the updated cursor position\n\n  // Handle the case where the cursor is at the beginning of the line and needs to move to the previous line\n  if (head.ch === 0 && head.line > 0) {\n    newHead.line--;  // Move up one line\n    newHead.ch = editor.getLine(newHead.line).length;  // Set the cursor to the end of the previous line\n  } else {\n    // Move left to the end of the next word or punctuation group\n    let ch = head.ch;\n    if (ch > 0 && !/[\\s]/.test(line[ch-1]) && !punctuationChars.test(line[ch-1])) {\n      // If the character is part of a word, move left until the end of the word is found\n      while (ch > 0 && !/[\\s]/.test(line[ch-1]) && !punctuationChars.test(line[ch-1])) ch--;\n    } else if (ch > 0 && punctuationChars.test(line[ch-1])) {\n      // If the character is a punctuation mark, move left until the end of the punctuation group is found\n      while (ch > 0 && punctuationChars.test(line[ch-1])) ch--;\n    } else {\n      ch--;  // If none of the above, simply move one character to the left\n    }\n    newHead.ch = ch;  // Update the head position with the new character position\n  }\n\n  // Adjust selection based on direction\n  if (editor.posToOffset(newHead) < editor.posToOffset(anchor)) {\n    // If the new head position is before the anchor, expand the selection to the left\n    return { anchor, head: newHead };\n  } else if (editor.posToOffset(newHead) > editor.posToOffset(anchor) && head.ch !== anchor.ch) {\n    // If the new head position is after the anchor and the selection is non-collapsed, shrink the selection from the right\n    return { anchor, head: newHead };\n  } else {\n    // If neither condition is met, maintain the current selection\n    return { anchor: newHead, head: anchor };\n  }\n};\n\nexport const select1WordRight = (\n  editor: Editor,\n  selection: EditorSelection\n) => {\n  const { anchor, head } = selection;\n  const line = editor.getLine(head.line);\n  let newHead = { ...head };\n\n  // Handle moving across lines\n  if (head.ch === line.length && head.line < editor.lineCount() - 1) {\n    newHead.line++;\n    newHead.ch = 0;\n  } else {\n    // Move right to the end of the next word or punctuation group\n    let ch = head.ch;\n    if (ch < line.length && !/[\\s]/.test(line[ch]) && !punctuationChars.test(line[ch])) {\n      while (ch < line.length && !/[\\s]/.test(line[ch]) && !punctuationChars.test(line[ch])) ch++;\n    } else if (ch < line.length && punctuationChars.test(line[ch])) {\n      while (ch < line.length && punctuationChars.test(line[ch])) ch++;\n    } else {\n      ch++;\n    }\n    newHead.ch = ch;\n  }\n\n  // Determine if we should shrink or expand the selection\n  if (editor.posToOffset(newHead) > editor.posToOffset(head)) {\n    // Shrink the selection from the left\n    return { anchor, head: newHead };\n  } else {\n    // Expand the selection to the right\n    return { anchor: newHead, head: anchor };\n  }\n};\n\nexport const moveWord = (editor: Editor, direction: 'left' | 'right') => {\n  switch (direction) {\n    case 'left':\n      // @ts-expect-error - command not defined in Obsidian API\n      editor.exec('goWordLeft');\n      break;\n    case 'right':\n      // @ts-expect-error - command not defined in Obsidian API\n      editor.exec('goWordRight');\n      break;\n  }\n};\n\nexport const transformCase = (\n  editor: Editor,\n  selection: EditorSelection,\n  caseType: CASE,\n) => {\n  let { from, to } = getSelectionBoundaries(selection);\n  let selectedText = editor.getRange(from, to);\n\n  // apply transform on word at cursor if nothing is selected\n  if (selectedText.length === 0) {\n    const pos = selection.head;\n    const { anchor, head } = wordRangeAtPos(pos, editor.getLine(pos.line));\n    [from, to] = [anchor, head];\n    selectedText = editor.getRange(anchor, head);\n  }\n\n  let replacementText = selectedText;\n\n  switch (caseType) {\n    case CASE.UPPER: {\n      replacementText = selectedText.toUpperCase();\n      break;\n    }\n    case CASE.LOWER: {\n      replacementText = selectedText.toLowerCase();\n      break;\n    }\n    case CASE.TITLE: {\n      replacementText = toTitleCase(selectedText);\n      break;\n    }\n    case CASE.NEXT: {\n      replacementText = getNextCase(selectedText);\n      break;\n    }\n  }\n\n  editor.replaceRange(replacementText, from, to);\n\n  return selection;\n};\n\nconst expandSelection = ({\n  editor,\n  selection,\n  openingCharacterCheck,\n  matchingCharacterMap,\n}: {\n  editor: Editor;\n  selection: EditorSelection;\n  openingCharacterCheck: CheckCharacter;\n  matchingCharacterMap: MatchingCharacterMap;\n}) => {\n  let { anchor, head } = selection;\n\n  // in case user selects upwards\n  if (anchor.line >= head.line && anchor.ch > anchor.ch) {\n    [anchor, head] = [head, anchor];\n  }\n\n  const newAnchor = findPosOfNextCharacter({\n    editor,\n    startPos: anchor,\n    checkCharacter: openingCharacterCheck,\n    searchDirection: SEARCH_DIRECTION.BACKWARD,\n  });\n  if (!newAnchor) {\n    return selection;\n  }\n\n  const newHead = findPosOfNextCharacter({\n    editor,\n    startPos: head,\n    checkCharacter: (char: string) =>\n      char === matchingCharacterMap[newAnchor.match],\n    searchDirection: SEARCH_DIRECTION.FORWARD,\n  });\n  if (!newHead) {\n    return selection;\n  }\n\n  return { anchor: newAnchor.pos, head: newHead.pos };\n};\n\nexport const expandSelectionToBrackets = (\n  editor: Editor,\n  selection: EditorSelection,\n) =>\n  expandSelection({\n    editor,\n    selection,\n    openingCharacterCheck: (char: string) => /[([{]/.test(char),\n    matchingCharacterMap: MATCHING_BRACKETS,\n  });\n\nexport const expandSelectionToQuotes = (\n  editor: Editor,\n  selection: EditorSelection,\n) =>\n  expandSelection({\n    editor,\n    selection,\n    openingCharacterCheck: (char: string) => /['\"`]/.test(char),\n    matchingCharacterMap: MATCHING_QUOTES,\n  });\n\nexport const expandSelectionToQuotesOrBrackets = (editor: Editor) => {\n  const selections = editor.listSelections();\n  const newSelection = expandSelection({\n    editor,\n    selection: selections[0],\n    openingCharacterCheck: (char: string) => /['\"`([{]/.test(char),\n    matchingCharacterMap: MATCHING_QUOTES_BRACKETS,\n  });\n  editor.setSelections([...selections, newSelection]);\n};\n\nconst insertCursor = (editor: Editor, lineOffset: number) => {\n  const selections = editor.listSelections();\n  const newSelections = [];\n  for (const selection of selections) {\n    const { line, ch } = selection.head;\n    if (\n      (line === 0 && lineOffset < 0) ||\n      (line === editor.lastLine() && lineOffset > 0)\n    ) {\n      break;\n    }\n    const targetLineLength = editor.getLine(line + lineOffset).length;\n    newSelections.push({\n      anchor: {\n        line: selection.anchor.line + lineOffset,\n        ch: Math.min(selection.anchor.ch, targetLineLength),\n      },\n      head: {\n        line: line + lineOffset,\n        ch: Math.min(ch, targetLineLength),\n      },\n    });\n  }\n  editor.setSelections([...editor.listSelections(), ...newSelections]);\n};\n\nexport const insertCursorAbove = (editor: Editor) => insertCursor(editor, -1);\n\nexport const insertCursorBelow = (editor: Editor) => insertCursor(editor, 1);\n\nexport const goToHeading = (\n  app: App,\n  editor: Editor,\n  boundary: 'prev' | 'next',\n) => {\n  const file = app.metadataCache.getFileCache(app.workspace.getActiveFile());\n  if (!file.headings || file.headings.length === 0) {\n    return;\n  }\n\n  const { line } = editor.getCursor('from');\n  let prevHeadingLine = 0;\n  let nextHeadingLine = editor.lastLine();\n\n  file.headings.forEach(({ position }) => {\n    const { end: headingPos } = position;\n    if (line > headingPos.line && headingPos.line > prevHeadingLine) {\n      prevHeadingLine = headingPos.line;\n    }\n    if (line < headingPos.line && headingPos.line < nextHeadingLine) {\n      nextHeadingLine = headingPos.line;\n    }\n  });\n\n  editor.setSelection(\n    boundary === 'prev'\n      ? getLineEndPos(prevHeadingLine, editor)\n      : getLineEndPos(nextHeadingLine, editor),\n  );\n};\n", "import { PluginSettingTab, App, Setting, ToggleComponent } from 'obsidian';\nimport CodeEditorShortcuts from './main';\n\nexport interface PluginSettings {\n  autoInsertListPrefix: boolean;\n}\n\nexport const DEFAULT_SETTINGS: PluginSettings = {\n  autoInsertListPrefix: true,\n};\n\nexport class SettingTab extends PluginSettingTab {\n  plugin: CodeEditorShortcuts;\n\n  constructor(app: App, plugin: CodeEditorShortcuts) {\n    super(app, plugin);\n    this.plugin = plugin;\n  }\n\n  display() {\n    const { containerEl } = this;\n\n    containerEl.empty();\n\n    containerEl.createEl('h2', { text: 'Code Editor Shortcuts' });\n\n    const listPrefixSetting = new Setting(containerEl)\n      .setName('Auto insert list prefix')\n      .setDesc(\n        'Automatically insert list prefix when inserting a line above or below',\n      )\n      .addToggle((toggle) =>\n        toggle\n          .setValue(this.plugin.settings.autoInsertListPrefix)\n          .onChange(async (value) => {\n            this.plugin.settings.autoInsertListPrefix = value;\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    new Setting(containerEl).setName('Reset defaults').addButton((btn) => {\n      btn.setButtonText('Reset').onClick(async () => {\n        this.plugin.settings = { ...DEFAULT_SETTINGS };\n        (listPrefixSetting.components[0] as ToggleComponent).setValue(\n          DEFAULT_SETTINGS.autoInsertListPrefix,\n        );\n        await this.plugin.saveSettings();\n      });\n    });\n  }\n}\n", "import { App, SuggestModal } from 'obsidian';\n\nexport class GoToLineModal extends SuggestModal<string> {\n  private lineCount;\n  private onSubmit;\n\n  constructor(\n    app: App,\n    lineCount: number,\n    onSubmit: (lineNumber: number) => void,\n  ) {\n    super(app);\n    this.lineCount = lineCount;\n    this.onSubmit = onSubmit;\n\n    const PROMPT_TEXT = `Enter a line number between 1 and ${lineCount}`;\n    this.limit = 1;\n    this.setPlaceholder(PROMPT_TEXT);\n    this.emptyStateText = PROMPT_TEXT;\n  }\n\n  getSuggestions(line: string): string[] {\n    const lineNumber = parseInt(line);\n    if (line.length > 0 && lineNumber > 0 && lineNumber <= this.lineCount) {\n      return [line];\n    }\n    return [];\n  }\n\n  renderSuggestion(line: string, el: HTMLElement) {\n    el.createEl('div', { text: line });\n  }\n\n  onChooseSuggestion(line: string) {\n    // Subtract 1 as line numbers are zero-indexed\n    this.onSubmit(parseInt(line) - 1);\n  }\n}\n"],
  "mappings": "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",
  "names": []
}
