-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'dev' into add_friend_link
- Loading branch information
Showing
11 changed files
with
208 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,154 @@ | ||
<script lang="ts"> | ||
import { Button, Icon, Text } from "$lib/elements" | ||
import { Appearance, CommunitySettingsRoute, Route, SettingsRoute } from "$lib/enums" | ||
import { VoiceRTCInstance } from "$lib/media/Voice" | ||
import { SettingsStore } from "$lib/state" | ||
import { Store } from "$lib/state/Store" | ||
import { UIStore } from "$lib/state/ui" | ||
import type { FriendRequest, NavRoute } from "$lib/types" | ||
import { checkMobile, isAndroidOriOS } from "$lib/utils/Mobile" | ||
import { createEventDispatcher, onDestroy } from "svelte" | ||
import { get } from "svelte/store" | ||
export let routes: NavRoute[] = [] | ||
export let activeRoute: Route | SettingsRoute | CommunitySettingsRoute = Route.Home | ||
export let icons: boolean = false | ||
export let vertical: boolean = false | ||
let incomingRequests: FriendRequest[] = [] | ||
let totalUnreads: number = 0 | ||
let badgeCounts: Record<string, number> = {} | ||
const dispatch = createEventDispatcher() | ||
// Subscribe to stores and update local state | ||
const unsubscribeStore = Store.state.activeRequests.subscribe(r => { | ||
incomingRequests = Store.inboundRequests(r) | ||
updateBadgeCounts() | ||
}) | ||
const unsubscribeUIStore = UIStore.state.chats.subscribe(() => { | ||
totalUnreads = UIStore.getTotalNotifications() | ||
updateBadgeCounts() | ||
}) | ||
function updateBadgeCounts() { | ||
badgeCounts = {} | ||
for (const route of routes) { | ||
badgeCounts[route.to] = calculateBadgeCount(route.to) | ||
} | ||
} | ||
function calculateBadgeCount(route: string): number { | ||
if (route === "/friends") { | ||
return incomingRequests.length | ||
} else if (route === "/chat") { | ||
return totalUnreads | ||
} | ||
return 0 | ||
} | ||
function handleNavigate(route: NavRoute) { | ||
if (checkMobile() && !overrides(route)) UIStore.state.sidebarOpen.set(false) | ||
if (isAndroidOriOS() && (route.to === Route.Settings || (route.to === Route.Chat && get(UIStore.state.chats).length > 0))) { | ||
if (get(UIStore.state.sidebarOpen) === false) { | ||
UIStore.toggleSidebar() | ||
// Avoid keyboard when navigate to chat preview list | ||
} | ||
} | ||
dispatch("navigate", route.to.toString()) | ||
} | ||
function overrides(route: NavRoute) { | ||
if (route.to === Route.Chat && $settings.messaging.quick) { | ||
return true | ||
} | ||
if (route.to === Route.Settings) return true | ||
} | ||
// Clean up subscriptions when component is destroyed | ||
onDestroy(() => { | ||
setTimeout(() => { | ||
if (get(Store.state.activeCall)) { | ||
Store.setActiveCall(Store.getCallingChat(VoiceRTCInstance.channel!)!) | ||
} | ||
}, 100) | ||
unsubscribeStore() | ||
unsubscribeUIStore() | ||
}) | ||
$: settings = SettingsStore.state | ||
</script> | ||
|
||
<div class="content"></div> | ||
|
||
<div class="navigation {vertical ? 'vertical' : 'horizontal'} {icons ? 'icons' : ''}"> | ||
{#each routes as route} | ||
<div class="navigation-control {!icons ? 'fill' : ''}"> | ||
<Button | ||
hook="button-{route.name}" | ||
badge={badgeCounts[route.to]} | ||
fill={!icons} | ||
tooltip={route.name} | ||
icon={icons} | ||
outline={activeRoute !== route.to && !icons} | ||
appearance={activeRoute === route.to ? Appearance.Primary : Appearance.Alt} | ||
on:click={() => handleNavigate(route)}> | ||
<Icon alt={activeRoute === route.to} icon={route.icon} /> | ||
{#if !icons} | ||
<Text appearance={activeRoute !== route.to ? Appearance.Default : Appearance.Alt}>{route.name}</Text> | ||
{/if} | ||
</Button> | ||
</div> | ||
{/each} | ||
</div> | ||
|
||
<style lang="scss"> | ||
.content { | ||
height: 60px; | ||
pointer-events: none; | ||
background: transparent; | ||
border: none; | ||
} | ||
.navigation { | ||
display: inline-flex; | ||
gap: var(--gap); | ||
justify-content: space-evenly; | ||
width: 100%; | ||
border-top: 1px solid var(--border-color, #ccc); | ||
box-shadow: 0px -4px 10px rgba(0, 0, 0, 0.1); | ||
backdrop-filter: blur(10px); | ||
-webkit-backdrop-filter: blur(10px); | ||
z-index: 1000; | ||
position: fixed; | ||
bottom: 0; | ||
left: 0; | ||
right: 0; | ||
padding: var(--padding, 10px); | ||
&.vertical { | ||
flex-direction: column; | ||
} | ||
.navigation-control { | ||
display: inline-flex; | ||
flex-direction: column; | ||
align-items: center; | ||
&.fill { | ||
flex: 1; | ||
:global(.text) { | ||
justify-content: space-between; | ||
} | ||
} | ||
} | ||
} | ||
@media only screen and (max-width: 600px) { | ||
.navigation { | ||
padding-bottom: var(--padding, 10px); | ||
} | ||
} | ||
</style> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters