Skip to content

Commit 133ae56

Browse files
authored
Improve Acquisition Report (#145)
* Have SkipAcquisition move to staging * Send Acquisition Report even when SkipAcquisition is clicked' * Improve rendering of settings on mobile
1 parent 40622cd commit 133ae56

File tree

6 files changed

+99
-74
lines changed

6 files changed

+99
-74
lines changed

feedingwebapp/src/Pages/GlobalState.jsx

+4-4
Original file line numberDiff line numberDiff line change
@@ -130,8 +130,8 @@ export const useGlobalState = create(
130130
// message received from the face detection node where a
131131
// face was detected and within the distance bounds of the camera.
132132
moveToMouthActionGoal: null,
133-
// Last RobotMotion action response
134-
lastMotionActionResponse: null,
133+
// Last RobotMotion action feedback message
134+
lastMotionActionFeedback: null,
135135
// Whether or not the currently-executing robot motion was paused by the user.
136136
// NOTE: `paused` may no longer need to be in global state now that we have
137137
// the `inNonMovingState` flag.
@@ -218,9 +218,9 @@ export const useGlobalState = create(
218218
set(() => ({
219219
biteAcquisitionActionGoal: biteAcquisitionActionGoal
220220
})),
221-
setLastMotionActionResponse: (lastMotionActionResponse) =>
221+
setLastMotionActionFeedback: (lastMotionActionFeedback) =>
222222
set(() => ({
223-
lastMotionActionResponse: lastMotionActionResponse
223+
lastMotionActionFeedback: lastMotionActionFeedback
224224
})),
225225
setMoveToMouthActionGoal: (moveToMouthActionGoal) =>
226226
set(() => {

feedingwebapp/src/Pages/Home/Home.jsx

+56-4
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11
// React imports
2-
import React, { useCallback, useEffect, useMemo } from 'react'
2+
import React, { useCallback, useEffect, useMemo, useRef } from 'react'
33
import PropTypes from 'prop-types'
4+
import { toast } from 'react-toastify'
45
import { View } from 'react-native'
56

67
// Local imports
78
import './Home.css'
9+
import { createROSService, createROSServiceRequest, useROS } from '../../ros/ros_helpers'
810
import { useGlobalState, MEAL_STATE } from '../GlobalState'
911
import BiteAcquisitionCheck from './MealStates/BiteAcquisitionCheck'
1012
import BiteDone from './MealStates/BiteDone'
@@ -13,7 +15,13 @@ import DetectingFace from './MealStates/DetectingFace'
1315
import PostMeal from './MealStates/PostMeal'
1416
import PreMeal from './MealStates/PreMeal'
1517
import RobotMotion from './MealStates/RobotMotion'
16-
import { getRobotMotionText, TIME_TO_RESET_MS } from '../Constants'
18+
import {
19+
ACQUISITION_REPORT_SERVICE_NAME,
20+
ACQUISITION_REPORT_SERVICE_TYPE,
21+
getRobotMotionText,
22+
REGULAR_CONTAINER_ID,
23+
TIME_TO_RESET_MS
24+
} from '../Constants'
1725

1826
/**
1927
* The Home component displays the state of the meal, solicits user input as
@@ -78,6 +86,47 @@ function Home(props) {
7886
const moveToMouthActionInput = useMemo(() => moveToMouthActionGoal, [moveToMouthActionGoal])
7987
const moveToStowPositionActionInput = useMemo(() => ({}), [])
8088

89+
/**
90+
* Create callbacks for acquisition success and failure. This is done here because these
91+
* callbacks can be called during BiteAcquisition or the BiteAcquisitionCheck.
92+
*/
93+
const lastMotionActionFeedback = useGlobalState((state) => state.lastMotionActionFeedback)
94+
const ros = useRef(useROS().ros)
95+
let acquisitionReportService = useRef(createROSService(ros.current, ACQUISITION_REPORT_SERVICE_NAME, ACQUISITION_REPORT_SERVICE_TYPE))
96+
let acquisitionResponse = useCallback(
97+
(success) => {
98+
if (!lastMotionActionFeedback.action_info_populated) {
99+
console.info('Cannot report acquisition success or failure without action_info_populated.')
100+
return
101+
}
102+
let msg, loss
103+
if (success) {
104+
msg = 'Reporting Food Acquisition Success!'
105+
loss = 0.0
106+
} else {
107+
msg = 'Reporting Food Acquisition Failure.'
108+
loss = 1.0
109+
}
110+
// NOTE: This uses the ToastContainer in Header
111+
console.log(msg)
112+
toast.info(msg, {
113+
containerId: REGULAR_CONTAINER_ID,
114+
toastId: msg
115+
})
116+
// Create a service request
117+
let request = createROSServiceRequest({
118+
loss: loss,
119+
action_index: lastMotionActionFeedback.action_index,
120+
posthoc: lastMotionActionFeedback.posthoc,
121+
id: lastMotionActionFeedback.selection_id
122+
})
123+
// Call the service
124+
let service = acquisitionReportService.current
125+
service.callService(request, (response) => console.log('Got acquisition report response', response))
126+
},
127+
[lastMotionActionFeedback]
128+
)
129+
81130
/**
82131
* Determines what screen to render based on the meal state.
83132
*/
@@ -119,7 +168,8 @@ function Home(props) {
119168
let nextMealState = MEAL_STATE.U_BiteAcquisitionCheck
120169
let backMealState = MEAL_STATE.R_MovingAbovePlate
121170
// TODO: Add an icon for this errorMealState!
122-
let errorMealState = MEAL_STATE.R_MovingToRestingPosition
171+
let errorMealState = MEAL_STATE.R_MovingToStagingConfiguration
172+
let errorCallback = () => acquisitionResponse(true) // Success if the user skips acquisition
123173
let errorMealStateDescription = 'Skip Acquisition'
124174
return (
125175
<RobotMotion
@@ -132,6 +182,7 @@ function Home(props) {
132182
waitingText={getRobotMotionText(currentMealState)}
133183
allowRetry={false} // Don't allow retrying bite acquisition
134184
errorMealState={errorMealState}
185+
errorCallback={errorCallback}
135186
errorMealStateDescription={errorMealStateDescription}
136187
/>
137188
)
@@ -153,7 +204,7 @@ function Home(props) {
153204
)
154205
}
155206
case MEAL_STATE.U_BiteAcquisitionCheck: {
156-
return <BiteAcquisitionCheck debug={props.debug} />
207+
return <BiteAcquisitionCheck debug={props.debug} acquisitionResponse={acquisitionResponse} />
157208
}
158209
case MEAL_STATE.R_MovingToStagingConfiguration: {
159210
/**
@@ -257,6 +308,7 @@ function Home(props) {
257308
props.debug,
258309
props.webrtcURL,
259310
biteAcquisitionActionInput,
311+
acquisitionResponse,
260312
mostRecentBiteDoneResponse,
261313
moveAbovePlateActionInput,
262314
moveToMouthActionInput,

feedingwebapp/src/Pages/Home/MealStates/BiteAcquisitionCheck.jsx

+17-50
Original file line numberDiff line numberDiff line change
@@ -2,28 +2,22 @@
22
import React, { useCallback, useEffect, useState, useRef } from 'react'
33
import Button from 'react-bootstrap/Button'
44
import { useMediaQuery } from 'react-responsive'
5-
import { toast } from 'react-toastify'
65
import { View } from 'react-native'
6+
// PropTypes is used to validate that the used props are in fact passed to this Component
7+
import PropTypes from 'prop-types'
78

89
// Local Imports
910
import '../Home.css'
1011
import { useGlobalState, MEAL_STATE } from '../../GlobalState'
1112
import { MOVING_STATE_ICON_DICT } from '../../Constants'
1213
import { useROS, createROSService, createROSServiceRequest, subscribeToROSTopic, unsubscribeFromROSTopic } from '../../../ros/ros_helpers'
13-
import {
14-
ACQUISITION_REPORT_SERVICE_NAME,
15-
ACQUISITION_REPORT_SERVICE_TYPE,
16-
FOOD_ON_FORK_DETECTION_TOPIC,
17-
FOOD_ON_FORK_DETECTION_TOPIC_MSG,
18-
REGULAR_CONTAINER_ID,
19-
ROS_SERVICE_NAMES
20-
} from '../../Constants'
14+
import { FOOD_ON_FORK_DETECTION_TOPIC, FOOD_ON_FORK_DETECTION_TOPIC_MSG, ROS_SERVICE_NAMES } from '../../Constants'
2115

2216
/**
2317
* The BiteAcquisitionCheck component appears after the robot has attempted to
2418
* acquire a bite, and asks the user whether it succeeded at acquiring the bite.
2519
*/
26-
const BiteAcquisitionCheck = () => {
20+
const BiteAcquisitionCheck = (props) => {
2721
// Store the remining time before auto-continuing
2822
const [remainingSeconds, setRemainingSeconds] = useState(null)
2923
// Get the relevant global variables
@@ -50,17 +44,11 @@ const BiteAcquisitionCheck = () => {
5044
let iconWidth = isPortrait ? '28vh' : '28vw'
5145
let iconHeight = isPortrait ? '18vh' : '18vw'
5246

53-
// Configure AcquisitionReport service
54-
const lastMotionActionResponse = useGlobalState((state) => state.lastMotionActionResponse)
5547
/**
5648
* Connect to ROS, if not already connected. Put this in useRef to avoid
5749
* re-connecting upon re-renders.
5850
*/
5951
const ros = useRef(useROS().ros)
60-
/**
61-
* Create the ROS Service Client for reporting success/failure
62-
*/
63-
let acquisitionReportService = useRef(createROSService(ros.current, ACQUISITION_REPORT_SERVICE_NAME, ACQUISITION_REPORT_SERVICE_TYPE))
6452
/**
6553
* Create the ROS Service. This is created in local state to avoid re-creating
6654
* it upon every re-render.
@@ -73,48 +61,20 @@ const BiteAcquisitionCheck = () => {
7361
* succeeded.
7462
*/
7563
const acquisitionSuccess = useCallback(() => {
76-
console.log('acquisitionSuccess')
77-
// NOTE: This uses the ToastContainer in Header
78-
toast.info('Reporting Food Acquisition Success!', {
79-
containerId: REGULAR_CONTAINER_ID,
80-
toastId: 'foodAcquisitionSuccess'
81-
})
82-
// Create a service request
83-
let request = createROSServiceRequest({
84-
loss: 0.0,
85-
action_index: lastMotionActionResponse.action_index,
86-
posthoc: lastMotionActionResponse.posthoc,
87-
id: lastMotionActionResponse.selection_id
88-
})
89-
// Call the service
90-
let service = acquisitionReportService.current
91-
service.callService(request, (response) => console.log('Got acquisition report response', response))
64+
let acquisitionResponse = props.acquisitionResponse
65+
acquisitionResponse(true)
9266
setMealState(MEAL_STATE.R_MovingToStagingConfiguration)
93-
}, [lastMotionActionResponse, setMealState])
67+
}, [props.acquisitionResponse, setMealState])
9468

9569
/**
9670
* Callback function for when the user indicates that the bite acquisition
9771
* failed.
9872
*/
9973
const acquisitionFailure = useCallback(() => {
100-
console.log('acquisitionFailure')
101-
// NOTE: This uses the ToastContainer in Header
102-
toast.info('Reporting Food Acquisition Failure.', {
103-
containerId: REGULAR_CONTAINER_ID,
104-
toastId: 'foodAcquisitionFailure'
105-
})
106-
// Create a service request
107-
let request = createROSServiceRequest({
108-
loss: 1.0,
109-
action_index: lastMotionActionResponse.action_index,
110-
posthoc: lastMotionActionResponse.posthoc,
111-
id: lastMotionActionResponse.selection_id
112-
})
113-
// Call the service
114-
let service = acquisitionReportService.current
115-
service.callService(request, (response) => console.log('Got acquisition report response', response))
74+
let acquisitionResponse = props.acquisitionResponse
75+
acquisitionResponse(false)
11676
setMealState(MEAL_STATE.R_MovingAbovePlate)
117-
}, [lastMotionActionResponse, setMealState])
77+
}, [props.acquisitionResponse, setMealState])
11878

11979
/*
12080
* Create refs to store the interval for the food-on-fork detection timers.
@@ -429,4 +389,11 @@ const BiteAcquisitionCheck = () => {
429389
return <>{fullPageView()}</>
430390
}
431391

392+
BiteAcquisitionCheck.propTypes = {
393+
debug: PropTypes.bool,
394+
// A function that takes a boolean indicating whether the robot succeeded at acquiring the bite,
395+
// and processes the response. Note that it does not transition to the next state.
396+
acquisitionResponse: PropTypes.func.isRequired
397+
}
398+
432399
export default BiteAcquisitionCheck

feedingwebapp/src/Pages/Home/MealStates/RobotMotion.jsx

+12-7
Original file line numberDiff line numberDiff line change
@@ -62,8 +62,8 @@ const RobotMotion = (props) => {
6262
const paused = useGlobalState((state) => state.paused)
6363
const setPaused = useGlobalState((state) => state.setPaused)
6464

65-
// Setter for last motion action response
66-
const setLastMotionActionResponse = useGlobalState((state) => state.setLastMotionActionResponse)
65+
// Setter for last motion action feedback msg
66+
const setLastMotionActionFeedback = useGlobalState((state) => state.setLastMotionActionFeedback)
6767

6868
/**
6969
* Connect to ROS, if not already connected. Put this in useRef to avoid
@@ -115,23 +115,27 @@ const RobotMotion = (props) => {
115115
const feedbackCallback = useCallback(
116116
(feedbackMsg) => {
117117
console.log('Got feedback message', feedbackMsg)
118+
setLastMotionActionFeedback(feedbackMsg.values.feedback)
118119
setActionStatus({
119120
actionStatus: ROS_ACTION_STATUS_EXECUTE,
120121
feedback: feedbackMsg.values.feedback
121122
})
122123
},
123-
[setActionStatus]
124+
[setActionStatus, setLastMotionActionFeedback]
124125
)
125126

126127
/**
127128
* Callback function to change the meal state.
128129
*/
129130
const changeMealState = useCallback(
130-
(nextMealState, msg = null) => {
131+
(nextMealState, msg = null, callback = null) => {
131132
if (msg) {
132133
console.log(msg)
133134
}
134135
setPaused(false)
136+
if (callback) {
137+
callback()
138+
}
135139
let setMealState = props.setMealState
136140
setMealState(nextMealState)
137141
},
@@ -164,7 +168,6 @@ const RobotMotion = (props) => {
164168
setActionStatus({
165169
actionStatus: ROS_ACTION_STATUS_SUCCEED
166170
})
167-
setLastMotionActionResponse(response.values)
168171
robotMotionDone()
169172
} else {
170173
if (
@@ -185,7 +188,7 @@ const RobotMotion = (props) => {
185188
}
186189
}
187190
},
188-
[setLastMotionActionResponse, setActionStatus, setPaused, robotMotionDone]
191+
[setActionStatus, setPaused, robotMotionDone]
189192
)
190193

191194
/**
@@ -327,7 +330,7 @@ const RobotMotion = (props) => {
327330
variant='warning'
328331
className='mx-2 btn-huge'
329332
size='lg'
330-
onClick={() => changeMealState(props.errorMealState, 'errorMealState')}
333+
onClick={() => changeMealState(props.errorMealState, 'errorMealState', props.errorCallback)}
331334
style={{
332335
width: '90%',
333336
height: '20%'
@@ -363,6 +366,7 @@ const RobotMotion = (props) => {
363366
props.waitingText,
364367
props.allowRetry,
365368
props.errorMealState,
369+
props.errorCallback,
366370
props.errorMealStateDescription,
367371
motionTextFontSize,
368372
waitingTextFontSize,
@@ -501,6 +505,7 @@ RobotMotion.propTypes = {
501505
allowRetry: PropTypes.bool,
502506
// If error, show the user the option to transition to this meal state
503507
errorMealState: PropTypes.string,
508+
errorCallback: PropTypes.func,
504509
errorMealStateDescription: PropTypes.string
505510
}
506511

0 commit comments

Comments
 (0)