feat: add relay feeds
This commit is contained in:
@@ -201,6 +201,44 @@ pub async fn get_all_events_by_hashtags(
|
|||||||
Ok(alt_events)
|
Ok(alt_events)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tauri::command]
|
||||||
|
#[specta::specta]
|
||||||
|
pub async fn get_all_events_from(
|
||||||
|
url: String,
|
||||||
|
until: Option<String>,
|
||||||
|
state: State<'_, Nostr>,
|
||||||
|
) -> Result<Vec<RichEvent>, String> {
|
||||||
|
let client = &state.client;
|
||||||
|
|
||||||
|
let _ = client.add_read_relay(&url).await;
|
||||||
|
let _ = client.connect_relay(&url).await;
|
||||||
|
|
||||||
|
let as_of = match until {
|
||||||
|
Some(until) => Timestamp::from_str(&until).map_err(|err| err.to_string())?,
|
||||||
|
None => Timestamp::now(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let filter = Filter::new()
|
||||||
|
.kinds(vec![Kind::TextNote, Kind::Repost])
|
||||||
|
.limit(FETCH_LIMIT)
|
||||||
|
.until(as_of);
|
||||||
|
|
||||||
|
let mut events = Events::new(&[filter.clone()]);
|
||||||
|
|
||||||
|
let mut rx = client
|
||||||
|
.stream_events_from(vec![url], vec![filter], Some(Duration::from_secs(3)))
|
||||||
|
.await
|
||||||
|
.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
|
while let Some(event) = rx.next().await {
|
||||||
|
events.insert(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
let alt_events = process_event(client, events, false).await;
|
||||||
|
|
||||||
|
Ok(alt_events)
|
||||||
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
#[specta::specta]
|
#[specta::specta]
|
||||||
pub async fn get_local_events(
|
pub async fn get_local_events(
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ use nostr_sdk::prelude::*;
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use specta::Type;
|
use specta::Type;
|
||||||
use std::{str::FromStr, time::Duration};
|
use std::{str::FromStr, time::Duration};
|
||||||
use tauri::{Emitter, Manager, State};
|
use tauri::State;
|
||||||
|
|
||||||
use crate::{common::process_event, Nostr, RichEvent, FETCH_LIMIT};
|
use crate::{common::process_event, Nostr, RichEvent, FETCH_LIMIT};
|
||||||
|
|
||||||
@@ -147,7 +147,6 @@ pub async fn set_group(
|
|||||||
image: Option<String>,
|
image: Option<String>,
|
||||||
users: Vec<String>,
|
users: Vec<String>,
|
||||||
state: State<'_, Nostr>,
|
state: State<'_, Nostr>,
|
||||||
handle: tauri::AppHandle,
|
|
||||||
) -> Result<String, String> {
|
) -> Result<String, String> {
|
||||||
let client = &state.client;
|
let client = &state.client;
|
||||||
let public_keys: Vec<PublicKey> = users
|
let public_keys: Vec<PublicKey> = users
|
||||||
@@ -181,25 +180,7 @@ pub async fn set_group(
|
|||||||
.map_err(|err| err.to_string())?;
|
.map_err(|err| err.to_string())?;
|
||||||
|
|
||||||
match client.send_event(event).await {
|
match client.send_event(event).await {
|
||||||
Ok(output) => {
|
Ok(output) => Ok(output.to_hex()),
|
||||||
// Sync event
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
let state = handle.state::<Nostr>();
|
|
||||||
let client = &state.client;
|
|
||||||
|
|
||||||
let filter = Filter::new()
|
|
||||||
.kinds(vec![Kind::TextNote, Kind::Repost])
|
|
||||||
.authors(public_keys)
|
|
||||||
.limit(500);
|
|
||||||
|
|
||||||
if let Ok(report) = client.sync(filter, &SyncOptions::default()).await {
|
|
||||||
println!("Received: {}", report.received.len());
|
|
||||||
handle.emit("synchronized", ()).unwrap();
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(output.to_hex())
|
|
||||||
}
|
|
||||||
Err(err) => Err(err.to_string()),
|
Err(err) => Err(err.to_string()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -296,7 +277,6 @@ pub async fn set_interest(
|
|||||||
image: Option<String>,
|
image: Option<String>,
|
||||||
hashtags: Vec<String>,
|
hashtags: Vec<String>,
|
||||||
state: State<'_, Nostr>,
|
state: State<'_, Nostr>,
|
||||||
handle: tauri::AppHandle,
|
|
||||||
) -> Result<String, String> {
|
) -> Result<String, String> {
|
||||||
let client = &state.client;
|
let client = &state.client;
|
||||||
let label = title.to_lowercase().replace(" ", "-");
|
let label = title.to_lowercase().replace(" ", "-");
|
||||||
@@ -320,25 +300,7 @@ pub async fn set_interest(
|
|||||||
.map_err(|err| err.to_string())?;
|
.map_err(|err| err.to_string())?;
|
||||||
|
|
||||||
match client.send_event(event).await {
|
match client.send_event(event).await {
|
||||||
Ok(output) => {
|
Ok(output) => Ok(output.to_hex()),
|
||||||
// Sync event
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
let state = handle.state::<Nostr>();
|
|
||||||
let client = &state.client;
|
|
||||||
|
|
||||||
let filter = Filter::new()
|
|
||||||
.kinds(vec![Kind::TextNote, Kind::Repost])
|
|
||||||
.hashtags(hashtags)
|
|
||||||
.limit(500);
|
|
||||||
|
|
||||||
if let Ok(report) = client.sync(filter, &SyncOptions::default()).await {
|
|
||||||
println!("Received: {}", report.received.len());
|
|
||||||
handle.emit("synchronized", ()).unwrap();
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(output.to_hex())
|
|
||||||
}
|
|
||||||
Err(err) => Err(err.to_string()),
|
Err(err) => Err(err.to_string()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
use crate::Nostr;
|
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use specta::Type;
|
use specta::Type;
|
||||||
@@ -9,6 +8,8 @@ use std::{
|
|||||||
};
|
};
|
||||||
use tauri::{path::BaseDirectory, Manager, State};
|
use tauri::{path::BaseDirectory, Manager, State};
|
||||||
|
|
||||||
|
use crate::{Nostr, FETCH_LIMIT};
|
||||||
|
|
||||||
#[derive(Serialize, Type)]
|
#[derive(Serialize, Type)]
|
||||||
pub struct Relays {
|
pub struct Relays {
|
||||||
connected: Vec<String>,
|
connected: Vec<String>,
|
||||||
@@ -94,31 +95,59 @@ pub async fn get_relays(id: String, state: State<'_, Nostr>) -> Result<Relays, S
|
|||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
#[specta::specta]
|
#[specta::specta]
|
||||||
pub async fn connect_relay(relay: &str, state: State<'_, Nostr>) -> Result<bool, String> {
|
pub async fn get_all_relays(
|
||||||
|
until: Option<String>,
|
||||||
|
state: State<'_, Nostr>,
|
||||||
|
) -> Result<Vec<String>, String> {
|
||||||
let client = &state.client;
|
let client = &state.client;
|
||||||
let status = client.add_relay(relay).await.map_err(|e| e.to_string())?;
|
|
||||||
|
|
||||||
if status {
|
let as_of = match until {
|
||||||
client
|
Some(until) => Timestamp::from_str(&until).unwrap_or(Timestamp::now()),
|
||||||
.connect_relay(relay)
|
None => Timestamp::now(),
|
||||||
.await
|
};
|
||||||
.map_err(|e| e.to_string())?;
|
|
||||||
}
|
let filter = Filter::new()
|
||||||
|
.kind(Kind::RelayList)
|
||||||
|
.limit(FETCH_LIMIT)
|
||||||
|
.until(as_of);
|
||||||
|
|
||||||
|
let events = client
|
||||||
|
.database()
|
||||||
|
.query(vec![filter])
|
||||||
|
.await
|
||||||
|
.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
|
let alt_events: Vec<String> = events.iter().map(|ev| ev.as_json()).collect();
|
||||||
|
|
||||||
|
Ok(alt_events)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tauri::command]
|
||||||
|
#[specta::specta]
|
||||||
|
pub async fn is_relay_connected(relay: String, state: State<'_, Nostr>) -> Result<bool, String> {
|
||||||
|
let client = &state.client;
|
||||||
|
let status = client.add_relay(&relay).await.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
Ok(status)
|
Ok(status)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
#[specta::specta]
|
#[specta::specta]
|
||||||
pub async fn remove_relay(relay: &str, state: State<'_, Nostr>) -> Result<bool, String> {
|
pub async fn connect_relay(relay: String, state: State<'_, Nostr>) -> Result<(), String> {
|
||||||
let client = &state.client;
|
let client = &state.client;
|
||||||
|
let _ = client.add_relay(&relay).await;
|
||||||
|
let _ = client.connect_relay(&relay).await;
|
||||||
|
|
||||||
client
|
Ok(())
|
||||||
.force_remove_relay(relay)
|
}
|
||||||
.await
|
|
||||||
.map_err(|e| e.to_string())?;
|
|
||||||
|
|
||||||
Ok(true)
|
#[tauri::command]
|
||||||
|
#[specta::specta]
|
||||||
|
pub async fn remove_relay(relay: String, state: State<'_, Nostr>) -> Result<(), String> {
|
||||||
|
let client = &state.client;
|
||||||
|
let _ = client.force_remove_relay(relay).await;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
@@ -140,7 +169,7 @@ pub fn get_bootstrap_relays(app: tauri::AppHandle) -> Result<Vec<String>, String
|
|||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
#[specta::specta]
|
#[specta::specta]
|
||||||
pub fn save_bootstrap_relays(relays: &str, app: tauri::AppHandle) -> Result<(), String> {
|
pub fn set_bootstrap_relays(relays: String, app: tauri::AppHandle) -> Result<(), String> {
|
||||||
let relays_path = app
|
let relays_path = app
|
||||||
.path()
|
.path()
|
||||||
.resolve("resources/relays.txt", BaseDirectory::Resource)
|
.resolve("resources/relays.txt", BaseDirectory::Resource)
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
use border::WebviewWindowExt as BorderWebviewWindowExt;
|
use border::WebviewWindowExt as BorderWebviewWindowExt;
|
||||||
use commands::{account::*, event::*, metadata::*, relay::*, sync::*, window::*};
|
use commands::{account::*, event::*, metadata::*, relay::*, window::*};
|
||||||
use common::{get_all_accounts, parse_event};
|
use common::{get_all_accounts, parse_event};
|
||||||
use nostr_sdk::prelude::{Profile as DatabaseProfile, *};
|
use nostr_sdk::prelude::{Profile as DatabaseProfile, *};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
@@ -71,13 +71,13 @@ fn main() {
|
|||||||
tracing_subscriber::fmt::init();
|
tracing_subscriber::fmt::init();
|
||||||
|
|
||||||
let builder = Builder::<tauri::Wry>::new().commands(collect_commands![
|
let builder = Builder::<tauri::Wry>::new().commands(collect_commands![
|
||||||
sync_account,
|
|
||||||
is_account_sync,
|
|
||||||
get_relays,
|
get_relays,
|
||||||
|
get_all_relays,
|
||||||
|
is_relay_connected,
|
||||||
connect_relay,
|
connect_relay,
|
||||||
remove_relay,
|
remove_relay,
|
||||||
get_bootstrap_relays,
|
get_bootstrap_relays,
|
||||||
save_bootstrap_relays,
|
set_bootstrap_relays,
|
||||||
get_accounts,
|
get_accounts,
|
||||||
watch_account,
|
watch_account,
|
||||||
import_account,
|
import_account,
|
||||||
@@ -116,6 +116,7 @@ fn main() {
|
|||||||
get_all_events_by_author,
|
get_all_events_by_author,
|
||||||
get_all_events_by_authors,
|
get_all_events_by_authors,
|
||||||
get_all_events_by_hashtags,
|
get_all_events_by_hashtags,
|
||||||
|
get_all_events_from,
|
||||||
get_local_events,
|
get_local_events,
|
||||||
get_global_events,
|
get_global_events,
|
||||||
search,
|
search,
|
||||||
|
|||||||
@@ -5,22 +5,6 @@
|
|||||||
|
|
||||||
|
|
||||||
export const commands = {
|
export const commands = {
|
||||||
async syncAccount(id: string, reader: TAURI_CHANNEL<number>) : Promise<Result<null, string>> {
|
|
||||||
try {
|
|
||||||
return { status: "ok", data: await TAURI_INVOKE("sync_account", { id, reader }) };
|
|
||||||
} catch (e) {
|
|
||||||
if(e instanceof Error) throw e;
|
|
||||||
else return { status: "error", error: e as any };
|
|
||||||
}
|
|
||||||
},
|
|
||||||
async isAccountSync(id: string) : Promise<Result<boolean, string>> {
|
|
||||||
try {
|
|
||||||
return { status: "ok", data: await TAURI_INVOKE("is_account_sync", { id }) };
|
|
||||||
} catch (e) {
|
|
||||||
if(e instanceof Error) throw e;
|
|
||||||
else return { status: "error", error: e as any };
|
|
||||||
}
|
|
||||||
},
|
|
||||||
async getRelays(id: string) : Promise<Result<Relays, string>> {
|
async getRelays(id: string) : Promise<Result<Relays, string>> {
|
||||||
try {
|
try {
|
||||||
return { status: "ok", data: await TAURI_INVOKE("get_relays", { id }) };
|
return { status: "ok", data: await TAURI_INVOKE("get_relays", { id }) };
|
||||||
@@ -29,7 +13,23 @@ async getRelays(id: string) : Promise<Result<Relays, string>> {
|
|||||||
else return { status: "error", error: e as any };
|
else return { status: "error", error: e as any };
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async connectRelay(relay: string) : Promise<Result<boolean, string>> {
|
async getAllRelays(until: string | null) : Promise<Result<string[], string>> {
|
||||||
|
try {
|
||||||
|
return { status: "ok", data: await TAURI_INVOKE("get_all_relays", { until }) };
|
||||||
|
} catch (e) {
|
||||||
|
if(e instanceof Error) throw e;
|
||||||
|
else return { status: "error", error: e as any };
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async isRelayConnected(relay: string) : Promise<Result<boolean, string>> {
|
||||||
|
try {
|
||||||
|
return { status: "ok", data: await TAURI_INVOKE("is_relay_connected", { relay }) };
|
||||||
|
} catch (e) {
|
||||||
|
if(e instanceof Error) throw e;
|
||||||
|
else return { status: "error", error: e as any };
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async connectRelay(relay: string) : Promise<Result<null, string>> {
|
||||||
try {
|
try {
|
||||||
return { status: "ok", data: await TAURI_INVOKE("connect_relay", { relay }) };
|
return { status: "ok", data: await TAURI_INVOKE("connect_relay", { relay }) };
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@@ -37,7 +37,7 @@ async connectRelay(relay: string) : Promise<Result<boolean, string>> {
|
|||||||
else return { status: "error", error: e as any };
|
else return { status: "error", error: e as any };
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async removeRelay(relay: string) : Promise<Result<boolean, string>> {
|
async removeRelay(relay: string) : Promise<Result<null, string>> {
|
||||||
try {
|
try {
|
||||||
return { status: "ok", data: await TAURI_INVOKE("remove_relay", { relay }) };
|
return { status: "ok", data: await TAURI_INVOKE("remove_relay", { relay }) };
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@@ -53,9 +53,9 @@ async getBootstrapRelays() : Promise<Result<string[], string>> {
|
|||||||
else return { status: "error", error: e as any };
|
else return { status: "error", error: e as any };
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async saveBootstrapRelays(relays: string) : Promise<Result<null, string>> {
|
async setBootstrapRelays(relays: string) : Promise<Result<null, string>> {
|
||||||
try {
|
try {
|
||||||
return { status: "ok", data: await TAURI_INVOKE("save_bootstrap_relays", { relays }) };
|
return { status: "ok", data: await TAURI_INVOKE("set_bootstrap_relays", { relays }) };
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if(e instanceof Error) throw e;
|
if(e instanceof Error) throw e;
|
||||||
else return { status: "error", error: e as any };
|
else return { status: "error", error: e as any };
|
||||||
@@ -360,6 +360,14 @@ async getAllEventsByHashtags(hashtags: string[], until: string | null) : Promise
|
|||||||
else return { status: "error", error: e as any };
|
else return { status: "error", error: e as any };
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
async getAllEventsFrom(url: string, until: string | null) : Promise<Result<RichEvent[], string>> {
|
||||||
|
try {
|
||||||
|
return { status: "ok", data: await TAURI_INVOKE("get_all_events_from", { url, until }) };
|
||||||
|
} catch (e) {
|
||||||
|
if(e instanceof Error) throw e;
|
||||||
|
else return { status: "error", error: e as any };
|
||||||
|
}
|
||||||
|
},
|
||||||
async getLocalEvents(until: string | null) : Promise<Result<RichEvent[], string>> {
|
async getLocalEvents(until: string | null) : Promise<Result<RichEvent[], string>> {
|
||||||
try {
|
try {
|
||||||
return { status: "ok", data: await TAURI_INVOKE("get_local_events", { until }) };
|
return { status: "ok", data: await TAURI_INVOKE("get_local_events", { until }) };
|
||||||
@@ -523,7 +531,6 @@ export type NewWindow = { label: string; title: string; url: string; width: numb
|
|||||||
export type Relays = { connected: string[]; read: string[] | null; write: string[] | null; both: string[] | null }
|
export type Relays = { connected: string[]; read: string[] | null; write: string[] | null; both: string[] | null }
|
||||||
export type RichEvent = { raw: string; parsed: Meta | null }
|
export type RichEvent = { raw: string; parsed: Meta | null }
|
||||||
export type Settings = { resize_service: boolean; content_warning: boolean; display_avatar: boolean; display_zap_button: boolean; display_repost_button: boolean; display_media: boolean }
|
export type Settings = { resize_service: boolean; content_warning: boolean; display_avatar: boolean; display_zap_button: boolean; display_repost_button: boolean; display_media: boolean }
|
||||||
export type TAURI_CHANNEL<TSend> = null
|
|
||||||
|
|
||||||
/** tauri-specta globals **/
|
/** tauri-specta globals **/
|
||||||
|
|
||||||
|
|||||||
@@ -40,7 +40,7 @@ export function Column({ column }: { column: LumeColumn }) {
|
|||||||
/>
|
/>
|
||||||
<div ref={ref} className="flex-1 size-full">
|
<div ref={ref} className="flex-1 size-full">
|
||||||
<div className="size-full flex flex-col items-center justify-center">
|
<div className="size-full flex flex-col items-center justify-center">
|
||||||
<div className="text-red-500 text-sm break-all">
|
<div className="invisible text-red-500 text-sm break-all">
|
||||||
{error?.length ? error : null}
|
{error?.length ? error : null}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -14,6 +14,8 @@ export const RepostNote = memo(function RepostNote({
|
|||||||
}) {
|
}) {
|
||||||
const { isLoading, isError, data } = useEvent(event.repostId, event.content);
|
const { isLoading, isError, data } = useEvent(event.repostId, event.content);
|
||||||
|
|
||||||
|
console.log("Repost: ", event);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Note.Root className={cn("", className)}>
|
<Note.Root className={cn("", className)}>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
|
|||||||
@@ -51,6 +51,9 @@ const ColumnsLayoutSearchLazyImport = createFileRoute(
|
|||||||
const ColumnsLayoutOnboardingLazyImport = createFileRoute(
|
const ColumnsLayoutOnboardingLazyImport = createFileRoute(
|
||||||
'/columns/_layout/onboarding',
|
'/columns/_layout/onboarding',
|
||||||
)()
|
)()
|
||||||
|
const ColumnsLayoutDiscoverRelaysLazyImport = createFileRoute(
|
||||||
|
'/columns/_layout/discover-relays',
|
||||||
|
)()
|
||||||
const ColumnsLayoutDiscoverNewsfeedsLazyImport = createFileRoute(
|
const ColumnsLayoutDiscoverNewsfeedsLazyImport = createFileRoute(
|
||||||
'/columns/_layout/discover-newsfeeds',
|
'/columns/_layout/discover-newsfeeds',
|
||||||
)()
|
)()
|
||||||
@@ -63,6 +66,9 @@ const ColumnsLayoutUsersIdLazyImport = createFileRoute(
|
|||||||
const ColumnsLayoutRepliesIdLazyImport = createFileRoute(
|
const ColumnsLayoutRepliesIdLazyImport = createFileRoute(
|
||||||
'/columns/_layout/replies/$id',
|
'/columns/_layout/replies/$id',
|
||||||
)()
|
)()
|
||||||
|
const ColumnsLayoutRelaysUrlLazyImport = createFileRoute(
|
||||||
|
'/columns/_layout/relays/$url',
|
||||||
|
)()
|
||||||
const ColumnsLayoutNotificationIdLazyImport = createFileRoute(
|
const ColumnsLayoutNotificationIdLazyImport = createFileRoute(
|
||||||
'/columns/_layout/notification/$id',
|
'/columns/_layout/notification/$id',
|
||||||
)()
|
)()
|
||||||
@@ -202,6 +208,17 @@ const ColumnsLayoutOnboardingLazyRoute =
|
|||||||
import('./routes/columns/_layout/onboarding.lazy').then((d) => d.Route),
|
import('./routes/columns/_layout/onboarding.lazy').then((d) => d.Route),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const ColumnsLayoutDiscoverRelaysLazyRoute =
|
||||||
|
ColumnsLayoutDiscoverRelaysLazyImport.update({
|
||||||
|
id: '/discover-relays',
|
||||||
|
path: '/discover-relays',
|
||||||
|
getParentRoute: () => ColumnsLayoutRoute,
|
||||||
|
} as any).lazy(() =>
|
||||||
|
import('./routes/columns/_layout/discover-relays.lazy').then(
|
||||||
|
(d) => d.Route,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
const ColumnsLayoutDiscoverNewsfeedsLazyRoute =
|
const ColumnsLayoutDiscoverNewsfeedsLazyRoute =
|
||||||
ColumnsLayoutDiscoverNewsfeedsLazyImport.update({
|
ColumnsLayoutDiscoverNewsfeedsLazyImport.update({
|
||||||
id: '/discover-newsfeeds',
|
id: '/discover-newsfeeds',
|
||||||
@@ -279,6 +296,16 @@ const ColumnsLayoutRepliesIdLazyRoute = ColumnsLayoutRepliesIdLazyImport.update(
|
|||||||
import('./routes/columns/_layout/replies.$id.lazy').then((d) => d.Route),
|
import('./routes/columns/_layout/replies.$id.lazy').then((d) => d.Route),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const ColumnsLayoutRelaysUrlLazyRoute = ColumnsLayoutRelaysUrlLazyImport.update(
|
||||||
|
{
|
||||||
|
id: '/relays/$url',
|
||||||
|
path: '/relays/$url',
|
||||||
|
getParentRoute: () => ColumnsLayoutRoute,
|
||||||
|
} as any,
|
||||||
|
).lazy(() =>
|
||||||
|
import('./routes/columns/_layout/relays.$url.lazy').then((d) => d.Route),
|
||||||
|
)
|
||||||
|
|
||||||
const ColumnsLayoutNotificationIdLazyRoute =
|
const ColumnsLayoutNotificationIdLazyRoute =
|
||||||
ColumnsLayoutNotificationIdLazyImport.update({
|
ColumnsLayoutNotificationIdLazyImport.update({
|
||||||
id: '/notification/$id',
|
id: '/notification/$id',
|
||||||
@@ -511,6 +538,13 @@ declare module '@tanstack/react-router' {
|
|||||||
preLoaderRoute: typeof ColumnsLayoutDiscoverNewsfeedsLazyImport
|
preLoaderRoute: typeof ColumnsLayoutDiscoverNewsfeedsLazyImport
|
||||||
parentRoute: typeof ColumnsLayoutImport
|
parentRoute: typeof ColumnsLayoutImport
|
||||||
}
|
}
|
||||||
|
'/columns/_layout/discover-relays': {
|
||||||
|
id: '/columns/_layout/discover-relays'
|
||||||
|
path: '/discover-relays'
|
||||||
|
fullPath: '/columns/discover-relays'
|
||||||
|
preLoaderRoute: typeof ColumnsLayoutDiscoverRelaysLazyImport
|
||||||
|
parentRoute: typeof ColumnsLayoutImport
|
||||||
|
}
|
||||||
'/columns/_layout/onboarding': {
|
'/columns/_layout/onboarding': {
|
||||||
id: '/columns/_layout/onboarding'
|
id: '/columns/_layout/onboarding'
|
||||||
path: '/onboarding'
|
path: '/onboarding'
|
||||||
@@ -595,6 +629,13 @@ declare module '@tanstack/react-router' {
|
|||||||
preLoaderRoute: typeof ColumnsLayoutNotificationIdLazyImport
|
preLoaderRoute: typeof ColumnsLayoutNotificationIdLazyImport
|
||||||
parentRoute: typeof ColumnsLayoutImport
|
parentRoute: typeof ColumnsLayoutImport
|
||||||
}
|
}
|
||||||
|
'/columns/_layout/relays/$url': {
|
||||||
|
id: '/columns/_layout/relays/$url'
|
||||||
|
path: '/relays/$url'
|
||||||
|
fullPath: '/columns/relays/$url'
|
||||||
|
preLoaderRoute: typeof ColumnsLayoutRelaysUrlLazyImport
|
||||||
|
parentRoute: typeof ColumnsLayoutImport
|
||||||
|
}
|
||||||
'/columns/_layout/replies/$id': {
|
'/columns/_layout/replies/$id': {
|
||||||
id: '/columns/_layout/replies/$id'
|
id: '/columns/_layout/replies/$id'
|
||||||
path: '/replies/$id'
|
path: '/replies/$id'
|
||||||
@@ -646,6 +687,7 @@ interface ColumnsLayoutRouteChildren {
|
|||||||
ColumnsLayoutGlobalRoute: typeof ColumnsLayoutGlobalRoute
|
ColumnsLayoutGlobalRoute: typeof ColumnsLayoutGlobalRoute
|
||||||
ColumnsLayoutDiscoverInterestsLazyRoute: typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
ColumnsLayoutDiscoverInterestsLazyRoute: typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
||||||
ColumnsLayoutDiscoverNewsfeedsLazyRoute: typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
ColumnsLayoutDiscoverNewsfeedsLazyRoute: typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
||||||
|
ColumnsLayoutDiscoverRelaysLazyRoute: typeof ColumnsLayoutDiscoverRelaysLazyRoute
|
||||||
ColumnsLayoutOnboardingLazyRoute: typeof ColumnsLayoutOnboardingLazyRoute
|
ColumnsLayoutOnboardingLazyRoute: typeof ColumnsLayoutOnboardingLazyRoute
|
||||||
ColumnsLayoutSearchLazyRoute: typeof ColumnsLayoutSearchLazyRoute
|
ColumnsLayoutSearchLazyRoute: typeof ColumnsLayoutSearchLazyRoute
|
||||||
ColumnsLayoutTrendingLazyRoute: typeof ColumnsLayoutTrendingLazyRoute
|
ColumnsLayoutTrendingLazyRoute: typeof ColumnsLayoutTrendingLazyRoute
|
||||||
@@ -656,6 +698,7 @@ interface ColumnsLayoutRouteChildren {
|
|||||||
ColumnsLayoutEventsIdLazyRoute: typeof ColumnsLayoutEventsIdLazyRoute
|
ColumnsLayoutEventsIdLazyRoute: typeof ColumnsLayoutEventsIdLazyRoute
|
||||||
ColumnsLayoutLaunchpadIdLazyRoute: typeof ColumnsLayoutLaunchpadIdLazyRoute
|
ColumnsLayoutLaunchpadIdLazyRoute: typeof ColumnsLayoutLaunchpadIdLazyRoute
|
||||||
ColumnsLayoutNotificationIdLazyRoute: typeof ColumnsLayoutNotificationIdLazyRoute
|
ColumnsLayoutNotificationIdLazyRoute: typeof ColumnsLayoutNotificationIdLazyRoute
|
||||||
|
ColumnsLayoutRelaysUrlLazyRoute: typeof ColumnsLayoutRelaysUrlLazyRoute
|
||||||
ColumnsLayoutRepliesIdLazyRoute: typeof ColumnsLayoutRepliesIdLazyRoute
|
ColumnsLayoutRepliesIdLazyRoute: typeof ColumnsLayoutRepliesIdLazyRoute
|
||||||
ColumnsLayoutUsersIdLazyRoute: typeof ColumnsLayoutUsersIdLazyRoute
|
ColumnsLayoutUsersIdLazyRoute: typeof ColumnsLayoutUsersIdLazyRoute
|
||||||
}
|
}
|
||||||
@@ -668,6 +711,7 @@ const ColumnsLayoutRouteChildren: ColumnsLayoutRouteChildren = {
|
|||||||
ColumnsLayoutDiscoverInterestsLazyRoute,
|
ColumnsLayoutDiscoverInterestsLazyRoute,
|
||||||
ColumnsLayoutDiscoverNewsfeedsLazyRoute:
|
ColumnsLayoutDiscoverNewsfeedsLazyRoute:
|
||||||
ColumnsLayoutDiscoverNewsfeedsLazyRoute,
|
ColumnsLayoutDiscoverNewsfeedsLazyRoute,
|
||||||
|
ColumnsLayoutDiscoverRelaysLazyRoute: ColumnsLayoutDiscoverRelaysLazyRoute,
|
||||||
ColumnsLayoutOnboardingLazyRoute: ColumnsLayoutOnboardingLazyRoute,
|
ColumnsLayoutOnboardingLazyRoute: ColumnsLayoutOnboardingLazyRoute,
|
||||||
ColumnsLayoutSearchLazyRoute: ColumnsLayoutSearchLazyRoute,
|
ColumnsLayoutSearchLazyRoute: ColumnsLayoutSearchLazyRoute,
|
||||||
ColumnsLayoutTrendingLazyRoute: ColumnsLayoutTrendingLazyRoute,
|
ColumnsLayoutTrendingLazyRoute: ColumnsLayoutTrendingLazyRoute,
|
||||||
@@ -678,6 +722,7 @@ const ColumnsLayoutRouteChildren: ColumnsLayoutRouteChildren = {
|
|||||||
ColumnsLayoutEventsIdLazyRoute: ColumnsLayoutEventsIdLazyRoute,
|
ColumnsLayoutEventsIdLazyRoute: ColumnsLayoutEventsIdLazyRoute,
|
||||||
ColumnsLayoutLaunchpadIdLazyRoute: ColumnsLayoutLaunchpadIdLazyRoute,
|
ColumnsLayoutLaunchpadIdLazyRoute: ColumnsLayoutLaunchpadIdLazyRoute,
|
||||||
ColumnsLayoutNotificationIdLazyRoute: ColumnsLayoutNotificationIdLazyRoute,
|
ColumnsLayoutNotificationIdLazyRoute: ColumnsLayoutNotificationIdLazyRoute,
|
||||||
|
ColumnsLayoutRelaysUrlLazyRoute: ColumnsLayoutRelaysUrlLazyRoute,
|
||||||
ColumnsLayoutRepliesIdLazyRoute: ColumnsLayoutRepliesIdLazyRoute,
|
ColumnsLayoutRepliesIdLazyRoute: ColumnsLayoutRepliesIdLazyRoute,
|
||||||
ColumnsLayoutUsersIdLazyRoute: ColumnsLayoutUsersIdLazyRoute,
|
ColumnsLayoutUsersIdLazyRoute: ColumnsLayoutUsersIdLazyRoute,
|
||||||
}
|
}
|
||||||
@@ -735,6 +780,7 @@ export interface FileRoutesByFullPath {
|
|||||||
'/settings/$id/wallet': typeof SettingsIdWalletRoute
|
'/settings/$id/wallet': typeof SettingsIdWalletRoute
|
||||||
'/columns/discover-interests': typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
'/columns/discover-interests': typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
||||||
'/columns/discover-newsfeeds': typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
'/columns/discover-newsfeeds': typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
||||||
|
'/columns/discover-relays': typeof ColumnsLayoutDiscoverRelaysLazyRoute
|
||||||
'/columns/onboarding': typeof ColumnsLayoutOnboardingLazyRoute
|
'/columns/onboarding': typeof ColumnsLayoutOnboardingLazyRoute
|
||||||
'/columns/search': typeof ColumnsLayoutSearchLazyRoute
|
'/columns/search': typeof ColumnsLayoutSearchLazyRoute
|
||||||
'/columns/trending': typeof ColumnsLayoutTrendingLazyRoute
|
'/columns/trending': typeof ColumnsLayoutTrendingLazyRoute
|
||||||
@@ -747,6 +793,7 @@ export interface FileRoutesByFullPath {
|
|||||||
'/columns/events/$id': typeof ColumnsLayoutEventsIdLazyRoute
|
'/columns/events/$id': typeof ColumnsLayoutEventsIdLazyRoute
|
||||||
'/columns/launchpad/$id': typeof ColumnsLayoutLaunchpadIdLazyRoute
|
'/columns/launchpad/$id': typeof ColumnsLayoutLaunchpadIdLazyRoute
|
||||||
'/columns/notification/$id': typeof ColumnsLayoutNotificationIdLazyRoute
|
'/columns/notification/$id': typeof ColumnsLayoutNotificationIdLazyRoute
|
||||||
|
'/columns/relays/$url': typeof ColumnsLayoutRelaysUrlLazyRoute
|
||||||
'/columns/replies/$id': typeof ColumnsLayoutRepliesIdLazyRoute
|
'/columns/replies/$id': typeof ColumnsLayoutRepliesIdLazyRoute
|
||||||
'/columns/users/$id': typeof ColumnsLayoutUsersIdLazyRoute
|
'/columns/users/$id': typeof ColumnsLayoutUsersIdLazyRoute
|
||||||
}
|
}
|
||||||
@@ -772,6 +819,7 @@ export interface FileRoutesByTo {
|
|||||||
'/settings/$id/wallet': typeof SettingsIdWalletRoute
|
'/settings/$id/wallet': typeof SettingsIdWalletRoute
|
||||||
'/columns/discover-interests': typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
'/columns/discover-interests': typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
||||||
'/columns/discover-newsfeeds': typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
'/columns/discover-newsfeeds': typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
||||||
|
'/columns/discover-relays': typeof ColumnsLayoutDiscoverRelaysLazyRoute
|
||||||
'/columns/onboarding': typeof ColumnsLayoutOnboardingLazyRoute
|
'/columns/onboarding': typeof ColumnsLayoutOnboardingLazyRoute
|
||||||
'/columns/search': typeof ColumnsLayoutSearchLazyRoute
|
'/columns/search': typeof ColumnsLayoutSearchLazyRoute
|
||||||
'/columns/trending': typeof ColumnsLayoutTrendingLazyRoute
|
'/columns/trending': typeof ColumnsLayoutTrendingLazyRoute
|
||||||
@@ -784,6 +832,7 @@ export interface FileRoutesByTo {
|
|||||||
'/columns/events/$id': typeof ColumnsLayoutEventsIdLazyRoute
|
'/columns/events/$id': typeof ColumnsLayoutEventsIdLazyRoute
|
||||||
'/columns/launchpad/$id': typeof ColumnsLayoutLaunchpadIdLazyRoute
|
'/columns/launchpad/$id': typeof ColumnsLayoutLaunchpadIdLazyRoute
|
||||||
'/columns/notification/$id': typeof ColumnsLayoutNotificationIdLazyRoute
|
'/columns/notification/$id': typeof ColumnsLayoutNotificationIdLazyRoute
|
||||||
|
'/columns/relays/$url': typeof ColumnsLayoutRelaysUrlLazyRoute
|
||||||
'/columns/replies/$id': typeof ColumnsLayoutRepliesIdLazyRoute
|
'/columns/replies/$id': typeof ColumnsLayoutRepliesIdLazyRoute
|
||||||
'/columns/users/$id': typeof ColumnsLayoutUsersIdLazyRoute
|
'/columns/users/$id': typeof ColumnsLayoutUsersIdLazyRoute
|
||||||
}
|
}
|
||||||
@@ -812,6 +861,7 @@ export interface FileRoutesById {
|
|||||||
'/settings/$id/wallet': typeof SettingsIdWalletRoute
|
'/settings/$id/wallet': typeof SettingsIdWalletRoute
|
||||||
'/columns/_layout/discover-interests': typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
'/columns/_layout/discover-interests': typeof ColumnsLayoutDiscoverInterestsLazyRoute
|
||||||
'/columns/_layout/discover-newsfeeds': typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
'/columns/_layout/discover-newsfeeds': typeof ColumnsLayoutDiscoverNewsfeedsLazyRoute
|
||||||
|
'/columns/_layout/discover-relays': typeof ColumnsLayoutDiscoverRelaysLazyRoute
|
||||||
'/columns/_layout/onboarding': typeof ColumnsLayoutOnboardingLazyRoute
|
'/columns/_layout/onboarding': typeof ColumnsLayoutOnboardingLazyRoute
|
||||||
'/columns/_layout/search': typeof ColumnsLayoutSearchLazyRoute
|
'/columns/_layout/search': typeof ColumnsLayoutSearchLazyRoute
|
||||||
'/columns/_layout/trending': typeof ColumnsLayoutTrendingLazyRoute
|
'/columns/_layout/trending': typeof ColumnsLayoutTrendingLazyRoute
|
||||||
@@ -824,6 +874,7 @@ export interface FileRoutesById {
|
|||||||
'/columns/_layout/events/$id': typeof ColumnsLayoutEventsIdLazyRoute
|
'/columns/_layout/events/$id': typeof ColumnsLayoutEventsIdLazyRoute
|
||||||
'/columns/_layout/launchpad/$id': typeof ColumnsLayoutLaunchpadIdLazyRoute
|
'/columns/_layout/launchpad/$id': typeof ColumnsLayoutLaunchpadIdLazyRoute
|
||||||
'/columns/_layout/notification/$id': typeof ColumnsLayoutNotificationIdLazyRoute
|
'/columns/_layout/notification/$id': typeof ColumnsLayoutNotificationIdLazyRoute
|
||||||
|
'/columns/_layout/relays/$url': typeof ColumnsLayoutRelaysUrlLazyRoute
|
||||||
'/columns/_layout/replies/$id': typeof ColumnsLayoutRepliesIdLazyRoute
|
'/columns/_layout/replies/$id': typeof ColumnsLayoutRepliesIdLazyRoute
|
||||||
'/columns/_layout/users/$id': typeof ColumnsLayoutUsersIdLazyRoute
|
'/columns/_layout/users/$id': typeof ColumnsLayoutUsersIdLazyRoute
|
||||||
}
|
}
|
||||||
@@ -852,6 +903,7 @@ export interface FileRouteTypes {
|
|||||||
| '/settings/$id/wallet'
|
| '/settings/$id/wallet'
|
||||||
| '/columns/discover-interests'
|
| '/columns/discover-interests'
|
||||||
| '/columns/discover-newsfeeds'
|
| '/columns/discover-newsfeeds'
|
||||||
|
| '/columns/discover-relays'
|
||||||
| '/columns/onboarding'
|
| '/columns/onboarding'
|
||||||
| '/columns/search'
|
| '/columns/search'
|
||||||
| '/columns/trending'
|
| '/columns/trending'
|
||||||
@@ -864,6 +916,7 @@ export interface FileRouteTypes {
|
|||||||
| '/columns/events/$id'
|
| '/columns/events/$id'
|
||||||
| '/columns/launchpad/$id'
|
| '/columns/launchpad/$id'
|
||||||
| '/columns/notification/$id'
|
| '/columns/notification/$id'
|
||||||
|
| '/columns/relays/$url'
|
||||||
| '/columns/replies/$id'
|
| '/columns/replies/$id'
|
||||||
| '/columns/users/$id'
|
| '/columns/users/$id'
|
||||||
fileRoutesByTo: FileRoutesByTo
|
fileRoutesByTo: FileRoutesByTo
|
||||||
@@ -888,6 +941,7 @@ export interface FileRouteTypes {
|
|||||||
| '/settings/$id/wallet'
|
| '/settings/$id/wallet'
|
||||||
| '/columns/discover-interests'
|
| '/columns/discover-interests'
|
||||||
| '/columns/discover-newsfeeds'
|
| '/columns/discover-newsfeeds'
|
||||||
|
| '/columns/discover-relays'
|
||||||
| '/columns/onboarding'
|
| '/columns/onboarding'
|
||||||
| '/columns/search'
|
| '/columns/search'
|
||||||
| '/columns/trending'
|
| '/columns/trending'
|
||||||
@@ -900,6 +954,7 @@ export interface FileRouteTypes {
|
|||||||
| '/columns/events/$id'
|
| '/columns/events/$id'
|
||||||
| '/columns/launchpad/$id'
|
| '/columns/launchpad/$id'
|
||||||
| '/columns/notification/$id'
|
| '/columns/notification/$id'
|
||||||
|
| '/columns/relays/$url'
|
||||||
| '/columns/replies/$id'
|
| '/columns/replies/$id'
|
||||||
| '/columns/users/$id'
|
| '/columns/users/$id'
|
||||||
id:
|
id:
|
||||||
@@ -926,6 +981,7 @@ export interface FileRouteTypes {
|
|||||||
| '/settings/$id/wallet'
|
| '/settings/$id/wallet'
|
||||||
| '/columns/_layout/discover-interests'
|
| '/columns/_layout/discover-interests'
|
||||||
| '/columns/_layout/discover-newsfeeds'
|
| '/columns/_layout/discover-newsfeeds'
|
||||||
|
| '/columns/_layout/discover-relays'
|
||||||
| '/columns/_layout/onboarding'
|
| '/columns/_layout/onboarding'
|
||||||
| '/columns/_layout/search'
|
| '/columns/_layout/search'
|
||||||
| '/columns/_layout/trending'
|
| '/columns/_layout/trending'
|
||||||
@@ -938,6 +994,7 @@ export interface FileRouteTypes {
|
|||||||
| '/columns/_layout/events/$id'
|
| '/columns/_layout/events/$id'
|
||||||
| '/columns/_layout/launchpad/$id'
|
| '/columns/_layout/launchpad/$id'
|
||||||
| '/columns/_layout/notification/$id'
|
| '/columns/_layout/notification/$id'
|
||||||
|
| '/columns/_layout/relays/$url'
|
||||||
| '/columns/_layout/replies/$id'
|
| '/columns/_layout/replies/$id'
|
||||||
| '/columns/_layout/users/$id'
|
| '/columns/_layout/users/$id'
|
||||||
fileRoutesById: FileRoutesById
|
fileRoutesById: FileRoutesById
|
||||||
@@ -1037,6 +1094,7 @@ export const routeTree = rootRoute
|
|||||||
"/columns/_layout/global",
|
"/columns/_layout/global",
|
||||||
"/columns/_layout/discover-interests",
|
"/columns/_layout/discover-interests",
|
||||||
"/columns/_layout/discover-newsfeeds",
|
"/columns/_layout/discover-newsfeeds",
|
||||||
|
"/columns/_layout/discover-relays",
|
||||||
"/columns/_layout/onboarding",
|
"/columns/_layout/onboarding",
|
||||||
"/columns/_layout/search",
|
"/columns/_layout/search",
|
||||||
"/columns/_layout/trending",
|
"/columns/_layout/trending",
|
||||||
@@ -1047,6 +1105,7 @@ export const routeTree = rootRoute
|
|||||||
"/columns/_layout/events/$id",
|
"/columns/_layout/events/$id",
|
||||||
"/columns/_layout/launchpad/$id",
|
"/columns/_layout/launchpad/$id",
|
||||||
"/columns/_layout/notification/$id",
|
"/columns/_layout/notification/$id",
|
||||||
|
"/columns/_layout/relays/$url",
|
||||||
"/columns/_layout/replies/$id",
|
"/columns/_layout/replies/$id",
|
||||||
"/columns/_layout/users/$id"
|
"/columns/_layout/users/$id"
|
||||||
]
|
]
|
||||||
@@ -1110,6 +1169,10 @@ export const routeTree = rootRoute
|
|||||||
"filePath": "columns/_layout/discover-newsfeeds.lazy.tsx",
|
"filePath": "columns/_layout/discover-newsfeeds.lazy.tsx",
|
||||||
"parent": "/columns/_layout"
|
"parent": "/columns/_layout"
|
||||||
},
|
},
|
||||||
|
"/columns/_layout/discover-relays": {
|
||||||
|
"filePath": "columns/_layout/discover-relays.lazy.tsx",
|
||||||
|
"parent": "/columns/_layout"
|
||||||
|
},
|
||||||
"/columns/_layout/onboarding": {
|
"/columns/_layout/onboarding": {
|
||||||
"filePath": "columns/_layout/onboarding.lazy.tsx",
|
"filePath": "columns/_layout/onboarding.lazy.tsx",
|
||||||
"parent": "/columns/_layout"
|
"parent": "/columns/_layout"
|
||||||
@@ -1158,6 +1221,10 @@ export const routeTree = rootRoute
|
|||||||
"filePath": "columns/_layout/notification.$id.lazy.tsx",
|
"filePath": "columns/_layout/notification.$id.lazy.tsx",
|
||||||
"parent": "/columns/_layout"
|
"parent": "/columns/_layout"
|
||||||
},
|
},
|
||||||
|
"/columns/_layout/relays/$url": {
|
||||||
|
"filePath": "columns/_layout/relays.$url.lazy.tsx",
|
||||||
|
"parent": "/columns/_layout"
|
||||||
|
},
|
||||||
"/columns/_layout/replies/$id": {
|
"/columns/_layout/replies/$id": {
|
||||||
"filePath": "columns/_layout/replies.$id.lazy.tsx",
|
"filePath": "columns/_layout/replies.$id.lazy.tsx",
|
||||||
"parent": "/columns/_layout"
|
"parent": "/columns/_layout"
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ function Screen() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const merged = relays.join("\r\n");
|
const merged = relays.join("\r\n");
|
||||||
const res = await commands.saveBootstrapRelays(merged);
|
const res = await commands.setBootstrapRelays(merged);
|
||||||
|
|
||||||
if (res.status === "ok") {
|
if (res.status === "ok") {
|
||||||
return await relaunch();
|
return await relaunch();
|
||||||
|
|||||||
154
src/routes/columns/_layout/discover-relays.lazy.tsx
Normal file
154
src/routes/columns/_layout/discover-relays.lazy.tsx
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
import { commands } from "@/commands.gen";
|
||||||
|
import { Spinner, User } from "@/components";
|
||||||
|
import { LumeWindow } from "@/system";
|
||||||
|
import type { NostrEvent } from "@/types";
|
||||||
|
import { ArrowDown } from "@phosphor-icons/react";
|
||||||
|
import * as ScrollArea from "@radix-ui/react-scroll-area";
|
||||||
|
import { useInfiniteQuery } from "@tanstack/react-query";
|
||||||
|
import { createLazyFileRoute } from "@tanstack/react-router";
|
||||||
|
import { type RefObject, useCallback, useRef } from "react";
|
||||||
|
import { Virtualizer } from "virtua";
|
||||||
|
|
||||||
|
export const Route = createLazyFileRoute("/columns/_layout/discover-relays")({
|
||||||
|
component: Screen,
|
||||||
|
});
|
||||||
|
|
||||||
|
function Screen() {
|
||||||
|
const {
|
||||||
|
isLoading,
|
||||||
|
isError,
|
||||||
|
error,
|
||||||
|
isFetchingNextPage,
|
||||||
|
hasNextPage,
|
||||||
|
fetchNextPage,
|
||||||
|
data,
|
||||||
|
} = useInfiniteQuery({
|
||||||
|
queryKey: ["discover-relays"],
|
||||||
|
initialPageParam: 0,
|
||||||
|
queryFn: async ({ pageParam }: { pageParam: number }) => {
|
||||||
|
const until = pageParam > 0 ? pageParam.toString() : null;
|
||||||
|
const res = await commands.getAllRelays(until);
|
||||||
|
|
||||||
|
if (res.status === "ok") {
|
||||||
|
const data: NostrEvent[] = res.data.map((item) => JSON.parse(item));
|
||||||
|
return data;
|
||||||
|
} else {
|
||||||
|
throw new Error(res.error);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getNextPageParam: (lastPage) => {
|
||||||
|
const lastEvent = lastPage.at(-1);
|
||||||
|
|
||||||
|
if (lastEvent) {
|
||||||
|
return lastEvent.created_at - 1;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
select: (data) => data?.pages.flat(),
|
||||||
|
refetchOnWindowFocus: false,
|
||||||
|
refetchOnReconnect: false,
|
||||||
|
refetchOnMount: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
const ref = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
const renderItem = useCallback(
|
||||||
|
(item: NostrEvent) => {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={item.id}
|
||||||
|
className="mb-3 flex flex-col rounded-xl overflow-hidden bg-white dark:bg-neutral-800/50 shadow-lg shadow-primary dark:ring-1 dark:ring-neutral-800"
|
||||||
|
>
|
||||||
|
<div className="flex flex-col gap-2 p-2">
|
||||||
|
{item.tags.map((tag) =>
|
||||||
|
tag[1]?.startsWith("wss://") ? (
|
||||||
|
<div
|
||||||
|
key={tag[1]}
|
||||||
|
className="group px-3 flex items-center justify-between h-11 rounded-lg bg-neutral-100 dark:bg-neutral-800"
|
||||||
|
>
|
||||||
|
<div className="flex-1 truncate select-text text-sm font-medium">
|
||||||
|
{tag[1]}
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() =>
|
||||||
|
LumeWindow.openColumn({
|
||||||
|
name: tag[1],
|
||||||
|
label: `relays_${tag[1].replace(/[^\w\s]/gi, "")}`,
|
||||||
|
url: `/columns/relays/${encodeURIComponent(tag[1])}`,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
className="hidden h-6 w-24 shrink-0 group-hover:inline-flex items-center justify-center gap-1 text-xs font-semibold rounded-full bg-neutral-200 dark:bg-neutral-700 hover:bg-blue-500 hover:text-white"
|
||||||
|
>
|
||||||
|
View event
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
) : null,
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="p-2 flex items-center">
|
||||||
|
<User.Provider pubkey={item.pubkey}>
|
||||||
|
<User.Root className="inline-flex items-center gap-2">
|
||||||
|
<User.Avatar className="size-7 rounded-full" />
|
||||||
|
<User.Name className="text-xs font-medium" />
|
||||||
|
</User.Root>
|
||||||
|
</User.Provider>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[data],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ScrollArea.Root
|
||||||
|
type={"scroll"}
|
||||||
|
scrollHideDelay={300}
|
||||||
|
className="overflow-hidden size-full"
|
||||||
|
>
|
||||||
|
<ScrollArea.Viewport ref={ref} className="relative h-full px-3 pb-3">
|
||||||
|
<Virtualizer scrollRef={ref as unknown as RefObject<HTMLElement>}>
|
||||||
|
{isLoading ? (
|
||||||
|
<div className="inline-flex items-center gap-1.5">
|
||||||
|
<Spinner className="size-4" />
|
||||||
|
Loading...
|
||||||
|
</div>
|
||||||
|
) : isError ? (
|
||||||
|
<div className="mb-3 flex flex-col items-center justify-center h-16 w-full rounded-xl overflow-hidden bg-neutral-200/50 dark:bg-neutral-800/50">
|
||||||
|
<p className="text-center">{error?.message ?? "Error"}</p>
|
||||||
|
</div>
|
||||||
|
) : !data?.length ? (
|
||||||
|
<div className="mb-3 flex flex-col items-center justify-center h-16 w-full rounded-xl overflow-hidden bg-neutral-200/50 dark:bg-neutral-800/50">
|
||||||
|
<p className="text-center">Empty.</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
data?.map((item) => renderItem(item))
|
||||||
|
)}
|
||||||
|
{hasNextPage ? (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => fetchNextPage()}
|
||||||
|
disabled={isFetchingNextPage || isLoading}
|
||||||
|
className="h-11 w-full px-3 flex items-center justify-center gap-1.5 bg-neutral-200/50 hover:bg-neutral-200 rounded-full text-sm font-medium text-blue-600 dark:hover:bg-neutral-800 dark:bg-neutral-800/50 dark:text-blue-400"
|
||||||
|
>
|
||||||
|
{isFetchingNextPage ? (
|
||||||
|
<Spinner className="size-4" />
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<ArrowDown className="size-4" />
|
||||||
|
Load more
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
) : null}
|
||||||
|
</Virtualizer>
|
||||||
|
</ScrollArea.Viewport>
|
||||||
|
<ScrollArea.Scrollbar
|
||||||
|
className="flex select-none touch-none p-0.5 duration-[160ms] ease-out data-[orientation=vertical]:w-2"
|
||||||
|
orientation="vertical"
|
||||||
|
>
|
||||||
|
<ScrollArea.Thumb className="flex-1 bg-black/10 dark:bg-white/10 rounded-full relative before:content-[''] before:absolute before:top-1/2 before:left-1/2 before:-translate-x-1/2 before:-translate-y-1/2 before:w-full before:h-full before:min-w-[44px] before:min-h-[44px]" />
|
||||||
|
</ScrollArea.Scrollbar>
|
||||||
|
<ScrollArea.Corner className="bg-transparent" />
|
||||||
|
</ScrollArea.Root>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -26,7 +26,7 @@ export function Screen() {
|
|||||||
hasNextPage,
|
hasNextPage,
|
||||||
fetchNextPage,
|
fetchNextPage,
|
||||||
} = useInfiniteQuery({
|
} = useInfiniteQuery({
|
||||||
queryKey: ["events", "groups", params.id],
|
queryKey: ["groups", params.id],
|
||||||
initialPageParam: 0,
|
initialPageParam: 0,
|
||||||
queryFn: async ({ pageParam }: { pageParam: number }) => {
|
queryFn: async ({ pageParam }: { pageParam: number }) => {
|
||||||
const until = pageParam > 0 ? pageParam.toString() : null;
|
const until = pageParam > 0 ? pageParam.toString() : null;
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ export function Screen() {
|
|||||||
hasNextPage,
|
hasNextPage,
|
||||||
fetchNextPage,
|
fetchNextPage,
|
||||||
} = useInfiniteQuery({
|
} = useInfiniteQuery({
|
||||||
queryKey: ["events", "hashtags", params.id],
|
queryKey: ["hashtags", params.id],
|
||||||
initialPageParam: 0,
|
initialPageParam: 0,
|
||||||
queryFn: async ({ pageParam }: { pageParam: number }) => {
|
queryFn: async ({ pageParam }: { pageParam: number }) => {
|
||||||
const tags = hashtags.map((tag) => tag.toLowerCase().replace("#", ""));
|
const tags = hashtags.map((tag) => tag.toLowerCase().replace("#", ""));
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ function Screen() {
|
|||||||
function Newsfeeds() {
|
function Newsfeeds() {
|
||||||
const { id } = Route.useParams();
|
const { id } = Route.useParams();
|
||||||
const { isLoading, isError, error, data, refetch, isRefetching } = useQuery({
|
const { isLoading, isError, error, data, refetch, isRefetching } = useQuery({
|
||||||
queryKey: ["others", "newsfeeds", id],
|
queryKey: ["newsfeeds", id],
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const res = await commands.getAllNewsfeeds(id);
|
const res = await commands.getAllNewsfeeds(id);
|
||||||
|
|
||||||
@@ -204,7 +204,7 @@ function Newsfeeds() {
|
|||||||
function Interests() {
|
function Interests() {
|
||||||
const { id } = Route.useParams();
|
const { id } = Route.useParams();
|
||||||
const { isLoading, isError, error, data, refetch, isRefetching } = useQuery({
|
const { isLoading, isError, error, data, refetch, isRefetching } = useQuery({
|
||||||
queryKey: ["others", "interests", id],
|
queryKey: ["interests", id],
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const res = await commands.getAllInterests(id);
|
const res = await commands.getAllInterests(id);
|
||||||
|
|
||||||
@@ -399,6 +399,22 @@ function Core() {
|
|||||||
Add
|
Add
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
<div className="px-3 flex items-center justify-between h-11 rounded-lg bg-neutral-100 dark:bg-neutral-800">
|
||||||
|
<div className="text-sm font-medium">Relays</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() =>
|
||||||
|
LumeWindow.openColumn({
|
||||||
|
name: "Relays",
|
||||||
|
label: "relays",
|
||||||
|
url: "/columns/discover-relays",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
className="h-6 w-16 inline-flex items-center justify-center gap-1 text-xs font-semibold rounded-full bg-neutral-200 dark:bg-neutral-700 hover:bg-blue-500 hover:text-white"
|
||||||
|
>
|
||||||
|
Add
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
{data?.map((column) => (
|
{data?.map((column) => (
|
||||||
<div
|
<div
|
||||||
key={column.label}
|
key={column.label}
|
||||||
|
|||||||
@@ -7,8 +7,7 @@ import { ArrowDown } from "@phosphor-icons/react";
|
|||||||
import * as ScrollArea from "@radix-ui/react-scroll-area";
|
import * as ScrollArea from "@radix-ui/react-scroll-area";
|
||||||
import { useInfiniteQuery } from "@tanstack/react-query";
|
import { useInfiniteQuery } from "@tanstack/react-query";
|
||||||
import { createLazyFileRoute } from "@tanstack/react-router";
|
import { createLazyFileRoute } from "@tanstack/react-router";
|
||||||
import { listen } from "@tauri-apps/api/event";
|
import { type RefObject, useCallback, useRef } from "react";
|
||||||
import { type RefObject, useCallback, useEffect, useRef } from "react";
|
|
||||||
import { Virtualizer } from "virtua";
|
import { Virtualizer } from "virtua";
|
||||||
|
|
||||||
export const Route = createLazyFileRoute("/columns/_layout/newsfeed/$id")({
|
export const Route = createLazyFileRoute("/columns/_layout/newsfeed/$id")({
|
||||||
@@ -18,7 +17,7 @@ export const Route = createLazyFileRoute("/columns/_layout/newsfeed/$id")({
|
|||||||
export function Screen() {
|
export function Screen() {
|
||||||
const contacts = Route.useLoaderData();
|
const contacts = Route.useLoaderData();
|
||||||
const search = Route.useSearch();
|
const search = Route.useSearch();
|
||||||
const { queryClient } = Route.useRouteContext();
|
|
||||||
const {
|
const {
|
||||||
data,
|
data,
|
||||||
isLoading,
|
isLoading,
|
||||||
@@ -83,18 +82,6 @@ export function Screen() {
|
|||||||
[data],
|
[data],
|
||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const unlisten = listen("synchronized", async () => {
|
|
||||||
await queryClient.refetchQueries({
|
|
||||||
queryKey: ["events", "newsfeed", search.label],
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
unlisten.then((f) => f());
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ScrollArea.Root
|
<ScrollArea.Root
|
||||||
type={"scroll"}
|
type={"scroll"}
|
||||||
|
|||||||
144
src/routes/columns/_layout/relays.$url.lazy.tsx
Normal file
144
src/routes/columns/_layout/relays.$url.lazy.tsx
Normal file
@@ -0,0 +1,144 @@
|
|||||||
|
import { commands } from "@/commands.gen";
|
||||||
|
import { toLumeEvents } from "@/commons";
|
||||||
|
import { RepostNote, Spinner, TextNote } from "@/components";
|
||||||
|
import type { LumeEvent } from "@/system";
|
||||||
|
import { Kind } from "@/types";
|
||||||
|
import { ArrowDown } from "@phosphor-icons/react";
|
||||||
|
import * as ScrollArea from "@radix-ui/react-scroll-area";
|
||||||
|
import { useInfiniteQuery } from "@tanstack/react-query";
|
||||||
|
import { createLazyFileRoute } from "@tanstack/react-router";
|
||||||
|
import { type RefObject, useCallback, useRef } from "react";
|
||||||
|
import { Virtualizer } from "virtua";
|
||||||
|
|
||||||
|
export const Route = createLazyFileRoute("/columns/_layout/relays/$url")({
|
||||||
|
component: Screen,
|
||||||
|
});
|
||||||
|
|
||||||
|
export function Screen() {
|
||||||
|
const { url } = Route.useParams();
|
||||||
|
const {
|
||||||
|
data,
|
||||||
|
isLoading,
|
||||||
|
isFetching,
|
||||||
|
isFetchingNextPage,
|
||||||
|
hasNextPage,
|
||||||
|
fetchNextPage,
|
||||||
|
} = useInfiniteQuery({
|
||||||
|
queryKey: ["relays", url],
|
||||||
|
initialPageParam: 0,
|
||||||
|
queryFn: async ({ pageParam }: { pageParam: number }) => {
|
||||||
|
const until = pageParam > 0 ? pageParam.toString() : null;
|
||||||
|
const relay = decodeURIComponent(url);
|
||||||
|
const res = await commands.getAllEventsFrom(relay, until);
|
||||||
|
|
||||||
|
if (res.status === "error") {
|
||||||
|
throw new Error(res.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
return toLumeEvents(res.data);
|
||||||
|
},
|
||||||
|
getNextPageParam: (lastPage) => {
|
||||||
|
const lastEvent = lastPage.at(-1);
|
||||||
|
|
||||||
|
if (lastEvent) {
|
||||||
|
return lastEvent.created_at - 1;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
select: (data) => data?.pages.flat(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const ref = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
const renderItem = useCallback(
|
||||||
|
(event: LumeEvent) => {
|
||||||
|
if (!event) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (event.kind) {
|
||||||
|
case Kind.Repost: {
|
||||||
|
const repostId = event.repostId;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<RepostNote
|
||||||
|
key={repostId + event.id}
|
||||||
|
event={event}
|
||||||
|
className="border-b-[.5px] border-neutral-300 dark:border-neutral-700"
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return (
|
||||||
|
<TextNote
|
||||||
|
key={event.id}
|
||||||
|
event={event}
|
||||||
|
className="border-b-[.5px] border-neutral-300 dark:border-neutral-700"
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[data],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ScrollArea.Root
|
||||||
|
type={"scroll"}
|
||||||
|
scrollHideDelay={300}
|
||||||
|
className="overflow-hidden size-full px-3"
|
||||||
|
>
|
||||||
|
<ScrollArea.Viewport
|
||||||
|
ref={ref}
|
||||||
|
className="relative h-full bg-white dark:bg-neutral-800 rounded-t-xl shadow shadow-neutral-300/50 dark:shadow-none border-[.5px] border-neutral-300 dark:border-neutral-700"
|
||||||
|
>
|
||||||
|
<Virtualizer scrollRef={ref as unknown as RefObject<HTMLElement>}>
|
||||||
|
{isFetching && !isLoading && !isFetchingNextPage ? (
|
||||||
|
<div className="z-50 fixed top-0 left-0 w-full h-14 flex items-center justify-center px-3">
|
||||||
|
<div className="w-max h-8 pl-2 pr-3 inline-flex items-center justify-center gap-1.5 rounded-full shadow-lg text-sm font-medium text-white bg-black dark:text-black dark:bg-white">
|
||||||
|
<Spinner className="size-4" />
|
||||||
|
Getting new notes...
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
|
{isLoading ? (
|
||||||
|
<div className="flex items-center justify-center w-full h-16 gap-2">
|
||||||
|
<Spinner className="size-4" />
|
||||||
|
<span className="text-sm font-medium">Loading...</span>
|
||||||
|
</div>
|
||||||
|
) : !data?.length ? (
|
||||||
|
<div className="mb-3 flex items-center justify-center h-20 text-sm">
|
||||||
|
🎉 Yo. You're catching up on all latest notes.
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
data.map((item) => renderItem(item))
|
||||||
|
)}
|
||||||
|
{hasNextPage ? (
|
||||||
|
<div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => fetchNextPage()}
|
||||||
|
disabled={isFetchingNextPage || isLoading}
|
||||||
|
className="inline-flex items-center justify-center w-full gap-2 px-3 text-sm font-medium text-blue-500 h-11 focus:outline-none"
|
||||||
|
>
|
||||||
|
{isFetchingNextPage ? (
|
||||||
|
<Spinner className="size-4" />
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<ArrowDown className="size-4" />
|
||||||
|
Load more
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
|
</Virtualizer>
|
||||||
|
</ScrollArea.Viewport>
|
||||||
|
<ScrollArea.Scrollbar
|
||||||
|
className="flex select-none touch-none p-0.5 duration-[160ms] ease-out data-[orientation=vertical]:w-2"
|
||||||
|
orientation="vertical"
|
||||||
|
>
|
||||||
|
<ScrollArea.Thumb className="flex-1 bg-black/10 dark:bg-white/10 rounded-full relative before:content-[''] before:absolute before:top-1/2 before:left-1/2 before:-translate-x-1/2 before:-translate-y-1/2 before:w-full before:h-full before:min-w-[44px] before:min-h-[44px]" />
|
||||||
|
</ScrollArea.Scrollbar>
|
||||||
|
<ScrollArea.Corner className="bg-transparent" />
|
||||||
|
</ScrollArea.Root>
|
||||||
|
);
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user