Use distance filter
parent
b89aee97d7
commit
c6628927ba
|
@ -4,8 +4,10 @@ import { CronHookT } from "../plugos/hooks/node_cron";
|
||||||
import { EventHookT } from "../plugos/hooks/event";
|
import { EventHookT } from "../plugos/hooks/event";
|
||||||
import { CommandHookT } from "../webapp/hooks/command";
|
import { CommandHookT } from "../webapp/hooks/command";
|
||||||
import { SlashCommandHookT } from "../webapp/hooks/slash_command";
|
import { SlashCommandHookT } from "../webapp/hooks/slash_command";
|
||||||
|
import { CompleterHookT } from "../webapp/hooks/completer";
|
||||||
|
|
||||||
export type SilverBulletHooks = CommandHookT &
|
export type SilverBulletHooks = CommandHookT &
|
||||||
|
CompleterHookT &
|
||||||
SlashCommandHookT &
|
SlashCommandHookT &
|
||||||
EndpointHookT &
|
EndpointHookT &
|
||||||
CronHookT &
|
CronHookT &
|
||||||
|
|
|
@ -44,8 +44,7 @@ return fn["default"].apply(null, arguments);`;
|
||||||
|
|
||||||
self.addEventListener("message", (event: { data: WorkerMessage }) => {
|
self.addEventListener("message", (event: { data: WorkerMessage }) => {
|
||||||
safeRun(async () => {
|
safeRun(async () => {
|
||||||
let messageEvent = event;
|
let data = event.data;
|
||||||
let data = messageEvent.data;
|
|
||||||
switch (data.type) {
|
switch (data.type) {
|
||||||
case "load":
|
case "load":
|
||||||
loadedFunctions.set(data.name!, new Function(wrapScript(data.code!)));
|
loadedFunctions.set(data.name!, new Function(wrapScript(data.code!)));
|
||||||
|
|
|
@ -27,7 +27,7 @@ export type EndPointDef = {
|
||||||
|
|
||||||
export class EndpointHook implements Hook<EndpointHookT> {
|
export class EndpointHook implements Hook<EndpointHookT> {
|
||||||
private app: Express;
|
private app: Express;
|
||||||
private prefix: string;
|
readonly prefix: string;
|
||||||
|
|
||||||
constructor(app: Express, prefix: string) {
|
constructor(app: Express, prefix: string) {
|
||||||
this.app = app;
|
this.app = app;
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import { Hook, Manifest } from "../types";
|
import { Hook, Manifest } from "../types";
|
||||||
import { System } from "../system";
|
import { System } from "../system";
|
||||||
|
import { safeRun } from "../util";
|
||||||
|
|
||||||
// System events:
|
// System events:
|
||||||
// - plug:load (plugName: string)
|
// - plug:load (plugName: string)
|
||||||
|
@ -11,11 +12,11 @@ export type EventHookT = {
|
||||||
export class EventHook implements Hook<EventHookT> {
|
export class EventHook implements Hook<EventHookT> {
|
||||||
private system?: System<EventHookT>;
|
private system?: System<EventHookT>;
|
||||||
|
|
||||||
async dispatchEvent(eventName: string, data?: any): Promise<any[]> {
|
async dispatchEvent(eventName: string, data?: any): Promise<void> {
|
||||||
if (!this.system) {
|
if (!this.system) {
|
||||||
throw new Error("Event hook is not initialized");
|
throw new Error("Event hook is not initialized");
|
||||||
}
|
}
|
||||||
let promises: Promise<any>[] = [];
|
let promises: Promise<void>[] = [];
|
||||||
for (const plug of this.system.loadedPlugs.values()) {
|
for (const plug of this.system.loadedPlugs.values()) {
|
||||||
for (const [name, functionDef] of Object.entries(
|
for (const [name, functionDef] of Object.entries(
|
||||||
plug.manifest!.functions
|
plug.manifest!.functions
|
||||||
|
@ -28,14 +29,16 @@ export class EventHook implements Hook<EventHookT> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Promise.all(promises);
|
await Promise.all(promises);
|
||||||
}
|
}
|
||||||
|
|
||||||
apply(system: System<EventHookT>): void {
|
apply(system: System<EventHookT>): void {
|
||||||
this.system = system;
|
this.system = system;
|
||||||
this.system.on({
|
this.system.on({
|
||||||
plugLoaded: (name) => {
|
plugLoaded: (name) => {
|
||||||
this.dispatchEvent("plug:load", name);
|
safeRun(async () => {
|
||||||
|
await this.dispatchEvent("plug:load", name);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,9 +29,8 @@ functions:
|
||||||
command:
|
command:
|
||||||
name: "Page: Rename"
|
name: "Page: Rename"
|
||||||
pageComplete:
|
pageComplete:
|
||||||
path: "./navigate.ts:pageComplete"
|
path: "./page.ts:pageComplete"
|
||||||
events:
|
isCompleter: true
|
||||||
- editor:complete
|
|
||||||
linkNavigate:
|
linkNavigate:
|
||||||
path: "./navigate.ts:linkNavigate"
|
path: "./navigate.ts:linkNavigate"
|
||||||
command:
|
command:
|
||||||
|
|
|
@ -38,18 +38,3 @@ export async function clickNavigate(event: ClickEvent) {
|
||||||
await navigate(syntaxNode);
|
await navigate(syntaxNode);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function pageComplete() {
|
|
||||||
let prefix = await syscall("editor.matchBefore", "\\[\\[[\\w\\s]*");
|
|
||||||
if (!prefix) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
let allPages = await syscall("space.listPages");
|
|
||||||
return {
|
|
||||||
from: prefix.from + 2,
|
|
||||||
options: allPages.map((pageMeta: any) => ({
|
|
||||||
label: pageMeta.name,
|
|
||||||
type: "page",
|
|
||||||
})),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
|
@ -110,6 +110,22 @@ export async function reindexCommand() {
|
||||||
await syscall("editor.flashNotification", "Reindexing done");
|
await syscall("editor.flashNotification", "Reindexing done");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Completion
|
||||||
|
export async function pageComplete() {
|
||||||
|
let prefix = await syscall("editor.matchBefore", "\\[\\[[\\w\\s]*");
|
||||||
|
if (!prefix) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
let allPages = await syscall("space.listPages");
|
||||||
|
return {
|
||||||
|
from: prefix.from + 2,
|
||||||
|
options: allPages.map((pageMeta: any) => ({
|
||||||
|
label: pageMeta.name,
|
||||||
|
type: "page",
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
// Server functions
|
// Server functions
|
||||||
export async function reindexSpace() {
|
export async function reindexSpace() {
|
||||||
console.log("Clearing page index...");
|
console.log("Clearing page index...");
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { syscall } from "../lib/syscall";
|
import { syscall } from "../lib/syscall";
|
||||||
|
|
||||||
function countWords(str: string): number {
|
function countWords(str: string): number {
|
||||||
var matches = str.match(/[\w\d\'\'-]+/gi);
|
const matches = str.match(/[\w\d\'-]+/gi);
|
||||||
return matches ? matches.length : 0;
|
return matches ? matches.length : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,5 +14,5 @@ export type IndexEvent = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface AppEventDispatcher {
|
export interface AppEventDispatcher {
|
||||||
dispatchAppEvent(name: AppEvent, data?: any): Promise<any[]>;
|
dispatchAppEvent(name: AppEvent, data?: any): Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,11 +2,11 @@ import React, { useEffect, useRef, useState } from "react";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
import { IconDefinition } from "@fortawesome/free-solid-svg-icons";
|
import { IconDefinition } from "@fortawesome/free-solid-svg-icons";
|
||||||
|
|
||||||
export interface Option {
|
export type Option = {
|
||||||
name: string;
|
name: string;
|
||||||
orderId?: number;
|
orderId?: number;
|
||||||
hint?: string;
|
hint?: string;
|
||||||
}
|
};
|
||||||
|
|
||||||
function magicSorter(a: Option, b: Option): number {
|
function magicSorter(a: Option, b: Option): number {
|
||||||
if (a.orderId && b.orderId) {
|
if (a.orderId && b.orderId) {
|
||||||
|
@ -15,6 +15,42 @@ function magicSorter(a: Option, b: Option): number {
|
||||||
return a.name.toLowerCase() < b.name.toLowerCase() ? -1 : 1;
|
return a.name.toLowerCase() < b.name.toLowerCase() ? -1 : 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function escapeRegExp(str: string): string {
|
||||||
|
return str.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
|
||||||
|
}
|
||||||
|
|
||||||
|
function fuzzyFilter(pattern: string, options: Option[]): Option[] {
|
||||||
|
let closeMatchRegex = escapeRegExp(pattern);
|
||||||
|
closeMatchRegex = closeMatchRegex.split(/\s+/).join(".*?");
|
||||||
|
closeMatchRegex = closeMatchRegex.replace(/\\\//g, ".*?\\/.*?");
|
||||||
|
const distantMatchRegex = escapeRegExp(pattern).split("").join(".*?");
|
||||||
|
const r1 = new RegExp(closeMatchRegex, "i");
|
||||||
|
const r2 = new RegExp(distantMatchRegex, "i");
|
||||||
|
let matches = [];
|
||||||
|
if (!pattern) {
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
for (let option of options) {
|
||||||
|
let m = r1.exec(option.name);
|
||||||
|
if (m) {
|
||||||
|
matches.push({
|
||||||
|
...option,
|
||||||
|
orderId: 100000 - (options.length - m[0].length - m.index),
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Let's try the distant matcher
|
||||||
|
var m2 = r2.exec(option.name);
|
||||||
|
if (m2) {
|
||||||
|
matches.push({
|
||||||
|
...option,
|
||||||
|
orderId: 10000 - (options.length - m2[0].length - m2.index),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return matches;
|
||||||
|
}
|
||||||
|
|
||||||
export function FilterList({
|
export function FilterList({
|
||||||
placeholder,
|
placeholder,
|
||||||
options,
|
options,
|
||||||
|
@ -51,12 +87,7 @@ export function FilterList({
|
||||||
|
|
||||||
if (searchPhrase) {
|
if (searchPhrase) {
|
||||||
let foundExactMatch = false;
|
let foundExactMatch = false;
|
||||||
let results = options.filter((option) => {
|
let results = fuzzyFilter(searchPhrase, options);
|
||||||
if (option.name.toLowerCase() === searchPhrase) {
|
|
||||||
foundExactMatch = true;
|
|
||||||
}
|
|
||||||
return option.name.toLowerCase().indexOf(searchPhrase) !== -1;
|
|
||||||
});
|
|
||||||
results = results.sort(magicSorter);
|
results = results.sort(magicSorter);
|
||||||
if (allowNew && !foundExactMatch) {
|
if (allowNew && !foundExactMatch) {
|
||||||
results.push({
|
results.push({
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
export const pageLinkRegex = /\[\[([\w\s\/\:,\.@\-]+)\]\]/;
|
export const pageLinkRegex = /\[\[([\w\s\/:,\.@\-]+)\]\]/;
|
||||||
|
|
|
@ -1,8 +1,4 @@
|
||||||
import {
|
import { autocompletion, completionKeymap } from "@codemirror/autocomplete";
|
||||||
autocompletion,
|
|
||||||
completionKeymap,
|
|
||||||
CompletionResult,
|
|
||||||
} from "@codemirror/autocomplete";
|
|
||||||
import { closeBrackets, closeBracketsKeymap } from "@codemirror/closebrackets";
|
import { closeBrackets, closeBracketsKeymap } from "@codemirror/closebrackets";
|
||||||
import { indentWithTab, standardKeymap } from "@codemirror/commands";
|
import { indentWithTab, standardKeymap } from "@codemirror/commands";
|
||||||
import { history, historyKeymap } from "@codemirror/history";
|
import { history, historyKeymap } from "@codemirror/history";
|
||||||
|
@ -47,6 +43,7 @@ import { systemSyscalls } from "./syscalls/system";
|
||||||
import { Panel } from "./components/panel";
|
import { Panel } from "./components/panel";
|
||||||
import { CommandHook } from "./hooks/command";
|
import { CommandHook } from "./hooks/command";
|
||||||
import { SlashCommandHook } from "./hooks/slash_command";
|
import { SlashCommandHook } from "./hooks/slash_command";
|
||||||
|
import { CompleterHook } from "./hooks/completer";
|
||||||
|
|
||||||
class PageState {
|
class PageState {
|
||||||
scrollTop: number;
|
scrollTop: number;
|
||||||
|
@ -60,15 +57,17 @@ class PageState {
|
||||||
|
|
||||||
export class Editor implements AppEventDispatcher {
|
export class Editor implements AppEventDispatcher {
|
||||||
private system = new System<SilverBulletHooks>("client");
|
private system = new System<SilverBulletHooks>("client");
|
||||||
|
readonly commandHook: CommandHook;
|
||||||
|
readonly slashCommandHook: SlashCommandHook;
|
||||||
|
readonly completerHook: CompleterHook;
|
||||||
|
|
||||||
openPages = new Map<string, PageState>();
|
openPages = new Map<string, PageState>();
|
||||||
commandHook: CommandHook;
|
|
||||||
editorView?: EditorView;
|
editorView?: EditorView;
|
||||||
viewState: AppViewState;
|
viewState: AppViewState;
|
||||||
viewDispatch: React.Dispatch<Action>;
|
viewDispatch: React.Dispatch<Action>;
|
||||||
space: Space;
|
space: Space;
|
||||||
pageNavigator: PathPageNavigator;
|
pageNavigator: PathPageNavigator;
|
||||||
eventHook: EventHook;
|
eventHook: EventHook;
|
||||||
private slashCommandHook: SlashCommandHook;
|
|
||||||
|
|
||||||
constructor(space: Space, parent: Element) {
|
constructor(space: Space, parent: Element) {
|
||||||
this.space = space;
|
this.space = space;
|
||||||
|
@ -95,6 +94,10 @@ export class Editor implements AppEventDispatcher {
|
||||||
this.slashCommandHook = new SlashCommandHook(this);
|
this.slashCommandHook = new SlashCommandHook(this);
|
||||||
this.system.addHook(this.slashCommandHook);
|
this.system.addHook(this.slashCommandHook);
|
||||||
|
|
||||||
|
// Completer hook
|
||||||
|
this.completerHook = new CompleterHook();
|
||||||
|
this.system.addHook(this.completerHook);
|
||||||
|
|
||||||
this.render(parent);
|
this.render(parent);
|
||||||
this.editorView = new EditorView({
|
this.editorView = new EditorView({
|
||||||
state: this.createEditorState(
|
state: this.createEditorState(
|
||||||
|
@ -192,7 +195,7 @@ export class Editor implements AppEventDispatcher {
|
||||||
}, 2000);
|
}, 2000);
|
||||||
}
|
}
|
||||||
|
|
||||||
async dispatchAppEvent(name: AppEvent, data?: any): Promise<any[]> {
|
async dispatchAppEvent(name: AppEvent, data?: any): Promise<void> {
|
||||||
return this.eventHook.dispatchEvent(name, data);
|
return this.eventHook.dispatchEvent(name, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -236,7 +239,7 @@ export class Editor implements AppEventDispatcher {
|
||||||
}),
|
}),
|
||||||
autocompletion({
|
autocompletion({
|
||||||
override: [
|
override: [
|
||||||
this.plugCompleter.bind(this),
|
this.completerHook.plugCompleter.bind(this.completerHook),
|
||||||
this.slashCommandHook.slashCommandCompleter.bind(
|
this.slashCommandHook.slashCommandCompleter.bind(
|
||||||
this.slashCommandHook
|
this.slashCommandHook
|
||||||
),
|
),
|
||||||
|
@ -330,19 +333,6 @@ export class Editor implements AppEventDispatcher {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async plugCompleter(): Promise<CompletionResult | null> {
|
|
||||||
let allCompletionResults = await this.dispatchAppEvent("editor:complete");
|
|
||||||
if (allCompletionResults.length === 1) {
|
|
||||||
return allCompletionResults[0];
|
|
||||||
} else if (allCompletionResults.length > 1) {
|
|
||||||
console.error(
|
|
||||||
"Got completion results from multiple sources, cannot deal with that",
|
|
||||||
allCompletionResults
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
focus() {
|
focus() {
|
||||||
this.editorView!.focus();
|
this.editorView!.focus();
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
import { Hook, Manifest } from "../../plugos/types";
|
||||||
|
import { System } from "../../plugos/system";
|
||||||
|
import { CompletionResult } from "@codemirror/autocomplete";
|
||||||
|
|
||||||
|
export type CompleterHookT = {
|
||||||
|
isCompleter?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export class CompleterHook implements Hook<CompleterHookT> {
|
||||||
|
private system?: System<CompleterHookT>;
|
||||||
|
|
||||||
|
public async plugCompleter(): Promise<CompletionResult | null> {
|
||||||
|
let completerPromises = [];
|
||||||
|
// TODO: Can be optimized (cache all functions)
|
||||||
|
for (const plug of this.system!.loadedPlugs.values()) {
|
||||||
|
if (!plug.manifest) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
for (const [functionName, functionDef] of Object.entries(
|
||||||
|
plug.manifest.functions
|
||||||
|
)) {
|
||||||
|
if (functionDef.isCompleter) {
|
||||||
|
completerPromises.push(plug.invoke(functionName, []));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let allCompletionResults = await Promise.all(completerPromises);
|
||||||
|
if (allCompletionResults.length === 1) {
|
||||||
|
return allCompletionResults[0];
|
||||||
|
} else if (allCompletionResults.length > 1) {
|
||||||
|
console.error(
|
||||||
|
"Got completion results from multiple sources, cannot deal with that",
|
||||||
|
allCompletionResults
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
apply(system: System<CompleterHookT>): void {
|
||||||
|
this.system = system;
|
||||||
|
}
|
||||||
|
|
||||||
|
validateManifest(manifest: Manifest<CompleterHookT>): string[] {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
export function countWords(str: string): number {
|
export function countWords(str: string): number {
|
||||||
var matches = str.match(/[\w\d\'\'-]+/gi);
|
const matches = str.match(/[\w\d\'-]+/gi);
|
||||||
return matches ? matches.length : 0;
|
return matches ? matches.length : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue