Skip to content

Commit 3c8687d

Browse files
authored
feat: support web (#75)
1 parent 5c5b197 commit 3c8687d

19 files changed

+707
-1258
lines changed

.editorconfig

+3
Original file line numberDiff line numberDiff line change
@@ -9,3 +9,6 @@ indent_size = 2
99
end_of_line = lf
1010
insert_final_newline = false
1111
trim_trailing_whitespace = true
12+
13+
[*.ts]
14+
quote_type = single

.vscode/launch.json

+15-2
Original file line numberDiff line numberDiff line change
@@ -10,12 +10,25 @@
1010
"type": "extensionHost",
1111
"request": "launch",
1212
"args": [
13-
"--extensionDevelopmentPath=${workspaceFolder}"
13+
"--extensionDevelopmentPath=${workspaceFolder}/out"
1414
],
1515
"outFiles": [
16-
"${workspaceFolder}/dist/**/*.js"
16+
"${workspaceFolder}/out/**/*.js"
1717
],
1818
"preLaunchTask": "${defaultBuildTask}"
19+
},
20+
{
21+
"name": "Run Web Extension",
22+
"type": "extensionHost",
23+
"debugWebWorkerHost": true,
24+
"request": "launch",
25+
"args": [
26+
"--extensionDevelopmentPath=${workspaceFolder}/out",
27+
"--extensionDevelopmentKind=web"
28+
],
29+
"outFiles": [
30+
"${workspaceFolder}/out/**/*.js"
31+
]
1932
}
2033
]
2134
}

.vscodeignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -15,4 +15,5 @@ vsc-extension-quickstart.md
1515
examples/**
1616
demo.gif
1717
.cssrem
18-
.yarnrc.yml
18+
.yarnrc.yml
19+
scripts/**

esbuild.js

-55
This file was deleted.

eslint.config.mjs

+11-11
Original file line numberDiff line numberDiff line change
@@ -1,34 +1,34 @@
1-
import typescriptEslint from '@typescript-eslint/eslint-plugin';
2-
import tsParser from '@typescript-eslint/parser';
1+
import typescriptEslint from "@typescript-eslint/eslint-plugin";
2+
import tsParser from "@typescript-eslint/parser";
33

44
export default [
55
{
6-
files: ['**/*.ts'],
6+
files: ["**/*.ts"],
77
},
88
{
99
plugins: {
10-
'@typescript-eslint': typescriptEslint,
10+
"@typescript-eslint": typescriptEslint,
1111
},
1212

1313
languageOptions: {
1414
parser: tsParser,
1515
ecmaVersion: 2022,
16-
sourceType: 'module',
16+
sourceType: "module",
1717
},
1818

1919
rules: {
20-
'@typescript-eslint/naming-convention': [
21-
'warn',
20+
"@typescript-eslint/naming-convention": [
21+
"warn",
2222
{
23-
selector: 'import',
24-
format: ['camelCase', 'PascalCase'],
23+
selector: "import",
24+
format: ["camelCase", "PascalCase"],
2525
},
2626
],
2727

2828
// curly: "warn",
2929
// eqeqeq: "warn",
30-
'no-throw-literal': 'warn',
31-
semi: 'warn',
30+
"no-throw-literal": "warn",
31+
semi: "warn",
3232
},
3333
},
3434
];

package.json

