ui: speed up page load

pull/533/head
undefined 2 years ago
parent cb741b346c
commit 67279c3745
No known key found for this signature in database

@ -36,8 +36,8 @@
"@types/node": "^18.14.1", "@types/node": "^18.14.1",
"@types/semver": "^7.3.13", "@types/semver": "^7.3.13",
"@types/supertest": "^2.0.12", "@types/supertest": "^2.0.12",
"@typescript-eslint/eslint-plugin": "^5.54.0", "@typescript-eslint/eslint-plugin": "^5.54.1",
"@typescript-eslint/parser": "^5.54.0", "@typescript-eslint/parser": "^5.54.1",
"autocannon": "^7.10.0", "autocannon": "^7.10.0",
"cac": "^6.7.14", "cac": "^6.7.14",
"chokidar": "^3.5.3", "chokidar": "^3.5.3",
@ -48,17 +48,17 @@
"css-loader": "^6.7.3", "css-loader": "^6.7.3",
"esbuild": "0.17.10", "esbuild": "0.17.10",
"esbuild-loader": "^2.21.0", "esbuild-loader": "^2.21.0",
"eslint": "^8.35.0", "eslint": "^8.36.0",
"eslint-import-resolver-typescript": "^3.5.3", "eslint-import-resolver-typescript": "^3.5.3",
"eslint-import-resolver-webpack": "^0.13.2", "eslint-import-resolver-webpack": "^0.13.2",
"fs-extra": "^11.1.0", "fs-extra": "^11.1.0",
"globby": "13.1.3", "globby": "13.1.3",
"inspectpack": "^4.7.1", "inspectpack": "^4.7.1",
"latest-version": "7.0.0", "latest-version": "7.0.0",
"mini-css-extract-plugin": "^2.7.2", "mini-css-extract-plugin": "^2.7.3",
"monaco-editor-webpack-plugin": "^7.0.1", "monaco-editor-webpack-plugin": "^7.0.1",
"mongodb": "^5.1.0", "mongodb": "^5.1.0",
"mongodb-memory-server": "^8.11.5", "mongodb-memory-server": "^8.12.0",
"nyc": "^15.1.0", "nyc": "^15.1.0",
"ora": "^6.1.2", "ora": "^6.1.2",
"postcss-loader": "^7.0.2", "postcss-loader": "^7.0.2",
@ -69,7 +69,7 @@
"supertest": "^6.3.3", "supertest": "^6.3.3",
"ts-loader": "^9.4.2", "ts-loader": "^9.4.2",
"typescript": "4.9.5", "typescript": "4.9.5",
"webpack": "^5.75.0", "webpack": "^5.76.1",
"webpack-bundle-analyzer": "^4.8.0", "webpack-bundle-analyzer": "^4.8.0",
"webpack-dev-server": "^4.11.1", "webpack-dev-server": "^4.11.1",
"webpack-manifest-plugin": "^5.0.0", "webpack-manifest-plugin": "^5.0.0",

@ -10,10 +10,10 @@
"cac": "^6.7.14", "cac": "^6.7.14",
"mongodb": "^5.1.0", "mongodb": "^5.1.0",
"p-queue": "^7.3.4", "p-queue": "^7.3.4",
"schemastery": "^3.7.1", "schemastery": "^3.7.2",
"shell-quote": "^1.8.0", "shell-quote": "^1.8.0",
"superagent": "^8.0.9", "superagent": "^8.0.9",
"ws": "^8.12.1" "ws": "^8.13.0"
}, },
"preferUnplugged": true, "preferUnplugged": true,
"license": "AGPL-3.0-or-later", "license": "AGPL-3.0-or-later",

