Compare commits

...

11 Commits

Author SHA1 Message Date
Jon Staab 68ebd32e15 Bump welshman 2025-05-09 12:41:02 -07:00
Jon Staab e94aa3c119 Bump version, fix new messages thing 2025-05-09 12:26:05 -07:00
Jon Staab 4d10fe7cc0 Handle broken supported_nips 2025-05-08 11:16:02 -07:00
Jon Staab 841928783b Re-introduce safe inset areas 2025-05-08 11:05:27 -07:00
Jon Staab 6e5e1a0846 Remove safe area inset stuff to re-apply later 2025-05-08 09:11:10 -07:00
Jon Staab d57f4747a6 Tweak errors so that actionable links are rendered 2025-05-07 15:04:35 -07:00
Jon Staab 94a0077b09 Use non-singleton broker 2025-05-07 13:53:58 -07:00
Jon Staab f2eb04adff Bump version 2025-05-07 09:12:17 -07:00
Jon Staab d4d5979a35 Fix missing room images and room overflow in nav 2025-05-07 09:11:00 -07:00
Jon Staab dde6e54657 Add build in production script 2025-05-06 18:26:48 -07:00
Jon Staab 698a7513b8 Tweak some gradle stuff 2025-05-06 18:07:30 -07:00
40 changed files with 310 additions and 152 deletions
+12
View File
@@ -1,5 +1,17 @@
# Changelog # Changelog
# 1.0.2
* Fix add relay button
* Fix safe inset areas
* Better rendering for errors from relays
* Improve remote signer login
# 1.0.1
* Fix relay images in nav
* Fix relay nav overflow
# 1.0.0 # 1.0.0
* Add alerts via Anchor * Add alerts via Anchor
+4 -4
View File
@@ -5,10 +5,10 @@ android {
compileSdk rootProject.ext.compileSdkVersion compileSdk rootProject.ext.compileSdkVersion
defaultConfig { defaultConfig {
applicationId "social.flotilla" applicationId "social.flotilla"
minSdkVersion rootProject.ext.minSdkVersion minSdk rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion targetSdk rootProject.ext.targetSdkVersion
versionCode 14 versionCode 16
versionName "1.0.0" versionName "1.0.2"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
aaptOptions { aaptOptions {
// Files and dirs to omit from the packaged assets dir, modified to accommodate modern web apps. // Files and dirs to omit from the packaged assets dir, modified to accommodate modern web apps.
+4 -2
View File
@@ -3,7 +3,9 @@ ext {
compileSdkVersion = 35 compileSdkVersion = 35
targetSdkVersion = 35 targetSdkVersion = 35
androidxActivityVersion = '1.9.2' androidxActivityVersion = '1.9.2'
androidxAppCompatVersion = '1.7.0' //https://github.com/ionic-team/capacitor/issues/7866
// androidxAppCompatVersion = '1.7.0'
androidxAppCompatVersion = '1.6.1'
androidxCoordinatorLayoutVersion = '1.2.0' androidxCoordinatorLayoutVersion = '1.2.0'
androidxCoreVersion = '1.15.0' androidxCoreVersion = '1.15.0'
androidxFragmentVersion = '1.8.4' androidxFragmentVersion = '1.8.4'
@@ -13,4 +15,4 @@ ext {
androidxJunitVersion = '1.2.1' androidxJunitVersion = '1.2.1'
androidxEspressoCoreVersion = '3.6.1' androidxEspressoCoreVersion = '3.6.1'
cordovaAndroidVersion = '10.1.1' cordovaAndroidVersion = '10.1.1'
} }
+19
View File
@@ -0,0 +1,19 @@
#!/usr/bin/env bash
# Fetch tags and set to env vars
git fetch --prune --unshallow --tags
git describe --tags --abbrev=0
export VITE_BUILD_VERSION=$RENDER_GIT_COMMIT
export VITE_BUILD_HASH=$RENDER_GIT_COMMIT
# Remove link overrides
node remove-pnpm-overrides.js package.json
# When CI=true as it is on render.com, removing link overrides breaks the lockfile
pnpm i --no-frozen-lockfile
# Rebuild sharp
pnpm rebuild
# The build runs out of memory at times
NODE_OPTIONS=--max_old_space_size=16384 pnpm run build
+1 -1
View File
@@ -18,7 +18,7 @@ const config: CapacitorConfig = {
}, },
// Use this for live reload https://capacitorjs.com/docs/guides/live-reload // Use this for live reload https://capacitorjs.com/docs/guides/live-reload
// server: { // server: {
// url: "http://192.168.1.250:1847", // url: "http://192.168.1.115:1847",
// cleartext: true // cleartext: true
// }, // },
}; };
+4 -4
View File
@@ -351,14 +351,14 @@
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic; CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = 8; CURRENT_PROJECT_VERSION = 10;
DEVELOPMENT_TEAM = S26U9DYW3A; DEVELOPMENT_TEAM = S26U9DYW3A;
INFOPLIST_FILE = App/Info.plist; INFOPLIST_FILE = App/Info.plist;
INFOPLIST_KEY_CFBundleDisplayName = "Flotilla Chat"; INFOPLIST_KEY_CFBundleDisplayName = "Flotilla Chat";
INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking"; INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking";
IPHONEOS_DEPLOYMENT_TARGET = 14.0; IPHONEOS_DEPLOYMENT_TARGET = 14.0;
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
MARKETING_VERSION = 1.0.0; MARKETING_VERSION = 1.0.2;
OTHER_SWIFT_FLAGS = "$(inherited) \"-D\" \"COCOAPODS\" \"-DDEBUG\""; OTHER_SWIFT_FLAGS = "$(inherited) \"-D\" \"COCOAPODS\" \"-DDEBUG\"";
PRODUCT_BUNDLE_IDENTIFIER = social.flotilla; PRODUCT_BUNDLE_IDENTIFIER = social.flotilla;
PRODUCT_NAME = "$(TARGET_NAME)"; PRODUCT_NAME = "$(TARGET_NAME)";
@@ -376,14 +376,14 @@
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic; CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = 8; CURRENT_PROJECT_VERSION = 10;
DEVELOPMENT_TEAM = S26U9DYW3A; DEVELOPMENT_TEAM = S26U9DYW3A;
INFOPLIST_FILE = App/Info.plist; INFOPLIST_FILE = App/Info.plist;
INFOPLIST_KEY_CFBundleDisplayName = "Flotilla Chat"; INFOPLIST_KEY_CFBundleDisplayName = "Flotilla Chat";
INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking"; INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking";
IPHONEOS_DEPLOYMENT_TARGET = 14.0; IPHONEOS_DEPLOYMENT_TARGET = 14.0;
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
MARKETING_VERSION = 1.0.0; MARKETING_VERSION = 1.0.2;
PRODUCT_BUNDLE_IDENTIFIER = social.flotilla; PRODUCT_BUNDLE_IDENTIFIER = social.flotilla;
PRODUCT_NAME = "$(TARGET_NAME)"; PRODUCT_NAME = "$(TARGET_NAME)";
PROVISIONING_PROFILE_SPECIFIER = ""; PROVISIONING_PROFILE_SPECIFIER = "";
+7 -7
View File
@@ -1,6 +1,6 @@
{ {
"name": "flotilla", "name": "flotilla",
"version": "1.0.0", "version": "1.0.2",
"private": true, "private": true,
"scripts": { "scripts": {
"dev": "vite dev", "dev": "vite dev",
@@ -51,16 +51,16 @@
"@types/qrcode": "^1.5.5", "@types/qrcode": "^1.5.5",
"@vite-pwa/assets-generator": "^0.2.6", "@vite-pwa/assets-generator": "^0.2.6",
"@vite-pwa/sveltekit": "^0.6.6", "@vite-pwa/sveltekit": "^0.6.6",
"@welshman/app": "^0.2.3", "@welshman/app": "^0.2.4",
"@welshman/content": "^0.2.0", "@welshman/content": "^0.2.1",
"@welshman/dvm": "^0.2.0", "@welshman/dvm": "^0.2.0",
"@welshman/editor": "^0.2.0", "@welshman/editor": "^0.2.1",
"@welshman/feeds": "^0.2.2", "@welshman/feeds": "^0.2.2",
"@welshman/lib": "^0.2.1", "@welshman/lib": "^0.2.2",
"@welshman/net": "^0.2.2", "@welshman/net": "^0.2.3",
"@welshman/relay": "^0.2.0", "@welshman/relay": "^0.2.0",
"@welshman/router": "^0.2.0", "@welshman/router": "^0.2.0",
"@welshman/signer": "^0.2.1", "@welshman/signer": "^0.2.3",
"@welshman/store": "^0.2.0", "@welshman/store": "^0.2.0",
"@welshman/util": "^0.2.2", "@welshman/util": "^0.2.2",
"daisyui": "^4.12.10", "daisyui": "^4.12.10",
+20
View File
@@ -0,0 +1,20 @@
// This script is necessary for installing stuff on a host, since our links don't exist there.
import fs from "fs"
const pkgName = process.argv[2]
if (!pkgName?.endsWith("package.json")) {
console.log("File passed was not a package.json file")
process.exit(1)
}
const pkg = JSON.parse(fs.readFileSync(pkgName, "utf8"))
if (pkg.pnpm && pkg.pnpm.overrides) {
delete pkg.pnpm.overrides
fs.writeFileSync(pkgName, JSON.stringify(pkg, null, 2) + "\n")
console.log("Removed pnpm.overrides from package.json")
} else {
console.log("No pnpm.overrides found in package.json")
}
+80 -36
View File
@@ -54,6 +54,10 @@
--primary-content: oklch(var(--pc)); --primary-content: oklch(var(--pc));
--secondary: oklch(var(--s)); --secondary: oklch(var(--s));
--secondary-content: oklch(var(--sc)); --secondary-content: oklch(var(--sc));
--sait: env(safe-area-inset-top);
--saib: env(safe-area-inset-bottom);
--sail: env(safe-area-inset-left);
--sair: env(safe-area-inset-right);
} }
:root, :root,
@@ -62,50 +66,80 @@ html {
@apply bg-base-300; @apply bg-base-300;
} }
/* ios */ /* safe area insets */
.sait { @layer components {
padding-top: env(safe-area-inset-top); .pt-sai {
} padding-top: var(--sait);
}
.sair { .pr-sai {
padding-right: env(safe-area-inset-right); padding-right: var(--sair);
} }
.saib { .pb-sai {
padding-bottom: env(safe-area-inset-bottom); padding-bottom: var(--saib);
} }
.sail { .pl-sai {
padding-left: env(safe-area-inset-left); padding-left: var(--sail);
} }
.saix { .px-sai {
@apply sail sair; @apply pl-sai pr-sai;
} }
.saiy { .py-sai {
@apply sait saib; @apply pt-sai pb-sai;
} }
.sai { .p-sai {
@apply saiy saix; @apply py-sai px-sai;
} }
.top-sai { .mt-sai {
top: env(safe-area-inset-top); padding-top: var(--sait);
} }
.right-sai { .mr-sai {
right: env(safe-area-inset-right); padding-right: var(--sair);
} }
.bottom-sai { .mb-sai {
bottom: env(safe-area-inset-bottom); padding-bottom: var(--saib);
} }
.left-sai { .ml-sai {
left: env(safe-area-inset-left); padding-left: var(--sail);
}
.mx-sai {
@apply ml-sai mr-sai;
}
.my-sai {
@apply mt-sai mb-sai;
}
.m-sai {
@apply my-sai mx-sai;
}
.top-sai {
top: var(--sait);
}
.right-sai {
right: var(--sair);
}
.bottom-sai {
bottom: var(--saib);
}
.left-sai {
left: var(--sail);
}
} }
/* utilities */ /* utilities */
@@ -294,6 +328,16 @@ html {
color: var(--base-content); color: var(--base-content);
} }
/* content rendered by welshman/content */
.welshman-content a {
@apply link;
}
.welshman-content-error a {
@apply underline;
}
/* date input */ /* date input */
.picker { .picker {
@@ -335,11 +379,11 @@ progress[value]::-webkit-progress-value {
/* content width for fixed elements */ /* content width for fixed elements */
.cw { .cw {
@apply w-full md:w-[calc(100%-18.5rem)]; @apply w-full md:left-[18.5rem] md:w-[calc(100%-18.5rem-var(--sair))];
} }
.cb { .cb {
@apply saib bottom-14 md:bottom-0; @apply md:bottom-sai bottom-[calc(var(--saib)+3.5rem)];
} }
/* chat view */ /* chat view */
@@ -349,5 +393,5 @@ progress[value]::-webkit-progress-value {
} }
.chat__scroll-down { .chat__scroll-down {
@apply saib fixed bottom-28 right-4 md:bottom-16; @apply fixed bottom-28 right-4 md:bottom-16;
} }
+10 -8
View File
@@ -1,6 +1,6 @@
import * as nip19 from "nostr-tools/nip19" import * as nip19 from "nostr-tools/nip19"
import {get} from "svelte/store" import {get} from "svelte/store"
import {randomId, ifLet, poll, uniq, equals, TIMEZONE, LOCALE} from "@welshman/lib" import {randomId, poll, uniq, equals, TIMEZONE, LOCALE} from "@welshman/lib"
import type {Feed} from "@welshman/feeds" import type {Feed} from "@welshman/feeds"
import type {TrustedEvent, EventContent} from "@welshman/util" import type {TrustedEvent, EventContent} from "@welshman/util"
import { import {
@@ -259,25 +259,27 @@ export const setInboxRelayPolicy = (url: string, enabled: boolean) => {
export const checkRelayAccess = async (url: string, claim = "") => { export const checkRelayAccess = async (url: string, claim = "") => {
const socket = Pool.get().get(url) const socket = Pool.get().get(url)
await socket.auth.attemptAuth(signer.get().sign) await socket.auth.attemptAuth(e => signer.get()?.sign(e))
const thunk = publishThunk({ const thunk = publishThunk({
event: createEvent(AUTH_JOIN, {tags: [["claim", claim]]}), event: createEvent(AUTH_JOIN, {tags: [["claim", claim]]}),
relays: [url], relays: [url],
}) })
ifLet(await getThunkError(thunk), error => { const error = await getThunkError(thunk)
if (error) {
const message = const message =
socket.auth.details?.replace(/^.*: /, "") || socket.auth.details?.replace(/^\w+: /, "") ||
error?.replace(/^.*: /, "") || error?.replace(/^\w+: /, "") ||
"join request rejected" "join request rejected"
// If it's a strict NIP 29 relay don't worry about requesting access // If it's a strict NIP 29 relay don't worry about requesting access
// TODO: remove this if relay29 ever gets less strict // TODO: remove this if relay29 ever gets less strict
if (message !== "missing group (`h`) tag") { if (message !== "missing group (`h`) tag") {
return `Failed to join relay (${message})` return message
} }
}) }
} }
export const checkRelayProfile = async (url: string) => { export const checkRelayProfile = async (url: string) => {
@@ -307,7 +309,7 @@ export const checkRelayAuth = async (url: string, timeout = 3000) => {
const socket = Pool.get().get(url) const socket = Pool.get().get(url)
const okStatuses = [AuthStatus.None, AuthStatus.Ok] const okStatuses = [AuthStatus.None, AuthStatus.Ok]
await socket.auth.attemptAuth(signer.get().sign) await socket.auth.attemptAuth(e => signer.get()?.sign(e))
// Only raise an error if it's not a timeout. // Only raise an error if it's not a timeout.
// If it is, odds are the problem is with our signer, not the relay // If it is, odds are the problem is with our signer, not the relay
+2 -1
View File
@@ -9,7 +9,7 @@
loading = $state(false) loading = $state(false)
clientSecret = makeSecret() clientSecret = makeSecret()
abortController = new AbortController() abortController = new AbortController()
broker = Nip46Broker.get({clientSecret: this.clientSecret, relays: SIGNER_RELAYS}) broker = new Nip46Broker({clientSecret: this.clientSecret, relays: SIGNER_RELAYS})
onNostrConnect: (response: Nip46ResponseWithResult) => void onNostrConnect: (response: Nip46ResponseWithResult) => void
constructor({onNostrConnect}: {onNostrConnect: (response: Nip46ResponseWithResult) => void}) { constructor({onNostrConnect}: {onNostrConnect: (response: Nip46ResponseWithResult) => void}) {
@@ -45,6 +45,7 @@
} }
stop() { stop() {
this.broker.cleanup()
this.abortController.abort() this.abortController.abort()
} }
} }
+2
View File
@@ -126,9 +126,11 @@
}) })
observer.observe(chatCompose!) observer.observe(chatCompose!)
observer.observe(dynamicPadding!)
return () => { return () => {
observer.unobserve(chatCompose!) observer.unobserve(chatCompose!)
observer.unobserve(dynamicPadding!)
} }
}) })
+2 -1
View File
@@ -46,12 +46,13 @@
try { try {
const {clientSecret} = controller const {clientSecret} = controller
const broker = Nip46Broker.get({relays, clientSecret, signerPubkey}) const broker = new Nip46Broker({relays, clientSecret, signerPubkey})
const result = await broker.connect(connectSecret, NIP46_PERMS) const result = await broker.connect(connectSecret, NIP46_PERMS)
const pubkey = await broker.getPublicKey() const pubkey = await broker.getPublicKey()
// TODO: remove ack result // TODO: remove ack result
if (pubkey && ["ack", connectSecret].includes(result)) { if (pubkey && ["ack", connectSecret].includes(result)) {
broker.cleanup()
controller.stop() controller.stop()
await loadUserData(pubkey) await loadUserData(pubkey)
+2 -2
View File
@@ -34,7 +34,7 @@
? [normalizeRelayUrl("ws://" + stripProtocol(BURROW_URL))] ? [normalizeRelayUrl("ws://" + stripProtocol(BURROW_URL))]
: [normalizeRelayUrl(BURROW_URL)] : [normalizeRelayUrl(BURROW_URL)]
const broker = Nip46Broker.get({clientSecret, relays}) const broker = new Nip46Broker({clientSecret, relays})
const back = () => history.back() const back = () => history.back()
@@ -89,7 +89,7 @@
await loadUserData(pubkey) await loadUserData(pubkey)
addSession({...session, email}) addSession({...session, email})
broker.cleanup()
setChecked("*") setChecked("*")
clearModals() clearModals()
} }
+15
View File
@@ -0,0 +1,15 @@
<script lang="ts">
import MenuSpacesItem from "@app/components/MenuSpacesItem.svelte"
type Props = {
urls: string[]
}
const {urls}: Props = $props()
</script>
<div class="column menu gap-2">
{#each urls as url (url)}
<MenuSpacesItem {url} />
{/each}
</div>
+34 -8
View File
@@ -1,6 +1,7 @@
<script lang="ts"> <script lang="ts">
import {page} from "$app/stores" import {page} from "$app/stores"
import {goto} from "$app/navigation" import {goto} from "$app/navigation"
import {splitAt} from "@welshman/lib"
import {userProfile} from "@welshman/app" import {userProfile} from "@welshman/app"
import Avatar from "@lib/components/Avatar.svelte" import Avatar from "@lib/components/Avatar.svelte"
import Divider from "@lib/components/Divider.svelte" import Divider from "@lib/components/Divider.svelte"
@@ -8,6 +9,7 @@
import SpaceAdd from "@app/components/SpaceAdd.svelte" import SpaceAdd from "@app/components/SpaceAdd.svelte"
import ChatEnable from "@app/components/ChatEnable.svelte" import ChatEnable from "@app/components/ChatEnable.svelte"
import MenuSpaces from "@app/components/MenuSpaces.svelte" import MenuSpaces from "@app/components/MenuSpaces.svelte"
import MenuOtherSpaces from "@app/components/MenuOtherSpaces.svelte"
import MenuSettings from "@app/components/MenuSettings.svelte" import MenuSettings from "@app/components/MenuSettings.svelte"
import PrimaryNavItemSpace from "@app/components/PrimaryNavItemSpace.svelte" import PrimaryNavItemSpace from "@app/components/PrimaryNavItemSpace.svelte"
import {userRoomsByUrl, canDecrypt, PLATFORM_RELAY, PLATFORM_LOGO} from "@app/state" import {userRoomsByUrl, canDecrypt, PLATFORM_RELAY, PLATFORM_LOGO} from "@app/state"
@@ -22,20 +24,35 @@
const addSpace = () => pushModal(SpaceAdd) const addSpace = () => pushModal(SpaceAdd)
const showSpacesMenu = () => (spacePaths.length > 0 ? pushModal(MenuSpaces) : pushModal(SpaceAdd)) const showSpacesMenu = () => (spaceUrls.length > 0 ? pushModal(MenuSpaces) : pushModal(SpaceAdd))
const showOtherSpacesMenu = () => pushModal(MenuOtherSpaces, {urls: secondarySpaceUrls})
const showSettingsMenu = () => pushModal(MenuSettings) const showSettingsMenu = () => pushModal(MenuSettings)
const openChat = () => ($canDecrypt ? goto("/chat") : pushModal(ChatEnable, {next: "/chat"})) const openChat = () => ($canDecrypt ? goto("/chat") : pushModal(ChatEnable, {next: "/chat"}))
const hasNotification = (url: string) => {
const path = makeSpacePath(url)
return !$page.url.pathname.startsWith(path) && $notifications.has(path)
}
let windowHeight = $state(0)
const itemHeight = 56
const navPadding = 6 * itemHeight
const itemLimit = $derived((windowHeight - navPadding) / itemHeight)
const spaceUrls = $derived(Array.from($userRoomsByUrl.keys())) const spaceUrls = $derived(Array.from($userRoomsByUrl.keys()))
const spacePaths = $derived(spaceUrls.map(url => makeSpacePath(url))) const [primarySpaceUrls, secondarySpaceUrls] = $derived(splitAt(itemLimit, spaceUrls))
const anySpaceNotifications = $derived( const anySpaceNotifications = $derived(spaceUrls.some(hasNotification))
spacePaths.some(path => !$page.url.pathname.startsWith(path) && $notifications.has(path)), const otherSpaceNotifications = $derived(secondarySpaceUrls.some(hasNotification))
)
</script> </script>
<div class="sail sait saib relative z-nav hidden w-14 flex-shrink-0 bg-base-200 pt-4 md:block"> <svelte:window bind:innerHeight={windowHeight} />
<div
class="ml-sai mt-sai mb-sai relative z-nav hidden w-14 flex-shrink-0 bg-base-200 pt-4 md:block">
<div class="flex h-full flex-col justify-between"> <div class="flex h-full flex-col justify-between">
<div> <div>
{#if PLATFORM_RELAY} {#if PLATFORM_RELAY}
@@ -45,9 +62,18 @@
<Avatar src={PLATFORM_LOGO} class="!h-10 !w-10" /> <Avatar src={PLATFORM_LOGO} class="!h-10 !w-10" />
</PrimaryNavItem> </PrimaryNavItem>
<Divider /> <Divider />
{#each spaceUrls as url (url)} {#each primarySpaceUrls as url (url)}
<PrimaryNavItemSpace {url} /> <PrimaryNavItemSpace {url} />
{/each} {/each}
{#if secondarySpaceUrls.length > 0}
<PrimaryNavItem
title="Other Spaces"
class="tooltip-right"
onclick={showOtherSpacesMenu}
notification={otherSpaceNotifications}>
<Avatar icon="widget" class="!h-10 !w-10" />
</PrimaryNavItem>
{/if}
<PrimaryNavItem title="Add Space" onclick={addSpace} class="tooltip-right"> <PrimaryNavItem title="Add Space" onclick={addSpace} class="tooltip-right">
<Avatar icon="settings-minimalistic" class="!h-10 !w-10" /> <Avatar icon="settings-minimalistic" class="!h-10 !w-10" />
</PrimaryNavItem> </PrimaryNavItem>
@@ -78,7 +104,7 @@
{@render children?.()} {@render children?.()}
<!-- a little extra something for ios --> <!-- a little extra something for ios -->
<div class="fixed bottom-0 left-0 right-0 z-nav h-14 bg-base-100 md:hidden"></div> <div class="fixed bottom-0 left-0 right-0 z-nav h-[var(--saib)] bg-base-100 md:hidden"></div>
<div <div
class="border-top bottom-sai fixed left-0 right-0 z-nav h-14 border border-base-200 bg-base-100 md:hidden"> class="border-top bottom-sai fixed left-0 right-0 z-nav h-14 border border-base-200 bg-base-100 md:hidden">
<div class="content-padding-x content-sizing flex justify-between px-2"> <div class="content-padding-x content-sizing flex justify-between px-2">
+1 -1
View File
@@ -66,7 +66,7 @@
</div> </div>
<ProfileInfo {pubkey} {url} /> <ProfileInfo {pubkey} {url} />
<ModalFooter> <ModalFooter>
<Button onclick={back} class="btn btn-link"> <Button onclick={back} class="hidden md:btn md:btn-link">
<Icon icon="alt-arrow-left" /> <Icon icon="alt-arrow-left" />
Go back Go back
</Button> </Button>
+2 -1
View File
@@ -45,7 +45,8 @@
e => e.id, e => e.id,
sortBy(e => -e.created_at, buffer), sortBy(e => -e.created_at, buffer),
) )
events = [...events, ...buffer.splice(0, 5)]
events = uniqBy(e => e.id, [...events, ...buffer.splice(0, 5)])
if (buffer.length < 50) { if (buffer.length < 50) {
ctrl.load(50) ctrl.load(50)
+2 -2
View File
@@ -4,7 +4,7 @@
import Button from "@lib/components/Button.svelte" import Button from "@lib/components/Button.svelte"
import {clip} from "@app/toast" import {clip} from "@app/toast"
const {code} = $props() const {code, ...props} = $props()
let canvas: Element | undefined = $state() let canvas: Element | undefined = $state()
let wrapper: Element | undefined = $state() let wrapper: Element | undefined = $state()
@@ -26,7 +26,7 @@
}) })
</script> </script>
<Button class="max-w-full" onclick={copy}> <Button class="max-w-full {props.class}" onclick={copy}>
<div bind:this={wrapper} style={`height: ${height}px`}> <div bind:this={wrapper} style={`height: ${height}px`}>
<canvas <canvas
class="rounded-box" class="rounded-box"
+1 -1
View File
@@ -29,7 +29,7 @@
>{displayUrl($relay.profile.contact)}</Link> >{displayUrl($relay.profile.contact)}</Link>
&bull; &bull;
{/if} {/if}
{#if $relay?.profile?.supported_nips} {#if Array.isArray($relay?.profile?.supported_nips)}
<span <span
class="tooltip cursor-pointer underline" class="tooltip cursor-pointer underline"
data-tip="NIPs supported: {$relay.profile.supported_nips.join(', ')}"> data-tip="NIPs supported: {$relay.profile.supported_nips.join(', ')}">
+9 -24
View File
@@ -1,10 +1,11 @@
<script lang="ts"> <script lang="ts">
import {displayRelayUrl} from "@welshman/util" import {displayRelayUrl} from "@welshman/util"
import {parse, renderAsHtml} from "@welshman/content"
import Spinner from "@lib/components/Spinner.svelte" import Spinner from "@lib/components/Spinner.svelte"
import Button from "@lib/components/Button.svelte" import Button from "@lib/components/Button.svelte"
import Field from "@lib/components/Field.svelte"
import Icon from "@lib/components/Icon.svelte" import Icon from "@lib/components/Icon.svelte"
import {preventDefault} from "@lib/html" import {preventDefault} from "@lib/html"
import {ucFirst} from "@lib/util"
import ModalHeader from "@lib/components/ModalHeader.svelte" import ModalHeader from "@lib/components/ModalHeader.svelte"
import ModalFooter from "@lib/components/ModalFooter.svelte" import ModalFooter from "@lib/components/ModalFooter.svelte"
import {pushToast} from "@app/toast" import {pushToast} from "@app/toast"
@@ -15,8 +16,8 @@
const back = () => history.back() const back = () => history.back()
const joinRelay = async (claim: string) => { const joinRelay = async () => {
const error = await attemptRelayAccess(url, claim) const error = await attemptRelayAccess(url)
if (error) { if (error) {
return pushToast({theme: "error", message: error}) return pushToast({theme: "error", message: error})
@@ -33,13 +34,12 @@
loading = true loading = true
try { try {
await joinRelay(claim) await joinRelay()
} finally { } finally {
loading = false loading = false
} }
} }
let claim = $state("")
let loading = $state(false) let loading = $state(false)
</script> </script>
@@ -53,32 +53,17 @@
{/snippet} {/snippet}
</ModalHeader> </ModalHeader>
<p> <p>
We received an error from the relay indicating you don't have access to {displayRelayUrl(url)}. We received an error from the relay indicating you don't have access to {displayRelayUrl(url)}:
</p> </p>
<p class="border-l border-solid border-error pl-4 text-error"> <p class="bg-alt card2 welshman-content">
{error} {@html renderAsHtml(parse({content: ucFirst(error)}))}
</p> </p>
<p>If you have one, you can try entering an invite code below to request access.</p>
<Field>
{#snippet label()}
<p>Invite code</p>
{/snippet}
{#snippet input()}
<label class="input input-bordered flex w-full items-center gap-2">
<Icon icon="link-round" />
<input bind:value={claim} class="grow" type="text" />
</label>
{/snippet}
{#snippet info()}
<p>Enter an invite code provided to you by the admin of the relay.</p>
{/snippet}
</Field>
<ModalFooter> <ModalFooter>
<Button class="btn btn-link" onclick={back}> <Button class="btn btn-link" onclick={back}>
<Icon icon="alt-arrow-left" /> <Icon icon="alt-arrow-left" />
Go back Go back
</Button> </Button>
<Button type="submit" class="btn btn-primary" disabled={!claim || loading}> <Button type="submit" class="btn btn-primary" disabled={loading}>
<Spinner {loading}>Request Access</Spinner> <Spinner {loading}>Request Access</Spinner>
<Icon icon="alt-arrow-right" /> <Icon icon="alt-arrow-right" />
</Button> </Button>
+1 -1
View File
@@ -23,7 +23,7 @@
const error = await attemptRelayAccess(url, claim) const error = await attemptRelayAccess(url, claim)
if (error) { if (error) {
return pushToast({theme: "error", message: error}) return pushToast({theme: "error", message: error, timeout: 30_000})
} }
const socket = Pool.get().get(url) const socket = Pool.get().get(url)
+5 -2
View File
@@ -1,4 +1,5 @@
<script lang="ts"> <script lang="ts">
import {parse, renderAsHtml} from "@welshman/content"
import {fly} from "@lib/transition" import {fly} from "@lib/transition"
import Icon from "@lib/components/Icon.svelte" import Icon from "@lib/components/Icon.svelte"
import Button from "@lib/components/Button.svelte" import Button from "@lib/components/Button.svelte"
@@ -7,7 +8,7 @@
{#if $toast} {#if $toast}
{@const theme = $toast.theme || "info"} {@const theme = $toast.theme || "info"}
<div transition:fly class="toast z-toast"> <div transition:fly class="bottom-sai right-sai toast z-toast">
{#key $toast.id} {#key $toast.id}
<div <div
role="alert" role="alert"
@@ -15,7 +16,9 @@
class:bg-base-100={theme === "info"} class:bg-base-100={theme === "info"}
class:text-base-content={theme === "info"} class:text-base-content={theme === "info"}
class:alert-error={theme === "error"}> class:alert-error={theme === "error"}>
{$toast.message} <p class="welshman-content-error">
{@html renderAsHtml(parse({content: $toast.message}))}
</p>
<Button class="flex items-center opacity-75" onclick={() => popToast($toast.id)}> <Button class="flex items-center opacity-75" onclick={() => popToast($toast.id)}>
<Icon icon="close-circle" /> <Icon icon="close-circle" />
</Button> </Button>
+24 -19
View File
@@ -11,26 +11,29 @@ import {makeMentionNodeView} from "./MentionNodeView"
import ProfileSuggestion from "./ProfileSuggestion.svelte" import ProfileSuggestion from "./ProfileSuggestion.svelte"
export const hasBlossomSupport = simpleCache(async ([url]: [string]) => { export const hasBlossomSupport = simpleCache(async ([url]: [string]) => {
try { const $signer = signer.get()
const event = await signer.get()!.sign( const headers: Record<string, string> = {
makeEvent(BLOSSOM_AUTH, { "X-Content-Type": "text/plain",
tags: [ "X-Content-Length": "1",
["t", "upload"], "X-SHA-256": "73cb3858a687a8494ca3323053016282f3dad39d42cf62ca4e79dda2aac7d9ac",
["server", url], }
["expiration", String(now() + 30)],
],
}),
)
const res = await fetch(normalizeUrl(url) + "/upload", { try {
method: "head", if ($signer) {
headers: { const event = await signer.get().sign(
Authorization: `Nostr ${btoa(JSON.stringify(event))}`, makeEvent(BLOSSOM_AUTH, {
"X-Content-Type": "text/plain", tags: [
"X-Content-Length": "1", ["t", "upload"],
"X-SHA-256": "73cb3858a687a8494ca3323053016282f3dad39d42cf62ca4e79dda2aac7d9ac", ["server", url],
}, ["expiration", String(now() + 30)],
}) ],
}),
)
headers.Authorization = `Nostr ${btoa(JSON.stringify(event))}`
}
const res = await fetch(normalizeUrl(url) + "/upload", {method: "head", headers})
return res.status === 200 return res.status === 200
} catch (e) { } catch (e) {
@@ -38,6 +41,8 @@ export const hasBlossomSupport = simpleCache(async ([url]: [string]) => {
console.error(e) console.error(e)
} }
} }
return false
}) })
export const getUploadUrl = async (spaceUrl?: string) => { export const getUploadUrl = async (spaceUrl?: string) => {
+3 -3
View File
@@ -485,7 +485,7 @@ export const messages = derived(
export const groupMeta = deriveEvents(repository, {filters: [{kinds: [GROUP_META]}]}) export const groupMeta = deriveEvents(repository, {filters: [{kinds: [GROUP_META]}]})
export const hasNip29 = (relay?: Relay) => export const hasNip29 = (relay?: Relay) =>
relay?.profile?.supported_nips?.map(String)?.includes("29") relay?.profile?.supported_nips?.map?.(String)?.includes?.("29")
// Channels // Channels
@@ -629,11 +629,11 @@ export const userRoomsByUrl = withGetter(
const $userRoomsByUrl = new Map<string, Set<string>>() const $userRoomsByUrl = new Map<string, Set<string>>()
for (const [_, room, url] of getGroupTags(tags)) { for (const [_, room, url] of getGroupTags(tags)) {
addToMapKey($userRoomsByUrl, url, room) addToMapKey($userRoomsByUrl, normalizeRelayUrl(url), room)
} }
for (const url of getRelayTagValues(tags)) { for (const url of getRelayTagValues(tags)) {
addToMapKey($userRoomsByUrl, url, GENERAL) addToMapKey($userRoomsByUrl, normalizeRelayUrl(url), GENERAL)
} }
return $userRoomsByUrl return $userRoomsByUrl
+4
View File
@@ -0,0 +1,4 @@
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M7.99986 12H8.00887M12.0044 12H12.0134M15.9908 12H15.9999" stroke="#1C274C" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="12" cy="12" r="10" stroke="#1C274C" stroke-width="1.5"/>
</svg>

After

Width:  |  Height:  |  Size: 323 B

+2 -2
View File
@@ -8,13 +8,13 @@
const {...props}: Props = $props() const {...props}: Props = $props()
</script> </script>
<div class="col-2 content-padding-t content-padding-x h-full {props.class}"> <div class="content-padding-t content-padding-x flex h-full flex-col gap-1 {props.class}">
<div class="z-feature"> <div class="z-feature">
<div class="content-sizing"> <div class="content-sizing">
{@render props.input?.()} {@render props.input?.()}
</div> </div>
</div> </div>
<div class="scroll-container overflow-auto pt-2"> <div class="scroll-container overflow-auto">
<div class="content-sizing"> <div class="content-sizing">
{@render props.content?.()} {@render props.content?.()}
</div> </div>
+4 -2
View File
@@ -1,9 +1,11 @@
<script lang="ts"> <script lang="ts">
import type {Snippet} from "svelte"
interface Props { interface Props {
children?: import("svelte").Snippet children?: Snippet
} }
const {children}: Props = $props() const {children, ...props}: Props = $props()
</script> </script>
<div class="flex items-center gap-2 p-2 text-xs uppercase opacity-50"> <div class="flex items-center gap-2 p-2 text-xs uppercase opacity-50">
+1 -1
View File
@@ -12,7 +12,7 @@
onclick={onClose}> onclick={onClose}>
</button> </button>
<div <div
class="scroll-container saiy sair absolute bottom-0 right-0 top-0 w-80 overflow-auto bg-base-200 text-base-content lg:w-96" class="scroll-container py-sai pr-sair absolute bottom-0 right-0 top-0 w-80 overflow-auto bg-base-200 text-base-content lg:w-96"
transition:translate={{axis: "x", duration: 300}}> transition:translate={{axis: "x", duration: 300}}>
{@render children?.()} {@render children?.()}
</div> </div>
+2
View File
@@ -58,6 +58,7 @@
import Mailbox from "@assets/icons/Mailbox.svg?dataurl" import Mailbox from "@assets/icons/Mailbox.svg?dataurl"
import MapPoint from "@assets/icons/Map Point.svg?dataurl" import MapPoint from "@assets/icons/Map Point.svg?dataurl"
import MenuDots from "@assets/icons/Menu Dots.svg?dataurl" import MenuDots from "@assets/icons/Menu Dots.svg?dataurl"
import MenuDotsCircle from "@assets/icons/Menu Dots Circle.svg?dataurl"
import NotesMinimalistic from "@assets/icons/Notes Minimalistic.svg?dataurl" import NotesMinimalistic from "@assets/icons/Notes Minimalistic.svg?dataurl"
import Pallete2 from "@assets/icons/Pallete 2.svg?dataurl" import Pallete2 from "@assets/icons/Pallete 2.svg?dataurl"
import Paperclip from "@assets/icons/Paperclip.svg?dataurl" import Paperclip from "@assets/icons/Paperclip.svg?dataurl"
@@ -149,6 +150,7 @@
mailbox: Mailbox, mailbox: Mailbox,
"map-point": MapPoint, "map-point": MapPoint,
"menu-dots": MenuDots, "menu-dots": MenuDots,
"menu-dots-circle": MenuDotsCircle,
"notes-minimalistic": NotesMinimalistic, "notes-minimalistic": NotesMinimalistic,
"pallete-2": Pallete2, "pallete-2": Pallete2,
paperclip: Paperclip, paperclip: Paperclip,
+2 -1
View File
@@ -8,6 +8,7 @@
</script> </script>
<div <div
class="sait saib sair scroll-container mb-14 max-h-screen flex-grow overflow-auto bg-base-200 md:mb-0 {props.class}"> data-component="Page"
class="scroll-container bottom-sai top-sai cw fixed mb-14 overflow-auto bg-base-200 md:mb-0 {props.class}">
{@render props.children?.()} {@render props.children?.()}
</div> </div>
+2 -2
View File
@@ -11,9 +11,9 @@
const {...props}: Props = $props() const {...props}: Props = $props()
</script> </script>
<div class="sait cw fixed top-2 z-feature rounded-xl px-2 pt-2 {props.class}"> <div data-component="PageBar" class="cw top-sai fixed z-feature p-2">
<div <div
class="flex min-h-12 items-center justify-between gap-4 rounded-xl bg-base-100 px-4 shadow-xl"> class="flex min-h-12 items-center justify-between gap-4 rounded-xl rounded-xl bg-base-100 px-4 shadow-xl">
<div class="ellipsize flex items-center gap-4 whitespace-nowrap"> <div class="ellipsize flex items-center gap-4 whitespace-nowrap">
{@render props.icon?.()} {@render props.icon?.()}
{@render props.title?.()} {@render props.title?.()}
+2 -1
View File
@@ -13,6 +13,7 @@
<div <div
{...props} {...props}
bind:this={element} bind:this={element}
class="scroll-container saib cw fixed top-12 h-[calc(100%-6.5rem)] overflow-y-auto overflow-x-hidden md:h-[calc(100%-3rem)] {props.class}"> data-component="PageContent"
class="scroll-container cw md:bottom-sai fixed bottom-[calc(var(--saib)+3.5rem)] top-[calc(var(--sait)+3rem)] overflow-y-auto overflow-x-hidden {props.class}">
{@render children?.()} {@render children?.()}
</div> </div>
+1 -1
View File
@@ -7,6 +7,6 @@
</script> </script>
<div <div
class="sail sait saib hidden max-h-screen w-60 flex-shrink-0 flex-col gap-1 bg-base-300 md:flex"> class="ml-sai mt-sai mb-sai hidden max-h-screen w-60 flex-shrink-0 flex-col gap-1 bg-base-300 md:flex">
{@render children?.()} {@render children?.()}
</div> </div>
+2
View File
@@ -15,3 +15,5 @@ export const nsecDecode = (nsec: string) => {
export const day = (seconds: number) => Math.floor(seconds / DAY) export const day = (seconds: number) => Math.floor(seconds / DAY)
export const daysBetween = (start: number, end: number) => [...range(start, end, DAY)].map(day) export const daysBetween = (start: number, end: number) => [...range(start, end, DAY)].map(day)
export const ucFirst = (s: string) => s.slice(0, 1).toUpperCase() + s.slice(1)
+2 -1
View File
@@ -96,7 +96,7 @@
if (login?.startsWith("bunker://")) { if (login?.startsWith("bunker://")) {
const clientSecret = makeSecret() const clientSecret = makeSecret()
const {signerPubkey, connectSecret, relays} = Nip46Broker.parseBunkerUrl(login) const {signerPubkey, connectSecret, relays} = Nip46Broker.parseBunkerUrl(login)
const broker = Nip46Broker.get({relays, clientSecret, signerPubkey}) const broker = new Nip46Broker({relays, clientSecret, signerPubkey})
const result = await broker.connect(connectSecret, appState.NIP46_PERMS) const result = await broker.connect(connectSecret, appState.NIP46_PERMS)
const pubkey = await broker.getPublicKey() const pubkey = await broker.getPublicKey()
@@ -105,6 +105,7 @@
await loadUserData(pubkey) await loadUserData(pubkey)
loginWithNip46(pubkey, clientSecret, signerPubkey, relays) loginWithNip46(pubkey, clientSecret, signerPubkey, relays)
broker.cleanup()
success = true success = true
} }
} else if (login) { } else if (login) {
-2
View File
@@ -58,8 +58,6 @@
let settings = $state({...$userSettingValues}) let settings = $state({...$userSettingValues})
let mutedPubkeys = $state(getPubkeyTagValues(getListTags($userMutes))) let mutedPubkeys = $state(getPubkeyTagValues(getListTags($userMutes)))
let blossomServers = $state(getTagValues("server", getListTags($userBlossomServers))) let blossomServers = $state(getTagValues("server", getListTags($userBlossomServers)))
$inspect(blossomServers)
</script> </script>
<form class="content column gap-4" {onsubmit}> <form class="content column gap-4" {onsubmit}>
+10 -7
View File
@@ -1,4 +1,5 @@
<script lang="ts"> <script lang="ts">
import {Capacitor} from "@capacitor/core"
import Link from "@lib/components/Link.svelte" import Link from "@lib/components/Link.svelte"
import Button from "@lib/components/Button.svelte" import Button from "@lib/components/Button.svelte"
import Icon from "@lib/components/Icon.svelte" import Icon from "@lib/components/Icon.svelte"
@@ -19,13 +20,15 @@
<p class="text-center text-2xl">Thanks for using</p> <p class="text-center text-2xl">Thanks for using</p>
<h1 class="mb-4 text-center text-5xl font-bold uppercase">{PLATFORM_NAME}</h1> <h1 class="mb-4 text-center text-5xl font-bold uppercase">{PLATFORM_NAME}</h1>
<div class="grid grid-cols-1 gap-8 lg:grid-cols-2"> <div class="grid grid-cols-1 gap-8 lg:grid-cols-2">
<div class="card2 bg-alt flex flex-col gap-2 text-center shadow-2xl"> {#if Capacitor.getPlatform() !== "ios"}
<h3 class="text-2xl sm:h-12">Donate</h3> <div class="card2 bg-alt flex flex-col gap-2 text-center shadow-2xl">
<p class="sm:h-16">Funds will be used to support development.</p> <h3 class="text-2xl sm:h-12">Donate</h3>
<Link external href="https://geyser.fund/project/flotilla" class="btn btn-primary"> <p class="sm:h-16">Funds will be used to support development.</p>
Support the Developer <Link external href="https://geyser.fund/project/flotilla" class="btn btn-primary">
</Link> Support the Developer
</div> </Link>
</div>
{/if}
<div class="card2 bg-alt flex flex-col gap-2 text-center shadow-2xl"> <div class="card2 bg-alt flex flex-col gap-2 text-center shadow-2xl">
<h3 class="text-2xl sm:h-12">Get in touch</h3> <h3 class="text-2xl sm:h-12">Get in touch</h3>
<p class="sm:h-16">Having problems? Let us know.</p> <p class="sm:h-16">Having problems? Let us know.</p>
+2 -2
View File
@@ -112,7 +112,7 @@
<span class="ellipsize">Requires PoW {limitation?.min_pow_difficulty}</span> <span class="ellipsize">Requires PoW {limitation?.min_pow_difficulty}</span>
</p> </p>
{/if} {/if}
{#if supported_nips} {#if Array.isArray(supported_nips)}
<p class="badge badge-neutral"> <p class="badge badge-neutral">
<span class="ellipsize">NIPs: {supported_nips.join(", ")}</span> <span class="ellipsize">NIPs: {supported_nips.join(", ")}</span>
</p> </p>
@@ -189,8 +189,8 @@
</Button> </Button>
</div> </div>
{#if pubkey} {#if pubkey}
<Divider>Recent posts from the relay admin</Divider>
<div class="hidden flex-col gap-2" class:!flex={relayAdminEvents.length > 0}> <div class="hidden flex-col gap-2" class:!flex={relayAdminEvents.length > 0}>
<Divider>Recent posts from the relay admin</Divider>
<ProfileFeed hideLoading {url} {pubkey} bind:events={relayAdminEvents} /> <ProfileFeed hideLoading {url} {pubkey} bind:events={relayAdminEvents} />
</div> </div>
{/if} {/if}
@@ -42,6 +42,7 @@
import {pushToast} from "@app/toast" import {pushToast} from "@app/toast"
const {room = GENERAL} = $page.params const {room = GENERAL} = $page.params
const mounted = now()
const lastChecked = $checked[$page.url.pathname] const lastChecked = $checked[$page.url.pathname]
const url = decodeRelay($page.params.relay) const url = decodeRelay($page.params.relay)
const filter = {kinds: [MESSAGE], "#h": [room]} const filter = {kinds: [MESSAGE], "#h": [room]}
@@ -170,7 +171,8 @@
!newMessagesSeen && !newMessagesSeen &&
adjustedLastChecked && adjustedLastChecked &&
event.pubkey !== $pubkey && event.pubkey !== $pubkey &&
event.created_at > adjustedLastChecked event.created_at > adjustedLastChecked &&
event.created_at < mounted
) { ) {
elements.push({type: "new-messages", id: "new-messages"}) elements.push({type: "new-messages", id: "new-messages"})
newMessagesSeen = true newMessagesSeen = true
@@ -213,13 +215,17 @@
})) }))
const observer = new ResizeObserver(() => { const observer = new ResizeObserver(() => {
dynamicPadding!.style.minHeight = `${chatCompose!.offsetHeight}px` if (dynamicPadding && chatCompose) {
dynamicPadding!.style.minHeight = `${chatCompose!.offsetHeight}px`
}
}) })
observer.observe(chatCompose!) observer.observe(chatCompose!)
observer.observe(dynamicPadding!)
return () => { return () => {
observer.unobserve(chatCompose!) observer.unobserve(chatCompose!)
observer.unobserve(dynamicPadding!)
} }
}) })