+6-7
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,10 @@
88
"license": "MIT",
99
"licenseUrl": "LICENSE",
1010
"scripts": {
11-
"vscode:prepublish": "yarn run package",
12-
"compile": "yarn run check-types && yarn run lint && node esbuild.js",
13-
"watch": "npm-run-all -p watch:*",
14-
"watch:esbuild": "node esbuild.js --watch",
15-
"watch:tsc": "tsc --noEmit --watch --project tsconfig.json",
16-
"package": "yarn run check-types && yarn run lint && node esbuild.js --production",
11+
"vscode:prepublish": "npm run check-types && tsx ./scripts/esbuild.js --production",
12+
"compile": "yarn run check-types && yarn run lint && tsx ./scripts/esbuild.js",
13+
"watch": "tsx ./scripts/esbuild.js --watch",
14+
"package": "yarn run lint && yarn run check-types && tsx ./scripts/esbuild.js --production",
1715
"compile-tests": "tsc -p . --outDir out",
1816
"watch-tests": "tsc -p . -w --outDir out",
1917
"pretest": "yarn run compile-tests && yarn run compile && yarn run lint",
@@ -25,6 +23,7 @@
2523
},
2624
"l10n": "./l10n",
2725
"main": "./out/extension",
26+
"browser": "./out/extension.web",
2827
"engines": {
2928
"vscode": "^1.90.0"
3029
},
@@ -272,7 +271,7 @@
272271
"conventional-changelog-cli": "^5.0.0",
273272
"esbuild": "^0.24.0",
274273
"eslint": "^9.13.0",
275-
"npm-run-all": "^4.1.5",
274+
"tsx": "^4.19.2",
276275
"typescript": "^5.6.3"
277276
},
278277
"dependencies": {

scripts/esbuild.ts

+78
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
import * as esbuild from "esbuild";
2+
3+
const production = process.argv.includes("--production");
4+
const watch = process.argv.includes("--watch");
5+
const buildType = watch ? "watch" : "build";
6+
7+
function esbuildProblemMatcherPlugin(type: "web" | "node"): esbuild.Plugin {
8+
const prefix = `[${buildType}/${type}]`;
9+
return {
10+
name: "esbuild-problem-matcher",
11+
setup(build) {
12+
build.onStart(() => {
13+
console.log(prefix + " started");
14+
});
15+
build.onEnd((result) => {
16+
result.errors.forEach(({ text, location }) => {
17+
console.error(`✘ [ERROR] ${text}`);
18+
if (location) {
19+
console.error(
20+
` ${location.file}:${location.line}:${location.column}:`
21+
);
22+
}
23+
});
24+
console.log(prefix + " finished");
25+
});
26+
},
27+
};
28+
}
29+
30+
const main = async () => {
31+
const nodeContext = await esbuild.context({
32+
entryPoints: ["src/extension.ts"],
33+
bundle: true,
34+
format: "cjs",
35+
minify: production,
36+
sourcemap: !production,
37+
sourcesContent: false,
38+
platform: "node",
39+
outfile: "out/extension.js",
40+
mainFields: ["module", "main"],
41+
external: ["vscode"],
42+
logLevel: "silent",
43+
plugins: [esbuildProblemMatcherPlugin("node")],
44+
});
45+
46+
const browserContext = await esbuild.context({
47+
entryPoints: ["src/extension.ts"],
48+
bundle: true,
49+
format: "cjs",
50+
minify: production,
51+
sourcemap: !production,
52+
sourcesContent: false,
53+
platform: "browser",
54+
outfile: "out/extension.web.js",
55+
mainFields: ["browser", "module", "main"],
56+
external: ["vscode"],
57+
logLevel: "silent",
58+
plugins: [esbuildProblemMatcherPlugin("web")],
59+
// Node.js global to browser globalThis
60+
define: {
61+
global: "globalThis",
62+
},
63+
});
64+
65+
if (watch) {
66+
await Promise.all([nodeContext.watch(), browserContext.watch()]);
67+
} else {
68+
await nodeContext.rebuild();
69+
await browserContext.rebuild();
70+
await nodeContext.dispose();
71+
await browserContext.dispose();
72+
}
73+
};
74+
75+
main().catch((e) => {
76+
console.error(e);
77+
process.exit(1);
78+
});

src/completion.ts

+18-5
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,30 @@
1-
import { CompletionItem, CompletionItemKind, CompletionItemProvider, MarkdownString, Position, Range, TextDocument } from 'vscode';
1+
import {
2+
CompletionItem,
3+
CompletionItemKind,
4+
CompletionItemProvider,
5+
MarkdownString,
6+
Position,
7+
Range,
8+
TextDocument,
9+
} from 'vscode';
210
import { cog, isIngore } from './config';
311
import { CssRemProcess } from './process';
412
import { isDisabledNextLine } from './ignore-comment';
513

614
export default class implements CompletionItemProvider {
715
constructor(private _: string, private process: CssRemProcess) {}
816

9-
provideCompletionItems(document: TextDocument, position: Position): Thenable<CompletionItem[]> {
17+
provideCompletionItems(
18+
document: TextDocument,
19+
position: Position
20+
): Thenable<CompletionItem[]> {
1021
if (isIngore(document.uri)) return Promise.resolve([]);
1122

12-
return new Promise(resolve => {
23+
return new Promise((resolve) => {
1324
if (isDisabledNextLine(document, position.line)) return null;
14-
const lineText = document.getText(new Range(position.with(undefined, 0), position));
25+
const lineText = document.getText(
26+
new Range(position.with(undefined, 0), position)
27+
);
1528
const res = this.process.convert(lineText);
1629
if (res == null || res.length === 0) {
1730
return resolve([]);
@@ -26,7 +39,7 @@ export default class implements CompletionItemProvider {
2639
item.preselect = idx === 0;
2740
item.insertText = i.value + (cog.addMark ? ` /* ${i.label} */` : ``);
2841
return item;
29-
}),
42+
})
3043
);
3144
});
3245
}

src/config.ts

+21-14
Original file line numberDiff line numberDiff line change
@@ -1,33 +1,40 @@
1-
import { existsSync, readFileSync } from 'fs';
21
import * as JSONC from 'jsonc-parser';
3-
import { join } from 'path';
42
import { Uri, workspace } from 'vscode';
53
import { Config } from './interface';
64
import { resetRules } from './rules';
75
import { minimatch } from 'minimatch';
6+
import { findUri, readFile } from './fs';
87

98
export let cog!: Config;
109
export const cssremConfigFileName = '.cssrem';
1110

12-
function loadConfigViaFile(): void {
13-
if (workspace.workspaceFolders == null || workspace.workspaceFolders?.length <= 0) {
11+
async function loadConfigViaFile(): Promise<void> {
12+
if (
13+
workspace.workspaceFolders == null ||
14+
workspace.workspaceFolders?.length <= 0
15+
) {
1416
return;
1517
}
1618

17-
const cssremConfigPath = join(workspace.workspaceFolders[0].uri.fsPath, cssremConfigFileName);
18-
if (!existsSync(cssremConfigPath)) {
19-
console.log(`Not found file: ${cssremConfigPath}`);
19+
const cssremConfigUri = await findUri(cssremConfigFileName);
20+
if (cssremConfigUri == null) {
21+
console.log(`Not found file: ${cssremConfigUri}`);
22+
return;
23+
}
24+
const cogText = await readFile(cssremConfigUri);
25+
if (cogText == null) {
26+
console.log(`Can't read file: ${cssremConfigUri}`);
2027
return;
2128
}
2229
try {
23-
const res = JSONC.parse(readFileSync(cssremConfigPath).toString('utf-8'));
30+
const res = JSONC.parse(cogText);
2431
cog = {
2532
...cog,
2633
...res,
2734
};
28-
console.warn(`Use override config via ${cssremConfigPath} file`);
35+
console.warn(`Use override config via ${cssremConfigUri} file`);
2936
} catch (ex) {
30-
console.warn(`Parse error in ${cssremConfigPath}`, ex);
37+
console.warn(`Parse error in ${cssremConfigUri}`, ex);
3138
}
3239
}
3340

@@ -57,18 +64,18 @@ function fixLanguages(): void {
5764
];
5865
}
5966

60-
export function loadConfig(): void {
67+
export async function loadConfig(): Promise<void> {
6168
cog = { ...(workspace.getConfiguration('cssrem') as any) };
62-
Object.keys(cog).forEach(key => {
69+
Object.keys(cog).forEach((key) => {
6370
if (typeof (cog as any)[key] === 'function') delete (cog as any)[key];
6471
});
65-
loadConfigViaFile();
72+
await loadConfigViaFile();
6673
fixIngores();
6774
fixLanguages();
6875
resetRules();
6976
console.log('Current config', cog);
7077
}
7178

7279
export function isIngore(uri: Uri) {
73-
return cog.ignores.some(p => minimatch(uri.path, p));
80+
return cog.ignores.some((p) => minimatch(uri.path, p));
7481
}

0 commit comments

Comments
 (0)