Compare commits
49 Commits
nim65s-mul
...
remove-mod
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
84dab9e329 | ||
|
|
ef02fca062 | ||
|
|
5f90d60bd6 | ||
|
|
02ed516a15 | ||
|
|
14a6af1259 | ||
|
|
fd26dcd0d0 | ||
|
|
3b316cbf3e | ||
|
|
d9657d8617 | ||
|
|
74fb95653a | ||
|
|
6073ab4a0b | ||
|
|
9e35d040da | ||
|
|
0e2904b34b | ||
|
|
71ba979120 | ||
|
|
fb4b8f2745 | ||
|
|
0028311805 | ||
|
|
abacd19c12 | ||
|
|
9446009738 | ||
|
|
dc7566951c | ||
|
|
67d2e18fb8 | ||
|
|
350b55ea16 | ||
|
|
ce50278621 | ||
|
|
d9262e47cb | ||
|
|
ce411b4784 | ||
|
|
55f0da45a8 | ||
|
|
dc7b10efe2 | ||
|
|
3687fd48ce | ||
|
|
a4d26497f9 | ||
|
|
1c14033d48 | ||
|
|
13ca0f5050 | ||
|
|
85f219ca1e | ||
|
|
503bb62864 | ||
|
|
c0225ea757 | ||
|
|
d6b27d8740 | ||
|
|
c09cb834fa | ||
|
|
91434044e0 | ||
|
|
56a76242ca | ||
|
|
7255673440 | ||
|
|
5b651752ba | ||
|
|
b1aa79b33f | ||
|
|
26fdfeec00 | ||
|
|
a0c3b6d33e | ||
|
|
8be8af9117 | ||
|
|
338a6ad340 | ||
|
|
38fafba6d8 | ||
|
|
2dcc4bba15 | ||
|
|
b57897c6c9 | ||
|
|
b640be3dd2 | ||
|
|
7266e4b273 | ||
|
|
546eeb22d0 |
@@ -39,7 +39,7 @@ in
|
|||||||
};
|
};
|
||||||
|
|
||||||
perInstance =
|
perInstance =
|
||||||
{ instanceName, settings, ... }:
|
{ settings, ... }:
|
||||||
{
|
{
|
||||||
nixosModule =
|
nixosModule =
|
||||||
{ pkgs, config, ... }:
|
{ pkgs, config, ... }:
|
||||||
@@ -86,7 +86,7 @@ in
|
|||||||
|
|
||||||
# service to generate the environment file containing all secrets, as
|
# service to generate the environment file containing all secrets, as
|
||||||
# expected by the nixos NetworkManager-ensure-profile service
|
# expected by the nixos NetworkManager-ensure-profile service
|
||||||
systemd.services."NetworkManager-setup-secrets-${instanceName}" = {
|
systemd.services.NetworkManager-setup-secrets = {
|
||||||
description = "Generate wifi secrets for NetworkManager";
|
description = "Generate wifi secrets for NetworkManager";
|
||||||
requiredBy = [ "NetworkManager-ensure-profiles.service" ];
|
requiredBy = [ "NetworkManager-ensure-profiles.service" ];
|
||||||
partOf = [ "NetworkManager-ensure-profiles.service" ];
|
partOf = [ "NetworkManager-ensure-profiles.service" ];
|
||||||
|
|||||||
@@ -7,16 +7,8 @@
|
|||||||
inventory = {
|
inventory = {
|
||||||
|
|
||||||
machines.test = { };
|
machines.test = { };
|
||||||
machines.second = { };
|
|
||||||
|
|
||||||
instances = {
|
instances = {
|
||||||
wg-test-all = {
|
|
||||||
module.name = "@clan/wifi";
|
|
||||||
module.input = "self";
|
|
||||||
roles.default.tags.all = { };
|
|
||||||
roles.default.settings.networks.all = { };
|
|
||||||
};
|
|
||||||
|
|
||||||
wg-test-one = {
|
wg-test-one = {
|
||||||
module.name = "@clan/wifi";
|
module.name = "@clan/wifi";
|
||||||
module.input = "self";
|
module.input = "self";
|
||||||
|
|||||||
6
flake.lock
generated
6
flake.lock
generated
@@ -31,11 +31,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1752541678,
|
"lastModified": 1752718651,
|
||||||
"narHash": "sha256-dyhGzkld6jPqnT/UfGV2oqe7tYn7hppAqFvF3GZTyXY=",
|
"narHash": "sha256-PkaR0qmyP9q/MDN3uYa+RLeBA0PjvEQiM0rTDDBXkL8=",
|
||||||
"owner": "nix-community",
|
"owner": "nix-community",
|
||||||
"repo": "disko",
|
"repo": "disko",
|
||||||
"rev": "2bf3421f7fed5c84d9392b62dcb9d76ef09796a7",
|
"rev": "d5ad4485e6f2edcc06751df65c5e16572877db88",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
|||||||
@@ -30,7 +30,6 @@
|
|||||||
inputs = {
|
inputs = {
|
||||||
flake-parts.follows = "flake-parts";
|
flake-parts.follows = "flake-parts";
|
||||||
nixpkgs.follows = "nixpkgs";
|
nixpkgs.follows = "nixpkgs";
|
||||||
systems.follows = "systems";
|
|
||||||
treefmt-nix.follows = "treefmt-nix";
|
treefmt-nix.follows = "treefmt-nix";
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -7,8 +7,29 @@
|
|||||||
}:
|
}:
|
||||||
rec {
|
rec {
|
||||||
buildClan =
|
buildClan =
|
||||||
# TODO: Once all templates and docs are migrated add: lib.warn "'buildClan' is deprecated. Use 'clan-core.lib.clan' instead"
|
module:
|
||||||
module: (clan module).config;
|
lib.warn ''
|
||||||
|
==================== DEPRECATION NOTICE ====================
|
||||||
|
Please migrate
|
||||||
|
from: 'clan = inputs.<clan-core>.lib.buildClan'
|
||||||
|
to : 'clan = inputs.<clan-core>.lib.clan'
|
||||||
|
in your flake.nix.
|
||||||
|
|
||||||
|
Please also migrate
|
||||||
|
from: 'inherit (clan) nixosConfigurations clanInternals; '
|
||||||
|
to : "
|
||||||
|
inherit (clan.config) nixosConfigurations clanInternals;
|
||||||
|
clan = clan.config;
|
||||||
|
"
|
||||||
|
in your flake.nix.
|
||||||
|
|
||||||
|
Reason:
|
||||||
|
- Improves consistency between flake-parts and non-flake-parts users.
|
||||||
|
|
||||||
|
- It also allows us to use the top level attribute 'clan' to expose
|
||||||
|
attributes that can be used for cross-clan functionality.
|
||||||
|
============================================================
|
||||||
|
'' (clan module).config;
|
||||||
|
|
||||||
clan =
|
clan =
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -48,6 +48,7 @@ in
|
|||||||
{
|
{
|
||||||
options = {
|
options = {
|
||||||
instances = lib.mkOption {
|
instances = lib.mkOption {
|
||||||
|
default = { };
|
||||||
# instances.<instanceName>...
|
# instances.<instanceName>...
|
||||||
type = types.attrsOf (submoduleWith {
|
type = types.attrsOf (submoduleWith {
|
||||||
modules = [
|
modules = [
|
||||||
@@ -57,6 +58,7 @@ in
|
|||||||
};
|
};
|
||||||
# instances.<machineName>...
|
# instances.<machineName>...
|
||||||
machines = lib.mkOption {
|
machines = lib.mkOption {
|
||||||
|
default = { };
|
||||||
type = types.attrsOf (submoduleWith {
|
type = types.attrsOf (submoduleWith {
|
||||||
modules = [
|
modules = [
|
||||||
config.exportsModule
|
config.exportsModule
|
||||||
|
|||||||
@@ -21,6 +21,12 @@ buildNpmPackage (_finalAttrs: {
|
|||||||
mkdir -p api
|
mkdir -p api
|
||||||
cp -r ${clan-ts-api}/* api
|
cp -r ${clan-ts-api}/* api
|
||||||
cp -r ${fonts} ".fonts"
|
cp -r ${fonts} ".fonts"
|
||||||
|
|
||||||
|
# only needed for the next couple weeks to make sure this file doesn't make it back into the git history
|
||||||
|
if [[ -f "${./ui}/src/routes/Onboarding/background.jpg" ]]; then
|
||||||
|
echo "background.jpg found, exiting"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
'';
|
'';
|
||||||
|
|
||||||
# todo figure out why this fails only inside of Nix
|
# todo figure out why this fails only inside of Nix
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import type { StorybookConfig } from "@kachurun/storybook-solid-vite";
|
|||||||
|
|
||||||
const config: StorybookConfig = {
|
const config: StorybookConfig = {
|
||||||
framework: "@kachurun/storybook-solid-vite",
|
framework: "@kachurun/storybook-solid-vite",
|
||||||
stories: ["../src/components/**/*.mdx", "../src/components/**/*.stories.tsx"],
|
stories: ["../src/**/*.mdx", "../src/**/*.stories.tsx"],
|
||||||
addons: [
|
addons: [
|
||||||
"@storybook/addon-links",
|
"@storybook/addon-links",
|
||||||
"@storybook/addon-docs",
|
"@storybook/addon-docs",
|
||||||
|
|||||||
@@ -138,6 +138,10 @@
|
|||||||
transition: all 0.5s ease;
|
transition: all 0.5s ease;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
& > span.typography {
|
||||||
|
@apply max-w-full overflow-hidden whitespace-nowrap text-ellipsis;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* button group */
|
/* button group */
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ export type Story = StoryObj<typeof meta>;
|
|||||||
export const Bare: Story = {
|
export const Bare: Story = {
|
||||||
args: {
|
args: {
|
||||||
onSelectFile: async () => {
|
onSelectFile: async () => {
|
||||||
return "/home/bob/clans/my-clan";
|
return "/home/github/clans/my-clan/foo/bar/baz/fizz/buzz";
|
||||||
},
|
},
|
||||||
input: {
|
input: {
|
||||||
placeholder: "e.g. 11/06/89",
|
placeholder: "e.g. 11/06/89",
|
||||||
|
|||||||
@@ -12,6 +12,8 @@ import { PolymorphicProps } from "@kobalte/core/polymorphic";
|
|||||||
import { FieldProps } from "./Field";
|
import { FieldProps } from "./Field";
|
||||||
import { Orienter } from "./Orienter";
|
import { Orienter } from "./Orienter";
|
||||||
import { createSignal } from "solid-js";
|
import { createSignal } from "solid-js";
|
||||||
|
import { Tooltip } from "@kobalte/core/tooltip";
|
||||||
|
import { Typography } from "@/src/components/Typography/Typography";
|
||||||
|
|
||||||
export type HostFileInputProps = FieldProps &
|
export type HostFileInputProps = FieldProps &
|
||||||
TextFieldRootProps & {
|
TextFieldRootProps & {
|
||||||
@@ -20,10 +22,21 @@ export type HostFileInputProps = FieldProps &
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const HostFileInput = (props: HostFileInputProps) => {
|
export const HostFileInput = (props: HostFileInputProps) => {
|
||||||
const [value, setValue] = createSignal<string | undefined>(undefined);
|
const [value, setValue] = createSignal<string>(props.value || "");
|
||||||
|
|
||||||
|
let actualInputElement: HTMLInputElement | undefined;
|
||||||
|
|
||||||
const selectFile = async () => {
|
const selectFile = async () => {
|
||||||
setValue(await props.onSelectFile());
|
try {
|
||||||
|
console.log("selecting file", props.onSelectFile);
|
||||||
|
setValue(await props.onSelectFile());
|
||||||
|
actualInputElement?.dispatchEvent(
|
||||||
|
new Event("input", { bubbles: true, cancelable: true }),
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error selecting file", error);
|
||||||
|
// todo work out how to display the error
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -33,8 +46,6 @@ export const HostFileInput = (props: HostFileInputProps) => {
|
|||||||
ghost: props.ghost,
|
ghost: props.ghost,
|
||||||
})}
|
})}
|
||||||
{...props}
|
{...props}
|
||||||
value={value()}
|
|
||||||
onChange={setValue}
|
|
||||||
>
|
>
|
||||||
<Orienter orientation={props.orientation} align={"start"}>
|
<Orienter orientation={props.orientation} align={"start"}>
|
||||||
<Label
|
<Label
|
||||||
@@ -43,16 +54,54 @@ export const HostFileInput = (props: HostFileInputProps) => {
|
|||||||
{...props}
|
{...props}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<TextField.Input {...props.input} hidden={true} />
|
<TextField.Input
|
||||||
|
{...props.input}
|
||||||
|
hidden={true}
|
||||||
|
value={value()}
|
||||||
|
ref={(el: HTMLInputElement) => {
|
||||||
|
actualInputElement = el; // Capture for local use
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
|
||||||
<Button
|
{!value() && (
|
||||||
hierarchy="secondary"
|
<Button
|
||||||
size={props.size}
|
hierarchy="secondary"
|
||||||
startIcon="Folder"
|
size={props.size}
|
||||||
onClick={selectFile}
|
startIcon="Folder"
|
||||||
>
|
onClick={selectFile}
|
||||||
{value() ? value() : "No Selection"}
|
disabled={props.disabled || props.readOnly}
|
||||||
</Button>
|
>
|
||||||
|
No Selection
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{value() && (
|
||||||
|
<Tooltip placement="top">
|
||||||
|
<Tooltip.Portal>
|
||||||
|
<Tooltip.Content class="tooltip-content">
|
||||||
|
<Typography
|
||||||
|
hierarchy="body"
|
||||||
|
size="xs"
|
||||||
|
weight="medium"
|
||||||
|
inverted={!props.inverted}
|
||||||
|
>
|
||||||
|
{value()}
|
||||||
|
</Typography>
|
||||||
|
<Tooltip.Arrow />
|
||||||
|
</Tooltip.Content>
|
||||||
|
</Tooltip.Portal>
|
||||||
|
<Tooltip.Trigger
|
||||||
|
as={Button}
|
||||||
|
hierarchy="secondary"
|
||||||
|
size={props.size}
|
||||||
|
startIcon="Folder"
|
||||||
|
onClick={selectFile}
|
||||||
|
disabled={props.disabled || props.readOnly}
|
||||||
|
>
|
||||||
|
{value()}
|
||||||
|
</Tooltip.Trigger>
|
||||||
|
</Tooltip>
|
||||||
|
)}
|
||||||
</Orienter>
|
</Orienter>
|
||||||
</TextField>
|
</TextField>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -22,40 +22,3 @@ div.form-label {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
div.tooltip-content {
|
|
||||||
@apply z-50 px-2 py-0.5 bg-inv-4 rounded-[0.125rem] leading-none;
|
|
||||||
|
|
||||||
max-width: min(calc(100vw - 16px), 380px);
|
|
||||||
transform-origin: var(--kb-tooltip-content-transform-origin);
|
|
||||||
animation: tooltipHide 250ms ease-in forwards;
|
|
||||||
|
|
||||||
&[data-expanded] {
|
|
||||||
animation: tooltipShow 250ms ease-out;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.inverted {
|
|
||||||
@apply bg-def-2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@keyframes tooltipShow {
|
|
||||||
from {
|
|
||||||
opacity: 0;
|
|
||||||
transform: scale(0.96);
|
|
||||||
}
|
|
||||||
to {
|
|
||||||
opacity: 1;
|
|
||||||
transform: scale(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@keyframes tooltipHide {
|
|
||||||
from {
|
|
||||||
opacity: 1;
|
|
||||||
transform: scale(1);
|
|
||||||
}
|
|
||||||
to {
|
|
||||||
opacity: 0;
|
|
||||||
transform: scale(0.96);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
import { Show } from "solid-js";
|
import { Show } from "solid-js";
|
||||||
import { Typography } from "@/src/components/Typography/Typography";
|
import { Typography } from "@/src/components/Typography/Typography";
|
||||||
import { Tooltip as KTooltip } from "@kobalte/core/tooltip";
|
import { Tooltip } from "@/src/components/Tooltip/Tooltip";
|
||||||
import Icon from "@/src/components/Icon/Icon";
|
import Icon from "@/src/components/Icon/Icon";
|
||||||
import { TextField } from "@kobalte/core/text-field";
|
import { TextField } from "@kobalte/core/text-field";
|
||||||
import { Checkbox } from "@kobalte/core/checkbox";
|
import { Checkbox } from "@kobalte/core/checkbox";
|
||||||
import { Combobox } from "@kobalte/core/combobox";
|
import { Combobox } from "@kobalte/core/combobox";
|
||||||
import "./Label.css";
|
import "./Label.css";
|
||||||
import cx from "classnames";
|
|
||||||
|
|
||||||
export type Size = "default" | "s";
|
export type Size = "default" | "s";
|
||||||
|
|
||||||
@@ -49,31 +48,27 @@ export const Label = (props: LabelProps) => {
|
|||||||
{props.label}
|
{props.label}
|
||||||
</Typography>
|
</Typography>
|
||||||
{props.tooltip && (
|
{props.tooltip && (
|
||||||
<KTooltip placement="top">
|
<Tooltip
|
||||||
<KTooltip.Trigger>
|
placement="top"
|
||||||
|
inverted={props.inverted}
|
||||||
|
trigger={
|
||||||
<Icon
|
<Icon
|
||||||
icon="Info"
|
icon="Info"
|
||||||
color="tertiary"
|
color="tertiary"
|
||||||
inverted={props.inverted}
|
inverted={props.inverted}
|
||||||
size={props.size == "default" ? "0.85em" : "0.75rem"}
|
size={props.size == "default" ? "0.85em" : "0.75rem"}
|
||||||
/>
|
/>
|
||||||
<KTooltip.Portal>
|
}
|
||||||
<KTooltip.Content
|
>
|
||||||
class={cx("tooltip-content", { inverted: props.inverted })}
|
<Typography
|
||||||
>
|
hierarchy="body"
|
||||||
<Typography
|
size="xs"
|
||||||
hierarchy="body"
|
weight="medium"
|
||||||
size="xs"
|
inverted={!props.inverted}
|
||||||
weight="medium"
|
>
|
||||||
inverted={!props.inverted}
|
{props.tooltip}
|
||||||
>
|
</Typography>
|
||||||
{props.tooltip}
|
</Tooltip>
|
||||||
</Typography>
|
|
||||||
<KTooltip.Arrow />
|
|
||||||
</KTooltip.Content>
|
|
||||||
</KTooltip.Portal>
|
|
||||||
</KTooltip.Trigger>
|
|
||||||
</KTooltip>
|
|
||||||
)}
|
)}
|
||||||
</props.labelComponent>
|
</props.labelComponent>
|
||||||
{props.description && (
|
{props.description && (
|
||||||
|
|||||||
9
pkgs/clan-app/ui/src/components/Tooltip/Tooltip.css
Normal file
9
pkgs/clan-app/ui/src/components/Tooltip/Tooltip.css
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
div.tooltip-content {
|
||||||
|
@apply z-50 px-2 py-0.5 bg-inv-4 rounded-[0.125rem] leading-none;
|
||||||
|
|
||||||
|
max-width: min(calc(100vw - 16px), 380px);
|
||||||
|
|
||||||
|
&.inverted {
|
||||||
|
@apply bg-def-2;
|
||||||
|
}
|
||||||
|
}
|
||||||
40
pkgs/clan-app/ui/src/components/Tooltip/Tooltip.stories.tsx
Normal file
40
pkgs/clan-app/ui/src/components/Tooltip/Tooltip.stories.tsx
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
import { Meta, StoryObj } from "@kachurun/storybook-solid";
|
||||||
|
import { Tooltip, TooltipProps } from "@/src/components/Tooltip/Tooltip";
|
||||||
|
import { Typography } from "@/src/components/Typography/Typography";
|
||||||
|
import { Button } from "@/src/components/Button/Button";
|
||||||
|
|
||||||
|
const meta: Meta<TooltipProps> = {
|
||||||
|
title: "Components/Tooltip",
|
||||||
|
component: Tooltip,
|
||||||
|
decorators: [
|
||||||
|
(Story: StoryObj<TooltipProps>) => (
|
||||||
|
<div class="p-16">
|
||||||
|
<Story />
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
type Story = StoryObj<TooltipProps>;
|
||||||
|
|
||||||
|
export const Default: Story = {
|
||||||
|
args: {
|
||||||
|
placement: "top",
|
||||||
|
inverted: false,
|
||||||
|
trigger: <Button hierarchy="primary">Trigger</Button>,
|
||||||
|
children: (
|
||||||
|
<Typography hierarchy="body" size="xs" inverted={true} weight="medium">
|
||||||
|
Your Clan is being created
|
||||||
|
</Typography>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const AnimateBounce: Story = {
|
||||||
|
args: {
|
||||||
|
...Default.args,
|
||||||
|
animation: "bounce",
|
||||||
|
},
|
||||||
|
};
|
||||||
34
pkgs/clan-app/ui/src/components/Tooltip/Tooltip.tsx
Normal file
34
pkgs/clan-app/ui/src/components/Tooltip/Tooltip.tsx
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import "./Tooltip.css";
|
||||||
|
import {
|
||||||
|
Tooltip as KTooltip,
|
||||||
|
TooltipRootProps as KTooltipRootProps,
|
||||||
|
} from "@kobalte/core/tooltip";
|
||||||
|
import cx from "classnames";
|
||||||
|
import { JSX } from "solid-js";
|
||||||
|
|
||||||
|
export interface TooltipProps extends KTooltipRootProps {
|
||||||
|
inverted?: boolean;
|
||||||
|
trigger: JSX.Element;
|
||||||
|
children: JSX.Element;
|
||||||
|
animation?: "bounce";
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Tooltip = (props: TooltipProps) => {
|
||||||
|
return (
|
||||||
|
<KTooltip {...props}>
|
||||||
|
<KTooltip.Trigger>{props.trigger}</KTooltip.Trigger>
|
||||||
|
<KTooltip.Portal>
|
||||||
|
<KTooltip.Content
|
||||||
|
class={cx("tooltip-content", {
|
||||||
|
inverted: props.inverted,
|
||||||
|
"animate-bounce": props.animation == "bounce",
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
{props.placement == "bottom" && <KTooltip.Arrow />}
|
||||||
|
{props.children}
|
||||||
|
{props.placement == "top" && <KTooltip.Arrow />}
|
||||||
|
</KTooltip.Content>
|
||||||
|
</KTooltip.Portal>
|
||||||
|
</KTooltip>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -42,7 +42,7 @@ interface BackendReturnType<K extends OperationNames> {
|
|||||||
* @property {Promise<BackendReturnType<K>>} result - A promise that resolves to the return type of the backend operation.
|
* @property {Promise<BackendReturnType<K>>} result - A promise that resolves to the return type of the backend operation.
|
||||||
* @property {() => Promise<void>} cancel - A function to cancel the API call, returning a promise that resolves when cancellation is completed.
|
* @property {() => Promise<void>} cancel - A function to cancel the API call, returning a promise that resolves when cancellation is completed.
|
||||||
*/
|
*/
|
||||||
interface ApiCall<K extends OperationNames> {
|
export interface ApiCall<K extends OperationNames> {
|
||||||
uuid: string;
|
uuid: string;
|
||||||
result: Promise<OperationResponse<K>>;
|
result: Promise<OperationResponse<K>>;
|
||||||
cancel: () => Promise<void>;
|
cancel: () => Promise<void>;
|
||||||
|
|||||||
@@ -1,6 +1,18 @@
|
|||||||
import { Component } from "solid-js";
|
import { Component } from "solid-js";
|
||||||
import { RouteSectionProps } from "@solidjs/router";
|
import { RouteSectionProps, useNavigate } from "@solidjs/router";
|
||||||
|
import { activeClanURI } from "@/src/stores/clan";
|
||||||
|
import { navigateToClan } from "@/src/hooks/clan";
|
||||||
|
|
||||||
export const Layout: Component<RouteSectionProps> = (props) => (
|
export const Layout: Component<RouteSectionProps> = (props) => {
|
||||||
<div class="size-full h-screen">{props.children}</div>
|
const navigate = useNavigate();
|
||||||
);
|
|
||||||
|
// check for an active clan uri and redirect to it on first load
|
||||||
|
const activeURI = activeClanURI();
|
||||||
|
if (!props.location.pathname.startsWith("/clan/") && activeURI) {
|
||||||
|
navigateToClan(navigate, activeURI);
|
||||||
|
} else {
|
||||||
|
navigate("/");
|
||||||
|
}
|
||||||
|
|
||||||
|
return <div class="size-full h-screen">{props.children}</div>;
|
||||||
|
};
|
||||||
|
|||||||
@@ -81,5 +81,10 @@ main#welcome {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
div.creating {
|
||||||
|
@apply w-[17.0625rem] h-[20.4375rem];
|
||||||
|
background: url(./cube.svg) center / cover no-repeat;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +1,30 @@
|
|||||||
import { Component, createSignal, Match, Setter, Show, Switch } from "solid-js";
|
import {
|
||||||
|
Accessor,
|
||||||
|
Component,
|
||||||
|
createSignal,
|
||||||
|
Match,
|
||||||
|
Setter,
|
||||||
|
Show,
|
||||||
|
Switch,
|
||||||
|
} from "solid-js";
|
||||||
import { RouteSectionProps, useNavigate } from "@solidjs/router";
|
import { RouteSectionProps, useNavigate } from "@solidjs/router";
|
||||||
import "./Onboarding.css";
|
import "./Onboarding.css";
|
||||||
import { Typography } from "@/src/components/Typography/Typography";
|
import { Typography } from "@/src/components/Typography/Typography";
|
||||||
import { Button } from "@/src/components/Button/Button";
|
import { Button } from "@/src/components/Button/Button";
|
||||||
|
import { Tooltip } from "@/src/components/Tooltip/Tooltip";
|
||||||
|
import { Alert } from "@/src/components/Alert/Alert";
|
||||||
|
|
||||||
import { Divider } from "@/src/components/Divider/Divider";
|
import { Divider } from "@/src/components/Divider/Divider";
|
||||||
import { Logo } from "@/src/components/Logo/Logo";
|
import { Logo } from "@/src/components/Logo/Logo";
|
||||||
import { navigateToClan, selectClanFolder } from "@/src/hooks/clan";
|
import { navigateToClan, selectClanFolder } from "@/src/hooks/clan";
|
||||||
import { activeClanURI } from "@/src/stores/clan";
|
import { activeClanURI, addClanURI, setActiveClanURI } from "@/src/stores/clan";
|
||||||
import {
|
import {
|
||||||
createForm,
|
createForm,
|
||||||
FormStore,
|
FormStore,
|
||||||
getError,
|
getError,
|
||||||
getErrors,
|
getErrors,
|
||||||
getValue,
|
getValue,
|
||||||
|
SubmitHandler,
|
||||||
valiForm,
|
valiForm,
|
||||||
} from "@modular-forms/solid";
|
} from "@modular-forms/solid";
|
||||||
import { TextInput } from "@/src/components/Form/TextInput";
|
import { TextInput } from "@/src/components/Form/TextInput";
|
||||||
@@ -20,23 +32,31 @@ import { TextArea } from "@/src/components/Form/TextArea";
|
|||||||
import { Fieldset } from "@/src/components/Form/Fieldset";
|
import { Fieldset } from "@/src/components/Form/Fieldset";
|
||||||
import * as v from "valibot";
|
import * as v from "valibot";
|
||||||
import { HostFileInput } from "@/src/components/Form/HostFileInput";
|
import { HostFileInput } from "@/src/components/Form/HostFileInput";
|
||||||
|
import { callApi } from "@/src/hooks/api";
|
||||||
|
|
||||||
type State = "welcome" | "setup";
|
type State = "welcome" | "setup" | "creating";
|
||||||
|
|
||||||
const SetupSchema = v.object({
|
const SetupSchema = v.object({
|
||||||
name: v.pipe(v.string(), v.nonEmpty("Please enter a name.")),
|
name: v.pipe(
|
||||||
|
v.string(),
|
||||||
|
v.nonEmpty("Please enter a name."),
|
||||||
|
v.regex(
|
||||||
|
new RegExp("^[a-zA-Z0-9_\\-]+$"),
|
||||||
|
"Name must be alphanumeric and can contain underscores and dashes, without spaces.",
|
||||||
|
),
|
||||||
|
),
|
||||||
description: v.pipe(v.string(), v.nonEmpty("Please describe your clan.")),
|
description: v.pipe(v.string(), v.nonEmpty("Please describe your clan.")),
|
||||||
directory: v.pipe(v.string(), v.nonEmpty("Please select a directory.")),
|
directory: v.pipe(
|
||||||
|
// initial value is undefined, and I can't see how to handle this better in valibot, so for now when the type
|
||||||
|
// is incorrect we treat it as empty
|
||||||
|
v.string("Please select a directory."),
|
||||||
|
v.nonEmpty("Please select a directory."),
|
||||||
|
),
|
||||||
});
|
});
|
||||||
|
|
||||||
type SetupForm = v.InferInput<typeof SetupSchema>;
|
type SetupForm = v.InferInput<typeof SetupSchema>;
|
||||||
|
|
||||||
interface backgroundProps {
|
const background = (props: { state: State; form: FormStore<SetupForm> }) => (
|
||||||
state: State;
|
|
||||||
form: FormStore<SetupForm>;
|
|
||||||
}
|
|
||||||
|
|
||||||
const background = (props: backgroundProps) => (
|
|
||||||
<div class="background">
|
<div class="background">
|
||||||
<div class="layer-1" />
|
<div class="layer-1" />
|
||||||
<div class="layer-2" />
|
<div class="layer-2" />
|
||||||
@@ -71,7 +91,11 @@ const background = (props: backgroundProps) => (
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
const welcome = (setState: Setter<State>) => {
|
const welcome = (props: {
|
||||||
|
setState: Setter<State>;
|
||||||
|
welcomeError: Accessor<string | undefined>;
|
||||||
|
setWelcomeError: Setter<string | undefined>;
|
||||||
|
}) => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const selectFolder = async () => {
|
const selectFolder = async () => {
|
||||||
@@ -91,7 +115,23 @@ const welcome = (setState: Setter<State>) => {
|
|||||||
Build your <br />
|
Build your <br />
|
||||||
own darknet
|
own darknet
|
||||||
</Typography>
|
</Typography>
|
||||||
<Button hierarchy="secondary" onClick={() => setState("setup")}>
|
{props.welcomeError() && (
|
||||||
|
<Alert
|
||||||
|
type="error"
|
||||||
|
icon="Info"
|
||||||
|
title="Your Clan creation failed"
|
||||||
|
description={props.welcomeError() || ""}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<Button
|
||||||
|
hierarchy="secondary"
|
||||||
|
onClick={() => {
|
||||||
|
// reset welcome error
|
||||||
|
props.setWelcomeError(undefined);
|
||||||
|
// move to next step
|
||||||
|
props.setState("setup");
|
||||||
|
}}
|
||||||
|
>
|
||||||
Start building
|
Start building
|
||||||
</Button>
|
</Button>
|
||||||
<div class="separator">
|
<div class="separator">
|
||||||
@@ -114,6 +154,21 @@ const welcome = (setState: Setter<State>) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const creating = () => (
|
||||||
|
<div class="animate-pulse">
|
||||||
|
<Tooltip
|
||||||
|
open={true}
|
||||||
|
placement="top"
|
||||||
|
animation="bounce"
|
||||||
|
trigger={<div class="creating" />}
|
||||||
|
>
|
||||||
|
<Typography hierarchy="body" size="xs" weight="medium" inverted={true}>
|
||||||
|
Your Clan is being created
|
||||||
|
</Typography>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
export const Onboarding: Component<RouteSectionProps> = (props) => {
|
export const Onboarding: Component<RouteSectionProps> = (props) => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@@ -126,13 +181,89 @@ export const Onboarding: Component<RouteSectionProps> = (props) => {
|
|||||||
|
|
||||||
const [state, setState] = createSignal<State>("welcome");
|
const [state, setState] = createSignal<State>("welcome");
|
||||||
|
|
||||||
|
// used to display an error in the welcome screen in the event of a failed
|
||||||
|
// clan creation
|
||||||
|
const [welcomeError, setWelcomeError] = createSignal<string | undefined>();
|
||||||
|
|
||||||
|
//
|
||||||
const [setupForm, { Form, Field }] = createForm<SetupForm>({
|
const [setupForm, { Form, Field }] = createForm<SetupForm>({
|
||||||
validate: valiForm(SetupSchema),
|
validate: valiForm(SetupSchema),
|
||||||
});
|
});
|
||||||
|
|
||||||
const metaError = () => {
|
const formError = () => {
|
||||||
const errors = getErrors(setupForm, ["name", "description"]);
|
const formErrors = getErrors(setupForm);
|
||||||
return errors ? errors.name || errors.description : undefined;
|
return (
|
||||||
|
formErrors.name ||
|
||||||
|
formErrors.description ||
|
||||||
|
formErrors.directory ||
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onSelectFile = async () => {
|
||||||
|
const req = callApi("get_system_file", {
|
||||||
|
file_request: {
|
||||||
|
mode: "select_folder",
|
||||||
|
title: "Select a folder for you new Clan",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const resp = await req.result;
|
||||||
|
|
||||||
|
if (resp.status === "error") {
|
||||||
|
// just throw the first error, I can't imagine why there would be multiple
|
||||||
|
// errors for this call
|
||||||
|
throw new Error(resp.errors[0].message);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resp.status === "success" && resp.data) {
|
||||||
|
return resp.data[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error("No data returned from api call");
|
||||||
|
};
|
||||||
|
|
||||||
|
const onSubmit: SubmitHandler<SetupForm> = async (
|
||||||
|
{ name, description, directory },
|
||||||
|
event,
|
||||||
|
) => {
|
||||||
|
const path = `${directory}/${name}`;
|
||||||
|
|
||||||
|
const req = callApi("create_clan", {
|
||||||
|
opts: {
|
||||||
|
dest: path,
|
||||||
|
// todo allow users to select a template
|
||||||
|
template: "minimal",
|
||||||
|
initial: {
|
||||||
|
meta: {
|
||||||
|
name: name,
|
||||||
|
description: description,
|
||||||
|
// todo it tries to 'delete' icon if it's not provided
|
||||||
|
// this logic is unexpected, and needs reviewed.
|
||||||
|
icon: null,
|
||||||
|
},
|
||||||
|
machines: {},
|
||||||
|
instances: {},
|
||||||
|
services: {},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
setState("creating");
|
||||||
|
|
||||||
|
const resp = await req.result;
|
||||||
|
|
||||||
|
if (resp.status === "error") {
|
||||||
|
setWelcomeError(resp.errors[0].message);
|
||||||
|
setState("welcome");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resp.status === "success") {
|
||||||
|
addClanURI(path);
|
||||||
|
setActiveClanURI(path);
|
||||||
|
navigateToClan(navigate, path);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -140,7 +271,13 @@ export const Onboarding: Component<RouteSectionProps> = (props) => {
|
|||||||
{background({ form: setupForm, state: state() })}
|
{background({ form: setupForm, state: state() })}
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<Switch>
|
<Switch>
|
||||||
<Match when={state() === "welcome"}>{welcome(setState)}</Match>
|
<Match when={state() === "welcome"}>
|
||||||
|
{welcome({
|
||||||
|
setState,
|
||||||
|
welcomeError,
|
||||||
|
setWelcomeError,
|
||||||
|
})}
|
||||||
|
</Match>
|
||||||
|
|
||||||
<Match when={state() === "setup"}>
|
<Match when={state() === "setup"}>
|
||||||
<div class="setup">
|
<div class="setup">
|
||||||
@@ -155,8 +292,16 @@ export const Onboarding: Component<RouteSectionProps> = (props) => {
|
|||||||
Setup
|
Setup
|
||||||
</Typography>
|
</Typography>
|
||||||
</div>
|
</div>
|
||||||
<Form>
|
<Form onSubmit={onSubmit}>
|
||||||
<Fieldset name="meta" error={metaError()}>
|
{formError() && (
|
||||||
|
<Alert
|
||||||
|
type="error"
|
||||||
|
icon="Info"
|
||||||
|
title="Form error"
|
||||||
|
description={formError() || ""}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<Fieldset name="meta">
|
||||||
<Field name="name">
|
<Field name="name">
|
||||||
{(field, input) => (
|
{(field, input) => (
|
||||||
<TextInput
|
<TextInput
|
||||||
@@ -195,15 +340,13 @@ export const Onboarding: Component<RouteSectionProps> = (props) => {
|
|||||||
</Field>
|
</Field>
|
||||||
</Fieldset>
|
</Fieldset>
|
||||||
|
|
||||||
<Fieldset
|
<Fieldset name="location">
|
||||||
name="location"
|
|
||||||
error={getError(setupForm, "directory")}
|
|
||||||
>
|
|
||||||
<Field name="directory">
|
<Field name="directory">
|
||||||
{(field, input) => (
|
{(field, input) => (
|
||||||
<HostFileInput
|
<HostFileInput
|
||||||
onSelectFile={async () => "test"}
|
onSelectFile={onSelectFile}
|
||||||
{...field}
|
{...field}
|
||||||
|
value={field.value}
|
||||||
label="Select directory"
|
label="Select directory"
|
||||||
orientation="horizontal"
|
orientation="horizontal"
|
||||||
required={true}
|
required={true}
|
||||||
@@ -228,6 +371,8 @@ export const Onboarding: Component<RouteSectionProps> = (props) => {
|
|||||||
</Form>
|
</Form>
|
||||||
</div>
|
</div>
|
||||||
</Match>
|
</Match>
|
||||||
|
|
||||||
|
<Match when={state() === "creating"}>{creating()}</Match>
|
||||||
</Switch>
|
</Switch>
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
|
|||||||
316
pkgs/clan-app/ui/src/routes/Onboarding/cube.svg
Normal file
316
pkgs/clan-app/ui/src/routes/Onboarding/cube.svg
Normal file
@@ -0,0 +1,316 @@
|
|||||||
|
<svg width="273" height="327" viewBox="0 0 273 327" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499716 -0.86619 0.499716 137.399 89.7148)" fill="url(#paint0_linear_4542_12187)" stroke="url(#paint1_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.8281 125.194)" fill="url(#paint2_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.8281 125.194)" fill="url(#paint3_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.8281 125.194)" stroke="url(#paint4_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.326 160.673)" fill="url(#paint5_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.326 160.673)" fill="url(#paint6_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.326 160.673)" stroke="url(#paint7_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499716 -0.86619 0.499716 62.5708 142.85)" fill="url(#paint8_linear_4542_12187)" stroke="url(#paint9_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 1 178.329)" fill="url(#paint10_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 1 178.329)" fill="url(#paint11_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 1 178.329)" stroke="url(#paint12_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 62.4976 213.808)" fill="url(#paint13_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 62.4976 213.808)" fill="url(#paint14_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 62.4976 213.808)" stroke="url(#paint15_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499715 -0.86619 0.499715 210.502 137.882)" fill="url(#paint16_linear_4542_12187)" stroke="url(#paint17_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 148.931 173.361)" fill="url(#paint18_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 148.931 173.361)" fill="url(#paint19_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 148.931 173.361)" stroke="url(#paint20_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499715 0 1 210.429 208.84)" fill="url(#paint21_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499715 0 1 210.429 208.84)" fill="url(#paint22_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499715 0 1 210.429 208.84)" stroke="url(#paint23_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499716 -0.86619 0.499716 137.158 185.084)" fill="url(#paint24_linear_4542_12187)" stroke="url(#paint25_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.5869 220.563)" fill="url(#paint26_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.5869 220.563)" fill="url(#paint27_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.5869 220.563)" stroke="url(#paint28_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.084 256.042)" fill="url(#paint29_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.084 256.042)" fill="url(#paint30_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.084 256.042)" stroke="url(#paint31_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499716 -0.86619 0.499716 135.915 0)" fill="url(#paint32_linear_4542_12187)" stroke="url(#paint33_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 74.3442 35.479)" fill="url(#paint34_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 74.3442 35.479)" fill="url(#paint35_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 74.3442 35.479)" stroke="url(#paint36_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 135.842 70.9579)" fill="url(#paint37_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 135.842 70.9579)" fill="url(#paint38_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 135.842 70.9579)" stroke="url(#paint39_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499716 -0.86619 0.499716 62.5708 47.2027)" fill="url(#paint40_linear_4542_12187)" stroke="url(#paint41_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 1 82.6817)" fill="url(#paint42_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 1 82.6817)" fill="url(#paint43_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 1 82.6817)" stroke="url(#paint44_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 62.4976 118.161)" fill="url(#paint45_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 62.4976 118.161)" fill="url(#paint46_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 62.4976 118.161)" stroke="url(#paint47_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499715 -0.86619 0.499715 210.502 42.234)" fill="url(#paint48_linear_4542_12187)" stroke="url(#paint49_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 148.931 77.713)" fill="url(#paint50_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 148.931 77.713)" fill="url(#paint51_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 148.931 77.713)" stroke="url(#paint52_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499715 0 1 210.429 113.192)" fill="url(#paint53_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499715 0 1 210.429 113.192)" fill="url(#paint54_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499715 0 1 210.429 113.192)" stroke="url(#paint55_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9982" rx="1" transform="matrix(0.86619 0.499716 -0.86619 0.499716 137.158 89.4367)" fill="url(#paint56_linear_4542_12187)" stroke="url(#paint57_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.5869 124.916)" fill="url(#paint58_linear_4542_12187)" fill-opacity="0.1"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.5869 124.916)" fill="url(#paint59_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 0.499716 0 1 75.5869 124.916)" stroke="url(#paint60_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.084 160.395)" fill="url(#paint61_linear_4542_12187)"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.084 160.395)" fill="url(#paint62_linear_4542_12187)" fill-opacity="0.24"/>
|
||||||
|
<rect width="70.9982" height="70.9578" rx="1" transform="matrix(0.86619 -0.499716 0 1 137.084 160.395)" stroke="url(#paint63_linear_4542_12187)" stroke-width="0.64"/>
|
||||||
|
<defs>
|
||||||
|
<linearGradient id="paint0_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint1_linear_4542_12187" x1="73.1289" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint2_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint3_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint4_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint5_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint6_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint7_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint8_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint9_linear_4542_12187" x1="73.1289" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint10_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint11_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint12_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint13_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint14_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint15_linear_4542_12187" x1="73.129" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint16_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint17_linear_4542_12187" x1="73.129" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint18_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint19_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42737" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint20_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint21_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint22_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint23_linear_4542_12187" x1="73.129" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint24_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint25_linear_4542_12187" x1="73.1289" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint26_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint27_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint28_linear_4542_12187" x1="73.129" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint29_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint30_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1743" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint31_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint32_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint33_linear_4542_12187" x1="73.129" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint34_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint35_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint36_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint37_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint38_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint39_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint40_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint41_linear_4542_12187" x1="73.1289" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint42_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint43_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint44_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint45_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint46_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint47_linear_4542_12187" x1="73.129" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint48_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint49_linear_4542_12187" x1="73.129" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint50_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint51_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint52_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint53_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint54_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint55_linear_4542_12187" x1="73.129" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint56_linear_4542_12187" x1="3.21525" y1="1.26987" x2="26.2123" y2="73.9372" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.94"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint57_linear_4542_12187" x1="73.1289" y1="21.7256" x2="19.1537" y2="-1.08702" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint58_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint59_linear_4542_12187" x1="71.0824" y1="68.7651" x2="43.5492" y2="9.42738" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint60_linear_4542_12187" x1="73.129" y1="21.7132" x2="19.163" y2="-1.10843" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint61_linear_4542_12187" x1="3.21525" y1="1.26914" x2="26.1885" y2="73.9026" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="white" stop-opacity="0.81"/>
|
||||||
|
<stop offset="1" stop-color="white" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint62_linear_4542_12187" x1="22.5463" y1="67.3059" x2="51.7875" y2="28.1744" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#40A2A6"/>
|
||||||
|
<stop offset="1" stop-color="#91ACAF" stop-opacity="0"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint63_linear_4542_12187" x1="73.1289" y1="21.7132" x2="19.163" y2="-1.10842" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#10191A"/>
|
||||||
|
<stop offset="1" stop-color="#2C4547"/>
|
||||||
|
</linearGradient>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 24 KiB |
15
pkgs/clan-app/ui/src/scene/cubes.css
Normal file
15
pkgs/clan-app/ui/src/scene/cubes.css
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
.cubes-scene-container {
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolbar-container {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 10%;
|
||||||
|
width: 100%;
|
||||||
|
z-index: 10;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
15
pkgs/clan-app/ui/src/scene/cubes.stories.tsx
Normal file
15
pkgs/clan-app/ui/src/scene/cubes.stories.tsx
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { Meta, StoryObj } from "@kachurun/storybook-solid";
|
||||||
|
import { CubeScene } from "./cubes";
|
||||||
|
|
||||||
|
const meta: Meta = {
|
||||||
|
title: "scene/cubes",
|
||||||
|
component: CubeScene,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
|
||||||
|
type Story = StoryObj;
|
||||||
|
|
||||||
|
export const Default: Story = {
|
||||||
|
args: {},
|
||||||
|
};
|
||||||
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user