WIP: chore: update to astro v5 #1
@ -2,13 +2,13 @@
|
|||||||
export default new Map([
|
export default new Map([
|
||||||
["src/content/articles/en/after-effects-expressions.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fafter-effects-expressions.mdx&astroContentModuleFlag=true")],
|
["src/content/articles/en/after-effects-expressions.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fafter-effects-expressions.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/articles/en/sci-hub-blocage.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fsci-hub-blocage.mdx&astroContentModuleFlag=true")],
|
["src/content/articles/en/sci-hub-blocage.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fsci-hub-blocage.mdx&astroContentModuleFlag=true")],
|
||||||
|
["src/content/fragments/en/image-full.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Fen%2Fimage-full.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/articles/en/the-day-I-jamd.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fthe-day-I-jamd.mdx&astroContentModuleFlag=true")],
|
["src/content/articles/en/the-day-I-jamd.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fthe-day-I-jamd.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/articles/en/video-compression.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fvideo-compression.mdx&astroContentModuleFlag=true")],
|
["src/content/articles/en/video-compression.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Fen%2Fvideo-compression.mdx&astroContentModuleFlag=true")],
|
||||||
|
["src/content/fragments/en/super-cookies.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Fen%2Fsuper-cookies.mdx&astroContentModuleFlag=true")],
|
||||||
|
["src/content/articles/fr/sci-hub-blocage.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Ffr%2Fsci-hub-blocage.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/fragments/fr/buttons.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Ffr%2Fbuttons.mdx&astroContentModuleFlag=true")],
|
["src/content/fragments/fr/buttons.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Ffr%2Fbuttons.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/fragments/fr/image-full.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Ffr%2Fimage-full.mdx&astroContentModuleFlag=true")],
|
["src/content/fragments/fr/image-full.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Ffr%2Fimage-full.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/fragments/fr/super-cookies.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Ffr%2Fsuper-cookies.mdx&astroContentModuleFlag=true")],
|
["src/content/fragments/fr/super-cookies.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Ffr%2Fsuper-cookies.mdx&astroContentModuleFlag=true")],
|
||||||
["src/content/articles/fr/sci-hub-blocage.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Ffr%2Fsci-hub-blocage.mdx&astroContentModuleFlag=true")],
|
|
||||||
["src/content/fragments/en/image-full.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Fen%2Fimage-full.mdx&astroContentModuleFlag=true")],
|
|
||||||
["src/content/fragments/en/super-cookies.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Ffragments%2Fen%2Fsuper-cookies.mdx&astroContentModuleFlag=true")],
|
|
||||||
["src/content/articles/fr/the-day-I-jamd.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Ffr%2Fthe-day-I-jamd.mdx&astroContentModuleFlag=true")]]);
|
["src/content/articles/fr/the-day-I-jamd.mdx", () => import("astro:content-layer-deferred-module?astro%3Acontent-layer-deferred-module=&fileName=src%2Fcontent%2Farticles%2Ffr%2Fthe-day-I-jamd.mdx&astroContentModuleFlag=true")]]);
|
||||||
|
|
206
.astro/content.d.ts
vendored
206
.astro/content.d.ts
vendored
@ -1,206 +0,0 @@
|
|||||||
declare module 'astro:content' {
|
|
||||||
interface Render {
|
|
||||||
'.mdx': Promise<{
|
|
||||||
Content: import('astro').MarkdownInstance<{}>['Content'];
|
|
||||||
headings: import('astro').MarkdownHeading[];
|
|
||||||
remarkPluginFrontmatter: Record<string, any>;
|
|
||||||
components: import('astro').MDXInstance<{}>['components'];
|
|
||||||
}>;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
declare module 'astro:content' {
|
|
||||||
export interface RenderResult {
|
|
||||||
Content: import('astro/runtime/server/index.js').AstroComponentFactory;
|
|
||||||
headings: import('astro').MarkdownHeading[];
|
|
||||||
remarkPluginFrontmatter: Record<string, any>;
|
|
||||||
}
|
|
||||||
interface Render {
|
|
||||||
'.md': Promise<RenderResult>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface RenderedContent {
|
|
||||||
html: string;
|
|
||||||
metadata?: {
|
|
||||||
imagePaths: Array<string>;
|
|
||||||
[key: string]: unknown;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
declare module 'astro:content' {
|
|
||||||
type Flatten<T> = T extends { [K: string]: infer U } ? U : never;
|
|
||||||
|
|
||||||
export type CollectionKey = keyof AnyEntryMap;
|
|
||||||
export type CollectionEntry<C extends CollectionKey> = Flatten<AnyEntryMap[C]>;
|
|
||||||
|
|
||||||
export type ContentCollectionKey = keyof ContentEntryMap;
|
|
||||||
export type DataCollectionKey = keyof DataEntryMap;
|
|
||||||
|
|
||||||
type AllValuesOf<T> = T extends any ? T[keyof T] : never;
|
|
||||||
type ValidContentEntrySlug<C extends keyof ContentEntryMap> = AllValuesOf<
|
|
||||||
ContentEntryMap[C]
|
|
||||||
>['slug'];
|
|
||||||
|
|
||||||
/** @deprecated Use `getEntry` instead. */
|
|
||||||
export function getEntryBySlug<
|
|
||||||
C extends keyof ContentEntryMap,
|
|
||||||
E extends ValidContentEntrySlug<C> | (string & {}),
|
|
||||||
>(
|
|
||||||
collection: C,
|
|
||||||
// Note that this has to accept a regular string too, for SSR
|
|
||||||
entrySlug: E,
|
|
||||||
): E extends ValidContentEntrySlug<C>
|
|
||||||
? Promise<CollectionEntry<C>>
|
|
||||||
: Promise<CollectionEntry<C> | undefined>;
|
|
||||||
|
|
||||||
/** @deprecated Use `getEntry` instead. */
|
|
||||||
export function getDataEntryById<C extends keyof DataEntryMap, E extends keyof DataEntryMap[C]>(
|
|
||||||
collection: C,
|
|
||||||
entryId: E,
|
|
||||||
): Promise<CollectionEntry<C>>;
|
|
||||||
|
|
||||||
export function getCollection<C extends keyof AnyEntryMap, E extends CollectionEntry<C>>(
|
|
||||||
collection: C,
|
|
||||||
filter?: (entry: CollectionEntry<C>) => entry is E,
|
|
||||||
): Promise<E[]>;
|
|
||||||
export function getCollection<C extends keyof AnyEntryMap>(
|
|
||||||
collection: C,
|
|
||||||
filter?: (entry: CollectionEntry<C>) => unknown,
|
|
||||||
): Promise<CollectionEntry<C>[]>;
|
|
||||||
|
|
||||||
export function getEntry<
|
|
||||||
C extends keyof ContentEntryMap,
|
|
||||||
E extends ValidContentEntrySlug<C> | (string & {}),
|
|
||||||
>(entry: {
|
|
||||||
collection: C;
|
|
||||||
slug: E;
|
|
||||||
}): E extends ValidContentEntrySlug<C>
|
|
||||||
? Promise<CollectionEntry<C>>
|
|
||||||
: Promise<CollectionEntry<C> | undefined>;
|
|
||||||
export function getEntry<
|
|
||||||
C extends keyof DataEntryMap,
|
|
||||||
E extends keyof DataEntryMap[C] | (string & {}),
|
|
||||||
>(entry: {
|
|
||||||
collection: C;
|
|
||||||
id: E;
|
|
||||||
}): E extends keyof DataEntryMap[C]
|
|
||||||
? Promise<DataEntryMap[C][E]>
|
|
||||||
: Promise<CollectionEntry<C> | undefined>;
|
|
||||||
export function getEntry<
|
|
||||||
C extends keyof ContentEntryMap,
|
|
||||||
E extends ValidContentEntrySlug<C> | (string & {}),
|
|
||||||
>(
|
|
||||||
collection: C,
|
|
||||||
slug: E,
|
|
||||||
): E extends ValidContentEntrySlug<C>
|
|
||||||
? Promise<CollectionEntry<C>>
|
|
||||||
: Promise<CollectionEntry<C> | undefined>;
|
|
||||||
export function getEntry<
|
|
||||||
C extends keyof DataEntryMap,
|
|
||||||
E extends keyof DataEntryMap[C] | (string & {}),
|
|
||||||
>(
|
|
||||||
collection: C,
|
|
||||||
id: E,
|
|
||||||
): E extends keyof DataEntryMap[C]
|
|
||||||
? string extends keyof DataEntryMap[C]
|
|
||||||
? Promise<DataEntryMap[C][E]> | undefined
|
|
||||||
: Promise<DataEntryMap[C][E]>
|
|
||||||
: Promise<CollectionEntry<C> | undefined>;
|
|
||||||
|
|
||||||
/** Resolve an array of entry references from the same collection */
|
|
||||||
export function getEntries<C extends keyof ContentEntryMap>(
|
|
||||||
entries: {
|
|
||||||
collection: C;
|
|
||||||
slug: ValidContentEntrySlug<C>;
|
|
||||||
}[],
|
|
||||||
): Promise<CollectionEntry<C>[]>;
|
|
||||||
export function getEntries<C extends keyof DataEntryMap>(
|
|
||||||
entries: {
|
|
||||||
collection: C;
|
|
||||||
id: keyof DataEntryMap[C];
|
|
||||||
}[],
|
|
||||||
): Promise<CollectionEntry<C>[]>;
|
|
||||||
|
|
||||||
export function render<C extends keyof AnyEntryMap>(
|
|
||||||
entry: AnyEntryMap[C][string],
|
|
||||||
): Promise<RenderResult>;
|
|
||||||
|
|
||||||
export function reference<C extends keyof AnyEntryMap>(
|
|
||||||
collection: C,
|
|
||||||
): import('astro/zod').ZodEffects<
|
|
||||||
import('astro/zod').ZodString,
|
|
||||||
C extends keyof ContentEntryMap
|
|
||||||
? {
|
|
||||||
collection: C;
|
|
||||||
slug: ValidContentEntrySlug<C>;
|
|
||||||
}
|
|
||||||
: {
|
|
||||||
collection: C;
|
|
||||||
id: keyof DataEntryMap[C];
|
|
||||||
}
|
|
||||||
>;
|
|
||||||
// Allow generic `string` to avoid excessive type errors in the config
|
|
||||||
// if `dev` is not running to update as you edit.
|
|
||||||
// Invalid collection names will be caught at build time.
|
|
||||||
export function reference<C extends string>(
|
|
||||||
collection: C,
|
|
||||||
): import('astro/zod').ZodEffects<import('astro/zod').ZodString, never>;
|
|
||||||
|
|
||||||
type ReturnTypeOrOriginal<T> = T extends (...args: any[]) => infer R ? R : T;
|
|
||||||
type InferEntrySchema<C extends keyof AnyEntryMap> = import('astro/zod').infer<
|
|
||||||
ReturnTypeOrOriginal<Required<ContentConfig['collections'][C]>['schema']>
|
|
||||||
>;
|
|
||||||
|
|
||||||
type ContentEntryMap = {
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
type DataEntryMap = {
|
|
||||||
"HPsections": Record<string, {
|
|
||||||
id: string;
|
|
||||||
body?: string;
|
|
||||||
collection: "HPsections";
|
|
||||||
data: InferEntrySchema<"HPsections">;
|
|
||||||
rendered?: RenderedContent;
|
|
||||||
filePath?: string;
|
|
||||||
}>;
|
|
||||||
"articles": Record<string, {
|
|
||||||
id: string;
|
|
||||||
body?: string;
|
|
||||||
collection: "articles";
|
|
||||||
data: InferEntrySchema<"articles">;
|
|
||||||
rendered?: RenderedContent;
|
|
||||||
filePath?: string;
|
|
||||||
}>;
|
|
||||||
"fragments": Record<string, {
|
|
||||||
id: string;
|
|
||||||
body?: string;
|
|
||||||
collection: "fragments";
|
|
||||||
data: InferEntrySchema<"fragments">;
|
|
||||||
rendered?: RenderedContent;
|
|
||||||
filePath?: string;
|
|
||||||
}>;
|
|
||||||
"references": Record<string, {
|
|
||||||
id: string;
|
|
||||||
body?: string;
|
|
||||||
collection: "references";
|
|
||||||
data: InferEntrySchema<"references">;
|
|
||||||
rendered?: RenderedContent;
|
|
||||||
filePath?: string;
|
|
||||||
}>;
|
|
||||||
"veille": Record<string, {
|
|
||||||
id: string;
|
|
||||||
body?: string;
|
|
||||||
collection: "veille";
|
|
||||||
data: InferEntrySchema<"veille">;
|
|
||||||
rendered?: RenderedContent;
|
|
||||||
filePath?: string;
|
|
||||||
}>;
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
type AnyEntryMap = ContentEntryMap & DataEntryMap;
|
|
||||||
|
|
||||||
export type ContentConfig = typeof import("../src/content.config.js");
|
|
||||||
}
|
|
File diff suppressed because one or more lines are too long
@ -1,30 +1,58 @@
|
|||||||
import { defineConfig } from 'astro/config'
|
import { defineConfig } from 'astro/config'
|
||||||
|
|
||||||
import { transformerMetaHighlight } from '@shikijs/transformers'
|
|
||||||
|
|
||||||
// https://astro.build/config
|
|
||||||
import mdx from '@astrojs/mdx'
|
import mdx from '@astrojs/mdx'
|
||||||
import sitemap from '@astrojs/sitemap'
|
import sitemap from '@astrojs/sitemap'
|
||||||
|
|
||||||
// https://astro.build/config
|
import { pluginLineNumbers } from '@expressive-code/plugin-line-numbers'
|
||||||
|
import expressiveCode from 'astro-expressive-code'
|
||||||
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
site: 'https://www.nardu.in',
|
site: 'https://www.nardu.in',
|
||||||
build: {
|
build: {
|
||||||
format: 'directory'
|
format: 'directory'
|
||||||
},
|
},
|
||||||
i18n: {
|
redirects: {
|
||||||
locales: ['fr', 'en'],
|
'/en/': {
|
||||||
defaultLocale: 'fr'
|
status: 308,
|
||||||
|
destination: '/articles/en-2025'
|
||||||
|
},
|
||||||
|
'/articles/en/': {
|
||||||
|
status: 308,
|
||||||
|
destination: '/articles/#en-articles'
|
||||||
|
},
|
||||||
|
'/articles/en/[...id]': {
|
||||||
|
status: 308,
|
||||||
|
destination: '/articles/en-[...id]'
|
||||||
|
},
|
||||||
|
'/snippets/en/': {
|
||||||
|
status: 308,
|
||||||
|
destination: '/fragments/#en-fragments'
|
||||||
|
},
|
||||||
|
'/snippets/en/[...id]': {
|
||||||
|
status: 308,
|
||||||
|
destination: '/fragments/en-[...id]'
|
||||||
|
}
|
||||||
},
|
},
|
||||||
image: {
|
image: {
|
||||||
domains: ['assets.nardu.in'],
|
domains: ['assets.nardu.in'],
|
||||||
remotePatterns: [{ protocol: 'https' }]
|
remotePatterns: [{ protocol: 'https' }]
|
||||||
},
|
},
|
||||||
integrations: [mdx(), sitemap()],
|
integrations: [
|
||||||
markdown: {
|
expressiveCode({
|
||||||
shikiConfig: {
|
|
||||||
theme: 'one-dark-pro',
|
theme: 'one-dark-pro',
|
||||||
transformers: [transformerMetaHighlight()]
|
plugins: [pluginLineNumbers()],
|
||||||
|
defaultProps: {
|
||||||
|
// Disable line numbers by default
|
||||||
|
showLineNumbers: false,
|
||||||
|
// But enable line numbers for certain languages
|
||||||
|
overridesByLang: {
|
||||||
|
'css,html,js,ts,vue': {
|
||||||
|
showLineNumbers: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
mdx(),
|
||||||
|
sitemap()
|
||||||
|
]
|
||||||
})
|
})
|
||||||
|
@ -14,7 +14,9 @@
|
|||||||
"@astrojs/rss": "4.0.10",
|
"@astrojs/rss": "4.0.10",
|
||||||
"@astrojs/sitemap": "3.2.1",
|
"@astrojs/sitemap": "3.2.1",
|
||||||
"@astrojs/ts-plugin": "^1.10.4",
|
"@astrojs/ts-plugin": "^1.10.4",
|
||||||
|
"@expressive-code/plugin-line-numbers": "^0.38.3",
|
||||||
"astro": "5.1.1",
|
"astro": "5.1.1",
|
||||||
|
"astro-expressive-code": "^0.38.3",
|
||||||
"sharp": "^0.33.5"
|
"sharp": "^0.33.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
13
src/content/articles/en/2025.md
Normal file
13
src/content/articles/en/2025.md
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
---
|
||||||
|
title: Nico v3.0
|
||||||
|
subtitle: Update 2025.
|
||||||
|
lang: en
|
||||||
|
slug: en-2023
|
||||||
|
excerpt: So long i18n
|
||||||
|
tags: ['Freelance']
|
||||||
|
type: articles
|
||||||
|
---
|
||||||
|
|
||||||
|
## This website no longer has an english version.
|
||||||
|
|
||||||
|
I have maintained an english version of my website for some years. But the
|
@ -13,7 +13,7 @@ updatedAt: '2022-12-27T12:08:00.000Z'
|
|||||||
|
|
||||||
import AstroImage from '../../../components/AstroImage.astro'
|
import AstroImage from '../../../components/AstroImage.astro'
|
||||||
|
|
||||||
The current sci-hub address is: <a href="https://sci-hub.se" rel="noreferer noopener">sci-hub.se</a>
|
The current sci-hub address is: <a href="https://www.sci-hub.st/" rel="noreferer noopener">sci-hub.st</a>
|
||||||
|
|
||||||
## What is this about?
|
## What is this about?
|
||||||
|
|
||||||
|
@ -12,7 +12,7 @@ updatedAt: '2022-12-27T12:08:00.000Z'
|
|||||||
|
|
||||||
import AstroImage from '../../../components/AstroImage.astro'
|
import AstroImage from '../../../components/AstroImage.astro'
|
||||||
|
|
||||||
L'adresse actuelle de sci-hub est : [sci-hub.se](https://sci-hub.se)
|
L'adresse actuelle de sci-hub est : <a href="https://www.sci-hub.st/" rel="noreferer noopener">sci-hub.st</a>
|
||||||
|
|
||||||
## Résumé de la situation
|
## Résumé de la situation
|
||||||
|
|
||||||
|
@ -89,7 +89,7 @@ Limit TLS version to 1.2 and 1.3 (or just 1.3 as there is only a [5% compatibili
|
|||||||
|
|
||||||
Use the [strict transport security](https://scotthelme.co.uk/hsts-the-missing-link-in-tls/) header.
|
Use the [strict transport security](https://scotthelme.co.uk/hsts-the-missing-link-in-tls/) header.
|
||||||
|
|
||||||
```
|
```http
|
||||||
Strict-Transport-Security: max-age=31536000; includeSubDomains
|
Strict-Transport-Security: max-age=31536000; includeSubDomains
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -101,12 +101,12 @@ I've based my initial choices of ciphers on [this list](https://tls.imirhil.fr/c
|
|||||||
|
|
||||||
I then asked [Aeris](https://twitter.com/aeris22), the creator of [tls.imirhil.fr](https://tls.imirhil.fr), about it and he advised me to use the following:
|
I then asked [Aeris](https://twitter.com/aeris22), the creator of [tls.imirhil.fr](https://tls.imirhil.fr), about it and he advised me to use the following:
|
||||||
|
|
||||||
```
|
```cypher
|
||||||
ECDHE+AES:ECDHE+CHACHA20
|
ECDHE+AES:ECDHE+CHACHA20
|
||||||
```
|
```
|
||||||
|
|
||||||
In order to achieve a perfect score, we can be a little more restrictive with:
|
In order to achieve a perfect score, we can be a little more restrictive with:
|
||||||
|
|
||||||
```
|
```cypher
|
||||||
ECDHE+AES256:ECDHE+CHACHA20
|
ECDHE+AES256:ECDHE+CHACHA20
|
||||||
```
|
```
|
||||||
|
@ -70,7 +70,7 @@ query {
|
|||||||
|
|
||||||
#### Inside a page
|
#### Inside a page
|
||||||
|
|
||||||
```javascript
|
```vue
|
||||||
// index.vue
|
// index.vue
|
||||||
<script>
|
<script>
|
||||||
import homepageQuery from '~/graphql/queries/singles/homepage'
|
import homepageQuery from '~/graphql/queries/singles/homepage'
|
||||||
@ -79,7 +79,7 @@ export default {
|
|||||||
async asyncData({ $graphql }) {
|
async asyncData({ $graphql }) {
|
||||||
const data = await $graphql.default.request(homepageQuery)
|
const data = await $graphql.default.request(homepageQuery)
|
||||||
return { data }
|
return { data }
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
```
|
```
|
||||||
@ -88,12 +88,10 @@ export default {
|
|||||||
|
|
||||||
It is safer to wait until `fetch` has received a response before displaying anything. You can use `$fetchState` to be sure ([documentation](https://nuxtjs.org/docs/2.x/components-glossary/pages-fetch 'Documentation on the fetch hook (new tab)')).
|
It is safer to wait until `fetch` has received a response before displaying anything. You can use `$fetchState` to be sure ([documentation](https://nuxtjs.org/docs/2.x/components-glossary/pages-fetch 'Documentation on the fetch hook (new tab)')).
|
||||||
|
|
||||||
```javascript
|
```vue
|
||||||
// Header.vue
|
// Header.vue
|
||||||
<template>
|
<template>
|
||||||
<header v-if="!$fetchState.pending">
|
<header v-if="!$fetchState.pending">…</header>
|
||||||
…
|
|
||||||
</header>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
@ -102,7 +100,7 @@ import headerQuery from '~/graphql/queries/singles/header'
|
|||||||
export default {
|
export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
data: {},
|
data: {}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async fetch() {
|
async fetch() {
|
||||||
@ -112,7 +110,7 @@ export default {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(JSON.stringify(error, undefined, 2))
|
console.error(JSON.stringify(error, undefined, 2))
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
```
|
```
|
||||||
@ -121,7 +119,7 @@ export default {
|
|||||||
|
|
||||||
To pass options to the request, for example for a multilingual version with [nuxt/i18n](https://i18n.nuxtjs.org/) and/or a url parameter in a dynamic page:
|
To pass options to the request, for example for a multilingual version with [nuxt/i18n](https://i18n.nuxtjs.org/) and/or a url parameter in a dynamic page:
|
||||||
|
|
||||||
```javascript
|
```vue
|
||||||
// _slug.vue
|
// _slug.vue
|
||||||
<script>
|
<script>
|
||||||
import articleQuery from '~/graphql/queries/articles'
|
import articleQuery from '~/graphql/queries/articles'
|
||||||
@ -131,10 +129,10 @@ export default {
|
|||||||
const locale = app.i18n.localeProperties.iso
|
const locale = app.i18n.localeProperties.iso
|
||||||
const data = await $graphql.default.request(articleQuery, {
|
const data = await $graphql.default.request(articleQuery, {
|
||||||
code: locale,
|
code: locale,
|
||||||
slug: params.slug,
|
slug: params.slug
|
||||||
})
|
})
|
||||||
return { data }
|
return { data }
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
```
|
```
|
||||||
|
@ -89,7 +89,7 @@ Limiter la version TLS à 1.2 et 1.3 (voire uniquement 1.3 vu [la différence de
|
|||||||
|
|
||||||
Utiliser le header [strict transport security](https://scotthelme.co.uk/hsts-the-missing-link-in-tls/).
|
Utiliser le header [strict transport security](https://scotthelme.co.uk/hsts-the-missing-link-in-tls/).
|
||||||
|
|
||||||
```
|
```http
|
||||||
Strict-Transport-Security: max-age=31536000; includeSubDomains
|
Strict-Transport-Security: max-age=31536000; includeSubDomains
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -101,12 +101,12 @@ J'avais basé ma première suite sur [cette liste](https://tls.imirhil.fr/cipher
|
|||||||
|
|
||||||
J'ai ensuite demandé à [Aeris](https://twitter.com/aeris22), le créateur de [tls.imirhil.fr](https://tls.imirhil.fr), ses conseils sur cette suite. Il m'a recommandé d'utiliser :
|
J'ai ensuite demandé à [Aeris](https://twitter.com/aeris22), le créateur de [tls.imirhil.fr](https://tls.imirhil.fr), ses conseils sur cette suite. Il m'a recommandé d'utiliser :
|
||||||
|
|
||||||
```
|
```cypher
|
||||||
ECDHE+AES:ECDHE+CHACHA20
|
ECDHE+AES:ECDHE+CHACHA20
|
||||||
```
|
```
|
||||||
|
|
||||||
Afin d'atteindre un score de 100/100, il est possible de restreindre un peu plus la suite comme ceci :
|
Afin d'atteindre un score de 100/100, il est possible de restreindre un peu plus la suite comme ceci :
|
||||||
|
|
||||||
```
|
```cypher
|
||||||
ECDHE+AES256:ECDHE+CHACHA20
|
ECDHE+AES256:ECDHE+CHACHA20
|
||||||
```
|
```
|
||||||
|
@ -27,6 +27,7 @@ On est parfois obligé d'utiliser des images dans des balises `img` plutôt que
|
|||||||
Considérons le html suivant :
|
Considérons le html suivant :
|
||||||
|
|
||||||
```html
|
```html
|
||||||
|
<!-- index.html -->
|
||||||
<section class="container">
|
<section class="container">
|
||||||
<div class="hero">
|
<div class="hero">
|
||||||
<img class="hero__image" src="hero.img" />
|
<img class="hero__image" src="hero.img" />
|
||||||
@ -37,15 +38,15 @@ Considérons le html suivant :
|
|||||||
Et le style suivant :
|
Et le style suivant :
|
||||||
|
|
||||||
```css
|
```css
|
||||||
|
/* style.css */
|
||||||
.container {
|
.container {
|
||||||
padding: 0 14px;
|
padding: 0 14px;
|
||||||
margin-left: auto;
|
margin-inline: auto;
|
||||||
margin-right: auto;
|
max-inline-size: 1040px;
|
||||||
max-width: 1040px;
|
|
||||||
}
|
}
|
||||||
img {
|
img {
|
||||||
max-width: 100%;
|
max-inline-size: 100%;
|
||||||
height: auto;
|
block-size: auto;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -60,10 +61,11 @@ img {
|
|||||||
Afin de faire prendre à l'image toute la largeur, on agit sur son conteneur :
|
Afin de faire prendre à l'image toute la largeur, on agit sur son conteneur :
|
||||||
|
|
||||||
```css
|
```css
|
||||||
|
/* style.css */
|
||||||
.hero {
|
.hero {
|
||||||
margin-left: calc(50% - 50vw);
|
margin-inline-start: calc(50% - 50vw);
|
||||||
position: relative;
|
position: relative;
|
||||||
width: 100vw;
|
inline-size: 100vw;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -77,12 +79,15 @@ Afin de faire prendre à l'image toute la largeur, on agit sur son conteneur&nbs
|
|||||||
|
|
||||||
On peut alors réduire la hauteur du conteneur pour obtenir une bannière plutôt que l'image entière et faire correspondre la hauteur de l'image à la hauteur du conteneur :
|
On peut alors réduire la hauteur du conteneur pour obtenir une bannière plutôt que l'image entière et faire correspondre la hauteur de l'image à la hauteur du conteneur :
|
||||||
|
|
||||||
```css
|
```css title='style.css' ins={5, 7-9}
|
||||||
.hero {
|
.hero {
|
||||||
height: 200px;
|
margin-inline-start: calc(50% - 50vw);
|
||||||
|
position: relative;
|
||||||
|
inline-size: 100vw;
|
||||||
|
block-size: 200px;
|
||||||
}
|
}
|
||||||
.hero__image {
|
.hero__image {
|
||||||
height: 100%;
|
block-size: 100%;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -94,9 +99,10 @@ On peut alors réduire la hauteur du conteneur pour obtenir une bannière plutô
|
|||||||
|
|
||||||
Il faut ensuite forcer l'image à prendre toute la largeur du conteneur :
|
Il faut ensuite forcer l'image à prendre toute la largeur du conteneur :
|
||||||
|
|
||||||
```css
|
```css title='style.css' ins={3}
|
||||||
.hero__image {
|
.hero__image {
|
||||||
width: 100%;
|
block-size: 100%;
|
||||||
|
inline-size: 100%;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -112,8 +118,10 @@ Pas top…
|
|||||||
|
|
||||||
**ENFIN** le code magique pour redonner son ratio à l'image sans la déformer :
|
**ENFIN** le code magique pour redonner son ratio à l'image sans la déformer :
|
||||||
|
|
||||||
```css
|
```css title='style.css' ins={4,5}
|
||||||
.hero__image {
|
.hero__image {
|
||||||
|
block-size: 100%;
|
||||||
|
inline-size: 100%;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
object-position: center; /* à positionner comme on veut */
|
object-position: center; /* à positionner comme on veut */
|
||||||
}
|
}
|
||||||
@ -125,22 +133,23 @@ Pas top…
|
|||||||
height='568'
|
height='568'
|
||||||
/>
|
/>
|
||||||
|
|
||||||
Cette technique s'apparente à l'utilisation d'une image de background mais en dur 😁
|
Cette technique s'apparente à l'utilisation d'une image de background.
|
||||||
|
|
||||||
## TL;DR
|
## TL;DR
|
||||||
|
|
||||||
Le code complet :
|
Le code complet :
|
||||||
|
|
||||||
```css
|
```css
|
||||||
|
/* style.css */
|
||||||
.hero {
|
.hero {
|
||||||
margin-left: calc(50% - 50vw);
|
margin-inline-start: calc(50% - 50vw);
|
||||||
position: relative;
|
position: relative;
|
||||||
width: 100vw;
|
inline-size: 100vw;
|
||||||
height: 200px;
|
block-size: 200px;
|
||||||
}
|
}
|
||||||
.hero__image {
|
.hero__image {
|
||||||
width: 100%;
|
inline-size: 100%;
|
||||||
height: 100%;
|
block-size: 100%;
|
||||||
-o-object-fit: cover;
|
-o-object-fit: cover;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
-o-object-position: center;
|
-o-object-position: center;
|
||||||
|
@ -70,7 +70,7 @@ query {
|
|||||||
|
|
||||||
#### Dans une page
|
#### Dans une page
|
||||||
|
|
||||||
```javascript
|
```vue
|
||||||
// index.vue
|
// index.vue
|
||||||
<script>
|
<script>
|
||||||
import homepageQuery from '~/graphql/queries/singles/homepage'
|
import homepageQuery from '~/graphql/queries/singles/homepage'
|
||||||
@ -79,7 +79,7 @@ export default {
|
|||||||
async asyncData({ $graphql }) {
|
async asyncData({ $graphql }) {
|
||||||
const data = await $graphql.default.request(homepageQuery)
|
const data = await $graphql.default.request(homepageQuery)
|
||||||
return { data }
|
return { data }
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
```
|
```
|
||||||
@ -88,12 +88,10 @@ export default {
|
|||||||
|
|
||||||
Il est plus prudent d'attendre que `fetch` ait reçu une réponse avant d'afficher quoi que ce soit. On peut utiliser `$fetchState` afin d'être tranquille ([documentation](https://fr.nuxtjs.org/docs/2.x/components-glossary/pages-fetch 'Documentation sur la méthode fetch (nouvel onglet)')).
|
Il est plus prudent d'attendre que `fetch` ait reçu une réponse avant d'afficher quoi que ce soit. On peut utiliser `$fetchState` afin d'être tranquille ([documentation](https://fr.nuxtjs.org/docs/2.x/components-glossary/pages-fetch 'Documentation sur la méthode fetch (nouvel onglet)')).
|
||||||
|
|
||||||
```javascript
|
```vue
|
||||||
// Header.vue
|
// Header.vue
|
||||||
<template>
|
<template>
|
||||||
<header v-if="!$fetchState.pending">
|
<header v-if="!$fetchState.pending">…</header>
|
||||||
…
|
|
||||||
</header>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
@ -102,7 +100,7 @@ import headerQuery from '~/graphql/queries/singles/header'
|
|||||||
export default {
|
export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
data: {},
|
data: {}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async fetch() {
|
async fetch() {
|
||||||
@ -112,7 +110,7 @@ export default {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(JSON.stringify(error, undefined, 2))
|
console.error(JSON.stringify(error, undefined, 2))
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
```
|
```
|
||||||
@ -121,7 +119,7 @@ export default {
|
|||||||
|
|
||||||
Pour passer des options à la requête, par exemple pour une version multilingue avec [nuxt/i18n](https://i18n.nuxtjs.org/ 'Documentation de nuxt i18n (nouvel onglet)') et/ou un paramètre d'url dans le cadre d'une page dynamique :
|
Pour passer des options à la requête, par exemple pour une version multilingue avec [nuxt/i18n](https://i18n.nuxtjs.org/ 'Documentation de nuxt i18n (nouvel onglet)') et/ou un paramètre d'url dans le cadre d'une page dynamique :
|
||||||
|
|
||||||
```javascript
|
```vue
|
||||||
// _slug.vue
|
// _slug.vue
|
||||||
<script>
|
<script>
|
||||||
import articleQuery from '~/graphql/queries/articles'
|
import articleQuery from '~/graphql/queries/articles'
|
||||||
@ -131,10 +129,10 @@ export default {
|
|||||||
const locale = app.i18n.localeProperties.iso
|
const locale = app.i18n.localeProperties.iso
|
||||||
const data = await $graphql.default.request(articleQuery, {
|
const data = await $graphql.default.request(articleQuery, {
|
||||||
code: locale,
|
code: locale,
|
||||||
slug: params.slug,
|
slug: params.slug
|
||||||
})
|
})
|
||||||
return { data }
|
return { data }
|
||||||
},
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
```
|
```
|
||||||
|
@ -45,7 +45,7 @@ J'ai personnellement choisi de faire différemment. J'ai décidé d'utiliser un
|
|||||||
|
|
||||||
### Le code
|
### Le code
|
||||||
|
|
||||||
```css {1,4-6}
|
```css {5,16} title="style.css"
|
||||||
a {
|
a {
|
||||||
position: relative;
|
position: relative;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -1,11 +1,13 @@
|
|||||||
---
|
---
|
||||||
|
import { getCollection } from 'astro:content'
|
||||||
|
|
||||||
|
import type { ArticleEntry } from 'src/content.config'
|
||||||
import EditorialContent from '../../components/EditorialContent.astro'
|
import EditorialContent from '../../components/EditorialContent.astro'
|
||||||
import BaseLayout from '../../layouts/BaseLayout.astro'
|
import BaseLayout from '../../layouts/BaseLayout.astro'
|
||||||
import { getCollection } from 'astro:content'
|
|
||||||
|
|
||||||
// 1. Generate a new path for every collection entry
|
// 1. Generate a new path for every collection entry
|
||||||
export async function getStaticPaths() {
|
export async function getStaticPaths() {
|
||||||
const articles = await getCollection('articles')
|
const articles: ArticleEntry[] = await getCollection('articles')
|
||||||
return articles.map((article) => ({
|
return articles.map((article) => ({
|
||||||
params: { id: article.data.slug },
|
params: { id: article.data.slug },
|
||||||
props: { article }
|
props: { article }
|
||||||
|
@ -34,7 +34,7 @@ const pageTitle = 'Articles'
|
|||||||
<ListCards list={frArticles} routeName='articles' />
|
<ListCards list={frArticles} routeName='articles' />
|
||||||
</section>
|
</section>
|
||||||
<section class='flow' lang='en'>
|
<section class='flow' lang='en'>
|
||||||
<h3>In english</h3>
|
<h3 id='en-articles'>In english</h3>
|
||||||
<ListCards list={enArticles} routeName='articles' />
|
<ListCards list={enArticles} routeName='articles' />
|
||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
---
|
---
|
||||||
import { getCollection } from 'astro:content'
|
import { getCollection } from 'astro:content'
|
||||||
import EditorialContent from '../../components/EditorialContent.astro'
|
|
||||||
|
|
||||||
|
import type { FragmentEntry } from 'src/content.config'
|
||||||
|
import EditorialContent from '../../components/EditorialContent.astro'
|
||||||
import BaseLayout from '../../layouts/BaseLayout.astro'
|
import BaseLayout from '../../layouts/BaseLayout.astro'
|
||||||
|
|
||||||
// 1. Generate a new path for every collection entry
|
// 1. Generate a new path for every collection entry
|
||||||
export async function getStaticPaths() {
|
export async function getStaticPaths() {
|
||||||
const fragments = await getCollection('fragments')
|
const fragments: FragmentEntry[] = await getCollection('fragments')
|
||||||
return fragments.map((fragment) => ({
|
return fragments.map((fragment) => ({
|
||||||
params: { id: fragment.data.slug },
|
params: { id: fragment.data.slug },
|
||||||
props: { fragment }
|
props: { fragment }
|
||||||
|
@ -34,7 +34,7 @@ const pageTitle = 'Fragments'
|
|||||||
<ListCards list={frFragments} routeName='fragments' />
|
<ListCards list={frFragments} routeName='fragments' />
|
||||||
</section>
|
</section>
|
||||||
<section class='flow' lang='en'>
|
<section class='flow' lang='en'>
|
||||||
<h3>In english</h3>
|
<h3 id='en-fragments'>In english</h3>
|
||||||
<ListCards list={enFragments} routeName='fragments' />
|
<ListCards list={enFragments} routeName='fragments' />
|
||||||
</section>
|
</section>
|
||||||
</section>
|
</section>
|
||||||
|
@ -140,7 +140,6 @@ const allReferences = await getCollection('references')
|
|||||||
}
|
}
|
||||||
.intro__subtitle {
|
.intro__subtitle {
|
||||||
margin: var(--space-s-m) 0;
|
margin: var(--space-s-m) 0;
|
||||||
font-family: 'wotfard';
|
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: var(--color-dark-blue);
|
color: var(--color-dark-blue);
|
||||||
|
@ -30,7 +30,7 @@ EXCEPTIONS
|
|||||||
/*
|
/*
|
||||||
A flipped version where the sidebar is on the right
|
A flipped version where the sidebar is on the right
|
||||||
*/
|
*/
|
||||||
.sidebar[data-direction="rtl"] {
|
.sidebar[data-direction='rtl'] {
|
||||||
flex-direction: row-reverse;
|
flex-direction: row-reverse;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,7 +44,9 @@
|
|||||||
border: 2px solid var(--dark);
|
border: 2px solid var(--dark);
|
||||||
color: var(--dark);
|
color: var(--dark);
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
transition: color 0.3s ease, border-color 0.3s ease;
|
transition:
|
||||||
|
color 0.3s ease,
|
||||||
|
border-color 0.3s ease;
|
||||||
}
|
}
|
||||||
.btn-rideau:hover {
|
.btn-rideau:hover {
|
||||||
color: white;
|
color: white;
|
||||||
|
@ -1,35 +0,0 @@
|
|||||||
@font-face {
|
|
||||||
font-family: "wotfard";
|
|
||||||
src: url("../../fonts/wotfard/wotfard-medium-webfont.woff2") format("woff2");
|
|
||||||
font-weight: 500;
|
|
||||||
font-style: normal;
|
|
||||||
font-display: swap;
|
|
||||||
}
|
|
||||||
@font-face {
|
|
||||||
font-family: "wotfard";
|
|
||||||
src: url("../../fonts/wotfard/wotfard-semibold-webfont.woff2") format("woff2");
|
|
||||||
font-weight: bold;
|
|
||||||
font-style: normal;
|
|
||||||
font-display: swap;
|
|
||||||
}
|
|
||||||
@font-face {
|
|
||||||
font-family: "wotfard";
|
|
||||||
src: url("../../fonts/wotfard/wotfard-regular-webfont.woff2") format("woff2");
|
|
||||||
font-weight: normal;
|
|
||||||
font-style: normal;
|
|
||||||
font-display: swap;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* reduces Cumulative Layout Shift
|
|
||||||
* https://www.24joursdeweb.fr/2021/performance-web-lintegrateur-ce-heros/
|
|
||||||
*/
|
|
||||||
@font-face {
|
|
||||||
font-family: "ArialReplace";
|
|
||||||
src: local("Arial");
|
|
||||||
font-weight: 400;
|
|
||||||
font-style: normal;
|
|
||||||
font-display: swap;
|
|
||||||
size-adjust: 96%;
|
|
||||||
letter-spacing: 1px;
|
|
||||||
}
|
|
@ -14,7 +14,7 @@
|
|||||||
body {
|
body {
|
||||||
font-family: var(--font-primary);
|
font-family: var(--font-primary);
|
||||||
font-size: var(--size-0);
|
font-size: var(--size-0);
|
||||||
line-height: 1.5;
|
line-height: 1.4;
|
||||||
color: var(--color-dark);
|
color: var(--color-dark);
|
||||||
background-color: var(--color-light-white);
|
background-color: var(--color-light-white);
|
||||||
accent-color: var(--color-brique);
|
accent-color: var(--color-brique);
|
||||||
@ -204,46 +204,8 @@ blockquote code {
|
|||||||
|
|
||||||
/* code highlight */
|
/* code highlight */
|
||||||
|
|
||||||
.astro-code,
|
|
||||||
code {
|
code {
|
||||||
font-size: var(--size--1);
|
font-size: var(--size--1);
|
||||||
font-family: var(--font-code);
|
font-family: var(--font-code);
|
||||||
}
|
background-color: var(--color-light-grey);
|
||||||
.astro-code {
|
|
||||||
position: relative;
|
|
||||||
border-radius: var(--radius-small);
|
|
||||||
padding-block: var(--space-xs);
|
|
||||||
direction: ltr;
|
|
||||||
text-align: left;
|
|
||||||
white-space: pre;
|
|
||||||
word-spacing: normal;
|
|
||||||
word-break: normal;
|
|
||||||
word-wrap: normal;
|
|
||||||
-moz-tab-size: 2;
|
|
||||||
-o-tab-size: 2;
|
|
||||||
tab-size: 2;
|
|
||||||
-webkit-hyphens: none;
|
|
||||||
-moz-hyphens: none;
|
|
||||||
-ms-hyphens: none;
|
|
||||||
hyphens: none;
|
|
||||||
|
|
||||||
code {
|
|
||||||
padding-inline: var(--space-s);
|
|
||||||
white-space: pre-wrap;
|
|
||||||
display: block;
|
|
||||||
inline-size: fit-content;
|
|
||||||
min-inline-size: 100%;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.line {
|
|
||||||
&.highlighted::before {
|
|
||||||
content: '';
|
|
||||||
position: absolute;
|
|
||||||
display: inline-block;
|
|
||||||
inline-size: 100%;
|
|
||||||
block-size: 1lh;
|
|
||||||
inset-inline-start: 0;
|
|
||||||
background-color: hsla(0, 0%, 80%, 0.1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/* RESET */
|
/* RESET */
|
||||||
|
|
||||||
:root {
|
:root {
|
||||||
--font-tnum: "tnum" on;
|
--font-tnum: 'tnum' on;
|
||||||
}
|
}
|
||||||
|
|
||||||
* {
|
* {
|
||||||
@ -29,7 +29,7 @@
|
|||||||
}
|
}
|
||||||
@supports (font-variant-numeric: tabular-nums) {
|
@supports (font-variant-numeric: tabular-nums) {
|
||||||
html {
|
html {
|
||||||
--font-tnum: "____";
|
--font-tnum: '____';
|
||||||
font-variant-numeric: tabular-nums;
|
font-variant-numeric: tabular-nums;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -87,7 +87,7 @@ body {
|
|||||||
|
|
||||||
/* Remove built-in form typography styles */
|
/* Remove built-in form typography styles */
|
||||||
:where(input, button, textarea, select),
|
:where(input, button, textarea, select),
|
||||||
:where(input[type="file"])::-webkit-file-upload-button {
|
:where(input[type='file'])::-webkit-file-upload-button {
|
||||||
color: inherit;
|
color: inherit;
|
||||||
font: inherit;
|
font: inherit;
|
||||||
font-size: inherit;
|
font-size: inherit;
|
||||||
@ -113,7 +113,7 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Remove list styles on ul, ol elements with a list role, which suggests default styling will be removed */
|
/* Remove list styles on ul, ol elements with a list role, which suggests default styling will be removed */
|
||||||
:where(ul, ol)[role="list"] {
|
:where(ul, ol)[role='list'] {
|
||||||
list-style: none;
|
list-style: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -132,16 +132,16 @@ a:not([class]) {
|
|||||||
select,
|
select,
|
||||||
summary,
|
summary,
|
||||||
textarea,
|
textarea,
|
||||||
[tabindex]:not([tabindex*="-"])
|
[tabindex]:not([tabindex*='-'])
|
||||||
) {
|
) {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
touch-action: manipulation;
|
touch-action: manipulation;
|
||||||
}
|
}
|
||||||
:where(input[type="file"]) {
|
:where(input[type='file']) {
|
||||||
cursor: auto;
|
cursor: auto;
|
||||||
}
|
}
|
||||||
:where(input[type="file"])::-webkit-file-upload-button,
|
:where(input[type='file'])::-webkit-file-upload-button,
|
||||||
:where(input[type="file"])::file-selector-button {
|
:where(input[type='file'])::file-selector-button {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -162,12 +162,12 @@ a:not([class]) {
|
|||||||
:where(
|
:where(
|
||||||
button,
|
button,
|
||||||
button[type],
|
button[type],
|
||||||
input[type="button"],
|
input[type='button'],
|
||||||
input[type="submit"],
|
input[type='submit'],
|
||||||
input[type="reset"]
|
input[type='reset']
|
||||||
),
|
),
|
||||||
:where(input[type="file"])::-webkit-file-upload-button,
|
:where(input[type='file'])::-webkit-file-upload-button,
|
||||||
:where(input[type="file"])::file-selector-button {
|
:where(input[type='file'])::file-selector-button {
|
||||||
-webkit-tap-highlight-color: transparent;
|
-webkit-tap-highlight-color: transparent;
|
||||||
-webkit-touch-callout: none;
|
-webkit-touch-callout: none;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
@ -178,9 +178,9 @@ a:not([class]) {
|
|||||||
:where(
|
:where(
|
||||||
button,
|
button,
|
||||||
button[type],
|
button[type],
|
||||||
input[type="button"],
|
input[type='button'],
|
||||||
input[type="submit"],
|
input[type='submit'],
|
||||||
input[type="reset"]
|
input[type='reset']
|
||||||
)[disabled] {
|
)[disabled] {
|
||||||
cursor: not-allowed;
|
cursor: not-allowed;
|
||||||
}
|
}
|
||||||
|
@ -51,10 +51,10 @@
|
|||||||
--space-l-3xl: clamp(2.25rem, calc(-0.08rem + 11.67vw), 7.5rem);
|
--space-l-3xl: clamp(2.25rem, calc(-0.08rem + 11.67vw), 7.5rem);
|
||||||
|
|
||||||
/* fonts */
|
/* fonts */
|
||||||
--font-primary: "wotfard", "ArialReplace", sans-serif;
|
--font-primary: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI',
|
||||||
--font-secondary: "recoleta", Palatino, serif;
|
Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
|
||||||
--font-code: Consolas, Monaco, "Andale Mono", "Ubuntu Mono", monospace;
|
--font-code: monospace;
|
||||||
--font-tnum: "tnum" on;
|
--font-tnum: 'tnum' on;
|
||||||
|
|
||||||
/* colors */
|
/* colors */
|
||||||
--color-dark: hsl(239, 57%, 15%);
|
--color-dark: hsl(239, 57%, 15%);
|
||||||
|
@ -1,18 +1,10 @@
|
|||||||
/* @import "open-props/style"; */
|
@import './global/reset.css';
|
||||||
/* @import "open-props/normalize"; */
|
@import './global/variables.css';
|
||||||
|
@import './global/global-styles.css';
|
||||||
|
|
||||||
@import "./global/reset.css";
|
@import './compositions/grid.css';
|
||||||
@import "./global/fonts.css";
|
@import './compositions/sidebar.css';
|
||||||
@import "./global/variables.css";
|
|
||||||
@import "./global/global-styles.css";
|
|
||||||
|
|
||||||
@import "./compositions/grid.css";
|
@import './utilities/flow.css';
|
||||||
@import "./compositions/sidebar.css";
|
@import './utilities/region.css';
|
||||||
|
@import './utilities/wrapper.css';
|
||||||
@import "./utilities/flow.css";
|
|
||||||
@import "./utilities/region.css";
|
|
||||||
@import "./utilities/wrapper.css";
|
|
||||||
|
|
||||||
/* @import-glob './blocks/*.css'; */
|
|
||||||
/* @import-glob './compositions/*.css'; */
|
|
||||||
/* @import-glob './utilities/*.css'; */
|
|
||||||
|
@ -1,71 +0,0 @@
|
|||||||
.waves {
|
|
||||||
background: transparent;
|
|
||||||
block-size: 4px;
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
.waves::before {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
bottom: 0;
|
|
||||||
right: 0;
|
|
||||||
background-repeat: repeat;
|
|
||||||
block-size: 10px;
|
|
||||||
background-size: 20px 20px;
|
|
||||||
background-image: radial-gradient(
|
|
||||||
circle at 10px -5px,
|
|
||||||
transparent 12px,
|
|
||||||
var(--waves-color, var(--color-white)) 13px
|
|
||||||
);
|
|
||||||
}
|
|
||||||
.waves::after {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
bottom: 0;
|
|
||||||
right: 0;
|
|
||||||
background-repeat: repeat;
|
|
||||||
block-size: 15px;
|
|
||||||
background-size: 40px 20px;
|
|
||||||
background-image: radial-gradient(
|
|
||||||
circle at 10px 15px,
|
|
||||||
var(--waves-color, var(--color-white)) 12px,
|
|
||||||
transparent 13px
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* LARGE */
|
|
||||||
.waves-container {
|
|
||||||
block-size: 30px;
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
.waves--large {
|
|
||||||
position: absolute;
|
|
||||||
block-size: 30px;
|
|
||||||
inline-size: 100%;
|
|
||||||
background: var(--waves-color, var(--color-light));
|
|
||||||
bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.waves--large::before,
|
|
||||||
.waves--large::after {
|
|
||||||
content: "";
|
|
||||||
display: block;
|
|
||||||
position: absolute;
|
|
||||||
border-radius: 100% 50%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.waves--large::before {
|
|
||||||
inline-size: 55%;
|
|
||||||
block-size: 100%;
|
|
||||||
background-color: var(--waves-color, var(--color-light));
|
|
||||||
left: -1.5%;
|
|
||||||
top: 40%;
|
|
||||||
}
|
|
||||||
.waves--large::after {
|
|
||||||
inline-size: 55%;
|
|
||||||
block-size: 109%;
|
|
||||||
background-color: var(--waves-bg-color, var(--color-white));
|
|
||||||
right: -1.5%;
|
|
||||||
top: 60%;
|
|
||||||
}
|
|
@ -32,6 +32,7 @@
|
|||||||
grid-column: full;
|
grid-column: full;
|
||||||
}
|
}
|
||||||
/* set full width color to full grid */
|
/* set full width color to full grid */
|
||||||
|
/* prettier-ignore */
|
||||||
.wrapper.full-width-color {
|
.wrapper.full-width-color {
|
||||||
/* https://codepen.io/t_afif/pen/oNEaqQX */
|
/* https://codepen.io/t_afif/pen/oNEaqQX */
|
||||||
border-image: conic-gradient(var(--color-full-width, var(--color-light)) 0 0)
|
border-image: conic-gradient(var(--color-full-width, var(--color-light)) 0 0)
|
||||||
|
271
src/styles/vendor/one-dark-pro.css
vendored
271
src/styles/vendor/one-dark-pro.css
vendored
@ -1,271 +0,0 @@
|
|||||||
code[class*="language-"],
|
|
||||||
pre[class*="language-"],
|
|
||||||
:not(pre) > code {
|
|
||||||
color: #abb2bf;
|
|
||||||
background: none;
|
|
||||||
font-family: Consolas, Monaco, "Andale Mono", "Ubuntu Mono", monospace;
|
|
||||||
font-size: var(--size--1);
|
|
||||||
text-align: left;
|
|
||||||
white-space: pre;
|
|
||||||
word-spacing: normal;
|
|
||||||
word-break: normal;
|
|
||||||
word-wrap: normal;
|
|
||||||
line-height: 1.5;
|
|
||||||
|
|
||||||
-moz-tab-size: 2;
|
|
||||||
-o-tab-size: 2;
|
|
||||||
tab-size: 2;
|
|
||||||
|
|
||||||
-webkit-hyphens: none;
|
|
||||||
-moz-hyphens: none;
|
|
||||||
-ms-hyphens: none;
|
|
||||||
hyphens: none;
|
|
||||||
}
|
|
||||||
/* Code blocks */
|
|
||||||
pre[class*="language-"] {
|
|
||||||
padding: var(--space-2xs-xs) var(--space-xs-s);
|
|
||||||
margin: var(--space-2xs-xs) 0;
|
|
||||||
overflow: auto;
|
|
||||||
border-radius: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Inline code */
|
|
||||||
:not(pre) > code[class*="language-"],
|
|
||||||
pre[class*="language-"] {
|
|
||||||
background: #282c34;
|
|
||||||
}
|
|
||||||
:not(pre) > code[class*="language-"],
|
|
||||||
:not(pre) > code {
|
|
||||||
padding: 0.2em;
|
|
||||||
white-space: normal;
|
|
||||||
color: var(--color-dark);
|
|
||||||
border-radius: 0.3em;
|
|
||||||
background-color: var(--color-light-grey);
|
|
||||||
}
|
|
||||||
pre[class*="language-"]::selection,
|
|
||||||
pre[class*="language-"] ::selection,
|
|
||||||
code[class*="language-"]::selection,
|
|
||||||
code[class*="language-"] ::selection {
|
|
||||||
background: rgba(148, 170, 209, 0.2);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* General styling */
|
|
||||||
.token.comment,
|
|
||||||
.token.prolog,
|
|
||||||
.token.doctype,
|
|
||||||
.token.cdata,
|
|
||||||
.token.punctuation {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
|
|
||||||
.token.namespace {
|
|
||||||
opacity: 0.7;
|
|
||||||
}
|
|
||||||
.token.keyword,
|
|
||||||
.token.regex,
|
|
||||||
.token.important {
|
|
||||||
color: #c678dd;
|
|
||||||
}
|
|
||||||
.token.property,
|
|
||||||
.token.attr-name,
|
|
||||||
.token.constant,
|
|
||||||
.token.symbol,
|
|
||||||
.token.deleted {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
|
|
||||||
.token.unit {
|
|
||||||
color: #2f02d1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.token.boolean,
|
|
||||||
.token.number,
|
|
||||||
.token.selector,
|
|
||||||
.token.tag,
|
|
||||||
.token.char,
|
|
||||||
.token.builtin,
|
|
||||||
.token.inserted,
|
|
||||||
.token.entity,
|
|
||||||
.token.url,
|
|
||||||
.language-css .token.string,
|
|
||||||
.style .token.string,
|
|
||||||
.token.variable {
|
|
||||||
color: #ce004b;
|
|
||||||
}
|
|
||||||
|
|
||||||
.token.operator,
|
|
||||||
.token.pseudo-element,
|
|
||||||
.token.pseudo-class,
|
|
||||||
.token.attr-value,
|
|
||||||
.token.function,
|
|
||||||
.token.class-name {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
|
|
||||||
.token.atrule {
|
|
||||||
color: #c678dd;
|
|
||||||
}
|
|
||||||
|
|
||||||
.token.string {
|
|
||||||
color: #10113a;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Specific styling */
|
|
||||||
.language-html .token.doctype-tag {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-html .token.name {
|
|
||||||
color: #d19a66;
|
|
||||||
}
|
|
||||||
.language-html .token.punctuation {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
.language-html .token.tag .token.tag {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-html .token.tag .token.attr-name {
|
|
||||||
color: #d19a66;
|
|
||||||
}
|
|
||||||
.language-html .token.tag .token.attr-value {
|
|
||||||
color: #98c379;
|
|
||||||
}
|
|
||||||
.language-html .token.tag .token.attr-value .token.punctuation {
|
|
||||||
color: #98c379;
|
|
||||||
}
|
|
||||||
.language-html .token.tag .token.attr-value .token.punctuation.attr-equals {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
.language-html .token.comment {
|
|
||||||
color: #7f848e;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
code[class*="language-css"],
|
|
||||||
pre[class*="language-css"] {
|
|
||||||
color: #d19a66;
|
|
||||||
}
|
|
||||||
.language-css .token.operator,
|
|
||||||
.language-css .token.punctuation,
|
|
||||||
.language-css .token.property,
|
|
||||||
.language-css .token.operator,
|
|
||||||
.language-css .token.combinator {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
.language-css .token.attr-name,
|
|
||||||
.language-css .token.color,
|
|
||||||
.language-css .token.number,
|
|
||||||
.language-css .token.class {
|
|
||||||
color: #d19a66;
|
|
||||||
}
|
|
||||||
.language-css .token.attr-value,
|
|
||||||
.language-css .token.string {
|
|
||||||
color: #98c379;
|
|
||||||
}
|
|
||||||
.language-css .token.selector,
|
|
||||||
.language-css .token.unit {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-css .token.property.prefix,
|
|
||||||
.language-css .token.pseudo-element,
|
|
||||||
.language-css .token.pseudo-class {
|
|
||||||
color: #56b6c2;
|
|
||||||
}
|
|
||||||
.language-css .token.atrule {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
.language-css .token.atrule .token.rule {
|
|
||||||
color: #c678dd;
|
|
||||||
}
|
|
||||||
.language-css .token.comment {
|
|
||||||
color: #7f848e;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
.language-css .token.keyword {
|
|
||||||
color: #56b6c2;
|
|
||||||
}
|
|
||||||
code[class*="language-javascript"],
|
|
||||||
pre[class*="language-javascript"] {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-javascript {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-javascript .token.punctuation {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
.language-javascript .token.comment {
|
|
||||||
color: #7f848e;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
.language-javascript .token.keyword.this,
|
|
||||||
.language-javascript .token.dom.variable,
|
|
||||||
.language-javascript .token.class-name {
|
|
||||||
color: #e5c07b;
|
|
||||||
}
|
|
||||||
.language-javascript .token.null,
|
|
||||||
.language-javascript .token.boolean,
|
|
||||||
.language-javascript .token.number {
|
|
||||||
color: #d19a66;
|
|
||||||
}
|
|
||||||
.language-javascript .token.property-access,
|
|
||||||
.language-javascript .token.imports,
|
|
||||||
.language-javascript .token.parameter {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-javascript .token.parameter {
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
.language-javascript .token.keyword {
|
|
||||||
color: #c678dd;
|
|
||||||
}
|
|
||||||
.language-javascript .token.function,
|
|
||||||
.language-javascript .token.property-access.function.method {
|
|
||||||
color: #61afef;
|
|
||||||
}
|
|
||||||
.language-javascript .token.regex-source,
|
|
||||||
.language-javascript .token.operator {
|
|
||||||
color: #56b6c2;
|
|
||||||
}
|
|
||||||
.language-javascript .token.regex-delimiter,
|
|
||||||
.language-javascript .token.string {
|
|
||||||
color: #98c379;
|
|
||||||
}
|
|
||||||
.language-json .token.punctuation,
|
|
||||||
.language-json .token.operator {
|
|
||||||
color: #abb2bf;
|
|
||||||
}
|
|
||||||
.language-json .token.string {
|
|
||||||
color: #98c379;
|
|
||||||
}
|
|
||||||
.language-json .token.boolean,
|
|
||||||
.language-json .token.number,
|
|
||||||
.language-json .token.null.keyword {
|
|
||||||
color: #d19a66;
|
|
||||||
}
|
|
||||||
.language-json .token.property {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
||||||
.language-bash .token.string,
|
|
||||||
.language-shell .token.string {
|
|
||||||
color: #98c379;
|
|
||||||
}
|
|
||||||
.language-bash .token.shebang.important,
|
|
||||||
.language-bash .token.comment,
|
|
||||||
.language-shell .token.shebang.important,
|
|
||||||
.language-shell .token.comment {
|
|
||||||
color: #7f848e;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
.language-bash .token.builtin.class-name,
|
|
||||||
.language-bash .token.entity,
|
|
||||||
.language-shell .token.builtin.class-name,
|
|
||||||
.language-shell .token.entity {
|
|
||||||
color: #56b6c2;
|
|
||||||
}
|
|
||||||
.language-bash .token.keyword,
|
|
||||||
.language-shell .token.keyword {
|
|
||||||
color: #c678dd;
|
|
||||||
}
|
|
||||||
.language-bash .token.variable,
|
|
||||||
.language-shell .token.variable {
|
|
||||||
color: #e06c75;
|
|
||||||
}
|
|
@ -1,28 +0,0 @@
|
|||||||
import { defaultLang, ui } from '../i18n/ui'
|
|
||||||
|
|
||||||
export function getLangFromUrl(url: URL) {
|
|
||||||
const [, lang] = url.pathname.split('/')
|
|
||||||
if (lang in ui) return lang as keyof typeof ui
|
|
||||||
return defaultLang
|
|
||||||
}
|
|
||||||
|
|
||||||
type NestedKeyOf<T> = {
|
|
||||||
[K in keyof T]: T[K] extends object
|
|
||||||
? `${K & string}.${NestedKeyOf<T[K]> & string}`
|
|
||||||
: K & string
|
|
||||||
}[keyof T]
|
|
||||||
|
|
||||||
export function useTranslations(lang: keyof typeof ui) {
|
|
||||||
return function t(key: NestedKeyOf<(typeof ui)[typeof defaultLang]>) {
|
|
||||||
const keys = key.split('.')
|
|
||||||
let value = ui[lang]
|
|
||||||
let fallback = ui[defaultLang]
|
|
||||||
|
|
||||||
for (const k of keys) {
|
|
||||||
value = value?.[k]
|
|
||||||
fallback = fallback?.[k]
|
|
||||||
}
|
|
||||||
|
|
||||||
return value || fallback
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user