mirror of
https://github.com/1f349/svelte-ssr-template.git
synced 2024-11-21 19:11:33 +00:00
First commit
This commit is contained in:
commit
88a66f6763
4
create.sh
Executable file
4
create.sh
Executable file
@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
git clone https://github.com/1f349/svelte-ssr-template svelte-ssr-template
|
||||||
|
cp -r svelte-ssr-template/template/* .
|
||||||
|
rm -r svelte-ssr-template
|
36
template/.editorconfig
Normal file
36
template/.editorconfig
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
# EditorConfig is awesome: https://EditorConfig.org
|
||||||
|
|
||||||
|
# top-most EditorConfig file
|
||||||
|
root = true
|
||||||
|
|
||||||
|
# Defaults
|
||||||
|
[*]
|
||||||
|
charset = utf-8
|
||||||
|
end_of_line = lf
|
||||||
|
insert_final_newline = true
|
||||||
|
|
||||||
|
# CSS
|
||||||
|
[*.css]
|
||||||
|
indent_size = 2
|
||||||
|
indent_style = space
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
|
# HTML
|
||||||
|
[*.{htm,html}]
|
||||||
|
indent_size = 2
|
||||||
|
indent_style = space
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
|
# GNU make
|
||||||
|
[Makefile]
|
||||||
|
indent_style = tab
|
||||||
|
|
||||||
|
# Svelte
|
||||||
|
[*.svelte]
|
||||||
|
indent_size = 2
|
||||||
|
indent_style = space
|
||||||
|
|
||||||
|
# YAML
|
||||||
|
[*.{yaml,yml}]
|
||||||
|
indent_size = 2
|
||||||
|
indent_style = space
|
25
template/.gitignore
vendored
Normal file
25
template/.gitignore
vendored
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
pnpm-debug.log*
|
||||||
|
lerna-debug.log*
|
||||||
|
|
||||||
|
node_modules
|
||||||
|
dist
|
||||||
|
dist-ssr
|
||||||
|
*.local
|
||||||
|
*.development
|
||||||
|
|
||||||
|
# Editor directories and files
|
||||||
|
.vscode/*
|
||||||
|
!.vscode/extensions.json
|
||||||
|
.idea
|
||||||
|
.DS_Store
|
||||||
|
*.suo
|
||||||
|
*.ntvs*
|
||||||
|
*.njsproj
|
||||||
|
*.sln
|
||||||
|
*.sw?
|
13
template/.prettierrc
Normal file
13
template/.prettierrc
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
"printWidth": 150,
|
||||||
|
"tabWidth": 2,
|
||||||
|
"useTabs": false,
|
||||||
|
"semi": true,
|
||||||
|
"singleQuote": false,
|
||||||
|
"trailingComma": "all",
|
||||||
|
"bracketSpacing": false,
|
||||||
|
"bracketSameLine": false,
|
||||||
|
"arrowParens": "avoid",
|
||||||
|
"requirePragma": false,
|
||||||
|
"insertPragma": false
|
||||||
|
}
|
3
template/.vscode/extensions.json
vendored
Normal file
3
template/.vscode/extensions.json
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"recommendations": ["svelte.svelte-vscode"]
|
||||||
|
}
|
15
template/README.md
Normal file
15
template/README.md
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# Svelte SSR Template
|
||||||
|
|
||||||
|
## Building
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn
|
||||||
|
yarn run build
|
||||||
|
```
|
||||||
|
|
||||||
|
## Development
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn
|
||||||
|
yarn run dev
|
||||||
|
```
|
38
template/package.json
Normal file
38
template/package.json
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
{
|
||||||
|
"name": "svelte-ssr-template",
|
||||||
|
"private": true,
|
||||||
|
"version": "0.0.0",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"prettier:check:ci": "./node_modules/.bin/prettier --check .",
|
||||||
|
"format": "./node_modules/.bin/prettier --write .",
|
||||||
|
"dev": "vite",
|
||||||
|
"build": "vite build",
|
||||||
|
"preview": "vite preview",
|
||||||
|
"check": "svelte-check --tsconfig ./tsconfig.json"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@svelte-parts/markdown": "^0.0.23",
|
||||||
|
"@sveltejs/vite-plugin-svelte": "^1.1.0",
|
||||||
|
"@tsconfig/svelte": "^3.0.0",
|
||||||
|
"@types/dompurify": "^2.4.0",
|
||||||
|
"@types/highlight.js": "^10.1.0",
|
||||||
|
"dompurify": "^2.4.1",
|
||||||
|
"highlight.js": "^11.7.0",
|
||||||
|
"prettier": "^2.7.1",
|
||||||
|
"prettier-plugin-svelte": "^2.8.0",
|
||||||
|
"sass": "^1.56.0",
|
||||||
|
"svelte": "^3.52.0",
|
||||||
|
"svelte-check": "^2.9.2",
|
||||||
|
"svelte-markdown": "^0.2.3",
|
||||||
|
"svelte-navigator": "^3.2.2",
|
||||||
|
"svelte-preprocess": "^4.10.7",
|
||||||
|
"tslib": "^2.4.0",
|
||||||
|
"typescript": "^4.6.4",
|
||||||
|
"vite": "^3.2.0",
|
||||||
|
"vite-plugin-ssr": "^0.4.90"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"semver": "^7.5.3"
|
||||||
|
}
|
||||||
|
}
|
BIN
template/public/fonts/Ubuntu.woff2
Normal file
BIN
template/public/fonts/Ubuntu.woff2
Normal file
Binary file not shown.
36
template/server.ts
Normal file
36
template/server.ts
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
const express = require("express");
|
||||||
|
const {renderPage} = require("vite-plugin-ssr");
|
||||||
|
|
||||||
|
const isProduction = process.env.NODE_ENV === "production";
|
||||||
|
const root = `${__dirname}/..`;
|
||||||
|
|
||||||
|
startServer();
|
||||||
|
|
||||||
|
async function startServer() {
|
||||||
|
const app = express();
|
||||||
|
|
||||||
|
if (isProduction) {
|
||||||
|
app.use(express.static(`${root}/client`));
|
||||||
|
} else {
|
||||||
|
const vite = require("vite");
|
||||||
|
const viteDevServer = await vite.createServer({
|
||||||
|
root,
|
||||||
|
server: {
|
||||||
|
middlewareMode: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
app.use(viteDevServer.middlewares);
|
||||||
|
}
|
||||||
|
|
||||||
|
app.get("*", async (req, res, next) => {
|
||||||
|
const pageContextInit = {urlOriginal: req.originalUrl};
|
||||||
|
const pageContext = await renderPage(pageContextInit);
|
||||||
|
if (pageContext.httpResponse === null) return next();
|
||||||
|
const {body, statusCode, contentType} = pageContext.httpResponse;
|
||||||
|
res.status(statusCode).type(contentType).send(body);
|
||||||
|
});
|
||||||
|
|
||||||
|
const port = 5173;
|
||||||
|
app.listen(port);
|
||||||
|
console.log(`Server running at http://localhost:${port}`);
|
||||||
|
}
|
27
template/src/components/MetaTags.svelte
Normal file
27
template/src/components/MetaTags.svelte
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
export let url: string;
|
||||||
|
export let title: string;
|
||||||
|
export let description: string;
|
||||||
|
export let keywords: string;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svelte:head>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no" />
|
||||||
|
<title>{title}</title>
|
||||||
|
|
||||||
|
<meta name="theme-color" content="#0000" />
|
||||||
|
<meta name="default-theme" content="auto" />
|
||||||
|
<meta name="author" content="Svelte SSR Template" />
|
||||||
|
<meta name="description" content={description} />
|
||||||
|
<meta name="keywords" content="svelte-ssr-template,{keywords}" />
|
||||||
|
<meta name="referrer" content="no-referrer" />
|
||||||
|
|
||||||
|
<meta property="og:title" content={title} />
|
||||||
|
<meta property="og:url" content={url.indexOf("https://") == 0 ? url : "https://example.com" + (url[0] == "/" ? url : "/" + url)} />
|
||||||
|
<meta property="og:type" content="object" />
|
||||||
|
<meta property="og:image" content="https://example.com/logo.png" />
|
||||||
|
<meta property="og:site_name" content="Svelte SSR Template" />
|
||||||
|
|
||||||
|
<link rel="icon" type="image/svg+xml" href="/logo.svg" />
|
||||||
|
<link rel="icon" type="image/png" href="/logo.png" />
|
||||||
|
</svelte:head>
|
9
template/src/pages/__layout.svelte
Normal file
9
template/src/pages/__layout.svelte
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import "~/styles/app.scss";
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<main>
|
||||||
|
<slot />
|
||||||
|
</main>
|
||||||
|
</div>
|
30
template/src/pages/_default/_default.page.client.js
Normal file
30
template/src/pages/_default/_default.page.client.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
export const clientRouting = true;
|
||||||
|
|
||||||
|
export const prefetchStaticAssets = {when: "HOVER"};
|
||||||
|
|
||||||
|
export async function render(pageContext) {
|
||||||
|
const app_el = document.getElementById("app");
|
||||||
|
new pageContext.Page({
|
||||||
|
target: app_el,
|
||||||
|
hydrate: true,
|
||||||
|
props: {
|
||||||
|
__: pageContext.__,
|
||||||
|
pageProps: pageContext.pageProps,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function onPageTransitionStart(pageContext) {
|
||||||
|
console.log("Page transition start");
|
||||||
|
// `pageContext.isBackwardNavigation` is also set at `render(pageContext)`
|
||||||
|
// and `onPageTransitionEnd(pageContext)`.
|
||||||
|
console.log("Is backwards navigation?", pageContext.isBackwardNavigation);
|
||||||
|
// For example:
|
||||||
|
document.body.classList.add("page-transition");
|
||||||
|
}
|
||||||
|
|
||||||
|
export function onPageTransitionEnd(pageContext) {
|
||||||
|
console.log("Page transition end");
|
||||||
|
// For example:
|
||||||
|
document.body.classList.remove("page-transition");
|
||||||
|
}
|
44
template/src/pages/_default/_default.page.server.js
Normal file
44
template/src/pages/_default/_default.page.server.js
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import {escapeInject, dangerouslySkipEscape} from "vite-plugin-ssr";
|
||||||
|
const base = import.meta.env.BASE_URL;
|
||||||
|
|
||||||
|
// See https://vite-plugin-ssr.com/data-fetching
|
||||||
|
export const passToClient = ["__", "pageProps", "routeParams", "urlOriginal"];
|
||||||
|
|
||||||
|
export async function onBeforeRender(pageContext) {
|
||||||
|
const {routeParams, urlOriginal} = pageContext;
|
||||||
|
|
||||||
|
return {
|
||||||
|
pageContext: {
|
||||||
|
__: {
|
||||||
|
routeParams,
|
||||||
|
urlOriginal,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function render(pageContext) {
|
||||||
|
const app = pageContext.Page.render(pageContext);
|
||||||
|
const appHtml = app.html;
|
||||||
|
const appCss = app.css.code;
|
||||||
|
const appHead = app.head;
|
||||||
|
|
||||||
|
// We are using Svelte's app.head variable rather than the Vite Plugin SSR
|
||||||
|
// technique described here: https://vite-plugin-ssr.com/html-head This seems
|
||||||
|
// easier for using data fetched from APIs and also allows us to input the
|
||||||
|
// data using our custom MetaTags Svelte component.
|
||||||
|
|
||||||
|
return escapeInject`<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<link rel="icon" href="${base}logo.svg" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
${dangerouslySkipEscape(appHead)}
|
||||||
|
<style>${appCss}</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="app">${dangerouslySkipEscape(appHtml)}</div>
|
||||||
|
</body>
|
||||||
|
</html>`;
|
||||||
|
}
|
24
template/src/pages/_error.page.svelte
Normal file
24
template/src/pages/_error.page.svelte
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import {navigate} from "vite-plugin-ssr/client/router";
|
||||||
|
import MetaTags from "~/components/MetaTags.svelte";
|
||||||
|
import Layout from "./__layout.svelte";
|
||||||
|
|
||||||
|
export let __;
|
||||||
|
export let pageProps;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<MetaTags url={__.urlOriginal} title="Error" description="" keywords="error,error page" />
|
||||||
|
|
||||||
|
<Layout>
|
||||||
|
<div>
|
||||||
|
{#if pageProps.is404}
|
||||||
|
<h1 class="title-text">404 Not Found</h1>
|
||||||
|
<p class="coming-soon">You have lost your way, please return to valid paths.</p>
|
||||||
|
<p><button on:click={() => navigate("/")} class="refresh-btn">Back to homepage</button></p>
|
||||||
|
{:else}
|
||||||
|
<h1 class="title-text">500 Internal Server Error</h1>
|
||||||
|
<p class="coming-soon">The server had an issue with this page.</p>
|
||||||
|
<p><button on:click={() => navigate("/")} class="refresh-btn">Back to homepage</button></p>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</Layout>
|
21
template/src/pages/index.page.svelte
Normal file
21
template/src/pages/index.page.svelte
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import MetaTags from "~/components/MetaTags.svelte";
|
||||||
|
import Layout from "./__layout.svelte";
|
||||||
|
|
||||||
|
export let __;
|
||||||
|
export let pageProps;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<MetaTags url={__.urlOriginal} title="Svelte SSR Template" description="Svelte SSR Template Home Page" keywords="" />
|
||||||
|
|
||||||
|
<Layout isHome={true}>
|
||||||
|
<h1 class="title-text">Svelte SSR Template</h1>
|
||||||
|
</Layout>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
.title-text {
|
||||||
|
margin: 0 0 24px 0;
|
||||||
|
font-size: 3.2em;
|
||||||
|
line-height: 1.1;
|
||||||
|
}
|
||||||
|
</style>
|
39
template/src/styles/app.scss
Normal file
39
template/src/styles/app.scss
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
@font-face {
|
||||||
|
font-family: 'Ubuntu';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
font-display: swap;
|
||||||
|
src: url('/fonts/Ubuntu.woff2') format('woff2');
|
||||||
|
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
|
||||||
|
}
|
||||||
|
|
||||||
|
:root {
|
||||||
|
font-family: Ubuntu, Inter, Avenir, Helvetica, Arial, sans-serif;
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 24px;
|
||||||
|
font-weight: 400;
|
||||||
|
|
||||||
|
color-scheme: light dark;
|
||||||
|
color: #e2e2e2;
|
||||||
|
background-color: #000000;
|
||||||
|
|
||||||
|
font-synthesis: none;
|
||||||
|
text-rendering: optimizeLegibility;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
-webkit-text-size-adjust: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
font-weight: 500;
|
||||||
|
color: tomato;
|
||||||
|
text-decoration: inherit;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: tomato;
|
||||||
|
}
|
||||||
|
}
|
9
template/src/vite-env.d.ts
vendored
Normal file
9
template/src/vite-env.d.ts
vendored
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
/// <reference types="svelte" />
|
||||||
|
/// <reference types="vite/client" />
|
||||||
|
|
||||||
|
interface ImportMetaEnv {
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ImportMeta {
|
||||||
|
readonly env: ImportMetaEnv;
|
||||||
|
}
|
7
template/svelte.config.js
Normal file
7
template/svelte.config.js
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
import sveltePreprocess from "svelte-preprocess";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
// Consult https://github.com/sveltejs/svelte-preprocess
|
||||||
|
// for more information about preprocessors
|
||||||
|
preprocess: sveltePreprocess(),
|
||||||
|
};
|
22
template/tsconfig.json
Normal file
22
template/tsconfig.json
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
"extends": "@tsconfig/svelte/tsconfig.json",
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "ESNext",
|
||||||
|
"useDefineForClassFields": true,
|
||||||
|
"module": "ESNext",
|
||||||
|
"resolveJsonModule": true,
|
||||||
|
"baseUrl": "./src",
|
||||||
|
"paths": {"~/*": ["./*"]},
|
||||||
|
/**
|
||||||
|
* Typecheck JS in `.svelte` and `.js` files by default.
|
||||||
|
* Disable checkJs if you'd like to use dynamic types in JS.
|
||||||
|
* Note that setting allowJs false does not prevent the use
|
||||||
|
* of JS in `.svelte` files.
|
||||||
|
*/
|
||||||
|
"allowJs": true,
|
||||||
|
"checkJs": true,
|
||||||
|
"isolatedModules": true
|
||||||
|
},
|
||||||
|
"include": ["src/**/*.d.ts", "src/**/*.ts", "src/**/*.js", "src/**/*.svelte", "src/**/*.mjs"],
|
||||||
|
"references": [{"path": "./tsconfig.node.json"}]
|
||||||
|
}
|
8
template/tsconfig.node.json
Normal file
8
template/tsconfig.node.json
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"composite": true,
|
||||||
|
"module": "ESNext",
|
||||||
|
"moduleResolution": "Node"
|
||||||
|
},
|
||||||
|
"include": ["vite.config.ts"]
|
||||||
|
}
|
38
template/vite.config.ts
Normal file
38
template/vite.config.ts
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
import {defineConfig} from "vite";
|
||||||
|
import {svelte} from "@sveltejs/vite-plugin-svelte";
|
||||||
|
import sveltePreprocess from "svelte-preprocess";
|
||||||
|
import {resolve as pathResolve} from "path";
|
||||||
|
import ssr from "vite-plugin-ssr/plugin";
|
||||||
|
|
||||||
|
// https://vitejs.dev/config/
|
||||||
|
export default defineConfig({
|
||||||
|
build: {
|
||||||
|
emptyOutDir: true,
|
||||||
|
},
|
||||||
|
server: {
|
||||||
|
port: 5173,
|
||||||
|
},
|
||||||
|
plugins: [
|
||||||
|
svelte({
|
||||||
|
preprocess: sveltePreprocess({
|
||||||
|
preserve: ["ld+json"],
|
||||||
|
scss: {
|
||||||
|
includePaths: ["src/"],
|
||||||
|
quietDeps: true,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
compilerOptions: {
|
||||||
|
hydratable: true,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
ssr({
|
||||||
|
prerender: true,
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
optimizeDeps: {exclude: ["svelte-navigator"]},
|
||||||
|
resolve: {
|
||||||
|
alias: {
|
||||||
|
"~": pathResolve(__dirname, "src"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
1056
template/yarn.lock
Normal file
1056
template/yarn.lock
Normal file
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user