Skip to content

Commit e5a4770

Browse files
authored
feat:Drawer (#152)
* feat:Drawer * feat: Drawer with clickoutside * fix: settings
1 parent 2d77b9c commit e5a4770

File tree

10 files changed

+299
-133
lines changed

10 files changed

+299
-133
lines changed

platforms/metagram/package.json

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@
3232
"@sveltejs/vite-plugin-svelte": "^5.0.0",
3333
"@tailwindcss/vite": "^4.0.0",
3434
"clsx": "^2.1.1",
35+
"cupertino-pane": "^1.4.22",
3536
"eslint": "^9.18.0",
3637
"eslint-config-prettier": "^10.0.1",
3738
"eslint-plugin-svelte": "^3.0.0",
@@ -42,6 +43,7 @@
4243
"storybook": "^8.6.12",
4344
"svelte": "^5.0.0",
4445
"svelte-check": "^4.0.0",
46+
"svelte-gestures": "^5.1.3",
4547
"tailwindcss": "^4.0.0",
4648
"typescript": "^5.0.0",
4749
"typescript-eslint": "^8.20.0",
Lines changed: 36 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
<script module>
2+
export {DrawerContent};
3+
</script>
4+
5+
{#snippet DrawerContent()}
6+
<p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Vero dolore, necessitatibus quam expedita repudiandae cum nulla totam officiis itaque earum explicabo vel soluta libero ea exercitationem iusto velit incidunt quibusdam.
7+
Incidunt maiores sed, voluptas magnam magni reiciendis ipsa. Totam molestias commodi incidunt obcaecati fugiat temporibus voluptatum ea excepturi cumque, nostrum quod quae aperiam, nemo asperiores consectetur doloremque repellat, alias consequatur.
8+
Inventore deleniti aliquam sapiente, vel ipsam tempore veritatis explicabo minus harum hic itaque dolorem nostrum tenetur dicta dolores delectus eveniet nisi accusantium aperiam cumque iure culpa, eos adipisci! Vel, iusto.
9+
Repellendus unde incidunt a saepe reiciendis? Nam, esse voluptas aliquam repellendus cupiditate placeat quis autem minima reiciendis ex unde eveniet neque qui dolorum. Dignissimos inventore quidem ipsum sequi non vel?
10+
Nemo aspernatur incidunt veniam quia animi vel asperiores quidem, expedita eum in qui provident, consequuntur officiis veritatis quisquam quo, necessitatibus similique autem odit officia numquam cum dolore pariatur eius! Cupiditate.
11+
Facilis aliquid nihil doloremque officiis fugit in minus dicta et, omnis nesciunt suscipit sequi atque perferendis alias exercitationem eaque aliquam provident praesentium quae sunt quas magni consequatur. Praesentium, dolor ipsa?
12+
Et, veritatis saepe animi harum eligendi fugit accusamus tempore, reiciendis eveniet, dolores rerum optio earum magnam nam laudantium necessitatibus incidunt minima labore laboriosam voluptatum mollitia quia? Consectetur, reprehenderit! Mollitia, neque.
13+
Optio nam eius voluptatem. Facilis hic cum non est quod modi sapiente ex asperiores earum, temporibus ipsum dolore optio aut mollitia fuga ducimus ipsam deleniti et adipisci maxime in velit.
14+
Praesentium doloribus earum sunt quasi voluptas quo expedita tempora consequatur ratione maiores accusantium dolorem sit corporis quibusdam esse molestias facilis nemo, non quam? Dolores recusandae neque ex consequatur in atque?
15+
Animi recusandae minus eaque nisi dicta earum sit tempore neque porro vero assumenda tempora alias, quis sunt iste illo fuga in inventore quaerat? Magnam cum vero fugiat quaerat provident necessitatibus.
16+
Consequatur iusto est sint doloribus quis porro quibusdam inventore numquam nisi nam ullam consectetur mollitia molestiae molestias necessitatibus recusandae voluptate cum aut pariatur, asperiores nobis et perspiciatis ipsam labore. Delectus.
17+
Laborum quisquam ea eligendi ut nesciunt aut minima iusto sint esse laboriosam! Doloremque deserunt enim et recusandae quo, nobis expedita maxime error inventore, officiis nulla ipsa tempora eos commodi beatae!
18+
Atque vel vitae perspiciatis quam soluta eaque sed illo autem ea ut, exercitationem sit ad quidem eius modi alias facilis labore porro esse. Vel nihil expedita reprehenderit culpa facilis omnis!
19+
Non eum minima eius sapiente quo nulla culpa maiores laudantium quos! Consequatur adipisci ad neque nemo! Iste aperiam, tenetur nisi distinctio saepe, reiciendis sed reprehenderit, eum assumenda vel provident in.
20+
Magni ex commodi soluta quod ipsum repellendus laborum assumenda velit, aperiam praesentium nemo, dolor aut similique libero tenetur eum! Atque maxime laborum tenetur similique nam ipsam possimus eos officia voluptates.
21+
Deleniti quas amet corporis enim possimus nostrum laboriosam nisi. Quisquam, perspiciatis. Quidem unde ut sequi maxime omnis inventore sed qui minima neque, accusamus officia, labore delectus et itaque repellendus. Quia.
22+
Quod unde id ipsum labore ex. Non quam, provident, excepturi expedita in, fuga repellendus quibusdam qui numquam similique eaque corrupti exercitationem dolorem mollitia quas! Eius facere nihil natus rerum saepe.
23+
Quos atque dolorem, eaque explicabo vel sequi, nam debitis nihil velit adipisci libero praesentium. Modi blanditiis odio, quis explicabo minus vel sunt eius! Odit, libero exercitationem dolore sequi odio assumenda.
24+
Cum nostrum iusto ratione excepturi mollitia reiciendis illo eius quisquam veniam non sunt ullam, porro suscipit beatae nesciunt temporibus optio quam tempora possimus. Facere, officia quasi tempora repudiandae quo rem!
25+
Explicabo cum ratione sapiente. Obcaecati at enim delectus aperiam iure ex, dolore, quae, id cupiditate quam facere similique perferendis voluptates vel ut. Dolorum delectus, officiis placeat amet rerum inventore nesciunt!
26+
Qui voluptas dolor similique id ratione magnam quas dolore ipsum dolores consectetur, error consequuntur nemo veniam tempora temporibus modi fugiat, quam officia exercitationem iusto veritatis dicta magni. Doloribus, harum provident.
27+
Veniam tempora quasi, pariatur omnis temporibus ipsam modi mollitia eum fuga tempore quia? Vel aliquam odit aut a! Perspiciatis sint nisi quidem, fuga eligendi architecto porro quos nemo ex provident!
28+
Modi nesciunt, aspernatur iure accusamus laborum incidunt? Suscipit amet dolor nihil odio id blanditiis beatae quos harum. Quasi dolorum voluptatum eaque sed vero cum. Impedit, esse. Voluptates tempora saepe repellendus!
29+
Quaerat quod animi suscipit error. Similique quae eaque accusantium fugit sint vero voluptas alias quidem? Animi cumque repudiandae quas deleniti ea repellat provident, esse, voluptatum nam dolore consequatur non blanditiis?
30+
Qui reiciendis quidem dicta iusto minima aliquam, iste itaque illo dolor earum eligendi, harum sed doloremque quo voluptatum amet consectetur at quod a officiis perspiciatis! Debitis perferendis pariatur cum odio!
31+
Est eveniet commodi dolorem numquam voluptatibus aliquid, unde accusamus, tempora laudantium consequatur autem repudiandae, quaerat magnam ducimus atque hic facilis officiis aspernatur. Ducimus necessitatibus, labore praesentium consectetur libero at voluptatibus.
32+
Consequatur repudiandae magnam sit blanditiis impedit fugiat totam reiciendis facilis porro sapiente harum veniam in, quos excepturi commodi perspiciatis ad voluptas! Perferendis cum nulla soluta ipsa at? Fugit, dignissimos mollitia!
33+
Adipisci voluptatem molestiae doloribus alias quisquam neque commodi incidunt libero maiores aut consequatur error corrupti, modi illo veritatis itaque numquam facere eveniet? Tenetur consectetur ipsum excepturi quaerat, facilis labore modi?
34+
Alias harum dolor tempora voluptatibus nam? Incidunt laborum debitis consequuntur, blanditiis laudantium ad facilis laboriosam, cupiditate optio similique saepe a, praesentium veniam dignissimos modi placeat reiciendis! Quia doloribus blanditiis labore.
35+
Suscipit dolorum nobis incidunt voluptatibus fugiat minima nostrum natus cum, quas magni obcaecati qui perspiciatis maxime, eligendi aperiam doloribus doloremque totam asperiores a labore! Explicabo quaerat doloremque labore harum vitae!</p>
36+
{/snippet}
Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
import type { ComponentProps } from 'svelte';
2+
import { Drawer } from '..';
3+
import { DrawerContent } from './Drawer.stories.snippet.svelte';
4+
5+
export default {
6+
title: 'UI/Drawer',
7+
component: Drawer,
8+
tags: ['autodocs'],
9+
render: (args: { Component: Drawer; props: ComponentProps<typeof Drawer> }) => ({
10+
Component: Drawer,
11+
props: args
12+
})
13+
};
14+
15+
export const Main = {
16+
args: {
17+
children: DrawerContent
18+
}
19+
};
Lines changed: 74 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
<script lang="ts">
2+
import { onMount, type Snippet } from 'svelte';
3+
import { CupertinoPane } from 'cupertino-pane';
4+
import type { HTMLAttributes } from 'svelte/elements';
5+
import { clickOutside, cn } from '$lib/utils';
6+
import { swipe } from 'svelte-gestures';
7+
import type { SwipeCustomEvent } from 'svelte-gestures';
8+
9+
interface IDrawerProps extends HTMLAttributes<HTMLDivElement> {
10+
isPaneOpen?: boolean;
11+
children?: Snippet;
12+
handleSwipe?: (isOpen: boolean | undefined) => void;
13+
}
14+
15+
let {
16+
isPaneOpen = $bindable(),
17+
children = undefined,
18+
handleSwipe,
19+
...restProps
20+
}: IDrawerProps = $props();
21+
22+
let drawerElement: HTMLElement;
23+
let drawer: CupertinoPane;
24+
25+
const handleClickOutside = () => {
26+
drawer?.destroy({ animate: true });
27+
isPaneOpen = false;
28+
};
29+
30+
const handleDrawerSwipe = (event: SwipeCustomEvent) => {
31+
if (event.detail.direction === 'down' as string) {
32+
handleSwipe?.(false);
33+
drawer?.destroy({ animate: true });
34+
isPaneOpen = false;
35+
}
36+
};
37+
38+
onMount(() => {
39+
if (!drawerElement) return;
40+
drawer = new CupertinoPane(drawerElement, {
41+
showDraggable: false,
42+
backdrop: true,
43+
backdropBlur: true,
44+
backdropOpacity: 0.4,
45+
animationType: 'ease',
46+
animationDuration: 300,
47+
bottomClose: true,
48+
buttonDestroy: false,
49+
cssClass: '',
50+
initialBreak: 'middle'
51+
});
52+
if (isPaneOpen) {
53+
drawer.present({ animate: true });
54+
} else {
55+
drawer.destroy({ animate: true });
56+
}
57+
58+
drawer.present();
59+
});
60+
</script>
61+
62+
<div
63+
bind:this={drawerElement}
64+
{...restProps}
65+
use:swipe={() => ({
66+
timeframe: 300,
67+
minSwipeDistance: 60
68+
})}
69+
onswipe={handleDrawerSwipe}
70+
use:clickOutside={handleClickOutside}
71+
class={cn(restProps.class)}
72+
>
73+
{@render children?.()}
74+
</div>

platforms/metagram/src/lib/fragments/Post/Post.stories.ts

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import type { ComponentProps } from 'svelte';
22
import Post from './Post.svelte';
33

44
export default {
5-
title: 'Fragments/Post',
5+
title: 'ui/Post',
66
component: Post,
77
tags: ['autodocs'],
88
render: (args: { Component: Post; props: ComponentProps<typeof Post> }) => ({

platforms/metagram/src/lib/fragments/index.ts

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2,3 +2,4 @@ export { default as BottomNav } from './BottomNav/BottomNav.svelte';
22
export { default as SettingsNavigationButton } from './SettingsNavigationButton/SettingsNavigationButton.svelte';
33
export { default as MessageInput } from './MessageInput/MessageInput.svelte';
44
export { default as InputFile } from './InputFile/InputFile.svelte';
5+
export { default as Drawer } from './Drawer/Drawer.svelte';
Lines changed: 33 additions & 37 deletions
Original file line numberDiff line numberDiff line change
@@ -1,58 +1,54 @@
1-
import { Input } from "..";
1+
import { Input } from '..';
22

33
export default {
4-
title: "UI/Input",
5-
component: Input,
6-
tags: ["autodocs"],
7-
render: (args: {
8-
type: string;
9-
placeholder: string;
10-
helperText: string;
11-
}) => ({
12-
Component: Input,
13-
props: args,
14-
}),
4+
title: 'UI/Input',
5+
component: Input,
6+
tags: ['autodocs'],
7+
render: (args: { type: string; placeholder: string; helperText: string }) => ({
8+
Component: Input,
9+
props: args
10+
})
1511
};
1612

1713
export const Text = {
18-
args: {
19-
type: "text",
20-
placeholder: "Joe Biden",
21-
},
14+
args: {
15+
type: 'text',
16+
placeholder: 'Joe Biden'
17+
}
2218
};
2319

2420
export const Tel = {
25-
args: {
26-
type: "tel",
27-
placeholder: "987654321",
28-
},
21+
args: {
22+
type: 'tel',
23+
placeholder: '987654321'
24+
}
2925
};
3026

3127
export const NumberInput = {
32-
args: {
33-
type: "number",
34-
placeholder: "Enter something",
35-
},
28+
args: {
29+
type: 'number',
30+
placeholder: 'Enter something'
31+
}
3632
};
3733

3834
export const Email = {
39-
args: {
40-
type: "email",
41-
placeholder: "[email protected]",
42-
},
35+
args: {
36+
type: 'email',
37+
placeholder: '[email protected]'
38+
}
4339
};
4440

4541
export const Invalid = {
46-
args: {
47-
type: "email",
48-
placeholder: "Invalid email",
49-
value: "not-an-email",
50-
},
42+
args: {
43+
type: 'email',
44+
placeholder: 'Invalid email',
45+
value: 'not-an-email'
46+
}
5147
};
5248

5349
export const Password = {
54-
args: {
55-
type: "password",
56-
placeholder: "Please enter password",
57-
},
50+
args: {
51+
type: 'password',
52+
placeholder: 'Please enter password'
53+
}
5854
};
Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
/** Dispatch event on click outside of node */
2+
export const clickOutside = (node: HTMLElement, callback: () => void) => {
3+
const handleClick = (event: Event) => {
4+
if (node && !node.contains(event.target as Node) && !event.defaultPrevented) {
5+
callback(); // Call the provided callback
6+
node.dispatchEvent(new CustomEvent('click_outside'));
7+
}
8+
};
9+
10+
document.addEventListener('click', handleClick, true);
11+
12+
return {
13+
destroy() {
14+
document.removeEventListener('click', handleClick, true);
15+
}
16+
};
17+
};
Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1 +1,2 @@
11
export * from './mergeClasses';
2+
export * from './clickOutside';

0 commit comments

Comments
 (0)