import {
  getSelection,
  getText,
  insertAtCursor,
  moveCursor,
  replaceRange,
  setSelection,
} from "@silverbulletmd/plugos-silverbullet-syscall/editor";

export async function quoteSelection() {
  let text = await getText();
  const selection = await getSelection();
  let from = selection.from;
  while (from >= 0 && text[from] !== "\n") {
    from--;
  }
  from++;
  if (text[from] === ">" && text[from + 1] === " ") {
    // Already quoted, we have to unquote
    text = text.slice(from + 2, selection.to);
    text = text.replaceAll("\n> ", "\n");
  } else {
    text = text.slice(from, selection.to);
    text = `> ${text.replaceAll("\n", "\n> ")}`;
  }
  await replaceRange(from, selection.to, text);
}

export async function listifySelection() {
  let text = await getText();
  const selection = await getSelection();
  let from = selection.from;
  while (from >= 0 && text[from] !== "\n") {
    from--;
  }
  from++;
  text = text.slice(from, selection.to);
  text = `* ${text.replaceAll(/\n(?!\n)/g, "\n* ")}`;
  await replaceRange(from, selection.to, text);
}

export async function numberListifySelection() {
  let text = await getText();
  const selection = await getSelection();
  let from = selection.from;
  while (from >= 0 && text[from] !== "\n") {
    from--;
  }
  from++;
  text = text.slice(from, selection.to);
  let counter = 1;
  text = `1. ${text.replaceAll(/\n(?!\n)/g, () => {
    counter++;
    return `\n${counter}. `;
  })}`;
  await replaceRange(from, selection.to, text);
}

export async function linkSelection() {
  const text = await getText();
  const selection = await getSelection();
  const textSelection = text.slice(selection.from, selection.to);
  let linkedText = `[]()`;
  let pos = 1;
  if (textSelection.length > 0) {
    try {
      new URL(textSelection);
      linkedText = `[](${textSelection})`;
    } catch {
      linkedText = `[${textSelection}]()`;
      pos = linkedText.length - 1;
    }
  }
  await replaceRange(selection.from, selection.to, linkedText)
  await moveCursor(selection.from + pos);
}

export function wrapSelection(cmdDef: any) {
  return insertMarker(cmdDef.wrapper);
}

async function insertMarker(marker: string) {
  let text = await getText();
  const selection = await getSelection();
  if (selection.from === selection.to) {
    // empty selection
    if (markerAt(selection.from)) {
      // Already there, skipping ahead
      await moveCursor(selection.from + marker.length);
    } else {
      // Not there, inserting
      await insertAtCursor(marker + marker);
      await moveCursor(selection.from + marker.length);
    }
  } else {
    let from = selection.from;
    let to = selection.to;
    let hasMarker = markerAt(from);
    if (!markerAt(from)) {
      // Maybe just before the cursor? We'll accept that
      from = selection.from - marker.length;
      to = selection.to + marker.length;
      hasMarker = markerAt(from);
    }

    if (!hasMarker) {
      // Adding
      await replaceRange(
        selection.from,
        selection.to,
        marker + text.slice(selection.from, selection.to) + marker
      );
      await setSelection(
        selection.from + marker.length,
        selection.to + marker.length
      );
    } else {
      // Removing
      await replaceRange(
        from,
        to,
        text.substring(from + marker.length, to - marker.length)
      );
      await setSelection(from, to - marker.length * 2);
    }
  }

  function markerAt(pos: number) {
    for (var i = 0; i < marker.length; i++) {
      if (text[pos + i] !== marker[i]) {
        return false;
      }
    }
    return true;
  }
}