import { t } from "../../services/i18n.js"; import noteAutocompleteService from "../../services/note_autocomplete.js"; import utils from "../../services/utils.js"; import appContext from "../../components/app_context.js"; import BasicWidget from "../basic_widget.js"; import shortcutService from "../../services/shortcuts.js"; import { Modal } from "bootstrap"; const TPL = ``; const KEEP_LAST_SEARCH_FOR_X_SECONDS = 120; export default class JumpToNoteDialog extends BasicWidget { private lastOpenedTs: number; private modal!: bootstrap.Modal; private $autoComplete!: JQuery; private $results!: JQuery; private $showInFullTextButton!: JQuery; constructor() { super(); this.lastOpenedTs = 0; } doRender() { this.$widget = $(TPL); this.modal = Modal.getOrCreateInstance(this.$widget[0]); this.$autoComplete = this.$widget.find(".jump-to-note-autocomplete"); this.$results = this.$widget.find(".jump-to-note-results"); this.$showInFullTextButton = this.$widget.find(".show-in-full-text-button"); this.$showInFullTextButton.on("click", (e) => this.showInFullText(e)); shortcutService.bindElShortcut(this.$widget, "ctrl+return", (e) => this.showInFullText(e)); } async jumpToNoteEvent() { const dialogPromise = utils.openDialog(this.$widget); if (utils.isMobile()) { dialogPromise.then(($dialog) => { const el = $dialog.find(">.modal-dialog")[0]; function reposition() { const offset = 100; const modalHeight = (window.visualViewport?.height ?? 0) - offset; const safeAreaInsetBottom = (window.visualViewport?.height ?? 0) - window.innerHeight; el.style.height = `${modalHeight}px`; el.style.bottom = `${(window.visualViewport?.height ?? 0) - modalHeight - safeAreaInsetBottom - offset}px`; } this.$autoComplete.on("focus", () => { reposition(); }); window.visualViewport?.addEventListener("resize", () => { reposition(); }); reposition(); }); } // first open dialog, then refresh since refresh is doing focus which should be visible this.refresh(); this.lastOpenedTs = Date.now(); } async refresh() { noteAutocompleteService .initNoteAutocomplete(this.$autoComplete, { allowCreatingNotes: true, hideGoToSelectedNoteButton: true, allowJumpToSearchNotes: true, container: this.$results[0] }) // clear any event listener added in previous invocation of this function .off("autocomplete:noteselected") .on("autocomplete:noteselected", function (event, suggestion, dataset) { if (!suggestion.notePath) { return false; } appContext.tabManager.getActiveContext()?.setNote(suggestion.notePath); }); // if you open the Jump To dialog soon after using it previously, it can often mean that you // actually want to search for the same thing (e.g., you opened the wrong note at first try) // so we'll keep the content. // if it's outside of this time limit, then we assume it's a completely new search and show recent notes instead. if (Date.now() - this.lastOpenedTs > KEEP_LAST_SEARCH_FOR_X_SECONDS * 1000) { noteAutocompleteService.showRecentNotes(this.$autoComplete); } else { this.$autoComplete // hack, the actual search value is stored in
 element next to the search input
                // this is important because the search input value is replaced with the suggestion note's title
                .autocomplete("val", this.$autoComplete.next().text())
                .trigger("focus")
                .trigger("select");
        }
    }

    showInFullText(e: JQuery.TriggeredEvent) {
        // stop from propagating upwards (dangerous, especially with ctrl+enter executable javascript notes)
        e.preventDefault();
        e.stopPropagation();

        const searchString = String(this.$autoComplete.val());

        this.triggerCommand("searchNotes", { searchString });
        this.modal.hide();
    }
}