mirror of
https://github.com/LukeHagar/website.git
synced 2025-12-06 12:57:48 +00:00
134 lines
5.4 KiB
Svelte
134 lines
5.4 KiB
Svelte
<script lang="ts">
|
|
import { MainFooter } from '$lib/components';
|
|
import { DEFAULT_HOST } from '$lib/utils/metadata';
|
|
import { DOCS_TITLE_SUFFIX } from '$routes/titles';
|
|
|
|
const title = 'Tutorials' + DOCS_TITLE_SUFFIX;
|
|
const description =
|
|
'Follow a simple tutorial to get started with Appwrite in your preferred framework quickly and easily.';
|
|
const ogImage = DEFAULT_HOST + '/images/open-graph/docs.png';
|
|
|
|
let { data } = $props();
|
|
|
|
type MappedTutorial = (typeof data.tutorials)[number]['tutorials'][number];
|
|
|
|
const iconMap: Record<string, string> = {
|
|
'react native': 'icon-react-native',
|
|
react: 'icon-react',
|
|
vue: 'web-icon-vue',
|
|
angular: 'icon-angular',
|
|
svelte: 'icon-svelte',
|
|
sveltekit: 'icon-svelte',
|
|
'sveltekit ssr': 'icon-svelte',
|
|
android: 'icon-android',
|
|
apple: 'icon-apple',
|
|
flutter: 'icon-flutter',
|
|
nuxt: 'icon-nuxt',
|
|
'nuxt ssr': 'icon-nuxt',
|
|
stripe: 'icon-stripe',
|
|
refine: 'web-icon-refine',
|
|
'next.js': 'icon-nextjs',
|
|
'next.js ssr': 'icon-nextjs',
|
|
astro: 'icon-astro',
|
|
'astro ssr': 'icon-astro'
|
|
};
|
|
|
|
const getIcon = (tutorial: MappedTutorial) => {
|
|
if (!tutorial.framework) return ''; // TODO: Default icon
|
|
return iconMap[tutorial.framework.toLowerCase()];
|
|
};
|
|
</script>
|
|
|
|
<svelte:head>
|
|
<!-- Titles -->
|
|
<title>{title}</title>
|
|
<meta property="og:title" content={title} />
|
|
<meta name="twitter:title" content={title} />
|
|
<!-- Description -->
|
|
<meta name="description" content={description} />
|
|
<meta property="og:description" content={description} />
|
|
<meta name="twitter:description" content={description} />
|
|
<!-- Image -->
|
|
<meta property="og:image" content={ogImage} />
|
|
<meta property="og:image:width" content="1200" />
|
|
<meta property="og:image:height" content="630" />
|
|
<meta name="twitter:image" content={ogImage} />
|
|
<meta name="twitter:card" content="summary_large_image" />
|
|
</svelte:head>
|
|
|
|
<main class="web-main-section" id="main">
|
|
<article class="web-article">
|
|
<header class="web-article-header">
|
|
<div class="web-article-header-start web-u-cross-start flex flex-col">
|
|
<div class="relative flex items-center">
|
|
<h1 class="text-title font-aeonik-pro">Tutorials</h1>
|
|
</div>
|
|
</div>
|
|
<div class="web-article-header-end"></div>
|
|
</header>
|
|
<div class="web-article-content web-u-gap-80">
|
|
{#each data.tutorials as category}
|
|
<section class="flex flex-col gap-6">
|
|
<h2 class="text-micro uppercase">{category.title}</h2>
|
|
<ul class="web-grid-row-4 web-grid-row-4-mobile-1">
|
|
{#each category.tutorials as tutorial}
|
|
<li>
|
|
{#if tutorial.draft === true}
|
|
<a
|
|
href={tutorial.href}
|
|
class="web-card is-normal draft"
|
|
aria-disabled="true"
|
|
tabindex="-1"
|
|
>
|
|
<header class="flex items-baseline gap-1">
|
|
<span
|
|
class="{getIcon(tutorial)} web-u-font-size-24"
|
|
aria-hidden="true"
|
|
></span>
|
|
<h3 class="text-sub-body text-primary font-medium">
|
|
{tutorial.framework}
|
|
</h3>
|
|
<span class="badge text-caption">Coming Soon</span>
|
|
</header>
|
|
</a>
|
|
{:else}
|
|
<a href={tutorial.href} class="web-card is-normal">
|
|
<header class="flex items-baseline gap-1">
|
|
<span
|
|
class="{getIcon(tutorial)} web-u-font-size-24"
|
|
aria-hidden="true"
|
|
></span>
|
|
<h3 class="text-sub-body text-primary font-medium">
|
|
{tutorial.framework}
|
|
</h3>
|
|
</header>
|
|
<p class="text-sub-body mt-1">
|
|
{tutorial.title}
|
|
</p>
|
|
</a>
|
|
{/if}
|
|
</li>
|
|
{/each}
|
|
</ul>
|
|
</section>
|
|
{/each}
|
|
</div>
|
|
</article>
|
|
|
|
<MainFooter variant="docs" />
|
|
</main>
|
|
|
|
<style lang="scss">
|
|
.badge {
|
|
border-radius: 0.25rem;
|
|
background: rgba(253, 54, 110, 0.24);
|
|
padding: 0.0625rem 0.375rem;
|
|
margin-inline-start: 0.25rem;
|
|
}
|
|
|
|
.draft {
|
|
opacity: 0.4;
|
|
pointer-events: none;
|
|
}
|
|
</style>
|