Compare commits
9 commits
Author | SHA1 | Date | |
---|---|---|---|
|
97fe45f6f9 | ||
|
9e2ee80699 | ||
|
62c25bc693 | ||
|
3b0b58e44b | ||
|
3fa4e39342 | ||
|
4d1ea863df | ||
|
b7ebfee84d | ||
|
a7513fab8c | ||
|
b61270e51e |
9 changed files with 1246 additions and 67 deletions
|
@ -2,6 +2,15 @@
|
||||||
|
|
||||||
All notable changes to this project will be documented in this file.
|
All notable changes to this project will be documented in this file.
|
||||||
|
|
||||||
|
## Unreleased Version 1.2.0 (2025-XX-XX)
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- New caching mechanism for references, tagRenderings and filters.
|
||||||
|
This allows the completions to show _all_ filters, not just the ones in `filters.json` and `questions.json`.
|
||||||
|
It's also possible to look up uses of a filter or a tagRendering.
|
||||||
|
The caching will take about 30 seconds to complete, but it will only run once per session, and will update individual files as they are saved or removed.
|
||||||
|
|
||||||
## Version 1.1.1 (2025-01-07)
|
## Version 1.1.1 (2025-01-07)
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
48
README.md
48
README.md
|
@ -5,15 +5,27 @@ It adds autocompletion and defintion support for the MapComplete theme and layer
|
||||||
|
|
||||||
Currently the following features are supported:
|
Currently the following features are supported:
|
||||||
|
|
||||||
- Autocompletion for the layer names
|
- Layers:
|
||||||
- Definition support for the layer names
|
- Autocompletion for the layer names
|
||||||
- Definintion support for icons
|
- Definition support for the layer names
|
||||||
- Autocompletion for tagRenderings in questions.json
|
- Implementation support for the layer names (showing where they are used)
|
||||||
- Definition support for tagRenderings
|
- TagRenderings:
|
||||||
- Autocompletion for filter keys in questions.json
|
- Autocompletion for _all_ tagRenderings
|
||||||
- Definition support for filter keys
|
- Definition support for tagRenderings
|
||||||
- Definition support for paths in license_info.json files
|
- Implementation support for tagRenderings (showing where they are used)
|
||||||
- Colour support for markers, lines and fills
|
- Filters:
|
||||||
|
- Autocompletion for _all_ filters
|
||||||
|
- Definition support for filters
|
||||||
|
- Implementation support for filters (showing where they are used)
|
||||||
|
- Icons:
|
||||||
|
- Autocompletion for icons
|
||||||
|
- Definition support for icons
|
||||||
|
- Colours:
|
||||||
|
- Support for colours in markers, lines and fills
|
||||||
|
- License info:
|
||||||
|
- Definition support for paths in license_info.json files
|
||||||
|
|
||||||
|
To achieve some of these feature, on startup the extension will parse all the files, which takes about 30 seconds. This will only happen once per session, and will update individual files as they are saved or removed.
|
||||||
|
|
||||||
![Demo showing autcomplete for layers and icon definition](images/demo.gif)
|
![Demo showing autcomplete for layers and icon definition](images/demo.gif)
|
||||||
|
|
||||||
|
@ -23,20 +35,24 @@ All notable changes to this project are documented in the [CHANGELOG](CHANGELOG.
|
||||||
|
|
||||||
The extension can be installed in several ways:
|
The extension can be installed in several ways:
|
||||||
|
|
||||||
### From the Visual Studio Code marketplace
|
### From the marketplace
|
||||||
|
|
||||||
You can install this extension from the [Visual Studio Code marketplace](https://marketplace.visualstudio.com/items?itemName=robin-van-der-linde.mapcompletevscode). Just search for "MapComplete" and you should find it.
|
The extension is available both in the [Visual Studio Code marketplace](https://marketplace.visualstudio.com/items?itemName=robin-van-der-linde.mapcompletevscode) and the [Open VSX registry](https://open-vsx.org/extension/robin-van-der-linde/mapcompletevscode).
|
||||||
|
|
||||||
Alternatively you van press `Ctrl+P` and paste the following command:
|
So for both Visual Studio Code and VSCodium, you should just be able to search for "MapComplete" in the extensions tab and install it from there.
|
||||||
|
|
||||||
|
Alternatively you can press `Ctrl+P` and paste the following command:
|
||||||
|
|
||||||
```cmd
|
```cmd
|
||||||
ext install robin-van-der-linde.mapcompletevscode
|
ext install robin-van-der-linde.mapcompletevscode
|
||||||
```
|
```
|
||||||
|
|
||||||
### From Open VSX
|
|
||||||
|
|
||||||
You can also install the extension from the [Open VSX registry](https://open-vsx.org/extension/robin-van-der-linde/mapcompletevscode). Just search for "MapComplete" and you should find it.
|
|
||||||
|
|
||||||
### From the .vsix file
|
### From the .vsix file
|
||||||
|
|
||||||
You can also install the extension from the .vsix file. You can download the latest version from the [releases page](https://github.com/RobinLinde/MapCompleteVScode/releases). After downloading the .vsix file, you should be able to install it by going to extensions in Visual Studio Code and clicking on the three dots in the top right corner. Then click on "Install from VSIX..." and select the downloaded .vsix file.
|
You can also install the extension from the .vsix file. You can download the latest version from the [releases page](https://github.com/RobinLinde/MapCompleteVScode/releases). After downloading the .vsix file, you should be able to install it by going to extensions in Visual Studio Code and clicking on the three dots in the top right corner. Then click on "Install from VSIX..." and select the downloaded .vsix file.
|
||||||
|
|
||||||
|
It's also possible to install builds for any commit in any branch by checking out the workflow run for the commit you want to install, and downloading the .vsix file from the artifacts.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
Most of the features should be pretty self-explanatory. As for the implementation support, the 'anchor' for this is the id property of the layer, tagRendering or filter. This means that if you want to see where a layer is used, you should be able to see all uses by using `CTRL+F12` on the id property of the layer, or by right-clicking on the id property and selecting "Go to Implementations".
|
||||||
|
|
4
package-lock.json
generated
4
package-lock.json
generated
|
@ -1,12 +1,12 @@
|
||||||
{
|
{
|
||||||
"name": "mapcompletevscode",
|
"name": "mapcompletevscode",
|
||||||
"version": "1.1.0",
|
"version": "1.2.0",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "mapcompletevscode",
|
"name": "mapcompletevscode",
|
||||||
"version": "1.1.0",
|
"version": "1.2.0",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"jsonc-parser": "^3.3.1"
|
"jsonc-parser": "^3.3.1"
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
{
|
{
|
||||||
"name": "mapcompletevscode",
|
"name": "mapcompletevscode",
|
||||||
"displayName": "MapComplete VScode",
|
"displayName": "MapComplete VScode",
|
||||||
"version": "1.1.1",
|
"version": "1.2.0",
|
||||||
"publisher": "robin-van-der-linde",
|
"publisher": "robin-van-der-linde",
|
||||||
"author": {
|
"author": {
|
||||||
"name": "Robin van der Linde",
|
"name": "Robin van der Linde",
|
||||||
|
|
|
@ -4,12 +4,16 @@ import { colorProvider, iconDefinitionProvider } from "./generic";
|
||||||
import {
|
import {
|
||||||
filterCompletionProvider,
|
filterCompletionProvider,
|
||||||
filterDefinitionProvider,
|
filterDefinitionProvider,
|
||||||
|
filterImplementationProvider,
|
||||||
|
layerImplementationProvider,
|
||||||
tagRenderingCompletionProvider,
|
tagRenderingCompletionProvider,
|
||||||
tagRenderingDefinitionProvider,
|
tagRenderingDefinitionProvider,
|
||||||
|
tagRenderingImplementationProvider,
|
||||||
} from "./layers";
|
} from "./layers";
|
||||||
import { pathDefinitionProvider } from "./license_info";
|
import { pathDefinitionProvider } from "./license_info";
|
||||||
|
import { CacheWorker } from "./utils/cache";
|
||||||
|
|
||||||
export function activate(context: vscode.ExtensionContext) {
|
export async function activate(context: vscode.ExtensionContext) {
|
||||||
// Activate all theme related features
|
// Activate all theme related features
|
||||||
context.subscriptions.push(layerCompletionProvider, layerDefinitionProvider);
|
context.subscriptions.push(layerCompletionProvider, layerDefinitionProvider);
|
||||||
|
|
||||||
|
@ -18,7 +22,10 @@ export function activate(context: vscode.ExtensionContext) {
|
||||||
tagRenderingCompletionProvider,
|
tagRenderingCompletionProvider,
|
||||||
tagRenderingDefinitionProvider,
|
tagRenderingDefinitionProvider,
|
||||||
filterCompletionProvider,
|
filterCompletionProvider,
|
||||||
filterDefinitionProvider
|
filterDefinitionProvider,
|
||||||
|
tagRenderingImplementationProvider,
|
||||||
|
filterImplementationProvider,
|
||||||
|
layerImplementationProvider
|
||||||
);
|
);
|
||||||
|
|
||||||
// Activate all license info related features
|
// Activate all license info related features
|
||||||
|
@ -26,4 +33,10 @@ export function activate(context: vscode.ExtensionContext) {
|
||||||
|
|
||||||
// Activate all generic features
|
// Activate all generic features
|
||||||
context.subscriptions.push(iconDefinitionProvider, colorProvider);
|
context.subscriptions.push(iconDefinitionProvider, colorProvider);
|
||||||
|
|
||||||
|
// Upon activation, we also scan the workspace for all themes and layers
|
||||||
|
// and save them in a cache, so we can quickly look up definitions and completions
|
||||||
|
// for each theme and layer
|
||||||
|
// We should also listen for changes in the workspace, so we can update the cache
|
||||||
|
CacheWorker.create(context);
|
||||||
}
|
}
|
||||||
|
|
223
src/layers.ts
223
src/layers.ts
|
@ -13,6 +13,7 @@ import { getCursorPath, getRawCursorPath, getStartEnd } from "./utils/cursor";
|
||||||
import { getFilters, getTagRenderings } from "./utils/mapcomplete";
|
import { getFilters, getTagRenderings } from "./utils/mapcomplete";
|
||||||
import { getValueFromPath } from "./utils/json";
|
import { getValueFromPath } from "./utils/json";
|
||||||
import { JSONPath } from "jsonc-parser";
|
import { JSONPath } from "jsonc-parser";
|
||||||
|
import { Cache } from "./utils/cache";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tag rendering completion provider
|
* Tag rendering completion provider
|
||||||
|
@ -172,6 +173,80 @@ export const tagRenderingDefinitionProvider =
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const tagRenderingImplementationProvider =
|
||||||
|
vscode.languages.registerImplementationProvider(
|
||||||
|
{
|
||||||
|
language: "json",
|
||||||
|
scheme: "file",
|
||||||
|
pattern: "**/assets/*/*/*.json",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
async provideImplementation(
|
||||||
|
document: vscode.TextDocument,
|
||||||
|
position: vscode.Position
|
||||||
|
) {
|
||||||
|
console.log("tagRenderingImplementationProvider");
|
||||||
|
const text = document.getText();
|
||||||
|
const jsonPath = getCursorPath(text, position);
|
||||||
|
const rawJsonPath = getRawCursorPath(text, position);
|
||||||
|
|
||||||
|
const regex = /^tagRenderings(\+)?\.\d+\.id$/;
|
||||||
|
|
||||||
|
if (regex.exec(jsonPath)) {
|
||||||
|
const tagRenderingId = getValueFromPath(text, rawJsonPath);
|
||||||
|
const layerName = document.fileName.split("/").pop()?.split(".")[0];
|
||||||
|
const to = `layers.${layerName}.tagRenderings.${tagRenderingId}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const cache = await Cache.create();
|
||||||
|
const references = cache.getReferences(to);
|
||||||
|
|
||||||
|
if (references.length === 0) {
|
||||||
|
return null;
|
||||||
|
} else {
|
||||||
|
console.log(`Found ${references.length} references to ${to}`);
|
||||||
|
|
||||||
|
const links: vscode.DefinitionLink[] = [];
|
||||||
|
for (const reference of references) {
|
||||||
|
console.log(
|
||||||
|
`Pushing link from ${document.fileName} to ${reference.reference?.from.uri?.fsPath} at ${reference.reference?.to.range?.[0]?.line}:${reference.reference?.to.range?.[0]?.character}`,
|
||||||
|
reference
|
||||||
|
);
|
||||||
|
|
||||||
|
// Check if we have a targetUri
|
||||||
|
if (reference.reference?.from.uri) {
|
||||||
|
links.push({
|
||||||
|
originSelectionRange: new vscode.Range(
|
||||||
|
reference.reference?.to?.range?.[0]?.line ?? 0,
|
||||||
|
reference.reference?.to?.range?.[0]?.character ?? 0,
|
||||||
|
reference.reference?.to?.range?.[1]?.line ?? 0,
|
||||||
|
reference.reference?.to?.range?.[1]?.character ?? 0
|
||||||
|
),
|
||||||
|
targetRange: new vscode.Range(
|
||||||
|
reference.reference?.from?.range?.[0]?.line ?? 0,
|
||||||
|
reference.reference?.from?.range?.[0]?.character ?? 0,
|
||||||
|
reference.reference?.from?.range?.[1]?.line ?? 0,
|
||||||
|
reference.reference?.from?.range?.[1]?.character ?? 0
|
||||||
|
),
|
||||||
|
targetUri: reference.reference?.from?.uri,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.error("Incomplete reference", reference);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log(`Found ${links.length} implementations`);
|
||||||
|
return links;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error get implementation", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Filter completion provider
|
* Filter completion provider
|
||||||
*
|
*
|
||||||
|
@ -329,3 +404,151 @@ export const filterDefinitionProvider =
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const filterImplementationProvider =
|
||||||
|
vscode.languages.registerImplementationProvider(
|
||||||
|
{
|
||||||
|
language: "json",
|
||||||
|
scheme: "file",
|
||||||
|
pattern: "**/assets/*/*/*.json",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
async provideImplementation(
|
||||||
|
document: vscode.TextDocument,
|
||||||
|
position: vscode.Position
|
||||||
|
) {
|
||||||
|
console.log("filterImplementationProvider");
|
||||||
|
const text = document.getText();
|
||||||
|
const jsonPath = getCursorPath(text, position);
|
||||||
|
const rawJsonPath = getRawCursorPath(text, position);
|
||||||
|
|
||||||
|
const regex = /^filter(\+)?\.\d+\.id$/;
|
||||||
|
|
||||||
|
if (regex.exec(jsonPath)) {
|
||||||
|
const filterId = getValueFromPath(text, rawJsonPath);
|
||||||
|
const layerName = document.fileName.split("/").pop()?.split(".")[0];
|
||||||
|
const to = `layers.${layerName}.filter.${filterId}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const cache = await Cache.create();
|
||||||
|
const references = cache.getReferences(to);
|
||||||
|
|
||||||
|
if (references.length === 0) {
|
||||||
|
console.log("No references found to", to);
|
||||||
|
return null;
|
||||||
|
} else {
|
||||||
|
console.log(`Found ${references.length} references to ${to}`);
|
||||||
|
|
||||||
|
const links: vscode.DefinitionLink[] = [];
|
||||||
|
for (const reference of references) {
|
||||||
|
console.log(
|
||||||
|
`Pushing link from ${document.fileName} to ${reference.reference?.from.uri?.fsPath} at ${reference.reference?.to.range?.[0]?.line}:${reference.reference?.to.range?.[0]?.character}`,
|
||||||
|
reference
|
||||||
|
);
|
||||||
|
|
||||||
|
// Check if we have a targetUri
|
||||||
|
if (reference.reference?.from.uri) {
|
||||||
|
links.push({
|
||||||
|
originSelectionRange: new vscode.Range(
|
||||||
|
reference.reference?.to?.range?.[0]?.line ?? 0,
|
||||||
|
reference.reference?.to?.range?.[0]?.character ?? 0,
|
||||||
|
reference.reference?.to?.range?.[1]?.line ?? 0,
|
||||||
|
reference.reference?.to?.range?.[1]?.character ?? 0
|
||||||
|
),
|
||||||
|
targetRange: new vscode.Range(
|
||||||
|
reference.reference?.from?.range?.[0]?.line ?? 0,
|
||||||
|
reference.reference?.from?.range?.[0]?.character ?? 0,
|
||||||
|
reference.reference?.from?.range?.[1]?.line ?? 0,
|
||||||
|
reference.reference?.from?.range?.[1]?.character ?? 0
|
||||||
|
),
|
||||||
|
targetUri: reference.reference?.from?.uri,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.error("Incomplete reference", reference);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log(`Found ${links.length} implementations`);
|
||||||
|
return links;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error get implementation", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
export const layerImplementationProvider =
|
||||||
|
vscode.languages.registerImplementationProvider(
|
||||||
|
{
|
||||||
|
language: "json",
|
||||||
|
scheme: "file",
|
||||||
|
pattern: "**/assets/layers/*/*.json",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
async provideImplementation(
|
||||||
|
document: vscode.TextDocument,
|
||||||
|
position: vscode.Position
|
||||||
|
) {
|
||||||
|
console.log("layerImplementationProvider");
|
||||||
|
const text = document.getText();
|
||||||
|
const jsonPath = getCursorPath(text, position);
|
||||||
|
const rawJsonPath = getRawCursorPath(text, position);
|
||||||
|
|
||||||
|
// Easiest regex in this package for sure
|
||||||
|
const regex = /^id$/;
|
||||||
|
|
||||||
|
if (regex.exec(jsonPath)) {
|
||||||
|
const layerId = getValueFromPath(text, rawJsonPath);
|
||||||
|
const to = `layers.${layerId}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const cache = await Cache.create();
|
||||||
|
const references = cache.getReferences(to);
|
||||||
|
|
||||||
|
if (references.length === 0) {
|
||||||
|
console.log("No references found to", to);
|
||||||
|
return null;
|
||||||
|
} else {
|
||||||
|
console.log(`Found ${references.length} references to ${to}`);
|
||||||
|
|
||||||
|
const links: vscode.DefinitionLink[] = [];
|
||||||
|
for (const reference of references) {
|
||||||
|
console.log(
|
||||||
|
`Pushing link from ${document.fileName} to ${reference.reference?.from.uri?.fsPath} at ${reference.reference?.to.range?.[0]?.line}:${reference.reference?.to.range?.[0]?.character}`,
|
||||||
|
reference
|
||||||
|
);
|
||||||
|
|
||||||
|
// Check if we have a targetUri
|
||||||
|
if (reference.reference?.from.uri) {
|
||||||
|
links.push({
|
||||||
|
originSelectionRange: new vscode.Range(
|
||||||
|
reference.reference?.to?.range?.[0]?.line ?? 0,
|
||||||
|
reference.reference?.to?.range?.[0]?.character ?? 0,
|
||||||
|
reference.reference?.to?.range?.[1]?.line ?? 0,
|
||||||
|
reference.reference?.to?.range?.[1]?.character ?? 0
|
||||||
|
),
|
||||||
|
targetRange: new vscode.Range(
|
||||||
|
reference.reference?.from?.range?.[0]?.line ?? 0,
|
||||||
|
reference.reference?.from?.range?.[0]?.character ?? 0,
|
||||||
|
reference.reference?.from?.range?.[1]?.line ?? 0,
|
||||||
|
reference.reference?.from?.range?.[1]?.character ?? 0
|
||||||
|
),
|
||||||
|
targetUri: reference.reference?.from?.uri,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.error("Incomplete reference", reference);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log(`Found ${links.length} implementations`);
|
||||||
|
return links;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error get implementation", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
890
src/utils/cache.ts
Normal file
890
src/utils/cache.ts
Normal file
|
@ -0,0 +1,890 @@
|
||||||
|
/**
|
||||||
|
* This file contains all functions related to caching the workspace
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as vscode from "vscode";
|
||||||
|
import { JSONPath } from "jsonc-parser";
|
||||||
|
import { getStartEnd } from "./cursor";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Worker class to handle the cache creation and updates
|
||||||
|
*/
|
||||||
|
export class CacheWorker {
|
||||||
|
/**
|
||||||
|
* The extension context
|
||||||
|
*/
|
||||||
|
private readonly context: vscode.ExtensionContext;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List of cache items
|
||||||
|
*/
|
||||||
|
private cache: CacheItem[] = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new cache
|
||||||
|
*
|
||||||
|
* @param context The extension context
|
||||||
|
* @returns Promise<Cache> The cache
|
||||||
|
*/
|
||||||
|
public static async create(
|
||||||
|
context: vscode.ExtensionContext
|
||||||
|
): Promise<CacheWorker> {
|
||||||
|
const cache = new CacheWorker(context);
|
||||||
|
await cache.scanWorkspace();
|
||||||
|
return cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new cache
|
||||||
|
*
|
||||||
|
* @param context The extension context
|
||||||
|
*/
|
||||||
|
private constructor(context: vscode.ExtensionContext) {
|
||||||
|
this.context = context;
|
||||||
|
// We probably want to create a fileSystemWatcher here
|
||||||
|
// to listen for changes in the workspace
|
||||||
|
this.createFileSystemWatcher();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves the current cache to the storage as JSON
|
||||||
|
* TODO: Find a more elegant way to do this
|
||||||
|
*/
|
||||||
|
private save() {
|
||||||
|
const jsonString = JSON.stringify(this.cache);
|
||||||
|
// Save it in the cache.json file in the .cache folder in the workspace
|
||||||
|
const workspaceFolder = vscode.workspace.workspaceFolders?.[0].uri.fsPath;
|
||||||
|
if (workspaceFolder) {
|
||||||
|
const cacheUri = vscode.Uri.file(`${workspaceFolder}/.cache/cache.json`);
|
||||||
|
vscode.workspace.fs.writeFile(cacheUri, Buffer.from(jsonString));
|
||||||
|
} else {
|
||||||
|
console.error("No workspace folder found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a file system watcher
|
||||||
|
*/
|
||||||
|
private createFileSystemWatcher() {
|
||||||
|
const watcher = vscode.workspace.createFileSystemWatcher(
|
||||||
|
"**/assets/**/*.json"
|
||||||
|
);
|
||||||
|
watcher.onDidChange(this.onChanged, this, this.context.subscriptions);
|
||||||
|
watcher.onDidCreate(this.onCreated, this, this.context.subscriptions);
|
||||||
|
watcher.onDidDelete(this.onDeleted, this, this.context.subscriptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
private onChanged(uri: vscode.Uri) {
|
||||||
|
this.saveFileToCache(uri);
|
||||||
|
}
|
||||||
|
|
||||||
|
private onCreated(uri: vscode.Uri) {
|
||||||
|
this.saveFileToCache(uri);
|
||||||
|
}
|
||||||
|
|
||||||
|
private onDeleted(uri: vscode.Uri) {
|
||||||
|
this.deleteFileFromCache(uri);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scans the workspace for all themes and layers
|
||||||
|
*/
|
||||||
|
private async scanWorkspace() {
|
||||||
|
const files = await vscode.workspace.findFiles("**/assets/**/*.json");
|
||||||
|
for (const file of files) {
|
||||||
|
this.saveFileToCache(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scans a file, extracts all relevant information and saves it to the cache
|
||||||
|
*
|
||||||
|
* @param uri File URI
|
||||||
|
*/
|
||||||
|
private saveFileToCache(uri: vscode.Uri) {
|
||||||
|
// First, we determine what kind of file it is, and whether we actually care about it
|
||||||
|
// If we don't care about it, we return early
|
||||||
|
// Possible relevant patterns:
|
||||||
|
// - ./assets/themes/{name}/{name}.json
|
||||||
|
// - ./assets/layers/{name}/{name}.json
|
||||||
|
|
||||||
|
const filePath = uri.fsPath;
|
||||||
|
const regex = /\/assets\/(?<asset>themes|layers)\/([^/]+)\/\2\.json/;
|
||||||
|
const match = regex.exec(filePath);
|
||||||
|
const asset = match?.groups?.asset;
|
||||||
|
if (!match) {
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
console.log("We care about this file", filePath);
|
||||||
|
this.deleteFileFromCache(uri);
|
||||||
|
|
||||||
|
// Determine what kind of file we're dealing with
|
||||||
|
switch (asset) {
|
||||||
|
case "themes":
|
||||||
|
console.log("Theme found", filePath);
|
||||||
|
this.saveThemeToCache(uri);
|
||||||
|
break;
|
||||||
|
case "layers":
|
||||||
|
console.log("Layer found", filePath);
|
||||||
|
this.saveLayerToCache(uri);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
console.error("Unknown asset type", filePath);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deletes a file from the cache
|
||||||
|
*
|
||||||
|
* @param uri File URI
|
||||||
|
*/
|
||||||
|
private deleteFileFromCache(uri: vscode.Uri) {
|
||||||
|
this.cache = this.cache.filter(
|
||||||
|
(item) => item.filePath.fsPath !== uri.fsPath
|
||||||
|
);
|
||||||
|
|
||||||
|
this.save();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves a theme to the cache
|
||||||
|
*
|
||||||
|
* @param filePath The file path
|
||||||
|
*/
|
||||||
|
private async saveThemeToCache(uri: vscode.Uri) {
|
||||||
|
const filePath = uri.fsPath;
|
||||||
|
console.log("Saving theme to cache", filePath);
|
||||||
|
/**
|
||||||
|
* For now, we only care about the layer references in the theme
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Read the file
|
||||||
|
const content = vscode.workspace.fs.readFile(uri);
|
||||||
|
const text = new TextDecoder().decode(await content);
|
||||||
|
const json = JSON.parse(text);
|
||||||
|
|
||||||
|
// Determine some locations
|
||||||
|
const from = `themes.${json.id}`;
|
||||||
|
const fromFile = uri;
|
||||||
|
|
||||||
|
// Look through the layers
|
||||||
|
for (const layer of json.layers) {
|
||||||
|
const layerIndex = json.layers.indexOf(layer);
|
||||||
|
|
||||||
|
// Reference if it's a string
|
||||||
|
if (typeof layer === "string") {
|
||||||
|
// It is a reference
|
||||||
|
console.log(`Reference found to layer ${layer} in ${filePath}`);
|
||||||
|
|
||||||
|
const fromStartEnd = getStartEnd(text, ["layers", layerIndex]);
|
||||||
|
const to = `layers.${layer}`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(
|
||||||
|
`**/assets/layers/${layer}/${layer}.json`
|
||||||
|
);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: layer,
|
||||||
|
filePath: uri,
|
||||||
|
jsonPath: ["layers"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromFile,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
},
|
||||||
|
type: "layer",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// Builtin layer if we have a builtin property
|
||||||
|
else if (layer.builtin) {
|
||||||
|
if (typeof layer.builtin === "string") {
|
||||||
|
// Single layer
|
||||||
|
console.log(
|
||||||
|
`Reference found to builtin layer ${layer.builtin} in ${filePath}`
|
||||||
|
);
|
||||||
|
|
||||||
|
const fromStartEnd = getStartEnd(text, [
|
||||||
|
"layers",
|
||||||
|
layerIndex,
|
||||||
|
"builtin",
|
||||||
|
]);
|
||||||
|
const to = `layers.${layer.builtin}`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(
|
||||||
|
`**/assets/layers/${layer.builtin}/${layer.builtin}.json`
|
||||||
|
);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: layer.builtin,
|
||||||
|
filePath: uri,
|
||||||
|
jsonPath: ["layers"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromFile,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
},
|
||||||
|
type: "layer",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Multiple layers
|
||||||
|
for (const builtinLayer of layer.builtin) {
|
||||||
|
console.log(
|
||||||
|
`Reference found to builtin layer ${builtinLayer} in ${filePath}`
|
||||||
|
);
|
||||||
|
|
||||||
|
const builtinLayerIndex = layer.builtin.indexOf(builtinLayer);
|
||||||
|
const fromStartEnd = getStartEnd(text, [
|
||||||
|
"layers",
|
||||||
|
layerIndex,
|
||||||
|
"builtin",
|
||||||
|
builtinLayerIndex,
|
||||||
|
]);
|
||||||
|
const to = `layers.${builtinLayer}`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(
|
||||||
|
`**/assets/layers/${builtinLayer}/${builtinLayer}.json`
|
||||||
|
);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: builtinLayer,
|
||||||
|
filePath: uri,
|
||||||
|
jsonPath: ["layers"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromFile,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
},
|
||||||
|
type: "layer",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Inline layer else
|
||||||
|
else {
|
||||||
|
console.log(`Found inline layer ${layer.id} in ${filePath}`);
|
||||||
|
const layerText = JSON.stringify(layer);
|
||||||
|
const from = `themes.${json.id}.layers.${layerIndex}`;
|
||||||
|
await this.saveLayerTextToCache(layerText, uri, from, true, text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.save();
|
||||||
|
this.printCache();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves a layer to the cache
|
||||||
|
*
|
||||||
|
* @param uri The URI of the layer
|
||||||
|
*/
|
||||||
|
private async saveLayerToCache(uri: vscode.Uri) {
|
||||||
|
if (uri.fsPath.endsWith("favourite.json")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read the file
|
||||||
|
const content = vscode.workspace.fs.readFile(uri);
|
||||||
|
const text = new TextDecoder().decode(await content);
|
||||||
|
const uriPath = uri.path;
|
||||||
|
const uriPathSplit = uriPath.split("/");
|
||||||
|
const uriFileName = uriPathSplit[uriPathSplit.length - 1];
|
||||||
|
const from = `layers.${uriFileName.split(".")[0]}`;
|
||||||
|
|
||||||
|
await this.saveLayerTextToCache(text, uri, from);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Save a layer to the cache, given the text of the layer
|
||||||
|
*
|
||||||
|
* @param text Text representation of layer
|
||||||
|
* @param uri The URI of the layer file
|
||||||
|
* @param from The theme or layer where the layer is from, e.g. layers.bicycle_rental or themes.cyclofix.layers.0
|
||||||
|
* @param referencesOnly Whether to only save references, or also the tagRenderings and filters. This is useful for inline layers, because their filters and tagRenderings can't be reused
|
||||||
|
* @param fullFileText The full text of the original theme file, used for calculating position
|
||||||
|
*/
|
||||||
|
private async saveLayerTextToCache(
|
||||||
|
text: string,
|
||||||
|
uri: vscode.Uri,
|
||||||
|
from: string,
|
||||||
|
referencesOnly = false,
|
||||||
|
fullFileText?: string
|
||||||
|
) {
|
||||||
|
const filePath = uri.fsPath;
|
||||||
|
console.log("Saving layer to cache", filePath);
|
||||||
|
/**
|
||||||
|
* For now, we only care about the tagRenderings and filters in the layer
|
||||||
|
*/
|
||||||
|
const json = JSON.parse(text);
|
||||||
|
|
||||||
|
// Check if this layer doesn't have a special source, or uses a geoJson source
|
||||||
|
if (json.source === "special" || json.source?.geoJson) {
|
||||||
|
console.log("Layer has a special source, only saving references");
|
||||||
|
referencesOnly = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look through the tagRenderings, if the layer has any
|
||||||
|
if (json.tagRenderings) {
|
||||||
|
try {
|
||||||
|
await this.saveTagRenderingsToCache(
|
||||||
|
text,
|
||||||
|
from,
|
||||||
|
uri,
|
||||||
|
referencesOnly,
|
||||||
|
fullFileText
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Error saving tagRenderings for ${from}`, error);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log("No tagRenderings found in", filePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (json.filter) {
|
||||||
|
// Look through the filters
|
||||||
|
try {
|
||||||
|
await this.saveFiltersToCache(
|
||||||
|
text,
|
||||||
|
from,
|
||||||
|
uri,
|
||||||
|
referencesOnly,
|
||||||
|
fullFileText
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Error saving filters for ${from}`, error);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log("No filters found in", filePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.save();
|
||||||
|
this.printCache();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Save tag renderings to cache
|
||||||
|
* TODO: references for tagRenderings can also be a label/group, which we don't support yet
|
||||||
|
*
|
||||||
|
* @param text Text representation of layer
|
||||||
|
* @param from The theme or layer where the layer is from, e.g. layers.bicycle_rental or themes.cyclofix.layers.0
|
||||||
|
* @param fromUri URI of the layer file
|
||||||
|
* @param referencesOnly Whether to only save references, or also the tagRenderings and filters. This is useful for inline layers, because their filters and tagRenderings can't be reused
|
||||||
|
* @param fullFileText The full text of the original theme file, used for calculating position
|
||||||
|
*/
|
||||||
|
private async saveTagRenderingsToCache(
|
||||||
|
text: string,
|
||||||
|
from: string,
|
||||||
|
fromUri: vscode.Uri,
|
||||||
|
referencesOnly = false,
|
||||||
|
fullFileText?: string
|
||||||
|
) {
|
||||||
|
const json = JSON.parse(text);
|
||||||
|
|
||||||
|
for (const tagRendering of json.tagRenderings) {
|
||||||
|
const tagRenderingReferenceIndex =
|
||||||
|
json.tagRenderings.indexOf(tagRendering);
|
||||||
|
|
||||||
|
// Check if it is a string and not an object
|
||||||
|
if (typeof tagRendering === "string") {
|
||||||
|
// It is a reference
|
||||||
|
console.log(
|
||||||
|
`Reference found to tagRendering ${tagRendering} in ${from}`
|
||||||
|
);
|
||||||
|
|
||||||
|
// The range is dependent on whether we're dealing with a full file or not
|
||||||
|
const path = fullFileText
|
||||||
|
? [
|
||||||
|
from.split(".")[2],
|
||||||
|
parseInt(from.split(".")[3]),
|
||||||
|
"tagRenderings",
|
||||||
|
tagRenderingReferenceIndex,
|
||||||
|
]
|
||||||
|
: ["tagRenderings", tagRenderingReferenceIndex];
|
||||||
|
const fromStartEnd = fullFileText
|
||||||
|
? getStartEnd(fullFileText, path)
|
||||||
|
: getStartEnd(text, path);
|
||||||
|
const to = tagRendering.includes(".")
|
||||||
|
? `layers.${tagRendering.split(".")[0]}.tagRenderings.${
|
||||||
|
tagRendering.split(".")[1]
|
||||||
|
}`
|
||||||
|
: `layers.questions.tagRenderings.${tagRendering}`;
|
||||||
|
const toFileName = tagRendering.includes(".")
|
||||||
|
? `**/assets/layers/${tagRendering.split(".")[0]}/${
|
||||||
|
tagRendering.split(".")[0]
|
||||||
|
}.json`
|
||||||
|
: `**/assets/layers/questions/questions.json`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(toFileName);
|
||||||
|
|
||||||
|
// Read toFile and get the text
|
||||||
|
const toContent = await vscode.workspace.fs.readFile(toFile[0]);
|
||||||
|
const toText = new TextDecoder().decode(toContent);
|
||||||
|
const toJson = JSON.parse(toText);
|
||||||
|
const trIndex = toJson.tagRenderings.findIndex(
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
(tr: any) => tr.id === tagRendering.split(".")?.pop()
|
||||||
|
);
|
||||||
|
const toRange = getStartEnd(toText, ["tagRenderings", trIndex]);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: tagRendering,
|
||||||
|
filePath: fromUri,
|
||||||
|
jsonPath: ["tagRenderings"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromUri,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
range: [toRange.start, toRange.end],
|
||||||
|
},
|
||||||
|
type: "tagRendering",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else if (typeof tagRendering === "object") {
|
||||||
|
// This is a tagRendering, or a reference to one, but with an override
|
||||||
|
if (tagRendering.builtin) {
|
||||||
|
// This is a reference to a built-in tagRendering (or multiple ones)
|
||||||
|
if (typeof tagRendering.builtin === "string") {
|
||||||
|
// Single tagRendering reference
|
||||||
|
console.log(
|
||||||
|
`Reference found to builtin tagRendering ${tagRendering.builtin} in ${from}`
|
||||||
|
);
|
||||||
|
// The range is dependent on whether we're dealing with a full file or not
|
||||||
|
const path = fullFileText
|
||||||
|
? [
|
||||||
|
from.split(".")[2],
|
||||||
|
parseInt(from.split(".")[3]),
|
||||||
|
"tagRenderings",
|
||||||
|
tagRenderingReferenceIndex,
|
||||||
|
"builtin",
|
||||||
|
]
|
||||||
|
: ["tagRenderings", tagRenderingReferenceIndex, "builtin"];
|
||||||
|
const fromStartEnd = fullFileText
|
||||||
|
? getStartEnd(fullFileText, path)
|
||||||
|
: getStartEnd(text, path);
|
||||||
|
const to = tagRendering.builtin.includes(".")
|
||||||
|
? `layers.${tagRendering.builtin.split(".")[0]}.tagRenderings.${
|
||||||
|
tagRendering.builtin.split(".")[1]
|
||||||
|
}`
|
||||||
|
: `layers.questions.tagRenderings.${tagRendering}`;
|
||||||
|
const toFileName = tagRendering.builtin.includes(".")
|
||||||
|
? `**/assets/layers/${tagRendering.builtin.split(".")[0]}/${
|
||||||
|
tagRendering.builtin.split(".")[0]
|
||||||
|
}.json`
|
||||||
|
: `**/assets/layers/questions/questions.json`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(toFileName);
|
||||||
|
|
||||||
|
// Read toFile and get the text
|
||||||
|
const toContent = await vscode.workspace.fs.readFile(toFile[0]);
|
||||||
|
const toText = new TextDecoder().decode(toContent);
|
||||||
|
const toJson = JSON.parse(toText);
|
||||||
|
const trIndex = toJson.tagRenderings.findIndex(
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
(tr: any) => tr.id === tagRendering.builtin.split(".")?.pop()
|
||||||
|
);
|
||||||
|
const toRange = getStartEnd(toText, ["tagRenderings", trIndex]);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: tagRendering.builtin,
|
||||||
|
filePath: fromUri,
|
||||||
|
jsonPath: ["tagRenderings"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromUri,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
range: [toRange.start, toRange.end],
|
||||||
|
},
|
||||||
|
type: "tagRendering",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Multiple tagRenderings
|
||||||
|
for (const builtinTagRendering of tagRendering.builtin) {
|
||||||
|
console.log(
|
||||||
|
`Reference found to builtin tagRendering ${builtinTagRendering} in ${from}`
|
||||||
|
);
|
||||||
|
// The range depends on whether we're dealing with a full file or not
|
||||||
|
const fromStartEnd = fullFileText
|
||||||
|
? getStartEnd(fullFileText, [
|
||||||
|
...from.split("."),
|
||||||
|
"tagRenderings",
|
||||||
|
json.tagRenderings.indexOf(tagRendering),
|
||||||
|
])
|
||||||
|
: getStartEnd(text, [
|
||||||
|
"tagRenderings",
|
||||||
|
json.tagRenderings.indexOf(tagRendering),
|
||||||
|
]);
|
||||||
|
const to = builtinTagRendering.includes(".")
|
||||||
|
? `layers.${builtinTagRendering.split(".")[0]}.tagRenderings.${
|
||||||
|
builtinTagRendering.split(".")[1]
|
||||||
|
}`
|
||||||
|
: `layers.questions.tagRenderings.${builtinTagRendering}`;
|
||||||
|
const toFileName = builtinTagRendering.includes(".")
|
||||||
|
? `**/assets/layers/${builtinTagRendering.split(".")[0]}/${
|
||||||
|
builtinTagRendering.split(".")[0]
|
||||||
|
}.json`
|
||||||
|
: `**/assets/layers/questions/questions.json`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(toFileName);
|
||||||
|
|
||||||
|
// Read toFile and get the text
|
||||||
|
const toContent = await vscode.workspace.fs.readFile(toFile[0]);
|
||||||
|
const toText = new TextDecoder().decode(toContent);
|
||||||
|
const toJson = JSON.parse(toText);
|
||||||
|
const trIndex = toJson.tagRenderings.findIndex(
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
(tr: any) => tr.id === builtinTagRendering.split(".")?.pop()
|
||||||
|
);
|
||||||
|
const toRange = getStartEnd(toText, ["tagRenderings", trIndex]);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: builtinTagRendering,
|
||||||
|
filePath: fromUri,
|
||||||
|
jsonPath: ["tagRenderings"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromUri,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
range: [toRange.start, toRange.end],
|
||||||
|
},
|
||||||
|
type: "tagRendering",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (!referencesOnly) {
|
||||||
|
// We've now had all possible references, so now we must have an acutal tagRendering
|
||||||
|
console.log(`TagRendering found in ${from}`);
|
||||||
|
this.cache.push({
|
||||||
|
id: `${json.id}.${tagRendering.id}`,
|
||||||
|
filePath: fromUri,
|
||||||
|
jsonPath: ["tagRenderings"],
|
||||||
|
type: "tagRendering",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Save filters to cache
|
||||||
|
*
|
||||||
|
* @param text Text representation of layer
|
||||||
|
* @param from The theme or layer where the layer is from, e.g. layers.bicycle_rental or themes.cyclofix.layers.0
|
||||||
|
* @param fromUri URI of the layer file
|
||||||
|
* @param referencesOnly Whether to only save references, or also the tagRenderings and filters. This is useful for inline layers, because their filters and tagRenderings can't be reused
|
||||||
|
* @param fullFileText The full text of the original theme file, used for calculating position
|
||||||
|
*/
|
||||||
|
private async saveFiltersToCache(
|
||||||
|
text: string,
|
||||||
|
from: string,
|
||||||
|
fromUri: vscode.Uri,
|
||||||
|
referencesOnly = false,
|
||||||
|
fullFileText?: string
|
||||||
|
) {
|
||||||
|
const json = JSON.parse(text);
|
||||||
|
|
||||||
|
for (const filter of json.filter) {
|
||||||
|
const filterReferenceIndex = json.filter.indexOf(filter);
|
||||||
|
|
||||||
|
// Check if it is a string and not an object
|
||||||
|
if (typeof filter === "string") {
|
||||||
|
// It is a reference
|
||||||
|
console.log(`Reference found to filter ${filter} in ${from}`);
|
||||||
|
|
||||||
|
// The range is dependent on whether we're dealing with a full file or not
|
||||||
|
const path = fullFileText
|
||||||
|
? [
|
||||||
|
from.split(".")[2],
|
||||||
|
parseInt(from.split(".")[3]),
|
||||||
|
"filter",
|
||||||
|
filterReferenceIndex,
|
||||||
|
]
|
||||||
|
: ["filter", filterReferenceIndex];
|
||||||
|
const fromStartEnd = fullFileText
|
||||||
|
? getStartEnd(fullFileText, path)
|
||||||
|
: getStartEnd(text, path);
|
||||||
|
const filterId = filter.includes(".") ? filter.split(".")[1] : filter;
|
||||||
|
const to = filter.includes(".")
|
||||||
|
? `layers.${filter.split(".")[0]}.filter.${filterId}`
|
||||||
|
: `layers.filters.filter.${filterId}`;
|
||||||
|
// Now we'll need to determine what file we need to look in
|
||||||
|
const toFileName = filter.includes(".")
|
||||||
|
? `**/assets/layers/${filter.split(".")[0]}/${
|
||||||
|
filter.split(".")[0]
|
||||||
|
}.json`
|
||||||
|
: `**/assets/layers/filters/filters.json`;
|
||||||
|
const toFile = await vscode.workspace.findFiles(toFileName);
|
||||||
|
const toContent = await vscode.workspace.fs.readFile(toFile[0]);
|
||||||
|
const toText = new TextDecoder().decode(toContent);
|
||||||
|
const toJson = JSON.parse(toText);
|
||||||
|
const toFilterIndex = toJson.filter.findIndex(
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
(f: any) => f.id === filterId
|
||||||
|
);
|
||||||
|
|
||||||
|
if (toFilterIndex === -1) {
|
||||||
|
console.error(`Filter ${filter} not found`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const toRange = getStartEnd(toText, ["filter", toFilterIndex]);
|
||||||
|
|
||||||
|
this.cache.push({
|
||||||
|
id: filter,
|
||||||
|
filePath: fromUri,
|
||||||
|
jsonPath: ["filters"],
|
||||||
|
type: "reference",
|
||||||
|
reference: {
|
||||||
|
from: {
|
||||||
|
id: from,
|
||||||
|
uri: fromUri,
|
||||||
|
range: [fromStartEnd.start, fromStartEnd.end],
|
||||||
|
},
|
||||||
|
to: {
|
||||||
|
id: to,
|
||||||
|
uri: toFile[0],
|
||||||
|
range: [toRange.start, toRange.end],
|
||||||
|
},
|
||||||
|
type: "filter",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else if (typeof filter === "object" && !referencesOnly) {
|
||||||
|
// This is a filter, which can be reused
|
||||||
|
console.log(`Filter found in ${from}`);
|
||||||
|
this.cache.push({
|
||||||
|
id: `${json.id}.${filter.id}`,
|
||||||
|
filePath: fromUri,
|
||||||
|
jsonPath: ["filters"],
|
||||||
|
type: "filter",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Print the current cache state
|
||||||
|
* TODO: This probably needs to be removed at some point
|
||||||
|
*/
|
||||||
|
public printCache() {
|
||||||
|
console.log("Current cache state:", this.cache);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache for interacting with the cache
|
||||||
|
*/
|
||||||
|
export class Cache {
|
||||||
|
private cache: CacheItem[] = [];
|
||||||
|
|
||||||
|
public static async create() {
|
||||||
|
const cache = new Cache();
|
||||||
|
await cache.loadCache();
|
||||||
|
return cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load the cache from the .cache/cache.json file
|
||||||
|
* TODO: Find a more elegant way to do this
|
||||||
|
*/
|
||||||
|
private async loadCache() {
|
||||||
|
// Get the cache from the .cache/cache.json file in the workspace folder
|
||||||
|
const workspaceFolder = vscode.workspace.workspaceFolders?.[0].uri.fsPath;
|
||||||
|
if (workspaceFolder) {
|
||||||
|
const cacheUri = vscode.Uri.file(`${workspaceFolder}/.cache/cache.json`);
|
||||||
|
const cache = await vscode.workspace.fs.readFile(cacheUri);
|
||||||
|
const cacheString = new TextDecoder().decode(cache);
|
||||||
|
this.cache = JSON.parse(cacheString);
|
||||||
|
} else {
|
||||||
|
console.error("No workspace folder found");
|
||||||
|
throw new Error("No workspace folder found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all tag renderings from the cache
|
||||||
|
*
|
||||||
|
* @returns List of CompletionItems for tagRenderings
|
||||||
|
*/
|
||||||
|
public getTagRenderings(): vscode.CompletionItem[] {
|
||||||
|
console.log("Getting tag renderings from cache");
|
||||||
|
const tagRenderings: vscode.CompletionItem[] = [];
|
||||||
|
for (const item of this.cache) {
|
||||||
|
if (item.type === "tagRendering") {
|
||||||
|
if (item.id.startsWith("questions.")) {
|
||||||
|
const completionItem = new vscode.CompletionItem(
|
||||||
|
item.id.replace("questions.", ""),
|
||||||
|
vscode.CompletionItemKind.Value
|
||||||
|
);
|
||||||
|
// To give built-in tagRenderings a higher priority, we sort them to the top
|
||||||
|
completionItem.sortText = `#${item.id}`;
|
||||||
|
tagRenderings.push(completionItem);
|
||||||
|
} else {
|
||||||
|
tagRenderings.push(
|
||||||
|
new vscode.CompletionItem(item.id, vscode.CompletionItemKind.Value)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return tagRenderings;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all filters from the cache
|
||||||
|
*/
|
||||||
|
public getFilters(): vscode.CompletionItem[] {
|
||||||
|
console.log("Getting filters from cache");
|
||||||
|
const filters: vscode.CompletionItem[] = [];
|
||||||
|
for (const item of this.cache) {
|
||||||
|
if (item.type === "filter") {
|
||||||
|
if (item.id.startsWith("filters.")) {
|
||||||
|
const completionItem = new vscode.CompletionItem(
|
||||||
|
item.id.replace("filters.", ""),
|
||||||
|
vscode.CompletionItemKind.Value
|
||||||
|
);
|
||||||
|
// To give built-in filters a higher priority, we sort them to the top
|
||||||
|
completionItem.sortText = `#${item.id}`;
|
||||||
|
filters.push(completionItem);
|
||||||
|
} else {
|
||||||
|
filters.push(
|
||||||
|
new vscode.CompletionItem(item.id, vscode.CompletionItemKind.Value)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return filters;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all references to a specific item
|
||||||
|
*
|
||||||
|
* @param to Item to get references for (e.g. layers.bicycle_rental)
|
||||||
|
* @returns List of references
|
||||||
|
*/
|
||||||
|
public getReferences(to: string): CacheItem[] {
|
||||||
|
return this.cache.filter((item) => {
|
||||||
|
if (item.type === "reference") {
|
||||||
|
return item.reference?.to.id === to;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A cached item
|
||||||
|
* Can be a tagRendering or filter from a(n) (inline) layer
|
||||||
|
* Can also be a reference between files
|
||||||
|
*/
|
||||||
|
interface CacheItem {
|
||||||
|
/**
|
||||||
|
* Where the item is defined in the workspace
|
||||||
|
*/
|
||||||
|
filePath: vscode.Uri;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The JSON path to the item in the file
|
||||||
|
*/
|
||||||
|
jsonPath: JSONPath;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* What kind of item it is
|
||||||
|
*/
|
||||||
|
type: "tagRendering" | "filter" | "reference";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The ID of the item
|
||||||
|
*
|
||||||
|
* When we need to reuse an items in a configuration, we can use the ID
|
||||||
|
* This does mean the ID definitely doesn't have to be unique, because when this is a reference,
|
||||||
|
* the ID is the same as the reference ID and items can be reused by multiple themes or layers
|
||||||
|
*/
|
||||||
|
id: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In case of a reference, this contains the reference
|
||||||
|
*/
|
||||||
|
reference?: Reference;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A reference between two items
|
||||||
|
*/
|
||||||
|
interface Reference {
|
||||||
|
/**
|
||||||
|
* The place where the item is being used (eg. themes.cyclofix)
|
||||||
|
*/
|
||||||
|
from: ReferenceDetail;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The place where the item is defined (eg. layers.bicycle_rental)
|
||||||
|
*/
|
||||||
|
to: ReferenceDetail;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The type of item being referenced/reused
|
||||||
|
*/
|
||||||
|
type: "tagRendering" | "filter" | "layer";
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ReferenceDetail {
|
||||||
|
/**
|
||||||
|
* The path of the file for this side of the reference
|
||||||
|
*
|
||||||
|
* @example layers.bicycle_rental
|
||||||
|
* @example layers.questions.tagRenderings.name
|
||||||
|
* @example themes.cyclofix
|
||||||
|
*/
|
||||||
|
id: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The URI of the file for this side of the reference
|
||||||
|
*/
|
||||||
|
uri?: vscode.Uri;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The range of the reference in the file
|
||||||
|
* Useful for highlighting the reference in the file
|
||||||
|
*
|
||||||
|
* Only defined when referencing to a part of a file
|
||||||
|
*/
|
||||||
|
range?: [ReferencePosition, ReferencePosition];
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ReferencePosition {
|
||||||
|
character: number;
|
||||||
|
line: number;
|
||||||
|
}
|
|
@ -13,7 +13,6 @@ import { JSONPath, findNodeAtLocation, parseTree } from "jsonc-parser";
|
||||||
*/
|
*/
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
export function getValueFromPath(json: string, path: JSONPath): any {
|
export function getValueFromPath(json: string, path: JSONPath): any {
|
||||||
console.log("getValueFromPath", path);
|
|
||||||
const rootNode = parseTree(json);
|
const rootNode = parseTree(json);
|
||||||
if (!rootNode) {
|
if (!rootNode) {
|
||||||
return undefined;
|
return undefined;
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
|
|
||||||
import * as vscode from "vscode";
|
import * as vscode from "vscode";
|
||||||
import * as path from "path";
|
import * as path from "path";
|
||||||
|
import { Cache } from "./cache";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function to get all available layers on disk
|
* Function to get all available layers on disk
|
||||||
|
@ -33,67 +34,95 @@ export async function getAvailableLayers(): Promise<string[]> {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility function to get the tagRenderings from the questions layer
|
* Utility function to get the tagRenderings from the questions layer
|
||||||
* //TODO: This should get ALL tagRenderings, not just from the questions layer
|
*
|
||||||
|
* This function first tries to get all of the tagRenderings from the cache, and if that fails, it falls back to the questions.json file
|
||||||
*
|
*
|
||||||
* @returns List of CompletionItems for tagRenderings
|
* @returns List of CompletionItems for tagRenderings
|
||||||
*/
|
*/
|
||||||
export async function getTagRenderings(): Promise<vscode.CompletionItem[]> {
|
export async function getTagRenderings(): Promise<vscode.CompletionItem[]> {
|
||||||
const tagRenderings: vscode.CompletionItem[] = [];
|
// First, we try to get the tagRenderings from the cache
|
||||||
|
// If the cache is not available, instead return the tagRenderings from the questions layer
|
||||||
|
|
||||||
// Open the questions layer file
|
try {
|
||||||
const questionsFile = await vscode.workspace.findFiles(
|
const cache = await Cache.create();
|
||||||
"assets/layers/questions/questions.json",
|
return cache.getTagRenderings();
|
||||||
"**/node_modules/**"
|
} catch (error) {
|
||||||
);
|
console.error(
|
||||||
|
"Error getting tagRenderings from cache, falling back to questions.json",
|
||||||
if (questionsFile.length === 0) {
|
error
|
||||||
console.error("questions.json not found");
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
const content = await vscode.workspace.fs.readFile(questionsFile[0]);
|
|
||||||
const questions = JSON.parse(new TextDecoder().decode(content));
|
|
||||||
|
|
||||||
for (const tagRendering of questions.tagRenderings) {
|
|
||||||
tagRenderings.push(
|
|
||||||
new vscode.CompletionItem(
|
|
||||||
tagRendering.id,
|
|
||||||
vscode.CompletionItemKind.Value
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
return tagRenderings;
|
const tagRenderings: vscode.CompletionItem[] = [];
|
||||||
|
|
||||||
|
// Open the questions layer file
|
||||||
|
const questionsFile = await vscode.workspace.findFiles(
|
||||||
|
"assets/layers/questions/questions.json",
|
||||||
|
"**/node_modules/**"
|
||||||
|
);
|
||||||
|
|
||||||
|
if (questionsFile.length === 0) {
|
||||||
|
console.error("questions.json not found");
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const content = await vscode.workspace.fs.readFile(questionsFile[0]);
|
||||||
|
const questions = JSON.parse(new TextDecoder().decode(content));
|
||||||
|
|
||||||
|
for (const tagRendering of questions.tagRenderings) {
|
||||||
|
tagRenderings.push(
|
||||||
|
new vscode.CompletionItem(
|
||||||
|
tagRendering.id,
|
||||||
|
vscode.CompletionItemKind.Value
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return tagRenderings;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility function to get the filters from the filters layer
|
* Utility function to get the filters from the filters layer
|
||||||
* //TODO: This should get ALL filters, not just from the filters layer
|
*
|
||||||
|
* This function first tries to get all of the filters from the cache, and if that fails, it falls back to the filters.json file
|
||||||
*
|
*
|
||||||
* @returns List of CompletionItems for tagRenderings
|
* @returns List of CompletionItems for tagRenderings
|
||||||
*/
|
*/
|
||||||
export async function getFilters(): Promise<vscode.CompletionItem[]> {
|
export async function getFilters(): Promise<vscode.CompletionItem[]> {
|
||||||
const filtersList: vscode.CompletionItem[] = [];
|
// First, we try to get the filters from the cache
|
||||||
|
// If the cache is not available, instead return the filters from the filters layer
|
||||||
|
|
||||||
// Open the filters layer file
|
try {
|
||||||
const filtersFile = await vscode.workspace.findFiles(
|
const cache = await Cache.create();
|
||||||
"assets/layers/filters/filters.json",
|
return cache.getFilters();
|
||||||
"**/node_modules/**"
|
} catch (error) {
|
||||||
);
|
console.error(
|
||||||
|
"Error getting filters from cache, falling back to filters.json",
|
||||||
if (filtersFile.length === 0) {
|
error
|
||||||
console.error("filters.json not found");
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
const content = await vscode.workspace.fs.readFile(filtersFile[0]);
|
|
||||||
const filters = JSON.parse(new TextDecoder().decode(content));
|
|
||||||
|
|
||||||
for (const filter of filters.filter) {
|
|
||||||
filtersList.push(
|
|
||||||
new vscode.CompletionItem(filter.id, vscode.CompletionItemKind.Value)
|
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
return filtersList;
|
const filtersList: vscode.CompletionItem[] = [];
|
||||||
|
|
||||||
|
// Open the filters layer file
|
||||||
|
const filtersFile = await vscode.workspace.findFiles(
|
||||||
|
"assets/layers/filters/filters.json",
|
||||||
|
"**/node_modules/**"
|
||||||
|
);
|
||||||
|
|
||||||
|
if (filtersFile.length === 0) {
|
||||||
|
console.error("filters.json not found");
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const content = await vscode.workspace.fs.readFile(filtersFile[0]);
|
||||||
|
const filters = JSON.parse(new TextDecoder().decode(content));
|
||||||
|
|
||||||
|
for (const filter of filters.filter) {
|
||||||
|
filtersList.push(
|
||||||
|
new vscode.CompletionItem(filter.id, vscode.CompletionItemKind.Value)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return filtersList;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue