cobalt/web/src/routes/+layout.svelte

319 lines
7.9 KiB
Svelte
Raw Normal View History

<script lang="ts">
2024-06-14 11:33:01 +01:00
import "@fontsource/ibm-plex-mono/400.css";
import "@fontsource/ibm-plex-mono/500.css";
2024-06-29 15:09:17 +01:00
import settings from "$lib/settings";
import { device, app } from "$lib/device";
import currentTheme, { statusBarColors } from "$lib/state/theme";
2024-06-29 15:09:17 +01:00
import Sidebar from "$components/sidebar/Sidebar.svelte";
import NotchSticker from "$components/misc/NotchSticker.svelte";
$: reduceMotion =
$settings.appearance.reduceMotion
|| device.prefers.reducedMotion;
$: reduceTransparency =
$settings.appearance.reduceTransparency
|| device.prefers.reducedTransparency;
2024-06-14 11:33:01 +01:00
</script>
<svelte:head>
{#if device.is.mobile}
<meta name="theme-color" content={statusBarColors[$currentTheme]}>
{/if}
</svelte:head>
2024-06-29 15:09:17 +01:00
<div style="display: contents" data-theme={$currentTheme}>
<div
id="cobalt"
data-iphone={device.is.iPhone}
data-reduce-motion={reduceMotion}
data-reduce-transparency={reduceTransparency}
>
{#if device.is.iPhone && app.is.installed}
<NotchSticker />
{/if}
2024-06-29 15:09:17 +01:00
<Sidebar />
<div id="content">
<slot></slot>
</div>
2024-06-14 11:33:01 +01:00
</div>
</div>
<style>
:global(:root) {
2024-06-14 16:48:57 +01:00
--primary: #ffffff;
--secondary: #000000;
--white: #ffffff;
--gray: #75757e;
--blue: #2f8af9;
--green: #51cf5e;
2024-06-14 16:48:57 +01:00
--button: #f4f4f4;
--button-hover: #e8e8e8;
--button-hover-transparent: rgba(0, 0, 0, 0.06);
--button-stroke: rgba(0, 0, 0, 0.05);
2024-06-17 13:46:52 +01:00
--button-text: #282828;
--button-box-shadow: 0 0 0 1.5px var(--button-stroke) inset;
2024-06-14 16:48:57 +01:00
--sidebar-bg: #000000;
--sidebar-highlight: #ffffff;
--sidebar-hover: rgba(255, 255, 255, 0.1);
--input-border: #adadb7;
--toggle-bg: var(--input-border);
--toggle-bg-enabled: var(--secondary);
2024-06-14 16:48:57 +01:00
--padding: 12px;
--border-radius: 11px;
2024-06-14 16:48:57 +01:00
2024-06-14 11:33:01 +01:00
--sidebar-width: 80px;
--sidebar-font-size: 11px;
--sidebar-inner-padding: 4px;
--sidebar-height-mobile: calc(
50px
+ calc(var(--sidebar-inner-padding) * 2)
+ env(safe-area-inset-bottom)
);
--safe-area-inset-top: env(safe-area-inset-top);
--switcher-padding: var(--sidebar-inner-padding);
--sidebar-mobile-gradient: linear-gradient(
90deg,
rgba(0, 0, 0, 0.9) 0%,
rgba(0, 0, 0, 0) 4%,
rgba(0, 0, 0, 0) 50%,
rgba(0, 0, 0, 0) 96%,
rgba(0, 0, 0, 0.9) 100%
);
}
:global([data-theme="dark"]) {
2024-06-29 15:09:17 +01:00
--primary: #000000;
--secondary: #e1e1e1;
--gray: #818181;
2024-06-29 15:09:17 +01:00
--blue: #2a7ce1;
--green: #37aa42;
--button: #191919;
--button-hover: #2a2a2a;
--button-hover-transparent: rgba(225, 225, 225, 0.1);
--button-stroke: rgba(255, 255, 255, 0.05);
--button-text: #e1e1e1;
--button-box-shadow: 0 0 0 1.5px var(--button-stroke) inset;
--sidebar-bg: #101010;
--sidebar-highlight: #f2f2f2;
--input-border: #383838;
--toggle-bg: var(--input-border);
--toggle-bg-enabled: #777777;
2024-06-29 15:09:17 +01:00
--sidebar-mobile-gradient: linear-gradient(
90deg,
rgba(16, 16, 16, 0.9) 0%,
rgba(16, 16, 16, 0) 4%,
rgba(16, 16, 16, 0) 50%,
rgba(16, 16, 16, 0) 96%,
rgba(16, 16, 16, 0.9) 100%
);
2024-06-14 11:33:01 +01:00
}
2024-06-14 11:38:10 +01:00
2024-06-14 11:33:01 +01:00
:global(html),
:global(body) {
margin: 0;
2024-06-25 11:14:54 +01:00
height: 100vh;
overflow: hidden;
overscroll-behavior-y: none;
2024-06-14 16:48:57 +01:00
}
#cobalt {
position: fixed;
height: 100%;
width: 100%;
display: grid;
grid-template-columns: calc(
var(--sidebar-width) + var(--sidebar-inner-padding) * 2
) 1fr;
overflow: hidden;
background-color: var(--sidebar-bg);
color: var(--secondary);
}
/* add padding for notch / dynamic island in landscape */
@media screen and (orientation: landscape) {
#cobalt[data-iphone="true"] {
grid-template-columns:
calc(
var(--sidebar-width)
+ var(--sidebar-inner-padding) * 2
+ env(safe-area-inset-left)
)
1fr;
}
#cobalt[data-iphone="true"] #content {
padding-right: env(safe-area-inset-right);
}
}
#content {
display: flex;
overflow: scroll;
background-color: var(--primary);
border-top-left-radius: var(--border-radius);
border-bottom-left-radius: var(--border-radius);
}
@media screen and (max-width: 535px) {
#cobalt {
display: grid;
grid-template-columns: unset;
grid-template-rows: 1fr var(--sidebar-height-mobile);
}
#content {
padding-top: env(safe-area-inset-top);
order: -1;
border-top-left-radius: 0;
border-bottom-left-radius: calc(var(--border-radius) * 2);
border-bottom-right-radius: calc(var(--border-radius) * 2);
}
}
2024-06-14 16:48:57 +01:00
:global(*) {
font-family: "IBM Plex Mono", "Noto Sans Mono Variable",
"Noto Sans Mono", monospace;
user-select: none;
2024-06-16 14:25:30 +01:00
scrollbar-width: none;
-webkit-user-select: none;
2024-06-14 16:48:57 +01:00
-webkit-user-drag: none;
2024-06-16 14:25:30 +01:00
-webkit-tap-highlight-color: transparent;
}
:global(::-webkit-scrollbar) {
display: none;
2024-06-14 11:33:01 +01:00
}
2024-06-14 11:38:10 +01:00
2024-06-14 11:33:01 +01:00
:global(a) {
color: inherit;
text-underline-offset: 3px;
-webkit-touch-callout: none;
2024-06-14 11:33:01 +01:00
}
2024-06-14 11:38:10 +01:00
:global(a:visited) {
color: inherit;
}
2024-06-14 16:48:57 +01:00
:global(svg),
:global(img) {
pointer-events: none;
}
:global(button, .button) {
2024-06-14 16:48:57 +01:00
display: flex;
align-items: center;
justify-content: center;
2024-06-17 13:46:21 +01:00
padding: 6px 13px;
gap: 6px;
2024-06-14 16:48:57 +01:00
border: none;
border-radius: var(--border-radius);
2024-06-17 13:46:21 +01:00
font-size: 14.5px;
2024-06-14 16:48:57 +01:00
cursor: pointer;
background-color: var(--button);
2024-06-17 13:46:52 +01:00
color: var(--button-text);
box-shadow: var(--button-box-shadow);
}
2024-06-24 18:22:19 +01:00
:global(:focus-visible) {
box-shadow: 0 0 0 2px var(--blue) inset;
outline: none;
z-index: 1;
}
:global(button:active, .button:active) {
background-color: var(--button-hover);
2024-06-14 16:48:57 +01:00
}
:global(.button.active) {
background: var(--secondary);
color: var(--primary);
}
2024-06-24 18:22:19 +01:00
/* important is used because active class is toggled by state */
/* and added to the end of the list, taking priority */
:global(.active:focus-visible) {
background: var(--blue) !important;
color: var(--sidebar-highlight) !important;
}
@media (hover: hover) {
:global(button:hover) {
background-color: var(--button-hover);
}
2024-06-14 16:48:57 +01:00
}
:global(.center-column-container) {
display: flex;
width: 100%;
flex-direction: column;
align-items: center;
justify-content: center;
}
:global(button) {
2024-06-24 15:26:45 +01:00
font-weight: 500;
}
:global(h1, h2, h3, h4, h5, h6) {
font-weight: 500;
margin-block: 0;
}
:global(h1) {
font-size: 24px;
letter-spacing: -1px;
}
:global(h2) {
font-size: 20px;
letter-spacing: -1px;
}
:global(h3) {
font-size: 16px;
}
:global(h4) {
font-size: 14.5px;
}
:global(h5) {
2024-06-24 15:26:45 +01:00
font-size: 12px;
}
:global(h6) {
font-size: 11px;
}
2024-06-24 15:26:45 +01:00
:global(.subtext) {
font-size: 12.5px;
font-weight: 500;
2024-06-24 15:26:45 +01:00
color: var(--gray);
line-height: 1.4;
padding: 0 var(--padding);
white-space: pre-line;
2024-06-24 15:26:45 +01:00
}
[data-reduce-motion="true"] :global(*) {
animation: none !important;
transition: none !important;
}
2024-06-14 11:33:01 +01:00
</style>