Files
billisdead ba9e4a5add fix: resolve TypeScript errors caught by typecheck
- _layout: replace invalid SFSymbols7_0 name "calendar.fill" with
  "calendar.circle.fill" (the fill variant of calendar in SF Symbols)
- useColors: remove unsafe cast through Record<string, palette> —
  colors.radius (number) is incompatible with the palette shape;
  simplify to a direct ternary since both light and dark palettes
  are always defined

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-16 12:34:09 +02:00

143 lines
4.2 KiB
TypeScript

import { BlurView } from "expo-blur";
import { isLiquidGlassAvailable } from "expo-glass-effect";
import { Tabs } from "expo-router";
import { Icon, Label, NativeTabs } from "expo-router/unstable-native-tabs";
import { SymbolView } from "expo-symbols";
import { Feather } from "@expo/vector-icons";
import React from "react";
import { Platform, StyleSheet, View, useColorScheme } from "react-native";
import { useSafeAreaInsets } from "react-native-safe-area-context";
import { useColors } from "@/hooks/useColors";
function NativeTabLayout() {
return (
<NativeTabs>
<NativeTabs.Trigger name="index">
<Icon sf={{ default: "calendar", selected: "calendar.circle.fill" }} />
<Label>Calendar</Label>
</NativeTabs.Trigger>
<NativeTabs.Trigger name="posts">
<Icon sf={{ default: "list.bullet", selected: "list.bullet" }} />
<Label>Posts</Label>
</NativeTabs.Trigger>
<NativeTabs.Trigger name="compose">
<Icon sf={{ default: "plus.circle", selected: "plus.circle.fill" }} />
<Label>Compose</Label>
</NativeTabs.Trigger>
<NativeTabs.Trigger name="settings">
<Icon sf={{ default: "gear", selected: "gear" }} />
<Label>Settings</Label>
</NativeTabs.Trigger>
</NativeTabs>
);
}
function ClassicTabLayout() {
const colors = useColors();
const colorScheme = useColorScheme();
const isDark = colorScheme === "dark";
const isIOS = Platform.OS === "ios";
const isWeb = Platform.OS === "web";
const insets = useSafeAreaInsets();
return (
<Tabs
screenOptions={{
tabBarActiveTintColor: colors.primary,
tabBarInactiveTintColor: colors.mutedForeground,
headerStyle: { backgroundColor: colors.background },
headerTitleStyle: {
fontFamily: "Inter_600SemiBold",
color: colors.foreground,
fontSize: 17,
},
headerShadowVisible: false,
headerTintColor: colors.primary,
tabBarStyle: {
position: "absolute",
backgroundColor: isIOS ? "transparent" : colors.background,
borderTopWidth: StyleSheet.hairlineWidth,
borderTopColor: colors.border,
elevation: 0,
...(isWeb ? { height: 84 } : {}),
},
tabBarBackground: () =>
isIOS ? (
<BlurView
intensity={80}
tint={isDark ? "dark" : "light"}
style={StyleSheet.absoluteFill}
/>
) : isWeb ? (
<View
style={[
StyleSheet.absoluteFill,
{ backgroundColor: colors.background },
]}
/>
) : null,
tabBarLabelStyle: {
fontFamily: "Inter_500Medium",
fontSize: 11,
},
}}
>
<Tabs.Screen
name="index"
options={{
title: "Calendar",
tabBarIcon: ({ color }) =>
isIOS ? (
<SymbolView name="calendar" tintColor={color} size={22} />
) : (
<Feather name="calendar" size={21} color={color} />
),
}}
/>
<Tabs.Screen
name="posts"
options={{
title: "Posts",
tabBarIcon: ({ color }) =>
isIOS ? (
<SymbolView name="list.bullet" tintColor={color} size={22} />
) : (
<Feather name="list" size={21} color={color} />
),
}}
/>
<Tabs.Screen
name="compose"
options={{
title: "Compose",
tabBarIcon: ({ color }) =>
isIOS ? (
<SymbolView name="plus.circle" tintColor={color} size={24} />
) : (
<Feather name="plus-circle" size={22} color={color} />
),
}}
/>
<Tabs.Screen
name="settings"
options={{
title: "Settings",
tabBarIcon: ({ color }) =>
isIOS ? (
<SymbolView name="gear" tintColor={color} size={22} />
) : (
<Feather name="settings" size={21} color={color} />
),
}}
/>
</Tabs>
);
}
export default function TabLayout() {
if (isLiquidGlassAvailable()) {
return <NativeTabLayout />;
}
return <ClassicTabLayout />;
}