|
| 1 | +import type { SharedValue } from "react-native-reanimated"; |
| 2 | +import { useCallback, useState } from "react"; |
| 3 | +import { View } from "react-native"; |
| 4 | +import { useDerivedValue } from "react-native-reanimated"; |
| 5 | +import { Inter_500Medium } from "@expo-google-fonts/inter"; |
| 6 | +import { Circle, Text as SKText, useFont } from "@shopify/react-native-skia"; |
| 7 | +import { CartesianChart, Line, useChartPressState } from "victory-native"; |
| 8 | + |
| 9 | +import { DATA, generateData } from "~/components/charts/data"; |
| 10 | +import { Button } from "~/components/ui/button"; |
| 11 | +import { Text } from "~/components/ui/text"; |
| 12 | +import { useColorScheme } from "~/lib/use-color-scheme"; |
| 13 | + |
| 14 | +export default function MyChart() { |
| 15 | + const { colorScheme } = useColorScheme(); |
| 16 | + const font = useFont(Inter_500Medium, 12); |
| 17 | + const inspectFont = useFont(Inter_500Medium, 30); |
| 18 | + const { state, isActive } = useChartPressState({ x: 0, y: { highTmp: 0 } }); |
| 19 | + |
| 20 | + const [chartData, setChartData] = useState(DATA); |
| 21 | + |
| 22 | + const value = useDerivedValue(() => { |
| 23 | + return "$" + state.y.highTmp.value.value.toFixed(2); |
| 24 | + }, [state]); |
| 25 | + |
| 26 | + const labelColor = colorScheme === "dark" ? "white" : "black"; |
| 27 | + const lineColor = colorScheme === "dark" ? "lightgray" : "black"; |
| 28 | + |
| 29 | + const refreshData = useCallback(() => { |
| 30 | + setChartData(generateData()); |
| 31 | + }, []); |
| 32 | + |
| 33 | + return ( |
| 34 | + <View className="h-80 w-full"> |
| 35 | + <CartesianChart |
| 36 | + data={chartData} |
| 37 | + xKey="day" |
| 38 | + yKeys={["highTmp"]} |
| 39 | + domainPadding={{ top: 30 }} |
| 40 | + axisOptions={{ font, labelColor, lineColor }} |
| 41 | + chartPressState={state} |
| 42 | + > |
| 43 | + {({ points, chartBounds }) => ( |
| 44 | + <> |
| 45 | + <SKText |
| 46 | + x={chartBounds.left + 10} |
| 47 | + y={40} |
| 48 | + font={inspectFont} |
| 49 | + text={value} |
| 50 | + color={labelColor} |
| 51 | + style={"fill"} |
| 52 | + /> |
| 53 | + <Line |
| 54 | + points={points.highTmp} |
| 55 | + color="lightgreen" |
| 56 | + strokeWidth={3} |
| 57 | + animate={{ type: "timing", duration: 500 }} |
| 58 | + connectMissingData={false} |
| 59 | + /> |
| 60 | + {isActive ? ( |
| 61 | + <ToolTip x={state.x.position} y={state.y.highTmp.position} /> |
| 62 | + ) : null} |
| 63 | + </> |
| 64 | + )} |
| 65 | + </CartesianChart> |
| 66 | + <Button onPress={refreshData}> |
| 67 | + <Text>Update Data</Text> |
| 68 | + </Button> |
| 69 | + </View> |
| 70 | + ); |
| 71 | +} |
| 72 | + |
| 73 | +function ToolTip({ x, y }: { x: SharedValue<number>; y: SharedValue<number> }) { |
| 74 | + return <Circle cx={x} cy={y} r={8} color="gray" opacity={0.8} />; |
| 75 | +} |
0 commit comments