Skip to content

Commit

Permalink
feat: support multiple languages (#616)
Browse files Browse the repository at this point in the history
  • Loading branch information
conwnet authored Jan 12, 2025
1 parent e5f22e9 commit 6b2463f
Show file tree
Hide file tree
Showing 19 changed files with 140 additions and 51 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/test-wtih-vscode-build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ jobs:
node-version: ${{ matrix.node-version }}

- run: npm install && cd vscode-web && npm install
- run: cd vscode-web && npm run clone && npm run build
- run: npm run link & npm run build
- run: cd vscode-web && npm run build
- run: npm run link && npm run build
- uses: microsoft/playwright-github-action@v1
- run: GITHUB_TOKEN=${{ secrets.GITHUB_TOKEN }} npm run test:ci
8 changes: 4 additions & 4 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@
"license": "ISC",
"devDependencies": {
"@cloudflare/workers-types": "^4.20250109.0",
"@github1s/vscode-web": "^0.23.1",
"@github1s/vscode-web": "0.23.2",
"chokidar": "^4.0.3",
"clean-css": "^5.3.3",
"copy-webpack-plugin": "^12.0.2",
Expand Down
6 changes: 3 additions & 3 deletions scripts/build.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,16 +3,16 @@
import path from 'path';
import fs from 'fs-extra';
import cp from 'child_process';
import { PROJECT_ROOT } from './utils.js';
import { executeCommand, PROJECT_ROOT } from './utils.js';

const main = () => {
for (const extension of fs.readdirSync('extensions')) {
const extensionPath = path.join(PROJECT_ROOT, 'extensions', extension);
if (fs.existsSync(path.join(extensionPath, 'package.json'))) {
cp.spawnSync('npm', ['run', 'compile'], { cwd: extensionPath, stdio: 'inherit' });
executeCommand('npm', ['run', 'compile'], extensionPath);
}
}
cp.spawnSync('npx', ['webpack', '--mode=production'], { cwd: PROJECT_ROOT, stdio: 'inherit' });
executeCommand('npx', ['webpack', '--mode=production'], PROJECT_ROOT);
};

main();
6 changes: 3 additions & 3 deletions scripts/link.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,12 @@

import path from 'path';
import cp from 'child_process';
import { PROJECT_ROOT } from './utils.js';
import { executeCommand, PROJECT_ROOT } from './utils.js';

const main = () => {
const distPath = path.join(PROJECT_ROOT, 'vscode-web/dist');
cp.spawnSync('npm', ['link'], { cwd: distPath, stdio: 'inherit' });
cp.spawnSync('npm', ['link', '@github1s/vscode-web'], { cwd: PROJECT_ROOT, stdio: 'inherit' });
executeCommand('npm', ['link'], distPath);
executeCommand('npm', ['link', '@github1s/vscode-web'], PROJECT_ROOT);
};

main();
4 changes: 2 additions & 2 deletions scripts/postinstall.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,13 @@
import path from 'path';
import fs from 'fs-extra';
import cp from 'child_process';
import { PROJECT_ROOT } from './utils.js';
import { executeCommand, PROJECT_ROOT } from './utils.js';

const main = () => {
const extensions = fs.readdirSync(path.join(PROJECT_ROOT, 'extensions'));
for (const extension of extensions) {
const extensionPath = path.join(PROJECT_ROOT, 'extensions', extension);
cp.spawnSync('npm', ['install', '--no-save'], { cwd: extensionPath, stdio: 'inherit' });
executeCommand('npm', ['install', '--no-save'], extensionPath);
}
};

Expand Down
8 changes: 8 additions & 0 deletions scripts/utils.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,11 @@
import path from 'path';
import cp from 'child_process';

export const PROJECT_ROOT = path.join(import.meta.dirname, '..');

export const executeCommand = (command, args, cwd) => {
const result = cp.spawnSync(command, args, { stdio: 'inherit', cwd });
if (result.error) {
throw result.error;
}
};
1 change: 1 addition & 0 deletions src/global.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ declare const DEV_VSCODE: boolean;
declare const GITHUB_ORIGIN: string;
declare const GITLAB_ORIGIN: string;
declare const GITHUB1S_EXTENSIONS: string;
declare const AVAILABLE_LANGUAGES: string[];

/* eslint-disable no-var */
declare var dynamicImport: (url: string) => Promise<any>;
Expand Down
11 changes: 7 additions & 4 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -58,13 +58,16 @@ globalThis._VSCODE_WEB = {

if (!DEV_VSCODE) {
const linkElement = document.createElement('link');
linkElement.rel = 'stylesheet';
linkElement.href = resolveVscodeUrl('vs/workbench/workbench.web.main.css');
linkElement.setAttribute('rel', 'stylesheet');
linkElement.setAttribute('href', resolveVscodeUrl('vs/workbench/workbench.web.main.css'));
document.head.appendChild(linkElement);

const languageId = document.cookie.match(/(^| )vscode.nls.locale=([^;]+)/)?.[2] || '';
const nlsUrl = AVAILABLE_LANGUAGES.includes(languageId)
? resolveVscodeUrl(`../nls/${languageId}/nls.messages.js`)
: resolveVscodeUrl('nls.messages.js');
const scriptElement = document.createElement('script');
scriptElement.type = 'text/javascript';
scriptElement.src = resolveVscodeUrl('nls.messages.js');
scriptElement.setAttribute('src', nlsUrl);
document.body.appendChild(scriptElement);
}

Expand Down
30 changes: 15 additions & 15 deletions vscode-web/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -3,23 +3,23 @@
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no" />
<title>VSCode Web</title>
<link rel="icon" href="/favicon.ico" type="image/x-icon" />
<link rel="stylesheet" href="/vscode/vs/workbench/workbench.web.main.css" />
<title>VSCode Web</title>
<link rel="icon" href="/favicon.ico" type="image/x-icon" />
<link rel="stylesheet" href="/vscode/vs/workbench/workbench.web.main.css" />
</head>
<body>
<script src="/vscode/nls.messages.js"></script>
<script>
globalThis._VSCODE_FILE_ROOT = new URL('/vscode/', window.location.origin).toString();
</script>
<script type="module">
import { create, URI } from '/vscode/vs/workbench/workbench.web.main.internal.js';
create(document.body, {
webviewEndpoint: '/vscode/vs/workbench/contrib/webview/browser/pre',
workspaceProvider: {
workspace: { workspaceUri: URI.from({ scheme: 'tmp', path: '/default.code-workspace' }) },
},
});
</script>
<script>
globalThis._VSCODE_FILE_ROOT = new URL('/vscode/', window.location.origin).toString();
</script>
<script type="module">
import { create, URI } from '/vscode/vs/workbench/workbench.web.main.internal.js';
create(document.body, {
webviewEndpoint: '/vscode/vs/workbench/contrib/webview/browser/pre',
workspaceProvider: {
workspace: { workspaceUri: URI.from({ scheme: 'tmp', path: '/default.code-workspace' }) },
},
});
</script>
</body>
</html>
4 changes: 2 additions & 2 deletions vscode-web/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 2 additions & 1 deletion vscode-web/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@github1s/vscode-web",
"version": "0.23.1",
"version": "0.23.2",
"description": "VS Code web for GitHub1s",
"author": "github1s",
"license": "MIT",
Expand All @@ -12,6 +12,7 @@
"build": "run-s clone patch build:*",
"build:vscode": "node scripts/build/vscode.js",
"build:package": "node scripts/build/package.js",
"build:nls": "node scripts/build/nls.js",
"watch": "run-s clone patch && run-p watch:*",
"watch:source": "node scripts/watch/source.js",
"watch:vscode": "node scripts/watch/vscode.js",
Expand Down
66 changes: 66 additions & 0 deletions vscode-web/scripts/build/nls.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
#!/usr/bin/env node

import path from 'path';
import fs from 'fs-extra';
import { PROJECT_ROOT } from '../utils.js';

const deepMerge = (target, source) => {
for (const key of Object.keys(source)) {
if (typeof source[key] === 'object' && target[key]) {
deepMerge(target[key], source[key]);
} else {
target[key] = source[key];
}
}
return target;
};

const main = () => {
const oputBuildPath = path.join(PROJECT_ROOT, 'lib/vscode/out-build');
const nlsKeys = JSON.parse(fs.readFileSync(path.join(oputBuildPath, 'nls.keys.json')));
const nlsMessages = JSON.parse(fs.readFileSync(path.join(oputBuildPath, 'nls.messages.json')));

const languageContentsMap = {};
const i18nPath = path.join(PROJECT_ROOT, 'lib/vscode-loc/i18n');

for (const languageDir of fs.readdirSync(i18nPath)) {
const languagePath = path.join(i18nPath, languageDir);
const packageJson = JSON.parse(fs.readFileSync(path.join(languagePath, 'package.json')));

for (const localization of packageJson.contributes.localizations) {
if (!languageContentsMap[localization.languageId]) {
languageContentsMap[localization.languageId] = {};
}

const contents = languageContentsMap[localization.languageId];
for (const translation of localization.translations) {
const translationPath = path.join(languagePath, translation.path);
deepMerge(contents, JSON.parse(fs.readFileSync(translationPath)).contents);
}
}
}

for (const languageId of Object.keys(languageContentsMap)) {
const langMessages = [];
const contents = languageContentsMap[languageId];

for (const [file, keys] of nlsKeys) {
for (const key of keys) {
langMessages.push(contents[file][key] || nlsMessages[langMessages.length]);
}
}
if (langMessages.length !== nlsMessages.length) {
throw new Error(`Invalid nls messages for ${languageId}`);
}

const nslDirPath = path.join(PROJECT_ROOT, `dist/nls/${languageId}`);
fs.ensureDirSync(nslDirPath);
fs.writeFileSync(
path.join(nslDirPath, 'nls.messages.js'),
`globalThis._VSCODE_NLS_MESSAGES=${JSON.stringify(langMessages)};` +
`globalThis._VSCODE_NLS_LANGUAGE=${JSON.stringify(languageId)};`,
);
}
};

main();
5 changes: 2 additions & 3 deletions vscode-web/scripts/build/vscode.js
Original file line number Diff line number Diff line change
@@ -1,12 +1,11 @@
#!/usr/bin/env node

import path from 'path';
import cp from 'child_process';
import { PROJECT_ROOT } from '../utils.js';
import { executeCommand, PROJECT_ROOT } from '../utils.js';

const main = () => {
const cwd = path.join(PROJECT_ROOT, 'lib/vscode');
cp.spawnSync('npm', ['run', 'gulp', 'vscode-web-min'], { stdio: 'inherit', cwd });
executeCommand('npm', ['run', 'gulp', 'vscode-web-min'], cwd);
};

main();
12 changes: 7 additions & 5 deletions vscode-web/scripts/clone.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import path from 'path';
import cp from 'child_process';
import fs from 'fs-extra';
import crypto from 'crypto';
import { PROJECT_ROOT, getAllFiles } from './utils.js';
import { executeCommand, PROJECT_ROOT, getAllFiles } from './utils.js';

const fixSourceFiles = () => {
const getDiffFiles = (base) => {
Expand Down Expand Up @@ -52,13 +52,15 @@ const main = () => {

const url = 'https://github.com/microsoft/vscode.git';
const ref = cp.execSync(`cat ${PROJECT_ROOT}/.VERSION`).toString();
const commands = ['git', ['clone', '--depth', '1', '-b', ref, url, 'lib/vscode']];
cp.spawnSync(...commands, { stdio: 'inherit', cwd: PROJECT_ROOT });
executeCommand('git', ['clone', '--depth', '1', '-b', ref, url, 'lib/vscode'], PROJECT_ROOT);

const vscodePath = path.join(PROJECT_ROOT, 'lib/vscode');
cp.spawnSync('npm', ['install'], { stdio: 'inherit', cwd: vscodePath });
const locUrl = 'https://github.com/microsoft/vscode-loc.git';
executeCommand('git', ['clone', '--depth', '1', locUrl, 'lib/vscode-loc'], PROJECT_ROOT);

fixSourceFiles();

const vscodePath = path.join(PROJECT_ROOT, 'lib/vscode');
executeCommand('npm', ['install'], vscodePath);
};

main();
8 changes: 8 additions & 0 deletions vscode-web/scripts/utils.js
Original file line number Diff line number Diff line change
@@ -1,8 +1,16 @@
import path from 'path';
import fs from 'fs-extra';
import cp from 'child_process';

export const PROJECT_ROOT = path.join(import.meta.dirname, '..');

export const executeCommand = (command, args, cwd) => {
const result = cp.spawnSync(command, args, { stdio: 'inherit', cwd });
if (result.error) {
throw result.error;
}
};

export const getAllFiles = (directory) => {
return fs.readdirSync(directory).flatMap((name) => {
const filePath = path.join(directory, name);
Expand Down
5 changes: 2 additions & 3 deletions vscode-web/scripts/watch/extensions.js
Original file line number Diff line number Diff line change
@@ -1,12 +1,11 @@
#!/usr/bin/env node

import path from 'path';
import cp from 'child_process';
import { PROJECT_ROOT } from '../utils.js';
import { executeCommand, PROJECT_ROOT } from '../utils.js';

const main = () => {
const cwd = path.join(PROJECT_ROOT, 'lib/vscode');
cp.spawnSync('npm', ['run', 'watch-web'], { stdio: 'inherit', cwd });
executeCommand('npm', ['run', 'watch-web'], cwd);
};

main();
5 changes: 2 additions & 3 deletions vscode-web/scripts/watch/vscode.js
Original file line number Diff line number Diff line change
@@ -1,12 +1,11 @@
#!/usr/bin/env node

import path from 'path';
import cp from 'child_process';
import { PROJECT_ROOT } from '../utils.js';
import { executeCommand, PROJECT_ROOT } from '../utils.js';

const main = () => {
const cwd = path.join(PROJECT_ROOT, 'lib/vscode');
cp.spawnSync('npm', ['run', 'watch'], { stdio: 'inherit', cwd });
executeCommand('npm', ['run', 'watch'], cwd);
};

main();
3 changes: 3 additions & 0 deletions webpack.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ const copyPluginPatterns = [
{ from: path.join(vscodeWebPath, 'vscode'), to: `${staticDir}/vscode`, ...skipMinified },
{ from: path.join(vscodeWebPath, 'extensions'), to: `${staticDir}/extensions`, ...skipMinified },
{ from: path.join(vscodeWebPath, 'dependencies'), to: `${staticDir}/dependencies`, ...skipMinified },
{ from: path.join(vscodeWebPath, 'nls'), to: `${staticDir}/nls`, ...skipMinified },
];

const devVscodeStatic = [
Expand All @@ -47,6 +48,7 @@ export default (env, argv) => {
const devVscode = !!process.env.DEV_VSCODE;
const minifyCSS = (code) => (devMode ? code : new CleanCSS().minify(code).styles);
const minifyJS = (code) => (devMode ? code : UglifyJS.minify(code).code);
const availableLanguages = devVscode ? [] : fs.readdirSync(path.join(vscodeWebPath, 'nls'));

return {
mode: env.mode || 'production',
Expand Down Expand Up @@ -84,6 +86,7 @@ export default (env, argv) => {
GITHUB_ORIGIN: JSON.stringify(process.env.GITHUB_DOMAIN || 'https://github.com'),
GITLAB_ORIGIN: JSON.stringify(process.env.GITLAB_DOMAIN || 'https://gitlab.com'),
GITHUB1S_EXTENSIONS: JSON.stringify(packUtils.getBuiltinExtensions(devVscode)),
AVAILABLE_LANGUAGES: JSON.stringify(availableLanguages),
}),
],
performance: false,
Expand Down

0 comments on commit 6b2463f

Please sign in to comment.