it-tools/src/layouts/base.layout.vue

307 lines
7.2 KiB
Vue
Raw Normal View History

2022-04-04 00:24:45 +02:00
<script lang="ts" setup>
import { NIcon } from 'naive-ui';
import { h, ref, type Component } from 'vue';
import { RouterLink, useRoute } from 'vue-router';
2022-04-15 12:21:09 +02:00
import { Heart, BrandGithub, BrandTwitter, Moon, Sun, Menu2, Home2 } from '@vicons/tabler'
2022-04-06 00:49:56 +02:00
import { toolsByCategory } from '@/tools';
2022-04-04 01:03:06 +02:00
import SearchBar from '../components/SearchBar.vue';
2022-04-04 01:52:59 +02:00
import { useStyleStore } from '@/stores/style.store';
2022-04-14 22:41:51 +02:00
import HeroGradient from '../assets/hero-gradient.svg?component'
import { useThemeVars } from 'naive-ui'
2022-04-15 12:21:09 +02:00
import MenuLayout from '../components/MenuLayout.vue'
2022-04-04 00:24:45 +02:00
2022-04-14 22:41:51 +02:00
const themeVars = useThemeVars()
2022-04-04 00:24:45 +02:00
const activeKey = ref(null)
const route = useRoute()
2022-04-04 01:52:59 +02:00
const styleStore = useStyleStore()
2022-04-04 00:24:45 +02:00
const makeLabel = (text: string, to: string) => () => h(RouterLink, { to }, { default: () => text })
const makeIcon = (icon: Component) => () => h(NIcon, null, { default: () => h(icon) })
2022-04-06 00:49:56 +02:00
const m = toolsByCategory.map(category => ({
label: category.name,
key: category.name,
type: 'group',
children: category.components.map(({ name, path, icon }) => ({
label: makeLabel(name, path),
2022-04-07 22:14:18 +02:00
icon: makeIcon(icon),
key: name
2022-04-06 00:49:56 +02:00
}))
2022-04-04 00:24:45 +02:00
}))
2022-04-06 00:49:56 +02:00
2022-03-31 00:33:29 +02:00
</script>
<template>
2022-04-15 23:10:47 +02:00
<menu-layout
class="menu-layout"
:class="{ isSmallScreen: styleStore.isSmallScreen }"
>
<template #sider>
<router-link
to="/"
class="hero-wrapper"
>
<hero-gradient class="gradient" />
<div class="text-wrapper">
<div class="title">
IT - TOOLS
</div>
<div class="divider" />
<div class="subtitle">
Handy tools for developers
</div>
</div>
</router-link>
<div class="sider-content">
<n-space
v-if="styleStore.isSmallScreen"
justify="center"
>
<n-button
size="large"
circle
quaternary
tag="a"
href="https://github.com/CorentinTh/it-tools"
rel="noopener"
target="_blank"
>
<n-icon
size="25"
:component="BrandGithub"
/>
</n-button>
<n-button
size="large"
circle
quaternary
tag="a"
href="https://twitter.com/cthmsst"
rel="noopener"
target="_blank"
>
<n-icon
size="25"
:component="BrandTwitter"
/>
</n-button>
<n-button
size="large"
circle
quaternary
@click="styleStore.isDarkTheme = !styleStore.isDarkTheme"
>
<n-icon
v-if="styleStore.isDarkTheme"
size="25"
:component="Sun"
/>
<n-icon
v-else
size="25"
:component="Moon"
/>
</n-button>
</n-space>
<n-menu
v-model:value="activeKey"
:value="route.name"
class="menu"
:collapsed-width="64"
:collapsed-icon-size="22"
:options="m"
:indent="20"
/>
</div>
</template>
<template #content>
<div class="navigation">
<n-button
:size="styleStore.isSmallScreen ? 'medium' : 'large'"
circle
quaternary
aria-label="Toogle menu"
@click="styleStore.isMenuCollapsed = !styleStore.isMenuCollapsed"
>
<n-icon
size="25"
:component="Menu2"
/>
</n-button>
<router-link
to="/"
#="{ navigate, href }"
custom
>
<n-button
tag="a"
:href="href"
:size="styleStore.isSmallScreen ? 'medium' : 'large'"
circle
quaternary
aria-label="Home"
@click="navigate"
>
<n-icon
size="25"
:component="Home2"
/>
</n-button>
</router-link>
<search-bar />
<n-button
type="primary"
tag="a"
href="https://github.com/sponsors/CorentinTh"
rel="noopener"
target="_blank"
>
<n-icon
v-if="!styleStore.isSmallScreen"
:component="Heart"
style="margin-right: 5px;"
/>
Sponsor
</n-button>
<n-button
v-if="!styleStore.isSmallScreen"
size="large"
circle
quaternary
tag="a"
href="https://github.com/CorentinTh/it-tools"
rel="noopener"
target="_blank"
aria-label="Github repository"
>
<n-icon
size="25"
:component="BrandGithub"
/>
</n-button>
<n-button
v-if="!styleStore.isSmallScreen"
size="large"
circle
quaternary
tag="a"
href="https://twitter.com/cthmsst"
rel="noopener"
target="_blank"
aria-label="Twitter account"
>
<n-icon
size="25"
:component="BrandTwitter"
/>
</n-button>
<n-button
v-if="!styleStore.isSmallScreen"
size="large"
circle
quaternary
aria-label="Toogle theme"
@click="styleStore.isDarkTheme = !styleStore.isDarkTheme"
>
<n-icon
v-if="styleStore.isDarkTheme"
size="25"
:component="Sun"
/>
<n-icon
v-else
size="25"
:component="Moon"
/>
</n-button>
</div>
<slot />
</template>
</menu-layout>
2022-03-31 00:33:29 +02:00
</template>
<style lang="less" scoped>
2022-04-14 23:12:36 +02:00
// ::v-deep(.n-layout-scroll-container) {
// @percent: 4%;
// @position: 25px;
// @size: 50px;
// @color: #eeeeee25;
// background-image: radial-gradient(@color @percent, transparent @percent),
// radial-gradient(@color @percent, transparent @percent);
// background-position: 0 0, @position @position;
// background-size: @size @size;
// }
2022-04-15 12:21:09 +02:00
.sider-content {
2022-04-14 22:41:51 +02:00
padding-top: 160px;
padding-bottom: 200px;
}
.hero-wrapper {
position: absolute;
display: block;
position: absolute;
left: 0;
width: 100%;
z-index: 10;
2022-04-15 12:21:09 +02:00
overflow: hidden;
2022-04-14 22:41:51 +02:00
.gradient {
2022-04-15 12:21:09 +02:00
margin-top: -65px;
2022-04-14 22:41:51 +02:00
}
.text-wrapper {
position: absolute;
left: 0;
width: 100%;
text-align: center;
top: 16px;
color: #fff;
.title {
font-size: 25px;
font-weight: 600;
}
.divider {
width: 50px;
height: 2px;
border-radius: 4px;
background-color: v-bind('themeVars.primaryColor');
margin: 0 auto 5px;
}
.subtitle {
font-size: 16px;
}
}
}
2022-04-15 12:21:09 +02:00
.navigation {
2022-04-04 00:24:45 +02:00
display: flex;
2022-04-14 22:41:51 +02:00
align-items: center;
justify-content: center;
2022-04-15 12:21:09 +02:00
flex-direction: row;
2022-04-14 18:45:47 +02:00
&>*:not(:first-child) {
2022-04-15 12:21:09 +02:00
margin-left: 10px;
2022-04-14 18:45:47 +02:00
2022-04-15 12:21:09 +02:00
.isSmallScreen & {
margin-left: 5px;
}
2022-04-04 00:24:45 +02:00
}
2022-04-14 18:45:47 +02:00
2022-04-15 12:21:09 +02:00
.search-bar {
// width: 100%;
flex-grow: 1;
2022-04-04 00:24:45 +02:00
}
}
2022-03-31 00:33:29 +02:00
</style>