Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
718 changes: 718 additions & 0 deletions packages/mobile/STREAMING_SPEC.md

Large diffs are not rendered by default.

212 changes: 169 additions & 43 deletions packages/mobile/app/(main)/_layout.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
import { useCallback, useEffect, useMemo, useState } from "react"
import { useCallback, useContext, useEffect, useMemo, useState } from "react"
import { View, StyleSheet, useWindowDimensions, Platform, type ViewProps } from "react-native"
import { Stack, useRouter } from "expo-router"
import { Drawer, useDrawerProgress } from "react-native-drawer-layout"
import { Drawer, DrawerGestureContext, useDrawerProgress } from "react-native-drawer-layout"
import Animated, { interpolate, useAnimatedStyle } from "react-native-reanimated"
import type { Session } from "@opencode-ai/sdk/client"
import { Sidebar } from "../../src/components/sidebar"
import { SessionDiffPanel } from "../../src/components/session-diff-panel"
import { ConnectionBanner } from "../../src/components/connection-banner"
import { useTheme } from "../../src/theme"
import { useSessions } from "../../src/store/sessions"
import { useConnection } from "../../src/store/connection"
import { useSidebar } from "../../src/store/sidebar"
import { addCrashBreadcrumb } from "../../src/perf/crash-breadcrumbs"
import { telemetry } from "../../src/perf/telemetry"

const AnimatedView = Animated.View as React.ComponentType<ViewProps & { style?: unknown; children?: React.ReactNode }>
const SWIPE_SURFACE_OVERLAY_OPACITY = 0.14
Expand All @@ -22,6 +24,7 @@ export default function MainLayout() {
const { width } = useWindowDimensions()
const isTablet = width >= 768
const [open, setOpen] = useState(isTablet)
const [diffOpen, setDiffOpen] = useState(false)
const currentSessionID = useSessions((s) => s.current)
const select = useSessions((s) => s.select)
const directory = useConnection((s) => s.directory)
Expand All @@ -32,6 +35,11 @@ export default function MainLayout() {
setOpen(isTablet)
}, [isTablet])

// Close diff panel when session changes
useEffect(() => {
setDiffOpen(false)
}, [currentSessionID])