@ -12,17 +12,17 @@
}, },
"preferUnplugged": true, "preferUnplugged": true,
"dependencies": { "dependencies": {
"@aws-sdk/client-s3": "^3.282.0", "@aws-sdk/client-s3": "^3.289.0",
"@aws-sdk/lib-storage": "^3.282.0", "@aws-sdk/lib-storage": "^3.289.0",
"@aws-sdk/middleware-endpoint": "^3.282.0", "@aws-sdk/middleware-endpoint": "^3.289.0",
"@aws-sdk/s3-presigned-post": "^3.282.0", "@aws-sdk/s3-presigned-post": "^3.289.0",
"@aws-sdk/s3-request-presigner": "^3.282.0", "@aws-sdk/s3-request-presigner": "^3.289.0",
"@graphql-tools/schema": "^9.0.16", "@graphql-tools/schema": "^9.0.17",
"@hydrooj/utils": "workspace:*", "@hydrooj/utils": "workspace:*",
"@simplewebauthn/server": "^7.0.1", "@simplewebauthn/server": "^7.0.1",
"adm-zip": "0.5.5", "adm-zip": "0.5.5",
"cac": "^6.7.14", "cac": "^6.7.14",
"cordis": "^2.7.3", "cordis": "^2.7.4",
"detect-browser": "^5.3.0", "detect-browser": "^5.3.0",
"emoji-regex": "^10.2.1", "emoji-regex": "^10.2.1",
"emojis-list": "2.1.0", "emojis-list": "2.1.0",
@ -48,13 +48,13 @@
"path-to-regexp": "^6.2.1", "path-to-regexp": "^6.2.1",
"require-resolve-hook": "^1.1.0", "require-resolve-hook": "^1.1.0",
"saslprep": "^1.0.3", "saslprep": "^1.0.3",
"schemastery": "^3.7.1", "schemastery": "^3.7.2",
"semver": "^7.3.8", "semver": "^7.3.8",
"serialize-javascript": "^6.0.1", "serialize-javascript": "^6.0.1",
"superagent": "^8.0.9", "superagent": "^8.0.9",
"tar": "^6.1.13", "tar": "^6.1.13",
"thirty-two": "^1.0.2", "thirty-two": "^1.0.2",
"ws": "^8.12.1" "ws": "^8.13.0"
}, },
"devDependencies": { "devDependencies": {
"@types/adm-zip": "^0.4.34", "@types/adm-zip": "^0.4.34",

@ -1,6 +1,4 @@
import { $ } from '@hydrooj/ui-default'; import { $, addPage, AutoloadPage } from '@hydrooj/ui-default';
const { AutoloadPage } = window.Hydro;
let loaded = false; let loaded = false;
async function load() { async function load() {
@ -64,18 +62,12 @@ const loader = (mode) => async (element) => {
}); });
}; };
const getEles = (types) => { const getEles = (types: string[]) => types.flatMap((type) => $(`div[data-${type}]`).get());
const eles = [];
for (const type of types) eles.push(...$(`div[data-${type}]`).get());
return eles;
};
const page = new AutoloadPage('onlyoffice', async () => { addPage(new AutoloadPage('onlyoffice', async () => {
const all = getEles(['doc', 'docx', 'cell', 'xls', 'xlsx', 'slide', 'ppt', 'pptx']); const all = getEles(['doc', 'docx', 'cell', 'xls', 'xlsx', 'slide', 'ppt', 'pptx']);
if (all.length) await load(); if (all.length) await load();
getEles(['doc', 'docx']).forEach(loader('word')); getEles(['doc', 'docx']).forEach(loader('word'));
getEles(['cell', 'xls', 'xlsx']).forEach(loader('cell')); getEles(['cell', 'xls', 'xlsx']).forEach(loader('cell'));
getEles(['slide', 'ppt', 'pptx']).forEach(loader('slide')); getEles(['slide', 'ppt', 'pptx']).forEach(loader('slide'));
}); }));
window.Hydro.extraPages.push(page);

@ -3,6 +3,6 @@
"version": "0.1.5", "version": "0.1.5",
"main": "index.ts", "main": "index.ts",
"dependencies": { "dependencies": {
"prom-client": "^14.1.1" "prom-client": "^14.2.0"
} }
} }

@ -0,0 +1,16 @@
import { addPage, NamedPage } from '@hydrooj/ui-default';
/* global grecaptcha */
addPage(new NamedPage('user_register', () => {
function captcha(event) {
event.preventDefault();
grecaptcha.ready(() => {
grecaptcha.execute(UiContext.recaptchaKey, { action: 'submit' }).then((token) => {
document.getElementById('_captcha').value = token;
document.getElementById('_submit').click();
});
});
}
const element = document.getElementById('submit');
if (element) element.onclick = captcha;
}));

@ -1,19 +0,0 @@
(() => {
const { NamedPage } = window.Hydro;
const page = new NamedPage('user_register', () => {
function captcha(event) {
event.preventDefault();
grecaptcha.ready(function () {
grecaptcha.execute(UiContext.recaptchaKey, { action: 'submit' }).then(function (token) {
document.getElementById('_captcha').value = token;
document.getElementById('_submit').click();
});
});
}
const element = document.getElementById('submit');
if (element) element.onclick = captcha;
});
window.Hydro.extraPages.push(page);
})();

@ -14,6 +14,7 @@ export { default as _ } from 'lodash';
export { default as React } from 'react'; export { default as React } from 'react';
export { default as ReactDOM } from 'react-dom/client'; export { default as ReactDOM } from 'react-dom/client';
export * from './misc/Page'; export * from './misc/Page';
export { initPageLoader } from './hydro';
const lazyModules = {}; const lazyModules = {};
export default async function load(name: string) { export default async function load(name: string) {

@ -1,11 +1,7 @@
/* eslint-disable camelcase */
import en_GB from 'monaco-editor-nls/locale/en-gb.json';
function format(message, args) { function format(message, args) {
let result; let result;
if (args.length === 0) { if (!args.length) result = message;
result = message; else {
} else {
result = String(message).replace(/\{(\d+)\}/g, (match, rest) => { result = String(message).replace(/\{(\d+)\}/g, (match, rest) => {
const index = rest[0]; const index = rest[0];
return typeof args[index] !== 'undefined' ? args[index] : match; return typeof args[index] !== 'undefined' ? args[index] : match;
@ -18,28 +14,12 @@ export const getConfiguredDefaultLocale = () => 'zh';
let CURRENT_LOCALE_DATA = {}; // eslint-disable-line @typescript-eslint/naming-convention let CURRENT_LOCALE_DATA = {}; // eslint-disable-line @typescript-eslint/naming-convention
function find(path, message) {
for (const key of Object.keys(CURRENT_LOCALE_DATA)) {
if (!CURRENT_LOCALE_DATA[key] || !en_GB[key]) continue;
if (CURRENT_LOCALE_DATA[key][path] && en_GB[key][path] === message) {
return CURRENT_LOCALE_DATA[key][path];
}
}
for (const key of Object.keys(CURRENT_LOCALE_DATA)) {
if (!CURRENT_LOCALE_DATA[key]) continue;
if (CURRENT_LOCALE_DATA[key][path]) {
return CURRENT_LOCALE_DATA[key][path];
}
}
return message;
}
export function localize(path, message, ...args) { export function localize(path, message, ...args) {
return format(find(path.key || path, message), args); return format(CURRENT_LOCALE_DATA[path.key || path] || message, args);
} }
export function setLocaleData(data) { export function setLocaleData(data) {
CURRENT_LOCALE_DATA = data; CURRENT_LOCALE_DATA = Object.assign(...Object.values(data));
} }
export function loadMessageBundle() { export function loadMessageBundle() {

@ -26,30 +26,30 @@ console.log(
); );
window.UiContext = JSON.parse(window.UiContext); window.UiContext = JSON.parse(window.UiContext);
window.UserContext = JSON.parse(window.UserContext);
try { __webpack_public_path__ = UiContext.cdn_prefix; } catch (e) { }
if ('serviceWorker' in navigator) { if ('serviceWorker' in navigator) {
window.addEventListener('load', () => { const encodedConfig = encodeURIComponent(JSON.stringify(UiContext.SWConfig));
const encodedConfig = encodeURIComponent(JSON.stringify(UiContext.SWConfig)); navigator.serviceWorker.register(`/service-worker.js?config=${encodedConfig}`).then((registration) => {
navigator.serviceWorker.register(`/service-worker.js?config=${encodedConfig}`).then((registration) => { console.log('SW registered: ', registration);
console.log('SW registered: ', registration); }).catch((registrationError) => {
}).catch((registrationError) => { console.log('SW registration failed: ', registrationError);
console.log('SW registration failed: ', registrationError);
});
}); });
} }
document.addEventListener('DOMContentLoaded', async () => { const PageLoader = '<div class="page-loader nojs--hide" style="display:none;"><div class="loader"></div></div>';
const PageLoader = '<div class="page-loader nojs--hide" style="display:none;"><div class="loader"></div></div>'; $('body').prepend(PageLoader);
$('body').prepend(PageLoader); $('.page-loader').fadeIn(500);
$('.page-loader').fadeIn(500);
// eslint-disable-next-line camelcase const prefetch = Promise.all([
try { __webpack_public_path__ = UiContext.cdn_prefix; } catch (e) { } fetch(`/constant/${UiContext.constantVersion}.js`).then((r) => r.text()),
import('./api'),
]);
const [data, HydroExports] = await Promise.all([ document.addEventListener('DOMContentLoaded', async () => {
fetch(`/constant/${UiContext.constantVersion}.js`).then((r) => r.text()), Object.assign(window.UiContext, JSON.parse(window.UiContextNew))
import('./api'), const [data, HydroExports] = await prefetch;
]);
Object.assign(window, { HydroExports }); Object.assign(window, { HydroExports });
eval(data); // eslint-disable-line no-eval eval(data); // eslint-disable-line no-eval
await HydroExports.initPageLoader();
import('./hydro');
}, false); }, false);

@ -17,7 +17,6 @@ declare global {
} }
const start = new Date(); const start = new Date();
window.UserContext = JSON.parse(window.UserContext);
function buildSequence(pages, type) { function buildSequence(pages, type) {
if (process.env.NODE_ENV !== 'production') { if (process.env.NODE_ENV !== 'production') {
@ -52,7 +51,7 @@ async function animate() {
} }
} }
async function load() { export async function initPageLoader() {
const pageLoader = new PageLoader(); const pageLoader = new PageLoader();
const currentPageName = document.documentElement.getAttribute('data-page'); const currentPageName = document.documentElement.getAttribute('data-page');
@ -99,5 +98,3 @@ async function load() {
$('.section').trigger('vjLayout'); $('.section').trigger('vjLayout');
$(document).trigger('vjPageFullyInitialized'); $(document).trigger('vjPageFullyInitialized');
} }
load();

@ -36,17 +36,17 @@
"@types/react": "^18.0.28", "@types/react": "^18.0.28",
"@types/react-dom": "^18.0.11", "@types/react-dom": "^18.0.11",
"@types/redux-logger": "^3.0.9", "@types/redux-logger": "^3.0.9",
"@types/serviceworker": "^0.0.62", "@types/serviceworker": "^0.0.64",
"@types/sharedworker": "^0.0.91", "@types/sharedworker": "^0.0.93",
"@types/webpack-env": "^1.18.0", "@types/webpack-env": "^1.18.0",
"@vscode/codicons": "^0.0.32", "@vscode/codicons": "^0.0.32",
"allotment": "^1.18.1", "allotment": "^1.18.1",
"autoprefixer": "^10.4.13", "autoprefixer": "^10.4.14",
"browser-update": "^3.3.44", "browser-update": "^3.3.44",
"chalk": "^5.2.0", "chalk": "^5.2.0",
"classnames": "^2.3.2", "classnames": "^2.3.2",
"clipboard": "^2.0.11", "clipboard": "^2.0.11",
"cordis": "^2.7.3", "cordis": "^2.7.4",
"diff": "^5.1.0", "diff": "^5.1.0",
"diff-dom": "^5.0.4", "diff-dom": "^5.0.4",
"echarts": "^5.4.1", "echarts": "^5.4.1",
@ -60,14 +60,14 @@
"gulp-if": "^3.0.0", "gulp-if": "^3.0.0",
"gulp-plumber": "^1.2.1", "gulp-plumber": "^1.2.1",
"gulp-svgmin": "^4.1.0", "gulp-svgmin": "^4.1.0",
"jquery": "^3.6.3", "jquery": "^3.6.4",
"jquery-scroll-lock": "^3.1.3", "jquery-scroll-lock": "^3.1.3",
"jquery.easing": "^1.4.1", "jquery.easing": "^1.4.1",
"jquery.transit": "^0.9.12", "jquery.transit": "^0.9.12",
"matchmedia-polyfill": "^0.3.2", "matchmedia-polyfill": "^0.3.2",
"moment": "^2.29.4", "moment": "^2.29.4",
"monaco-editor": "0.36.1", "monaco-editor": "0.36.1",
"monaco-editor-nls": "^2.0.0", "monaco-editor-nls": "^3.0.0",
"monaco-themes": "^0.4.3", "monaco-themes": "^0.4.3",
"monaco-yaml": "^4.0.4", "monaco-yaml": "^4.0.4",
"nanoid": "^4.0.1", "nanoid": "^4.0.1",
@ -94,13 +94,13 @@
"redux-promise-middleware": "^6.1.3", "redux-promise-middleware": "^6.1.3",
"redux-thunk": "^2.4.2", "redux-thunk": "^2.4.2",
"rupture": "^0.7.1", "rupture": "^0.7.1",
"schemastery": "^3.7.1", "schemastery": "^3.7.2",
"slideout": "^1.0.1", "slideout": "^1.0.1",
"sticky-kit": "^1.1.3", "sticky-kit": "^1.1.3",
"tether": "1.4.7", "tether": "1.4.7",
"tether-drop": "^1.4.2", "tether-drop": "^1.4.2",
"through2": "^4.0.2", "through2": "^4.0.2",
"timeago-react": "^3.0.5", "timeago-react": "^3.0.6",
"timeago.js": "^4.0.2", "timeago.js": "^4.0.2",
"vditor": "^3.9.0", "vditor": "^3.9.0",
"vinyl-buffer": "^1.0.1", "vinyl-buffer": "^1.0.1",

@ -58,12 +58,12 @@
{% endif %} {% endif %}
</head> </head>
<body> <body>
{% block body %}{% endblock %}
{% if not isIE(handler.request.headers['user-agent']) and not handler.session.legacy %} {% if not isIE(handler.request.headers['user-agent']) and not handler.session.legacy %}
<script> <script>
var UiContext = '{{ UiContext|json|jsesc|safe }}'; var UiContext = '{{ UiContext|json|jsesc|safe }}';
var UserContext = '{{ UserContext|json|jsesc|safe }}'; var UserContext = '{{ UserContext|json|jsesc|safe }}';
</script> </script>
{% set UiContext = Object.create(UiContext) %}
{% if process.env.DEV %} {% if process.env.DEV %}
<script type="text/javascript" src="/resource/hash/lang-{{ _('__id') }}.js?version={{ global.Hydro.version['ui-default'] }}"></script> <script type="text/javascript" src="/resource/hash/lang-{{ _('__id') }}.js?version={{ global.Hydro.version['ui-default'] }}"></script>
<script type="text/javascript" src="/hydro.js?version={{ global.Hydro.version['ui-default'] }}"></script> <script type="text/javascript" src="/hydro.js?version={{ global.Hydro.version['ui-default'] }}"></script>
@ -72,6 +72,10 @@
<script type="text/javascript" src="{{ UiContext.cdn_prefix }}hydro-{{ global.Hydro.version['ui-default'] }}.js"></script> <script type="text/javascript" src="{{ UiContext.cdn_prefix }}hydro-{{ global.Hydro.version['ui-default'] }}.js"></script>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% block body %}{% endblock %}
{% block script %}{% endblock %} {% block script %}{% endblock %}
{% if not isIE(handler.request.headers['user-agent']) and not handler.session.legacy %}
<script>var UiContextNew = '{{ UiContext|json|jsesc|safe }}';</script>
{% endif %}
</body> </body>
</html> </html>

@ -1,5 +1,7 @@
import $ from 'jquery'; import $ from 'jquery';
import _ from 'lodash'; import _ from 'lodash';
import React from 'react';
import DOMServer from 'react-dom/server.browser';
export function substitute(str: string, obj: any) { export function substitute(str: string, obj: any) {
return str.replace(/\{([^{}]+)\}/g, (match, key) => { return str.replace(/\{([^{}]+)\}/g, (match, key) => {
@ -32,6 +34,7 @@ export function secureRandomString(digit = 32, dict = defaultDict) {
type Substitution = string | number | { templateRaw: true, html: string }; type Substitution = string | number | { templateRaw: true, html: string };
export function tpl(pieces: TemplateStringsArray, ...substitutions: Substitution[]) { export function tpl(pieces: TemplateStringsArray, ...substitutions: Substitution[]) {
if (React.isValidElement(pieces)) return DOMServer.renderToStaticMarkup(pieces);
let result = pieces[0]; let result = pieces[0];
for (let i = 0; i < substitutions.length; ++i) { for (let i = 0; i < substitutions.length; ++i) {
const subst = substitutions[i]; const subst = substitutions[i];

@ -9,7 +9,7 @@
"preferUnplugged": true, "preferUnplugged": true,
"dependencies": { "dependencies": {
"@hydrooj/utils": "workspace:*", "@hydrooj/utils": "workspace:*",
"jsdom": "^21.1.0", "jsdom": "^21.1.1",
"superagent-proxy": "^3.0.0" "superagent-proxy": "^3.0.0"
}, },
"devDependencies": { "devDependencies": {

Loading…
Cancel
Save