Compare commits
11 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 68ebd32e15 | |||
| e94aa3c119 | |||
| 4d10fe7cc0 | |||
| 841928783b | |||
| 6e5e1a0846 | |||
| d57f4747a6 | |||
| 94a0077b09 | |||
| f2eb04adff | |||
| d4d5979a35 | |||
| dde6e54657 | |||
| 698a7513b8 |
@@ -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
|
||||||
|
|||||||
@@ -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.
|
||||||
|
|||||||
@@ -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'
|
||||||
}
|
}
|
||||||
|
|||||||
Executable
+19
@@ -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
@@ -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
|
||||||
// },
|
// },
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -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
@@ -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",
|
||||||
|
|||||||
@@ -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
@@ -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
@@ -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
|
||||||
|
|||||||
@@ -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()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -126,9 +126,11 @@
|
|||||||
})
|
})
|
||||||
|
|
||||||
observer.observe(chatCompose!)
|
observer.observe(chatCompose!)
|
||||||
|
observer.observe(dynamicPadding!)
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
observer.unobserve(chatCompose!)
|
observer.unobserve(chatCompose!)
|
||||||
|
observer.unobserve(dynamicPadding!)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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>
|
||||||
@@ -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">
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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"
|
||||||
|
|||||||
@@ -29,7 +29,7 @@
|
|||||||
>{displayUrl($relay.profile.contact)}</Link>
|
>{displayUrl($relay.profile.contact)}</Link>
|
||||||
•
|
•
|
||||||
{/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(', ')}">
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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
@@ -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
@@ -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
|
||||||
|
|||||||
@@ -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 |
@@ -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>
|
||||||
|
|||||||
@@ -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">
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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?.()}
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -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}>
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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!)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user