30 #include <VirtualRobot/math/Helpers.h>
42 using namespace ProsthesisKinestheticTeachIn;
45 CyberGloveProsthesisControl::SubClassRegistry
58 CyberGloveProsthesisControl::shapeHand(
float fingers,
float thumb)
60 getHandUnit()->setJointAngles({{
"Fingers", fingers}, {
"Thumb", thumb}});
124 .
addChild(
new RemoteGui::VSpacer());
126 getRemoteGui()->createTab(
"CyberGloveProsthesisControl", rootLayoutBuilder);
132 std::filesystem::path myPath = std::filesystem::current_path();
133 std::string dataDir = myPath.string() +
"/prosthesisKinestheticTeaching/recordings/";
136 std::ofstream logger;
137 std::string timestamp =
"";
138 std::string latest_timestamp =
"";
142 int lastClickCountCalibrateCyberGlove = guiTab.
getValue<
int>(
"calibrate_cyber_glove").get();
143 int lastClickCountRecordOn = guiTab.
getValue<
int>(
"start_recording").get();
144 int lastClickCountRecordOff = guiTab.
getValue<
int>(
"stop_recording").get();
145 int lastClickCountLatestTimestamp = guiTab.
getValue<
int>(
"set_latest_timestamp").get();
151 bool storeCalibrationValuesForCyberGlove =
false;
153 std::chrono::time_point<std::chrono::system_clock> start, end;
157 while (!isRunningTaskStopped())
165 int calibrateCyberGloveClickCount = guiTab.
getValue<
int>(
"calibrate_cyber_glove").get();
166 bool calibrateCyberGloveButtonWasClicked =
167 calibrateCyberGloveClickCount > lastClickCountCalibrateCyberGlove;
168 lastClickCountCalibrateCyberGlove = calibrateCyberGloveClickCount;
170 int startRecordingClickCount = guiTab.
getValue<
int>(
"start_recording").get();
171 bool startRecordingButtonClicked = startRecordingClickCount > lastClickCountRecordOn;
172 lastClickCountRecordOn = startRecordingClickCount;
174 int stopRecordingClickCount = guiTab.
getValue<
int>(
"stop_recording").get();
175 bool stopRecordingButtonClicked = stopRecordingClickCount > lastClickCountRecordOff;
176 lastClickCountRecordOff = stopRecordingClickCount;
178 int setLatestTimestampClickCount = guiTab.
getValue<
int>(
"set_latest_timestamp").get();
179 bool setLatestTimeStampButtonClicked =
180 setLatestTimestampClickCount > lastClickCountLatestTimestamp;
181 lastClickCountLatestTimestamp = setLatestTimestampClickCount;
182 ARMARX_IMPORTANT <<
"Timestamp click count: " << setLatestTimestampClickCount;
184 CyberGloveValues currentValues = getCyberGloveObserver()->getLatestValues();
186 float thumbValue = 0;
187 float indexValue = 0;
190 {
return v.thumbAbd +
v.thumbCMC +
v.thumbIP +
v.thumbMCP; };
192 {
return v.indexDIP +
v.indexMCP +
v.indexPIP; };
194 if (setLatestTimeStampButtonClicked)
196 if (!latest_timestamp.empty())
198 timestamp = latest_timestamp;
200 getMessageDisplay()->setMessage(
"Resume latest timestamp",
201 "Latest timestamp is " + latest_timestamp);
206 getMessageDisplay()->setMessage(
"No timestamp stored.",
207 "Take a point cloud to generate a timestamp.");
211 if (startRecordingButtonClicked)
213 if (timestamp.empty())
216 time_t timer = now.toSeconds();
219 ts = localtime(&timer);
220 strftime(buffer, 80,
"%Y-%m-%d-%H-%M-%S", ts);
221 std::string
str(buffer);
223 latest_timestamp = timestamp;
225 std::string fileName = guiTab.
getValue<std::string>(
"file_name").get();
227 logger.open(dataDir + fileName +
"-temp-" + timestamp +
".csv");
228 start = std::chrono::system_clock::now();
229 logger <<
"Timestamp, indexTarget, thumbTarget, indexActual, thumbActual" << std::endl;
231 storeCalibrationValuesForCyberGlove =
true;
232 getMessageDisplay()->setMessage(
"Start Recording Trajectory",
"");
235 if (stopRecordingButtonClicked)
242 getMessageDisplay()->setMessage(
"Stop Recording Trajectory",
"");
247 getMessageDisplay()->setMessage(
"Calibrating open hand",
"");
248 guiTab.
getValue<std::string>(
"message").
set(
"Put your hand flat on the table!");
249 if (calibrateCyberGloveButtonWasClicked)
251 openValues = currentValues;
257 getMessageDisplay()->setMessage(
"Calibrating closed hand",
"");
258 guiTab.
getValue<std::string>(
"message").
set(
"Make a fist!");
259 if (calibrateCyberGloveButtonWasClicked)
261 closedValues = currentValues;
267 guiTab.
getValue<std::string>(
"message").
set(
"Teach-In!");
268 thumbValue = math::Helpers::Clamp(0,
270 math::Helpers::ILerp(calcThumbSum(openValues),
271 calcThumbSum(closedValues),
272 calcThumbSum(currentValues)));
273 indexValue = math::Helpers::Clamp(0,
275 math::Helpers::ILerp(calcIndexSum(openValues),
276 calcIndexSum(closedValues),
277 calcIndexSum(currentValues)));
289 shapeHand(indexValue, thumbValue);
292 NameValueMap handMotorValues = getHandUnit()->getCurrentJointValues();
301 end = std::chrono::system_clock::now();
302 double elapsed_milliseconds =
303 std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();
304 logger << elapsed_milliseconds <<
"," << indexValue <<
"," << thumbValue <<
","
305 << handMotorValues[
"Fingers"] <<
"," << handMotorValues[
"Thumb"]
323 std::stringstream ss;
324 ss <<
"index: " << currentValues.
indexMCP <<
" " << currentValues.
indexPIP <<
" "
326 ss <<
"thumb: " << currentValues.
thumbMCP <<
" " << currentValues.
thumbIP <<
" "
328 ss << std::floor(indexValue * 100) <<
"% " << std::floor(thumbValue * 100) <<
"%\n";
329 guiTab.
getValue<std::string>(
"status").
set(ss.str());
335 currentPhase = nextPhase;