const setOpenIfChanged = useCallback((next: boolean) => {
setOpen((current) => (current === next ? current : next))
}, [])
Expand All @@ -50,6 +58,7 @@ export default function MainLayout() {
return
}
if (!isTablet) setOpenIfChanged(false)
telemetry.track("session", "session:switch", { sessionID: session.id })
addCrashBreadcrumb("session-switch:start", {
sessionID: session.id,
toDirectory: session.directory,
Expand All @@ -61,14 +70,18 @@ export default function MainLayout() {
[currentSessionID, directory, isTablet, router, setOpenIfChanged],
)

const onNewSession = useCallback(async (worktree?: string) => {
if (worktree && worktree !== directory) {
await switchDirectory(worktree)
}
select(null)
if (!isTablet) setOpenIfChanged(false)
router.replace("/(main)/session")
}, [directory, isTablet, router, select, setOpenIfChanged, switchDirectory])
const onNewSession = useCallback(
async (worktree?: string) => {
telemetry.track("session", "session:new", { worktree: worktree ?? null })
if (worktree && worktree !== directory) {
await switchDirectory(worktree)
}
select(null)
if (!isTablet) setOpenIfChanged(false)
router.replace("/(main)/session")
},
[directory, isTablet, router, select, setOpenIfChanged, switchDirectory],
)

const onSettings = useCallback(() => {
if (!isTablet) setOpenIfChanged(false)
Expand All @@ -80,15 +93,23 @@ export default function MainLayout() {
setOpenIfChanged(false)
}, [isTablet, setOpenIfChanged])

const drawerStyle = useMemo(
const sidebarDrawerStyle = useMemo(
() => ({
width: isTablet ? 320 : width,
backgroundColor: "transparent",
}),
[isTablet, width],
)

const renderDrawerContent = useCallback(
const diffDrawerStyle = useMemo(
() => ({
width: isTablet ? width - 320 : width,
backgroundColor: "transparent",
}),
[isTablet, width],
)

const renderSidebarContent = useCallback(
() => (
<Sidebar
onSelect={onSelectSession}
Expand All @@ -105,30 +126,129 @@ export default function MainLayout() {
<Drawer
open={isTablet ? true : open}
onOpen={() => {
if (!isTablet) setOpenIfChanged(true)
if (!isTablet) {
telemetry.track("drawer", "drawer:left:open")
setOpenIfChanged(true)
}
}}
onClose={() => {
if (!isTablet) setOpenIfChanged(false)
if (!isTablet) {
telemetry.track("drawer", "drawer:left:close")
setOpenIfChanged(false)
}
}}
drawerType={isTablet ? "permanent" : "slide"}
swipeEnabled={!isTablet}
swipeEnabled={!isTablet && !diffOpen}
swipeEdgeWidth={isTablet ? 0 : width}
swipeMinDistance={DRAWER_SWIPE_MIN_DISTANCE}
swipeMinVelocity={DRAWER_SWIPE_MIN_VELOCITY}
overlayStyle={styles.drawerOverlay}
drawerStyle={drawerStyle}
renderDrawerContent={renderDrawerContent}
drawerStyle={sidebarDrawerStyle}
renderDrawerContent={renderSidebarContent}
>
<SlidingContent isTablet={isTablet} />
<LeftOverlay isTablet={isTablet}>
<RightDrawer
isTablet={isTablet}
diffOpen={diffOpen}
setDiffOpen={setDiffOpen}
currentSessionID={currentSessionID}
sidebarOpen={open}
diffDrawerStyle={diffDrawerStyle}
/>
</LeftOverlay>
</Drawer>
)
}

function SlidingContent({ isTablet }: { isTablet: boolean }) {
function RightDrawer({
isTablet,
diffOpen,
setDiffOpen,
currentSessionID,
sidebarOpen,
diffDrawerStyle,
}: {
isTablet: boolean
diffOpen: boolean
setDiffOpen: (open: boolean) => void
currentSessionID: string | null
sidebarOpen: boolean
diffDrawerStyle: { width: number; backgroundColor: string }
}) {
const { width } = useWindowDimensions()
const parentGesture = useContext(DrawerGestureContext as React.Context<unknown>)

const renderDiffContent = useCallback(
() =>
currentSessionID ? (
<SessionDiffPanel sessionId={currentSessionID} />
) : (
<View style={{ flex: 1, backgroundColor: "#18181b" }} />
),
[currentSessionID],
)

// Disable right drawer swipe when left sidebar is open (gesture conflict)
const swipeEnabled = !!currentSessionID && !sidebarOpen && !isTablet

// Let both drawer gestures run simultaneously so each direction works independently
const configureGesture = useCallback(
(gesture: Parameters<NonNullable<React.ComponentProps<typeof Drawer>["configureGestureHandler"]>>[0]) => {
if (parentGesture) return gesture.simultaneousWithExternalGesture(parentGesture as never)
return gesture
},
[parentGesture],
)

return (
<Drawer
drawerPosition="right"
open={diffOpen}
onOpen={() => {
telemetry.track("drawer", "drawer:right:open")
setDiffOpen(true)
}}
onClose={() => {
telemetry.track("drawer", "drawer:right:close")
setDiffOpen(false)
}}
drawerType="slide"
swipeEnabled={swipeEnabled}
swipeEdgeWidth={width}
swipeMinDistance={DRAWER_SWIPE_MIN_DISTANCE}
swipeMinVelocity={DRAWER_SWIPE_MIN_VELOCITY}
configureGestureHandler={configureGesture}
overlayStyle={styles.drawerOverlay}
drawerStyle={diffDrawerStyle}
renderDrawerContent={renderDiffContent}
>
<RightOverlay isTablet={isTablet}>
<ConnectionBanner />
<Stack screenOptions={{ headerShown: false, gestureEnabled: false }}>
<Stack.Screen name="session" />
<Stack.Screen
name="settings"
options={{
presentation: "formSheet",
headerShown: false,
gestureDirection: "vertical",
sheetGrabberVisible: true,
sheetCornerRadius: 20,
sheetExpandsWhenScrolledToEdge: false,
gestureEnabled: true,
}}
/>
</Stack>
</RightOverlay>
</Drawer>
)
}

function LeftOverlay({ isTablet, children }: { isTablet: boolean; children: React.ReactNode }) {
const theme = useTheme()
const progress = useDrawerProgress()

const blurOverlayStyle = useAnimatedStyle(
const overlayStyle = useAnimatedStyle(
() => ({
opacity: isTablet ? 0 : interpolate(progress.value, [0, 1], [0, SWIPE_SURFACE_OVERLAY_OPACITY]),
}),
Expand All @@ -137,28 +257,34 @@ function SlidingContent({ isTablet }: { isTablet: boolean }) {

return (
<View style={[styles.content, { backgroundColor: theme.colors.background }]}>
<ConnectionBanner />
<Stack screenOptions={{ headerShown: false, gestureEnabled: false }}>
<Stack.Screen name="session" />
<Stack.Screen
name="settings"
options={{
presentation: "formSheet",
headerShown: false,
gestureDirection: "vertical",
sheetGrabberVisible: true,
sheetCornerRadius: 20,
sheetExpandsWhenScrolledToEdge: false,
gestureEnabled: true,
}}
/>
</Stack>
{children}
{!isTablet && Platform.OS === "ios" ? (
<AnimatedView pointerEvents="none" style={[styles.blurOverlay, overlayStyle]}>
<View style={[styles.blurTint, { backgroundColor: theme.colors.background }]} />
</AnimatedView>
) : null}
</View>
)
}

function RightOverlay({ isTablet, children }: { isTablet: boolean; children: React.ReactNode }) {
const theme = useTheme()
const progress = useDrawerProgress()

const overlayStyle = useAnimatedStyle(
() => ({
opacity: isTablet ? 0 : interpolate(progress.value, [0, 1], [0, SWIPE_SURFACE_OVERLAY_OPACITY]),
}),
[isTablet],
)

return (
<View style={[styles.content, { backgroundColor: theme.colors.background }]}>
{children}
{!isTablet && Platform.OS === "ios" ? (
<>
<AnimatedView pointerEvents="none" style={[styles.mainBlurOverlay, blurOverlayStyle]}>
<View style={[styles.mainBlurTint, { backgroundColor: theme.colors.background }]} />
</AnimatedView>
</>
<AnimatedView pointerEvents="none" style={[styles.blurOverlay, overlayStyle]}>
<View style={[styles.blurTint, { backgroundColor: theme.colors.background }]} />
</AnimatedView>
) : null}
</View>
)
Expand All @@ -168,10 +294,10 @@ const styles = StyleSheet.create({
content: {
flex: 1,
},
mainBlurOverlay: {
blurOverlay: {
...StyleSheet.absoluteFillObject,
},
mainBlurTint: {
blurTint: {
...StyleSheet.absoluteFillObject,
},
drawerOverlay: {
Expand Down
Loading
Loading