Component.cpp
Go to the documentation of this file.
1 /**
2  * This file is part of ArmarX.
3  *
4  * ArmarX is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * ArmarX is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11  * GNU General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program. If not, see <http://www.gnu.org/licenses/>.
15  *
16  * @package control::ArmarXObjects::example_component_plugin_user
17  * @author Fabian Reister ( fabian dot reister at kit dot edu )
18  * @date 2022
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 
24 #include "Component.h"
25 
26 #include <thread>
27 
29 
31 
32 // Include headers you only need in function definitions in the .cpp.
33 
34 // #include <Eigen/Core>
35 
36 // #include <SimoxUtility/color/Color.h>
37 
38 
40 {
41 
42  const std::string Component::defaultName = "example_component_plugin_user";
43 
46  {
49 
50  // Publish to a topic (passing the TopicListenerPrx).
51  // def->topic(myTopicListener);
52 
53  // Subscribe to a topic (passing the topic name).
54  // def->topic<PlatformUnitListener>("MyTopic");
55 
56  // Use (and depend on) another component (passing the ComponentInterfacePrx).
57  // def->component(myComponentProxy)
58 
59 
60  // Add a required property. (The component won't start without a value being set.)
61  // def->required(properties.boxLayerName, "p.box.LayerName", "Name of the box layer in ArViz.");
62 
63  // Add an optionalproperty.
64  // def->optional(
65  // properties.boxLayerName, "p.box.LayerName", "Name of the box layer in ArViz.");
66  // def->optional(properties.numBoxes, "p.box.Number", "Number of boxes to draw in ArViz.");
67 
68  return def;
69  }
70 
71  void
73  {
74  // Topics and properties defined above are automagically registered.
75 
76  // Keep debug observer data until calling `sendDebugObserverBatch()`.
77  // (Requies the armarx::DebugObserverComponentPluginUser.)
78  // setDebugObserverBatchModeEnabled(true);
79  }
80 
81  void
83  {
84 
85  // Load the controllers that we are using here.
87  // robotUnit->loadLibFromPackage("armarx_control", "libarmarx_control_njoint_controller.so");
88  // robotUnit->loadLibFromPackage("armarx_control", "libarmarx_control_njoint_qp_controller.so");
89 
90  //
91  // Example:
92  //
93  // use of controller builder with defaults
94  //
95  {
96  ARMARX_INFO << "Example 1";
97 
98  auto builder = controllerBuilder<common::ControllerType::TSImp>();
99 
100  auto controller = builder.withDefaultConfig()
101  .withNodeSet("RightArm")
102  .create("MyTaskSpaceImpedanceController1");
104 
105  std::this_thread::sleep_for(std::chrono::seconds(1));
106 
107 
108  controller->activate();
109  std::this_thread::sleep_for(std::chrono::seconds(2));
110  }
111 
112 
113  //
114  // Example:
115  //
116  // use of controller builder with defaults (short version)
117  //
118  {
119  ARMARX_INFO << "Example 2";
120 
121  auto builder = controllerBuilder<common::ControllerType::TSImp>();
122 
123  auto controller =
124  builder.withNodeSet("RightArm")
125  .create("MyTaskSpaceImpedanceController2"); // implicit withDefaultConfig()
127 
128  controller->activate();
129 
130  std::this_thread::sleep_for(std::chrono::seconds(2));
131  }
132 
133 
134  //
135  // Example:
136  //
137  // use of controller builder with config from file
138  //
139  {
140  ARMARX_INFO << "Example 3";
141 
142  auto builder = controllerBuilder<common::ControllerType::TSImp>();
143 
144 
145  auto controller = builder.withDefaultConfig()
146  .withConfig("file provided by Jeff")
147  .withNodeSet("RightArm")
148  .create("MyTaskSpaceImpedanceController3");
150 
151  std::this_thread::sleep_for(std::chrono::seconds(2));
152 
153  controller->updateConfig();
154 
155  std::this_thread::sleep_for(std::chrono::seconds(2));
156  }
157 
158 
159  //
160  // Example:
161  //
162  // use of controller builder with config from memory
163  //
164  // {
165  // ARMARX_INFO << "Example 4";
166 
167  // auto builder = controllerBuilder<common::ControllerType::TSImp>();
168 
169  // auto controller = builder.withNodeSet("RightArm")
170  // .withConfig(armarx::armem::MemoryID("memorylink", "to", "config"))
171  // .create("MyTaskSpaceImpedanceController4");
172  // ARMARX_CHECK_NOT_NULL(controller);
173  // }
174 
175 
176  //
177  // Example:
178  //
179  // use of controller builder with config update
180  //
181  {
182  ARMARX_INFO << "Example 5";
183 
184  auto builder = controllerBuilder<common::ControllerType::TSImp>();
185 
186  auto controller =
187  builder.withNodeSet("RightArm").create("MyTaskSpaceImpedanceController5");
189 
190  // update some parameters
191  controller->config.kdImpedance *= 2;
192 
193  // send new config to robot unit / the controller
194  controller->updateConfig();
195 
196  // again, update some parameters
197  controller->config.kdImpedance.setOnes();
198  controller->config.kdImpedance *= 42;
199 
200  controller->updateConfig();
201 
202 
203  std::this_thread::sleep_for(std::chrono::seconds(2));
204 
205  controller->activate();
206 
207  std::this_thread::sleep_for(std::chrono::seconds(2));
208 
209  // send new config to robot unit / the controller
210  controller->updateConfig();
211 
212  std::this_thread::sleep_for(std::chrono::seconds(2));
213 
214  controller->deactivate();
215  }
216 
217  //
218  // Example:
219  //
220  // use of controller builder with config access before creation
221  //
222  {
223  ARMARX_INFO << "Example 6";
224 
225  auto builder = controllerBuilder<common::ControllerType::TSImp>();
226 
227  // access config before creation
228  builder.config().kdImpedance *= 3;
229 
230  auto controller =
231  builder.withNodeSet("RightArm").create("MyTaskSpaceImpedanceController6");
233 
234  std::this_thread::sleep_for(std::chrono::seconds(2));
235 
236  controller->activate();
237 
238  std::this_thread::sleep_for(std::chrono::seconds(2));
239 
240  controller->deactivate();
241  }
242 
243 
244  //
245  // Example:
246  //
247  // use of controller builder with config access before creation
248  //
249  {
250  ARMARX_INFO << "Example 7";
251 
252  auto builder = controllerBuilder<common::ControllerType::TSImp>();
253 
254  // access config before creation
255  builder.config().kdImpedance *= 3;
256 
257  auto controller =
258  builder.withNodeSet("RightArm").create("MyTaskSpaceImpedanceController7");
260 
261  // controller->daemonize();
262 
263  std::this_thread::sleep_for(std::chrono::seconds(2));
264  }
265 
266 
267  ARMARX_INFO << "done.";
268  }
269 
270  void
272  {
273  }
274 
275  void
277  {
278  }
279 
280  std::string
282  {
283  return Component::defaultName;
284  }
285 
286  std::string
288  {
289  return Component::defaultName;
290  }
291 
292  /* (Requires the armarx::LightweightRemoteGuiComponentPluginUser.)
293  void
294  Component::createRemoteGuiTab()
295  {
296  using namespace armarx::RemoteGui::Client;
297 
298  // Setup the widgets.
299 
300  tab.boxLayerName.setValue(properties.boxLayerName);
301 
302  tab.numBoxes.setValue(properties.numBoxes);
303  tab.numBoxes.setRange(0, 100);
304 
305  tab.drawBoxes.setLabel("Draw Boxes");
306 
307  // Setup the layout.
308 
309  GridLayout grid;
310  int row = 0;
311  {
312  grid.add(Label("Box Layer"), {row, 0}).add(tab.boxLayerName, {row, 1});
313  ++row;
314 
315  grid.add(Label("Num Boxes"), {row, 0}).add(tab.numBoxes, {row, 1});
316  ++row;
317 
318  grid.add(tab.drawBoxes, {row, 0}, {2, 1});
319  ++row;
320  }
321 
322  VBoxLayout root = {grid, VSpacer()};
323  RemoteGui_createTab(getName(), root, &tab);
324  }
325 
326 
327  void
328  Component::RemoteGui_update()
329  {
330  if (tab.boxLayerName.hasValueChanged() || tab.numBoxes.hasValueChanged())
331  {
332  std::scoped_lock lock(propertiesMutex);
333  properties.boxLayerName = tab.boxLayerName.getValue();
334  properties.numBoxes = tab.numBoxes.getValue();
335 
336  {
337  setDebugObserverDatafield("numBoxes", properties.numBoxes);
338  setDebugObserverDatafield("boxLayerName", properties.boxLayerName);
339  sendDebugObserverBatch();
340  }
341  }
342  if (tab.drawBoxes.wasClicked())
343  {
344  // Lock shared variables in methods running in seperate threads
345  // and pass them to functions. This way, the called functions do
346  // not need to think about locking.
347  std::scoped_lock lock(propertiesMutex, arvizMutex);
348  drawBoxes(properties, arviz);
349  }
350  }
351  */
352 
353 
354  /* (Requires the armarx::ArVizComponentPluginUser.)
355  void
356  Component::drawBoxes(const Component::Properties& p, viz::Client& arviz)
357  {
358  // Draw something in ArViz (requires the armarx::ArVizComponentPluginUser.
359  // See the ArVizExample in RobotAPI for more examples.
360 
361  viz::Layer layer = arviz.layer(p.boxLayerName);
362  for (int i = 0; i < p.numBoxes; ++i)
363  {
364  layer.add(viz::Box("box_" + std::to_string(i))
365  .position(Eigen::Vector3f(i * 100, 0, 0))
366  .size(20).color(simox::Color::blue()));
367  }
368  arviz.commit(layer);
369  }
370  */
371 
372 
374 
375 } // namespace armarx::control::components::example_component_plugin_user
armarx::control::components::example_component_plugin_user::Component::getDefaultName
std::string getDefaultName() const override
Definition: Component.cpp:281
ARMARX_CHECK_NOT_NULL
#define ARMARX_CHECK_NOT_NULL(ptr)
This macro evaluates whether ptr is not null and if it turns out to be false it will throw an Express...
Definition: ExpressionException.h:206
armarx::control::components::example_component_plugin_user::Component::createPropertyDefinitions
armarx::PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: Component.cpp:45
armarx::control::components::example_component_plugin_user::Component::GetDefaultName
static std::string GetDefaultName()
Get the component's default name.
Definition: Component.cpp:287
controller
Definition: AddOperation.h:39
armarx::control::components::example_component_plugin_user::Component::onDisconnectComponent
void onDisconnectComponent() override
Definition: Component.cpp:271
armarx::control::components::example_component_plugin_user::ARMARX_REGISTER_COMPONENT_EXECUTABLE
ARMARX_REGISTER_COMPONENT_EXECUTABLE(Component, Component::GetDefaultName())
Component.h
armarx::control::components::example_component_plugin_user::Component::onExitComponent
void onExitComponent() override
Definition: Component.cpp:276
armarx::control::client::ComponentPlugin::getRobotUnitPlugin
armarx::plugins::RobotUnitComponentPlugin & getRobotUnitPlugin()
Definition: ComponentPlugin.h:104
armarx::control::components::example_component_plugin_user
Definition: Component.cpp:39
armarx::control::client::ComponentPluginUser::getControlComponentPlugin
ComponentPlugin & getControlComponentPlugin()
Definition: ComponentPlugin.cpp:68
armarx::Component::getConfigIdentifier
std::string getConfigIdentifier()
Retrieve config identifier for this component as set in constructor.
Definition: Component.cpp:79
Decoupled.h
armarx::control::components::example_component_plugin_user::Component::onConnectComponent
void onConnectComponent() override
Definition: Component.cpp:82
armarx::ComponentPropertyDefinitions
Default component property definition container.
Definition: Component.h:69
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
IceUtil::Handle< class PropertyDefinitionContainer >
armarx::control::components::example_component_plugin_user::Component
Definition: Component.h:46
armarx::control::components::example_component_plugin_user::Component::onInitComponent
void onInitComponent() override
Definition: Component.cpp:72
armarx::plugins::RobotUnitComponentPlugin::getRobotUnit
RobotUnitInterfacePrx getRobotUnit() const
Definition: RobotUnitComponentPlugin.cpp:112
controller_descriptions.h