Code complete fixes and changelog
parent
371932e085
commit
d8797aa805
|
@ -27,108 +27,68 @@ import {
|
||||||
} from "./deps.ts";
|
} from "./deps.ts";
|
||||||
import { highlightingDirectiveParser } from "./markdown_parser/parser.ts";
|
import { highlightingDirectiveParser } from "./markdown_parser/parser.ts";
|
||||||
|
|
||||||
const languageCache = new Map<string, Language>();
|
export const builtinLanguages: Record<string, Language> = {
|
||||||
|
"meta": StreamLanguage.define(yamlLanguage),
|
||||||
|
"yaml": StreamLanguage.define(yamlLanguage),
|
||||||
|
"template": StreamLanguage.define(yamlLanguage),
|
||||||
|
"embed": StreamLanguage.define(yamlLanguage),
|
||||||
|
"data": StreamLanguage.define(yamlLanguage),
|
||||||
|
"javascript": javascriptLanguage,
|
||||||
|
"js": javascriptLanguage,
|
||||||
|
"typescript": typescriptLanguage,
|
||||||
|
"ts": typescriptLanguage,
|
||||||
|
"sql": StreamLanguage.define(sqlLanguage),
|
||||||
|
"postgresql": StreamLanguage.define(postgresqlLanguage),
|
||||||
|
"pgsql": StreamLanguage.define(postgresqlLanguage),
|
||||||
|
"postgres": StreamLanguage.define(postgresqlLanguage),
|
||||||
|
"rust": StreamLanguage.define(rustLanguage),
|
||||||
|
"rs": StreamLanguage.define(rustLanguage),
|
||||||
|
"css": StreamLanguage.define(sqlLanguage),
|
||||||
|
"html": htmlLanguage,
|
||||||
|
"python": StreamLanguage.define(pythonLanguage),
|
||||||
|
"py": StreamLanguage.define(pythonLanguage),
|
||||||
|
"protobuf": StreamLanguage.define(protobufLanguage),
|
||||||
|
"proto": StreamLanguage.define(protobufLanguage),
|
||||||
|
"shell": StreamLanguage.define(shellLanguage),
|
||||||
|
"sh": StreamLanguage.define(shellLanguage),
|
||||||
|
"bash": StreamLanguage.define(shellLanguage),
|
||||||
|
"zsh": StreamLanguage.define(shellLanguage),
|
||||||
|
"fish": StreamLanguage.define(shellLanguage),
|
||||||
|
"swift": StreamLanguage.define(rustLanguage),
|
||||||
|
"toml": StreamLanguage.define(tomlLanguage),
|
||||||
|
"json": StreamLanguage.define(jsonLanguage),
|
||||||
|
"xml": StreamLanguage.define(xmlLanguage),
|
||||||
|
"c": StreamLanguage.define(cLanguage),
|
||||||
|
"cpp": StreamLanguage.define(cppLanguage),
|
||||||
|
"c++": StreamLanguage.define(cppLanguage),
|
||||||
|
"cxx": StreamLanguage.define(cppLanguage),
|
||||||
|
"java": StreamLanguage.define(javaLanguage),
|
||||||
|
"csharp": StreamLanguage.define(csharpLanguage),
|
||||||
|
"cs": StreamLanguage.define(csharpLanguage),
|
||||||
|
"c#": StreamLanguage.define(csharpLanguage),
|
||||||
|
"scala": StreamLanguage.define(scalaLanguage),
|
||||||
|
"kotlin": StreamLanguage.define(kotlinLanguage),
|
||||||
|
"objc": StreamLanguage.define(objectiveCLanguage),
|
||||||
|
"objective-c": StreamLanguage.define(objectiveCLanguage),
|
||||||
|
"objectivec": StreamLanguage.define(objectiveCLanguage),
|
||||||
|
"objcpp": StreamLanguage.define(objectiveCppLanguage),
|
||||||
|
"objective-cpp": StreamLanguage.define(objectiveCppLanguage),
|
||||||
|
"objectivecpp": StreamLanguage.define(objectiveCppLanguage),
|
||||||
|
"objective-c++": StreamLanguage.define(objectiveCppLanguage),
|
||||||
|
"objectivec++": StreamLanguage.define(objectiveCppLanguage),
|
||||||
|
"dart": StreamLanguage.define(dartLanguage),
|
||||||
|
"query": LRLanguage.define({
|
||||||
|
name: "query",
|
||||||
|
parser: highlightingDirectiveParser,
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
export function languageFor(name: string): Language | null {
|
export function languageFor(name: string): Language | null {
|
||||||
if (languageCache.has(name)) {
|
if (builtinLanguages[name]) {
|
||||||
return languageCache.get(name)!;
|
return builtinLanguages[name];
|
||||||
}
|
}
|
||||||
const language = languageLookup(name);
|
if (name.startsWith("#")) {
|
||||||
if (!language) {
|
return StreamLanguage.define(yamlLanguage);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
languageCache.set(name, language);
|
|
||||||
return language;
|
|
||||||
}
|
|
||||||
|
|
||||||
function languageLookup(name: string): Language | null {
|
|
||||||
switch (name) {
|
|
||||||
case "meta":
|
|
||||||
case "yaml":
|
|
||||||
case "template":
|
|
||||||
case "embed":
|
|
||||||
case "data":
|
|
||||||
return StreamLanguage.define(yamlLanguage);
|
|
||||||
|
|
||||||
case "javascript":
|
|
||||||
case "js":
|
|
||||||
return javascriptLanguage;
|
|
||||||
case "typescript":
|
|
||||||
case "ts":
|
|
||||||
return typescriptLanguage;
|
|
||||||
case "sql":
|
|
||||||
return StreamLanguage.define(sqlLanguage);
|
|
||||||
case "postgresql":
|
|
||||||
case "pgsql":
|
|
||||||
case "postgres":
|
|
||||||
return StreamLanguage.define(postgresqlLanguage);
|
|
||||||
case "rust":
|
|
||||||
case "rs":
|
|
||||||
return StreamLanguage.define(rustLanguage);
|
|
||||||
case "css":
|
|
||||||
return StreamLanguage.define(sqlLanguage);
|
|
||||||
case "html":
|
|
||||||
return htmlLanguage;
|
|
||||||
case "python":
|
|
||||||
case "py":
|
|
||||||
return StreamLanguage.define(pythonLanguage);
|
|
||||||
case "protobuf":
|
|
||||||
case "proto":
|
|
||||||
return StreamLanguage.define(protobufLanguage);
|
|
||||||
case "shell":
|
|
||||||
case "sh":
|
|
||||||
case "bash":
|
|
||||||
case "zsh":
|
|
||||||
case "fish":
|
|
||||||
return StreamLanguage.define(shellLanguage);
|
|
||||||
case "swift":
|
|
||||||
return StreamLanguage.define(rustLanguage);
|
|
||||||
case "toml":
|
|
||||||
return StreamLanguage.define(tomlLanguage);
|
|
||||||
case "json":
|
|
||||||
return StreamLanguage.define(jsonLanguage);
|
|
||||||
case "xml":
|
|
||||||
return StreamLanguage.define(xmlLanguage);
|
|
||||||
case "c":
|
|
||||||
return StreamLanguage.define(cLanguage);
|
|
||||||
case "cpp":
|
|
||||||
case "c++":
|
|
||||||
case "cxx":
|
|
||||||
return StreamLanguage.define(cppLanguage);
|
|
||||||
case "java":
|
|
||||||
return StreamLanguage.define(javaLanguage);
|
|
||||||
case "csharp":
|
|
||||||
case "cs":
|
|
||||||
case "c#":
|
|
||||||
return StreamLanguage.define(csharpLanguage);
|
|
||||||
case "scala":
|
|
||||||
return StreamLanguage.define(scalaLanguage);
|
|
||||||
case "kotlin":
|
|
||||||
return StreamLanguage.define(kotlinLanguage);
|
|
||||||
case "objc":
|
|
||||||
case "objective-c":
|
|
||||||
case "objectivec":
|
|
||||||
return StreamLanguage.define(objectiveCLanguage);
|
|
||||||
case "objcpp":
|
|
||||||
case "objective-cpp":
|
|
||||||
case "objectivecpp":
|
|
||||||
case "objective-c++":
|
|
||||||
case "objectivec++":
|
|
||||||
return StreamLanguage.define(objectiveCppLanguage);
|
|
||||||
|
|
||||||
case "dart":
|
|
||||||
return StreamLanguage.define(dartLanguage);
|
|
||||||
|
|
||||||
case "query":
|
|
||||||
return LRLanguage.define({
|
|
||||||
name: "query",
|
|
||||||
parser: highlightingDirectiveParser,
|
|
||||||
});
|
|
||||||
|
|
||||||
default:
|
|
||||||
if (name.startsWith("#")) {
|
|
||||||
return StreamLanguage.define(yamlLanguage);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { SysCallMapping } from "../../plugos/system.ts";
|
import { SysCallMapping } from "../../plugos/system.ts";
|
||||||
import { parse } from "../markdown_parser/parse_tree.ts";
|
import { parse } from "../markdown_parser/parse_tree.ts";
|
||||||
import type { ParseTree } from "$sb/lib/tree.ts";
|
import type { ParseTree } from "$sb/lib/tree.ts";
|
||||||
import { languageFor } from "../languages.ts";
|
import { builtinLanguages, languageFor } from "../languages.ts";
|
||||||
|
|
||||||
export function languageSyscalls(): SysCallMapping {
|
export function languageSyscalls(): SysCallMapping {
|
||||||
return {
|
return {
|
||||||
|
@ -16,5 +16,10 @@ export function languageSyscalls(): SysCallMapping {
|
||||||
}
|
}
|
||||||
return parse(lang, code);
|
return parse(lang, code);
|
||||||
},
|
},
|
||||||
|
"language.listLanguages": (
|
||||||
|
_ctx,
|
||||||
|
): string[] => {
|
||||||
|
return Object.keys(builtinLanguages);
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,3 +14,7 @@ export function parseLanguage(
|
||||||
): Promise<ParseTree> {
|
): Promise<ParseTree> {
|
||||||
return syscall("language.parseLanguage", language, code);
|
return syscall("language.parseLanguage", language, code);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function listLanguages(): Promise<string[]> {
|
||||||
|
return syscall("language.listLanguages");
|
||||||
|
}
|
||||||
|
|
|
@ -34,14 +34,3 @@ functions:
|
||||||
path: command.ts:convertToLiveQuery
|
path: command.ts:convertToLiveQuery
|
||||||
command:
|
command:
|
||||||
name: "Directive: Convert Query to Live Query"
|
name: "Directive: Convert Query to Live Query"
|
||||||
|
|
||||||
# Templates
|
|
||||||
insertEvalTemplate:
|
|
||||||
redirect: template.insertTemplateText
|
|
||||||
slashCommand:
|
|
||||||
name: eval
|
|
||||||
description: Evaluate a JavaScript expression
|
|
||||||
value: |
|
|
||||||
<!-- #eval |^| -->
|
|
||||||
|
|
||||||
<!-- /eval -->
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { CompleteEvent } from "$sb/app_event.ts";
|
import { CompleteEvent } from "$sb/app_event.ts";
|
||||||
import { events } from "$sb/syscalls.ts";
|
import { events, language } from "$sb/syscalls.ts";
|
||||||
import {
|
import {
|
||||||
AttributeCompleteEvent,
|
AttributeCompleteEvent,
|
||||||
AttributeCompletion,
|
AttributeCompletion,
|
||||||
|
@ -7,11 +7,12 @@ import {
|
||||||
|
|
||||||
export async function queryComplete(completeEvent: CompleteEvent) {
|
export async function queryComplete(completeEvent: CompleteEvent) {
|
||||||
const fencedParent = completeEvent.parentNodes.find((node) =>
|
const fencedParent = completeEvent.parentNodes.find((node) =>
|
||||||
node === "FencedCode:query"
|
node.startsWith("FencedCode:query")
|
||||||
);
|
);
|
||||||
if (!fencedParent) {
|
if (!fencedParent) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
// First let's try to match the query source
|
||||||
let querySourceMatch = /^\s*([\w\-_]*)$/.exec(
|
let querySourceMatch = /^\s*([\w\-_]*)$/.exec(
|
||||||
completeEvent.linePrefix,
|
completeEvent.linePrefix,
|
||||||
);
|
);
|
||||||
|
@ -41,8 +42,10 @@ export async function queryComplete(completeEvent: CompleteEvent) {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
querySourceMatch = /^\s*([\w\-_]*)/.exec(
|
// If that doesn't work, let's try to match other bits of the query
|
||||||
completeEvent.linePrefix,
|
// For this we do need to find the query source, though, so let's look for it in fencedParent
|
||||||
|
querySourceMatch = /^[\n\r\s]*([\w\-_]+)/.exec(
|
||||||
|
fencedParent.slice("FencedCode:query".length),
|
||||||
);
|
);
|
||||||
const whereMatch =
|
const whereMatch =
|
||||||
/(where|order\s+by|and|or|select(\s+[\w\s,]+)?)\s+([\w\-_]*)$/
|
/(where|order\s+by|and|or|select(\s+[\w\s,]+)?)\s+([\w\-_]*)$/
|
||||||
|
@ -78,3 +81,23 @@ function attributeCompletionsToCMCompletion(
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function languageComplete(completeEvent: CompleteEvent) {
|
||||||
|
const languagePrefix = /^```(\w*)$/.exec(
|
||||||
|
completeEvent.linePrefix,
|
||||||
|
);
|
||||||
|
if (!languagePrefix) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const allLanguages = await language.listLanguages();
|
||||||
|
return {
|
||||||
|
from: completeEvent.pos - languagePrefix[1].length,
|
||||||
|
options: allLanguages.map(
|
||||||
|
(lang) => ({
|
||||||
|
label: lang,
|
||||||
|
type: "language",
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
|
@ -15,6 +15,11 @@ functions:
|
||||||
events:
|
events:
|
||||||
- editor:complete
|
- editor:complete
|
||||||
|
|
||||||
|
languageComplete:
|
||||||
|
path: complete.ts:languageComplete
|
||||||
|
events:
|
||||||
|
- editor:complete
|
||||||
|
|
||||||
# Slash commands
|
# Slash commands
|
||||||
insertQuery:
|
insertQuery:
|
||||||
redirect: template.insertTemplateText
|
redirect: template.insertTemplateText
|
||||||
|
|
|
@ -46,10 +46,10 @@ functions:
|
||||||
redirect: insertTemplateText
|
redirect: insertTemplateText
|
||||||
slashCommand:
|
slashCommand:
|
||||||
name: code
|
name: code
|
||||||
description: Insert code block
|
description: Insert fenced code block
|
||||||
value: |
|
value: |
|
||||||
```
|
```|^|
|
||||||
|^|
|
|
||||||
```
|
```
|
||||||
|
|
||||||
insertHRTemplate:
|
insertHRTemplate:
|
||||||
|
|
|
@ -640,9 +640,8 @@ export class Client {
|
||||||
let node: SyntaxNode | null = currentNode;
|
let node: SyntaxNode | null = currentNode;
|
||||||
do {
|
do {
|
||||||
if (node.name === "FencedCode") {
|
if (node.name === "FencedCode") {
|
||||||
const code = editorState.sliceDoc(node.from + 3, node.to);
|
const body = editorState.sliceDoc(node.from + 3, node.to - 3);
|
||||||
const fencedCodeLanguage = code.split("\n")[0];
|
parentNodes.push(`FencedCode:${body}`);
|
||||||
parentNodes.push(`FencedCode:${fencedCodeLanguage}`);
|
|
||||||
} else {
|
} else {
|
||||||
parentNodes.push(node.name);
|
parentNodes.push(node.name);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,15 @@ release.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## Next
|
||||||
|
* Fixes to auto-sizing of [[Live Queries]] and [[Live Templates]] widgets
|
||||||
|
* Fixed the combination of `limit` and `order by` not working well
|
||||||
|
* Auto complete for queries now works for queries split across multiple lines
|
||||||
|
* Auto complete for fenced code block languages (use the `/code` slash command)
|
||||||
|
* Slightly tweaked semantics for the `=` operator on arrays, see [[Live Queries]] for details
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## 0.5.0
|
## 0.5.0
|
||||||
Oh boy, this is a big one. This release brings you the following:
|
Oh boy, this is a big one. This release brings you the following:
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue