Skip to content

Commit 3c2f06a

Browse files
committed
Fixes #357
1 parent 822261a commit 3c2f06a

File tree

5 files changed

+82
-5
lines changed

5 files changed

+82
-5
lines changed

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,8 @@
11
# Changelog
22

3+
## [2.7.0] - 2025-XX-XX
4+
- Fix issue [#357](https://github.com/intersystems/language-server/issues/357): Add completion for globals and routines
5+
36
## [2.6.5] - 2024-11-13
47
- Fix issue [#356](https://github.com/intersystems/language-server/issues/356): Unexpected new dialog during password retrieval using Server Manager authprovider
58

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ This is a [LSP](https://microsoft.github.io/language-server-protocol/) compliant
99
- [Semantic token](https://code.visualstudio.com/api/language-extensions/semantic-highlight-guide)-based coloring for InterSystems ObjectScript classes, routines and CSP files, with support for embedded languages like SQL, Python, HTML, XML, Java, JavaScript and CSS.
1010
- Hover information for ObjectScript commands, system functions, system variables, classes, class members, macros, preprocessor directives, class keywords, Parameter types, Storage definition keywords, and embedded SQL tables, fields and class methods and queries invoked as SQL procedures.
1111
- [Go to definition](https://code.visualstudio.com/docs/editor/editingevolved#_go-to-definition) for ObjectScript classes, class members, macros, routines, routine labels, class name parameters, `##super()`, and embedded SQL tables, fields and class methods and queries invoked as SQL procedures.
12-
- Code completion for ObjectScript classes, class members, system functions, system variables, macros, include files, package imports, preprocessor directives, class keywords, class keyword values, class Parameter types and Storage definition keywords. Code completion for properties referenced using instance variable syntax (`i%PropertyName`) must be triggered manually using `Ctrl-Space` with the cursor immediately after the `i%`.
12+
- Code completion for ObjectScript classes, class members, system functions, system variables, macros, include files, package imports, preprocessor directives, class keywords, class keyword values, class Parameter types, Storage definition keywords, routines and globals. Code completion for properties referenced using instance variable syntax (`i%PropertyName`) must be triggered manually using `Ctrl-Space` with the cursor immediately after the `i%`.
1313
- Code completion for XML Element names, Attribute names and Attribute values within XData blocks that have the XMLNamespace keyword set to a URL that corresponds to a Studio Assist Schema (SASchema).
1414
- Signature help for ObjectScript methods and macros that accept arguments.
1515
- Document symbols for ObjectScript classes, routines and include files.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -241,7 +241,7 @@
241241
"scope": "window",
242242
"type": "boolean",
243243
"default": false,
244-
"markdownDescription": "Controls whether class members with the `Internal` keyword are shown in completion lists."
244+
"markdownDescription": "Controls whether class members with the `Internal` keyword and system globals are shown in completion lists."
245245
},
246246
"intersystems.language-server.completion.showGenerated": {
247247
"scope": "window",

server/src/providers/completion.ts

+76-2
Original file line numberDiff line numberDiff line change
@@ -657,6 +657,69 @@ async function completionInclude(server: ServerSpec): Promise<CompletionItem[]>
657657
return result;
658658
};
659659

660+
/** Return a list of globals or routines. Optionally filter using `prefix`. */
661+
async function globalsOrRoutines(
662+
doc: TextDocument, parsed: compressedline[], line: number, token: number,
663+
settings: LanguageServerConfiguration, server: ServerSpec, lineText: string, prefix: string = ""
664+
): Promise<CompletionItem[]> {
665+
// Determine if this is a routine or global, and return null if we're in $BITLOGIC
666+
let brk = false, parenLevel = 0, isBitlogic = false, lastCmd = "";
667+
for (let ln = line; ln >= 0; ln--) {
668+
if (!parsed[ln]?.length) continue;
669+
for (let tkn = (ln == line ? token : parsed[ln].length - 1); tkn >= 0; tkn--) {
670+
if (parsed[ln][tkn].l == ld.cos_langindex && parsed[ln][tkn].s == ld.cos_delim_attrindex) {
671+
const delimtext = doc.getText(Range.create(ln,parsed[ln][tkn].p,ln,parsed[ln][tkn].p+parsed[ln][tkn].c));
672+
if (delimtext == "(") {
673+
parenLevel++;
674+
if (parenLevel > 0 && tkn > 0 && parsed[ln][tkn-1].l == ld.cos_langindex && parsed[ln][tkn-1].s == ld.cos_sysf_attrindex &&
675+
doc.getText(Range.create(ln,parsed[ln][tkn-1].p,ln,parsed[ln][tkn-1].p+parsed[ln][tkn-1].c)).toLowerCase() == "$bitlogic"
676+
) {
677+
isBitlogic = true;
678+
brk = true;
679+
break;
680+
}
681+
}
682+
else if (delimtext == ")") {
683+
parenLevel--;
684+
}
685+
} else if (parsed[ln][tkn].l == ld.cos_langindex && parsed[ln][tkn].s == ld.cos_command_attrindex) {
686+
lastCmd = doc.getText(Range.create(ln,parsed[ln][tkn].p,ln,parsed[ln][tkn].p+parsed[ln][tkn].c)).toLowerCase();
687+
brk = true;
688+
break;
689+
}
690+
}
691+
if (brk) break;
692+
}
693+
if (isBitlogic) return null;
694+
const isRoutine =
695+
// The character before the caret is part of a label or extrinsic function syntax
696+
/[%$\d\p{L}]/u.test(lineText.slice(-2,-1)) ||
697+
// Routine syntax without a label or extrinsic function syntax can only appear as an argument for a DO or JOB command
698+
(["d","do","j","job"].includes(lastCmd) && parenLevel == 0) ||
699+
// Special case needed since this DO command will be an error token instead of a command token
700+
/(^|\s+)do?\s+\^$/i.test(lineText);
701+
const respdata = await makeRESTRequest("POST",1,"/action/query",server,
702+
isRoutine ? {
703+
query: `SELECT DISTINCT $PIECE(Name,'.',1,$LENGTH(Name,'.')-1) AS Name FROM %Library.RoutineMgr_StudioOpenDialog(?,1,1,1,1,1,0,'NOT (Name %PATTERN ''.E1"."0.1"G"1N1".obj"'' AND $LENGTH(Name,''.'') > 3)')`,
704+
parameters: [`${prefix.length ? `${prefix.slice(0,-1)}/` : ""}*.mac,*.int,*.obj`]
705+
} : {
706+
query: "SELECT Name FROM %SYS.GlobalQuery_NameSpaceList(,?,?,,,1,0)",
707+
parameters: [`${prefix}*`,settings.completion.showInternal ? 1 : 0]
708+
}
709+
);
710+
if (Array.isArray(respdata?.data?.result?.content) && respdata.data.result.content.length > 0) {
711+
return respdata.data.result.content.map((item: { Name: string; }) => {
712+
return {
713+
label: item.Name.slice(prefix.length),
714+
kind: isRoutine ? CompletionItemKind.Function : CompletionItemKind.Variable,
715+
data: isRoutine ? "routine" : "global"
716+
};
717+
});
718+
} else {
719+
return null;
720+
}
721+
}
722+
660723
export async function onCompletion(params: CompletionParams): Promise<CompletionItem[] | null> {
661724
var result: CompletionItem[] = [];
662725
const doc = documents.get(params.textDocument.uri);
@@ -1063,6 +1126,7 @@ export async function onCompletion(params: CompletionParams): Promise<Completion
10631126
}
10641127
if (isCls) prevtokentype = "class";
10651128
}
1129+
const globalOrRoutineMatch = prevline.match(/\^(%?[\d\p{L}.]+)$/u);
10661130
if (prevtokentype === "class" || prevtokentype === "system") {
10671131
// This is a partial class name
10681132

@@ -1100,6 +1164,11 @@ export async function onCompletion(params: CompletionParams): Promise<Completion
11001164
}
11011165
}
11021166
}
1167+
else if (globalOrRoutineMatch && triggerlang == ld.cos_langindex) {
1168+
// This might be a routine or global
1169+
1170+
result = await globalsOrRoutines(doc,parsed,params.position.line,thistoken,settings,server,prevline.slice(0,-(globalOrRoutineMatch[1].length)),globalOrRoutineMatch[1]);
1171+
}
11031172
else {
11041173
// This is a class member
11051174

@@ -2150,11 +2219,16 @@ export async function onCompletion(params: CompletionParams): Promise<Completion
21502219
}
21512220
}
21522221
}
2222+
else if (prevline.slice(-1) == "^" && triggerlang == ld.cos_langindex) {
2223+
// This might be a routine or global
2224+
2225+
result = await globalsOrRoutines(doc,parsed,params.position.line,thistoken,settings,server,prevline);
2226+
}
21532227
return result;
21542228
}
21552229

21562230
export async function onCompletionResolve(item: CompletionItem): Promise<CompletionItem> {
2157-
if (item.data instanceof Array && item.data[0] === "class") {
2231+
if (Array.isArray(item.data) && item.data[0] === "class") {
21582232
// Get the description for this class from the server
21592233
const server: ServerSpec = await getServerSpec(item.data[2]);
21602234
const querydata: QueryData = {
@@ -2170,7 +2244,7 @@ export async function onCompletionResolve(item: CompletionItem): Promise<Complet
21702244
};
21712245
}
21722246
}
2173-
else if (item.data instanceof Array && item.data[0] === "macro" && item.documentation === undefined) {
2247+
else if (Array.isArray(item.data) && item.data[0] === "macro" && !item.documentation) {
21742248
// Get the macro definition from the server
21752249
const server: ServerSpec = await getServerSpec(item.data[1]);
21762250
const querydata = {

server/src/server.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ connection.onInitialize(() => {
3939
textDocumentSync: TextDocumentSyncKind.Full,
4040
completionProvider: {
4141
resolveProvider: true,
42-
triggerCharacters: [".","$","("," ","<",'"',"#"]
42+
triggerCharacters: [".","$","("," ","<",'"',"#","^"]
4343
},
4444
hoverProvider: true,
4545
definitionProvider: true,

0 commit comments

Comments
 (0)