mirror of
https://github.com/go-gitea/gitea.git
synced 2026-04-02 00:01:32 +02:00
Replace webpack with Vite 8 as the frontend bundler. Frontend build is around 3-4 times faster than before. Will work on all platforms including riscv64 (via wasm). `iife.js` is a classic render-blocking script in `<head>` (handles web components/early DOM setup). `index.js` is loaded as a `type="module"` script in the footer. All other JS chunks are also module scripts (supported in all browsers since 2018). Entry filenames are content-hashed (e.g. `index.C6Z2MRVQ.js`) and resolved at runtime via the Vite manifest, eliminating the `?v=` cache busting (which was unreliable in some scenarios like vscode dev build). Replaces: https://github.com/go-gitea/gitea/pull/36896 Fixes: https://github.com/go-gitea/gitea/issues/17793 Signed-off-by: silverwind <me@silverwind.io> Signed-off-by: wxiaoguang <wxiaoguang@gmail.com> Co-authored-by: Claude (Opus 4.6) <noreply@anthropic.com> Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
59 lines
2.8 KiB
TypeScript
59 lines
2.8 KiB
TypeScript
/* To manually test:
|
|
|
|
[markup.in-iframe]
|
|
ENABLED = true
|
|
FILE_EXTENSIONS = .in-iframe
|
|
RENDER_CONTENT_MODE = iframe
|
|
RENDER_COMMAND = `echo '<div style="width: 100%; height: 2000px; border: 10px solid red; box-sizing: border-box;"><a href="/">a link</a> <a target="_blank" href="//gitea.com">external link</a></div>'`
|
|
|
|
;RENDER_COMMAND = cat /path/to/file.pdf
|
|
;RENDER_CONTENT_SANDBOX = disabled
|
|
|
|
*/
|
|
|
|
import '../../css/standalone/external-render-iframe.css';
|
|
|
|
function mainExternalRenderIframe() {
|
|
const u = new URL(window.location.href);
|
|
const iframeId = u.searchParams.get('gitea-iframe-id');
|
|
|
|
// iframe is in different origin, so we need to use postMessage to communicate
|
|
const postIframeMsg = (cmd: string, data: Record<string, any> = {}) => {
|
|
window.parent.postMessage({giteaIframeCmd: cmd, giteaIframeId: iframeId, ...data}, '*');
|
|
};
|
|
|
|
const updateIframeHeight = () => {
|
|
// Don't use integer heights from the DOM node.
|
|
// Use getBoundingClientRect(), then ceil the height to avoid fractional pixels which causes incorrect scrollbars.
|
|
const rect = document.documentElement.getBoundingClientRect();
|
|
postIframeMsg('resize', {iframeHeight: Math.ceil(rect.height)});
|
|
// As long as the parent page is responsible for the iframe height, the iframe itself doesn't need scrollbars.
|
|
// This style should only be dynamically set here when our code can run.
|
|
document.documentElement.style.overflowY = 'hidden';
|
|
};
|
|
const resizeObserver = new ResizeObserver(() => updateIframeHeight());
|
|
resizeObserver.observe(window.document.documentElement);
|
|
|
|
updateIframeHeight();
|
|
window.addEventListener('DOMContentLoaded', updateIframeHeight);
|
|
// the easiest way to handle dynamic content changes and easy to debug, can be fine-tuned in the future
|
|
setInterval(updateIframeHeight, 1000);
|
|
|
|
// no way to open an absolute link with CSP frame-src, it needs some tricks like "postMessage" (let parent window to handle) or "copy the link to clipboard" (let users manually paste it to open).
|
|
// here we choose "postMessage" way for better user experience.
|
|
const openIframeLink = (link: string, target: string | null) => postIframeMsg('open-link', {openLink: link, anchorTarget: target});
|
|
document.addEventListener('click', (e) => {
|
|
const el = e.target as HTMLAnchorElement;
|
|
if (el.nodeName !== 'A') return;
|
|
const href = el.getAttribute('href') ?? '';
|
|
// safe links: "./any", "../any", "/any", "//host/any", "http://host/any", "https://host/any"
|
|
if (href.startsWith('.') || href.startsWith('/') || href.startsWith('http://') || href.startsWith('https://')) {
|
|
e.preventDefault();
|
|
const forceTarget = (e.metaKey || e.ctrlKey) ? '_blank' : null;
|
|
openIframeLink(href, forceTarget ?? el.getAttribute('target'));
|
|
}
|
|
});
|
|
}
|
|
|
|
mainExternalRenderIframe();
|