Amélioration Géolocalisation

This commit is contained in:
Emmy D'Anello 2024-12-19 16:47:41 +01:00
parent abb5c3c584
commit 590539a979
Signed by: ynerant
GPG Key ID: 3A75C55819C8CF85
7 changed files with 84 additions and 19 deletions

View File

@ -1,10 +1,12 @@
import { useGameRepairMutation, useGameResetMutation, useGameStartMutation, useGameStopMutation, useGameSwitchPlayerMutation } from '@/hooks/mutations/useGameMutation' import { useGameRepairMutation, useGameResetMutation, useGameStartMutation, useGameStopMutation, useGameSwitchPlayerMutation } from '@/hooks/mutations/useGameMutation'
import { useAuth } from '@/hooks/useAuth' import { useAuth } from '@/hooks/useAuth'
import { useGame, useUpdateGameState } from '@/hooks/useGame' import { useGame, useSetLocationAccuracy, useUpdateGameState } from '@/hooks/useGame'
import { useQueryClient } from '@tanstack/react-query' import { useQueryClient } from '@tanstack/react-query'
import { useRouter } from 'expo-router' import { useRouter } from 'expo-router'
import { useState } from 'react' import { useState } from 'react'
import { Button, Dialog, FAB, List, MD3Colors, Portal, Snackbar, Surface, Text } from 'react-native-paper' import { Button, Dialog, List, MD3Colors, Portal, Snackbar, Surface, Text } from 'react-native-paper'
import { Dropdown } from 'react-native-paper-dropdown'
import { Accuracy } from 'expo-location'
export default function HistoryScreen() { export default function HistoryScreen() {
const [successVisible, setSuccessVisible] = useState(false) const [successVisible, setSuccessVisible] = useState(false)
@ -17,6 +19,17 @@ export default function HistoryScreen() {
const auth = useAuth() const auth = useAuth()
const game = useGame() const game = useGame()
const updateGameState = useUpdateGameState() const updateGameState = useUpdateGameState()
const setLocationAccuracy = useSetLocationAccuracy()
const accuracyArrayList = [
{ value: Accuracy.BestForNavigation.toString(), label: "Navigation" },
{ value: Accuracy.Highest.toString(), label: "Plus haute" },
{ value: Accuracy.High.toString(), label: "Haute" },
{ value: Accuracy.Balanced.toString(), label: "Équilibrée" },
{ value: Accuracy.Low.toString(), label: "Basse" },
{ value: Accuracy.Lowest.toString(), label: "Plus basse" },
{ value: 'null', label: "Désactivée" },
]
const gameStartMutation = useGameStartMutation({ const gameStartMutation = useGameStartMutation({
auth, auth,
@ -112,6 +125,11 @@ export default function HistoryScreen() {
description={auth.loggedIn ? "Vous êtes déjà connecté⋅e" : "Vous n'êtes pas connecté⋅e"} description={auth.loggedIn ? "Vous êtes déjà connecté⋅e" : "Vous n'êtes pas connecté⋅e"}
right={() => <List.Icon icon="login" />} right={() => <List.Icon icon="login" />}
onPress={() => router.navigate('/login')} /> onPress={() => router.navigate('/login')} />
<List.Item
key={"location-accuracy"}
title="Précision de la géolocalisation"
description="Réglez le niveau de précision de la géolocalisation. Une valeur élevée indique une consommation de batterie accrue."
right={() => <Dropdown label={"Géolocalisation"} hideMenuHeader={true} options={accuracyArrayList} value={game.settings.locationAccuracy?.toString() ?? 'null'} onSelect={(value) => setLocationAccuracy(!value || value == 'null' ? null : +value)} />} />
</List.Section> </List.Section>
<List.Section title={"Gestion du jeu"}> <List.Section title={"Gestion du jeu"}>
<List.Item <List.Item

View File

@ -1,7 +1,6 @@
import { Accuracy } from 'expo-location' import { Accuracy } from 'expo-location'
export const Constants = { export const Constants = {
LOCATION_ACCURACY: Accuracy.BestForNavigation,
MIN_DELAY_LOCATION_SENT: 20, MIN_DELAY_LOCATION_SENT: 20,
QUERY_REFETCH_INTERVAL: 15, QUERY_REFETCH_INTERVAL: 15,
} }

View File

@ -1,7 +1,9 @@
import { Accuracy } from "expo-location"
import { useAppDispatch, useAppSelector } from "./useStore" import { useAppDispatch, useAppSelector } from "./useStore"
import { GamePayload, PenaltyPayload, setPlayerId, updateActiveChallengeId, updateGameState, updateMoney, updatePenalty } from "@/utils/features/game/gameSlice" import { GamePayload, PenaltyPayload, setLocationAccuracy, setPlayerId, updateActiveChallengeId, updateGameState, updateMoney, updatePenalty } from "@/utils/features/game/gameSlice"
export const useGame = () => useAppSelector((state) => state.game) export const useGame = () => useAppSelector((state) => state.game)
export const useSettings = () => useAppSelector((state) => state.game.settings)
export const useSetPlayerId = () => { export const useSetPlayerId = () => {
const dispath = useAppDispatch() const dispath = useAppDispatch()
return (playerId: number) => dispath(setPlayerId(playerId)) return (playerId: number) => dispath(setPlayerId(playerId))
@ -22,3 +24,7 @@ export const useUpdatePenalty = () => {
const dispatch = useAppDispatch() const dispatch = useAppDispatch()
return (penalty: PenaltyPayload) => dispatch(updatePenalty(penalty)) return (penalty: PenaltyPayload) => dispatch(updatePenalty(penalty))
} }
export const useSetLocationAccuracy = () => {
const dispatch = useAppDispatch()
return (accuracy: Accuracy | null) => dispatch(setLocationAccuracy(accuracy))
}

View File

@ -1,12 +1,12 @@
{ {
"name": "traintrape-moi-client", "name": "traintrape-moi-client",
"version": "1.0.0", "version": "1.0.1",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "traintrape-moi-client", "name": "traintrape-moi-client",
"version": "1.0.0", "version": "1.0.1",
"dependencies": { "dependencies": {
"@dev-plugins/react-navigation": "^0.1.0", "@dev-plugins/react-navigation": "^0.1.0",
"@dev-plugins/react-query": "^0.1.0", "@dev-plugins/react-query": "^0.1.0",
@ -47,6 +47,7 @@
"react-native": "0.76.3", "react-native": "0.76.3",
"react-native-gesture-handler": "~2.20.2", "react-native-gesture-handler": "~2.20.2",
"react-native-paper": "^5.12.5", "react-native-paper": "^5.12.5",
"react-native-paper-dropdown": "^2.3.1",
"react-native-reanimated": "~3.16.1", "react-native-reanimated": "~3.16.1",
"react-native-safe-area-context": "~4.12.0", "react-native-safe-area-context": "~4.12.0",
"react-native-screens": "~4.1.0", "react-native-screens": "~4.1.0",
@ -13631,6 +13632,23 @@
"react-native-vector-icons": "*" "react-native-vector-icons": "*"
} }
}, },
"node_modules/react-native-paper-dropdown": {
"version": "2.3.1",
"resolved": "https://registry.npmjs.org/react-native-paper-dropdown/-/react-native-paper-dropdown-2.3.1.tgz",
"integrity": "sha512-IvcHTucAV5+fiX2IVMiVdBDKT6KHxycW0o9QzZe7bpmeZWmuCajHDnwG3OSBGlXhUxrrM3TC0/HJZHwORWGgQg==",
"license": "MIT",
"workspaces": [
"example"
],
"dependencies": {
"react-native-paper": "^5.12.3"
},
"peerDependencies": {
"react": "*",
"react-native": "*",
"react-native-paper": "*"
}
},
"node_modules/react-native-paper/node_modules/color": { "node_modules/react-native-paper/node_modules/color": {
"version": "3.2.1", "version": "3.2.1",
"resolved": "https://registry.npmjs.org/color/-/color-3.2.1.tgz", "resolved": "https://registry.npmjs.org/color/-/color-3.2.1.tgz",

View File

@ -53,6 +53,7 @@
"react-native": "0.76.3", "react-native": "0.76.3",
"react-native-gesture-handler": "~2.20.2", "react-native-gesture-handler": "~2.20.2",
"react-native-paper": "^5.12.5", "react-native-paper": "^5.12.5",
"react-native-paper-dropdown": "^2.3.1",
"react-native-reanimated": "~3.16.1", "react-native-reanimated": "~3.16.1",
"react-native-safe-area-context": "~4.12.0", "react-native-safe-area-context": "~4.12.0",
"react-native-screens": "~4.1.0", "react-native-screens": "~4.1.0",

View File

@ -1,4 +1,5 @@
import { createSlice, PayloadAction } from '@reduxjs/toolkit' import { createSlice, PayloadAction } from '@reduxjs/toolkit'
import { Accuracy } from 'expo-location'
export interface RunPayload { export interface RunPayload {
id: number id: number
@ -37,6 +38,11 @@ export interface GameState {
chaseFreeTime: number | null // date chaseFreeTime: number | null // date
penaltyStart: number | null // date penaltyStart: number | null // date
penaltyEnd: number | null //date penaltyEnd: number | null //date
settings: Settings
}
export interface Settings {
locationAccuracy: Accuracy | null
} }
const initialState: GameState = { const initialState: GameState = {
@ -49,6 +55,9 @@ const initialState: GameState = {
chaseFreeTime: null, chaseFreeTime: null,
penaltyStart: null, penaltyStart: null,
penaltyEnd: null, penaltyEnd: null,
settings: {
locationAccuracy: Accuracy.Highest,
}
} }
export const gameSlice = createSlice({ export const gameSlice = createSlice({
@ -77,10 +86,13 @@ export const gameSlice = createSlice({
updatePenalty: (state, action: PayloadAction<PenaltyPayload>) => { updatePenalty: (state, action: PayloadAction<PenaltyPayload>) => {
state.penaltyStart = action.payload.penaltyStart state.penaltyStart = action.payload.penaltyStart
state.penaltyEnd = action.payload.penaltyEnd state.penaltyEnd = action.payload.penaltyEnd
},
setLocationAccuracy: (state, action: PayloadAction<Accuracy | null>) => {
state.settings.locationAccuracy = action.payload
} }
}, },
}) })
export const { setPlayerId, updateMoney, updateActiveChallengeId, updateGameState, updatePenalty } = gameSlice.actions export const { setLocationAccuracy, setPlayerId, updateMoney, updateActiveChallengeId, updateGameState, updatePenalty } = gameSlice.actions
export default gameSlice.reducer export default gameSlice.reducer

View File

@ -5,9 +5,11 @@ import { PlayerLocation, setLastLocation } from './features/location/locationSli
import store from './store' import store from './store'
import { useEffect } from 'react' import { useEffect } from 'react'
import { socket } from './socket' import { socket } from './socket'
import { Constants } from '@/constants/Constants' import { useSettings } from '@/hooks/useGame'
import { useAuth } from '@/hooks/useAuth'
import { isAuthValid } from './features/auth/authSlice'
const LOCATION_TASK = "fetch-geolocation" const LOCATION_TASK = "TRAINTRAPE_MOI_GEOLOCATION"
TaskManager.defineTask(LOCATION_TASK, async ({ data, error }: any) => { TaskManager.defineTask(LOCATION_TASK, async ({ data, error }: any) => {
if (error) { if (error) {
@ -33,9 +35,12 @@ TaskManager.defineTask(LOCATION_TASK, async ({ data, error }: any) => {
} }
}) })
export async function startGeolocationService(): Promise<void | (() => void)> { export async function startGeolocationService(locationAccuracy: Location.Accuracy | null): Promise<void | (() => void)> {
if (Platform.OS !== "web" && await Location.hasStartedLocationUpdatesAsync(LOCATION_TASK)) if (Platform.OS !== "web" && await Location.hasStartedLocationUpdatesAsync(LOCATION_TASK))
return async () => await Location.stopLocationUpdatesAsync(LOCATION_TASK) await Location.stopLocationUpdatesAsync(LOCATION_TASK)
if (locationAccuracy === null)
return
await Location.enableNetworkProviderAsync().catch(error => alert(error)) await Location.enableNetworkProviderAsync().catch(error => alert(error))
@ -49,9 +54,9 @@ export async function startGeolocationService(): Promise<void | (() => void)> {
if (Platform.OS !== "web") { if (Platform.OS !== "web") {
await Location.startLocationUpdatesAsync(LOCATION_TASK, { await Location.startLocationUpdatesAsync(LOCATION_TASK, {
accuracy: Constants.LOCATION_ACCURACY, accuracy: locationAccuracy,
activityType: Location.ActivityType.OtherNavigation, activityType: Location.ActivityType.OtherNavigation,
deferredUpdatesInterval: 1000, distanceInterval: 10,
timeInterval: 1000, timeInterval: 1000,
foregroundService: { foregroundService: {
killServiceOnDestroy: false, killServiceOnDestroy: false,
@ -63,13 +68,19 @@ export async function startGeolocationService(): Promise<void | (() => void)> {
return async () => await Location.stopLocationUpdatesAsync(LOCATION_TASK) return async () => await Location.stopLocationUpdatesAsync(LOCATION_TASK)
} }
else { else {
const locationSubscription = await Location.watchPositionAsync({ accuracy: Constants.LOCATION_ACCURACY }, location_nouveau => store.dispatch(setLastLocation(location_nouveau))) const locationSubscription = await Location.watchPositionAsync({ accuracy: locationAccuracy }, location_nouveau => store.dispatch(setLastLocation(location_nouveau)))
return locationSubscription.remove return locationSubscription.remove
} }
} }
export const useStartGeolocationServiceEffect = () => useEffect(() => { export const useStartGeolocationServiceEffect = () => {
let cleanup: void | (() => void) = () => {} const auth = useAuth()
startGeolocationService().then(result => cleanup = result) const settings = useSettings()
return cleanup return useEffect(() => {
}, []) if (!isAuthValid(auth))
return
let cleanup: void | (() => void) = () => {}
startGeolocationService(settings.locationAccuracy).then(result => cleanup = result)
return cleanup
}, [auth, settings.locationAccuracy])
}