Urcap Tutorial HTML
Urcap Tutorial HTML
Abstract
URCaps make it possible to seamlessly extend any Universal Robot with customized func-
tionality. Using the URCap Software Platform, a URCap developer can define customized
installation screens and program nodes for the end user. These can, for example, encapsu-
late new complex robot programming concepts, or provide friendly hardware configuration
interfaces.
This tutorial explains how to use the URCap Software Platform version 1.14.0 to develop
and deploy URCaps with HTML-based user interfaces for PolyScope version 5.15.0 running
on e-Series robots. Older PolyScope versions can be targeted using older versions of the
URCap Software Platform. URCaps can be developed for CB3 robots with PolyScope
version 3.3.x to 3.15.x using version 1.12.0 or older of the URCap Software Platform.
Contents
1 Introduction 3
1.1 Features in URCap Software Platform 1.14.0 . . . . . . . . . . . . . . . . . . . . 3
2 Prerequisites 5
3 URCap SDK 6
9 Contribution of a Daemon 27
9.1 Daemon Service . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 27
9.2 Interaction with the Daemon . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28
9.3 C/C++ Daemon Executables . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30
9.4 Tying the different Contributions together . . . . . . . . . . . . . . . . . . . . . . 31
12 Compatibility 44
12.1 Robot Series Compatibility . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 44
12.1.1 Compatibility Flags . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 44
12.1.2 Installation and Startup of URCaps . . . . . . . . . . . . . . . . . . . . . 45
12.1.3 Updating Existing URCaps . . . . . . . . . . . . . . . . . . . . . . . . . . 45
12.2 API Compatibility . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 46
12.2.1 Advanced compatibility . . . . . . . . . . . . . . . . . . . . . . . . . . . . 47
13 Exception Handling 49
14 Troubleshooting 50
1 Introduction
The first official version of the URCap Software Platform (version 1.0.0) was released with
PolyScope version 3.3.0 running on CB3 robots. This tutorial describes features supported in
version 1.14.0 of the URCap Software Platform which is released together with PolyScope ver-
sion 5.15.0 for e-Series robots. To create URCaps for PolyScope version 3.3.x to 3.15.x running
on CB3 robots, older versions of the URCap Software Platform must be used.
This platform can be used to develop external contributions to PolyScope that are loaded when
PolyScope starts up. This makes it possible for a URCap developer to provide customized func-
tionality within PolyScope.
For example, a customized installation screen can serve the end user to comfortably configure a
new tool. Similarly, a customized program node may serve as a way to perform complex tasks
while hiding unnecessary detail.
The layout of a customized screen is defined using a subset of HTML and CSS. The behaviour
of a customized node, data persistence and script code generation is implemented in Java. The
URCap along with its resources is packaged and distributed as a single Java Jar file with the
.urcap file extension. A URCap can be installed from the URCap Settings/Setup screen in
PolyScope.
• Script code that a program node contributes to the script code generated by the robot
program.
• Behaviour for widgets and other elements on the customized screens.
2 Prerequisites
A working version of Java SDK 6 is required for URCap development along with Apache Maven
3.0.5. You will also need PolyScope version 5.15.0 in order to install the developed URCap, if it
is using URCap API version 1.14.0. Previous versions of the API will have lower requirements
for the PolyScope version. Any Universal Robots robot (e.g., a UR3, UR5, UR10, etc.) can be
used for that purpose or the Universal Robots offline simulator software (URSim). PolyScope
and the offline simulator can be found in the download area of the tech support website at:
www.universal-robots.com/support
Select the applicable version and follow the given installation instructions. The offline simulator
is available for Linux and non-Linux operating systems through a virtual Linux machine.
The script language and pre-defined script functions are defined in the script manual, which can
also be found in the download area of the tech support website.
The URCap SDK is freely available on the Universal Robots+ website at:
https://plus.universal-robots.com
The following section 3. URCap SDK describes the content of the URCap SDK.
3 URCap SDK
The URCap SDK provides the basics to create a URCap. It contains a Java package with
the API that the developer will program against, documentation, the Hello World and other
URCap examples, the urtool3 cross-compiler toolchain and means of easily creating a new empty
Maven-based template URCap project (See section 11. Creating new thin Projects using a Maven
Archetype) as well as easily updating an existing URCap.
com.ur.urcap.sdk
artifacts
api
.
.
.
1.14.0
com.ur.urcap.api-1.14.0.jar
com.ur.urcap.api-1.14.0-javadoc.jar
com.ur.urcap.api-1.14.0-sources.jar
archetype
com.ur.urcap.archetype-1.14.0.jar
other
commons-httpclient-3.1.0.0.jar
ws-commons-util-1.0.2.0.jar
The URCap SDK is distributed as a single ZIP file. Figure 1 shows the structure of the file.
A description of the directories and files contained in this file is given below:
/artifacts/: This directory holds all released versions of the URCap API in separate folders, the
Maven archetype and other necessary files. Each folder named with a version number holds
the Java packages, (e.g. com.ur.urcap.api-1.14.0*.jar files), that contains Java interfaces,
Javadoc and sources of the URCap API that are necessary to implement the Java portion
of a URCap. The Maven archetype folder holds the com.ur.urcap.archetype-1.14.0.jar
file, that can be used to create a new empty template URCap project.
/doc/: The directory contains this tutorial (both in a HTML-based version and a Swing-based
version) as well as a document describing how to configure child program nodes in a sub-
tree and a document explaining how to work with variables. Included is also a document
with a guide on how to convert an existing URCap with HTML-based user interface to a
Swing-based one.
/samples/: A folder containing example projects demonstrating different features of the soft-
ware framework. A description of the examples is found in section 10. URCap Examples
Overview.
/urtool/: Contains the urtool3 cross-compiler toolchain that should be used when building
C/C++ daemon executables for the CB3.0/3.1 and CB5.0 control boxes.
upgradeURCap.sh: A script which can be used to update an existing URCap (project) to spec-
ify which robot series the URCap is compatible with (see section 12.1. Robot Series
Compatibility).
readme.txt: A readme file describing the content of the SDK.
5. Select a .urcap file, e.g. helloworld-1.0-SNAPSHOT.urcap and tap the Open button.
6. Restart PolyScope using the Restart button in the bottom of the screen:
CB3 robot
CB3 robot
CB3 robot
On e-Series robots, the node is visible on the left of the screen after tapping URCaps:
CB3 robot
com.ur.urcap.examples.helloworld
pom.xml
assembly.xml
src
main
java
com
ur
urcap
examples
helloworld
impl
Activator.java
HelloWorldInstallationNodeService.java
HelloWorldInstallationNodeContribution.java
HelloWorldProgramNodeService.java
Figure 2 shows the structure of the Hello World URCap project. This project consists of the
following parts:
1. A view part consisting of two screens with the layout specified in the files
installation.html and programnode.html.
3. A license META-INF/LICENSE with the license information that are shown to the user when
the URCap is installed.
4. Maven configuration files pom.xml and assembly.xml for building the project.
The My Daemon URCap is an extended version of the Hello World URCap, that exemplifies
the integration of an external daemon process. Figure 3 shows the structure of the My Daemon
URCap project. Compared to the Hello World project it additionally offers the following parts:
1. A Python 2.5 daemon executable in the file hello-world.py.
com.ur.urcap.examples.mydaemon
pom.xml
assembly.xml
daemon
.
.
.(See Figure 6, page 31)
src
main
java
com
The services:
• HelloWorldInstallationNodeService.java
• HelloWorldProgramNodeService.java
are registered in Activator.java and thereby a new installation node and program node are
offered to PolyScope. The My Daemon additionally registers its MyDaemonDaemonService.java
service to make the daemon executable available to PolyScope.
The file pom.xml contains a subsection (under the <properties> section) with a set of properties
for the URCap with meta-data specifying the vendor, contact address, copyright, description,
and short license information which will be displayed to the user when the URCap is installed
in PolyScope. See Listing 1 for the Hello World version of these properties.
Listing 1: Section with meta-data properties inside the pom.xml file for the Hello World URCap
1 <! - -******************************************************************** - - >
2 <! - - Note : Update this section with relevant meta data -->
3 <! - - that comes along with your URCap -->
4 <! - -******************************************************************** - - >
5 <! - -******************* BEGINNING OF URCAP META DATA ******************* - - >
6 < urcap . symbolicname > com . ur . urcap . examples . helloworld </ urcap . symbolicname >
7 < urcap . vendor > Universal Robots </ urcap . vendor >
8 < urcap . contactAddress > Energivej 25 , 5260 Odense S , Denmark </ urcap .
The URCap meta-data subsection also contains two boolean compatibility flag properties (urcap.
compatibility.CB3 and urcap.compatibility.eSeries) which specify the URCap’s compatibility with
the CB3 and the e-Series robot series, respectively. See section 12.1. Robot Series Compatibility
for more information about these compatibility flags.
Deployment to a robot with Maven Given the IP address of the robot, e.g. 10.2.128.64,
go to your URCap project folder and type:
1 $ cd samples / html / com . ur . urcap . examples . helloworld
2 $ mvn install - Premote - Durcap . install . host =10.2.128.64
and the URCap is deployed and installed on the robot. During this process PolyScope will be
restarted.
You can also specify the IP address of the robot via the property urcap.install.host inside the
pom.xml file. Then you can deploy by typing:
Deployment to URSim If you are running Linux then URSim can be installed locally.
Otherwise it needs to run in a virtual machine (VM). It is possible to deploy to both environments
with Maven. As shown above parameters can be supplied either directly on the command line
• To deploy to a locally running URSim specify the path to the extracted URSim with the
property ursim.home.
• To deploy to a URSim running in a VM specify the IP address of the VM using the
property ursimvm.install.host.
Once the properties are configured you can deploy to a local URSim by using the ursim profile:
1 $ cd samples / html / com . ur . urcap . examples . helloworld
2 $ mvn install -P ursim
Note, if you are using VirtualBox to run the VM you should make sure that the network of the
VM is operating in bridged mode.
Listing 2: HTML document that specifies the layout of the customized Hello World installation
screen
1 <! DOCTYPE html >
2 < html >
3 < head >
4 < title > Hello World </ title >
5 < style >
6 input {
7 display : inline - block ;
8 width : 200 px ;
9
Listing 2 shows the content of the installation.html file which defines the layout of the screen
used for the Hello World installation node. The listing begins with CSS styling properties (de-
fined within the <style> tag). After that follows a simple document with a heading, a paragraph,
a label and a single text input widget.
In order to connect the HTML GUI specification to your Java implementation an id attribute
is specified for the text field to access its value. This creates a model-view separation where
the id is used to wire a Java object with the HTML widgets. In this way, methods in the Java
implementation can be used to define behaviour for widgets identified by id attributes. The
corresponding Java code is presented in the following two sections.
• getHTML() returns an input stream with the HTML which is passed into PolyScope.
• getTitle() returns the title for the node, to be shown on the left side of the Installation tab
to access the customized installation screen. For simplicity, the title is specified simply
as "Hello World". In a more realistic example, the return value of the getTitle() method
would be translated into the language specified by standard Java localization, based on
the locale provided by Locale.getDefault().
• createInstallationNode(URCapAPI, DataModel) is called by PolyScope when it needs to create
an instance of the installation node. Both a URCapAPI and a DataModel object is passed in
as arguments. The first gives access to the domain of PolyScope and the second provides
a data model with automatic persistence. The constructor used in the implementation
of the method createInstallationNode(...) is discussed in section 7.3. Functionality of the
Installation Node. All modifications to the supplied data model from the installation node
constructor are ignored when existing installation is loaded. This means that ideally the
installation node constructor should not set anything in the data model.
Listing 4: Java class defining functionality for the Hello World installation node
1 package com . ur . urcap . examples . helloworld . impl ;
2
3 import com . ur . urcap . api . contribution . I n s t a l l a t i o n N o d e C o n t r i b u t i o n ;
4 import com . ur . urcap . api . domain . data . DataModel ;
The data model which was mentioned in section 7.2. Making the customized Installation Node
available to PolyScope is passed into the constructor through a DataModel object. All data that
needs to be saved and loaded along with a robot installation must be stored in and retrieved
The HTML text input widget has an id attribute equal to "popuptitle". This attribute maps the
HTML widget to an object of type InputTextField which permits basic operations on text fields.
This is achieved using the annotation @Input by specifying its argument id.
Particularly, when the user interacts with the widget by, e.g., entering some text, the method
onMessageChange(InputEvent) is called. Its argument indicates what kind of interaction has oc-
curred. The code within that method takes care of storing the contents of the text input widget
in the data model under the key POPUPTITLE_KEY whenever the content of the widget changes. By
saving and loading the robot installation you will notice that values are stored and read again
from and back to the popupTitleField.
The openView() method is called whenever the user enters the screen. It sets the contents of the
text input widget to the current value stored within the member field message. The closeView()
method is called when the user leaves the screen.
Finally, the preamble of each program run with this URCap installed will contain an assignment
in its preamble, as specified in the generateScript(ScriptWriter) method. In the assignment, the
script variable named "hello_world_popup_title" is assigned to a string that contains the popup
title stored within the data model object.
The getId() method returns the unique identifier for this type of program node. The identifier
will be used when storing programs that contain these program nodes. Do not change the return
value of this method in released URCaps, since it will break backwards compatibility for existing
programs. URCap program nodes in such existing programs will not be loaded properly and
the program can not run anymore.
Its getTitle() method supplies the text for the button in the Structure Tab that corresponds
to this type of program node. It is also used as the heading on the Command tab for such nodes.
Letting the method isDeprecated() return true makes it impossible to create new program nodes
of this type, but still support loading program nodes of this type in existing programs.
If the method isChildrenAllowed() returns true, it signals that it is possible for the program node
to contain other (child) program nodes.
Finally, createNode(URCapAPI, DataModel) creates program nodes with references to the URCapAPI and
the DataModel. The first gives access to the domain of PolyScope and the second gives the user a
The returned object is used when interacting with widgets on the customized program node
screen for the particular node selected in the program tree. It must use the supplied data model
object to retrieve and store data that should be saved and loaded within the robot program
along with the corresponding node occurrence. Please note that only data related to the current
configuration of that particular program node instance should be stored in the data model, i.e.
no global or shared state, state irrevelant to this node instance, etc. should be stored.
Listing 6: Java class defining how Hello World program nodes are created
1 package com . ur . urcap . examples . helloworld . impl ;
2
3 import com . ur . urcap . api . contribution . P r o g r a m N o d e C o n t r i b u t i o n ;
4 import com . ur . urcap . api . contribution . P r o g r a m N o d e S e r v i c e ;
5 import com . ur . urcap . api . domain . URCapAPI ;
6 import com . ur . urcap . api . domain . data . DataModel ;
7
8 import java . io . InputStream ;
9
10 public class H e l l o W o r l d P r o g r a m N o d e S e r v i c e implements P r o g r a m N o d e S e r v i c e {
11
12 public H e l l o W o r l d P r o g r a m N o d e S e r v i c e () {
13 }
14
15 @Override
16 public String getId () {
17 return " Hel loWorldN ode " ;
18 }
19
20 @Override
21 public String getTitle () {
22 return " Hello World " ;
23 }
24
25 @Override
26 public InputStream getHTML () {
27 InputStream is = this . getClass () . g e t R e s o u r c e A s S t r e a m ( " / com / ur / urcap /
examples / helloworld / impl / programnode . html " ) ;
28 return is ;
29 }
30
31 @Override
32 public boolean isDeprecated () {
33 return false ;
34 }
35
36 @Override
37 public boolean i s C h i l d r e n A l l o w e d () {
38 return true ;
39 }
40
41 @Override
42 public P r o g r a m N o d e C o n t r i b u t i o n createNode ( URCapAPI api , DataModel model ) {
43 return new H e l l o W o r l d P r o g r a m N o d e C o n t r i b u t i o n ( api , model ) ;
44 }
45 }
Listing 7: Java class defining functionality for the Hello World program node
1 package com . ur . urcap . examples . helloworld . impl ;
2
3 import com . ur . urcap . api . contribution . P r o g r a m N o d e C o n t r i b u t i o n ;
4 import com . ur . urcap . api . domain . URCapAPI ;
5 import com . ur . urcap . api . domain . data . DataModel ;
6 import com . ur . urcap . api . domain . script . ScriptWriter ;
7 import com . ur . urcap . api . ui . annotation . Input ;
8 import com . ur . urcap . api . ui . annotation . Label ;
9 import com . ur . urcap . api . ui . component . InputEvent ;
10 import com . ur . urcap . api . ui . component . In putTextF ield ;
11 import com . ur . urcap . api . ui . component . La belCompo nent ;
12
13 public class H e l l o W o r l d P r o g r a m N o d e C o n t r i b u t i o n implements
ProgramNodeContribution {
14 private static final String NAME = " name " ;
15
16 private final DataModel model ;
17 private final URCapAPI api ;
18
19 public H e l l o W o r l d P r o g r a m N o d e C o n t r i b u t i o n ( URCapAPI api , DataModel model ) {
20 this . api = api ;
21 this . model = model ;
22 }
23
24 @Input ( id = " yourname " )
25 private In putTextF ield nameTextField ;
26
90 return api . g e t I n s t a l l a t i o n N o d e ( H e l l o W o r l d I n s t a l l a t i o n N o d e C o n t r i b u t i o n .
class ) ;
91 }
92
93 }
The openView() and closeView() methods specify what happens when the user selects and unse-
lects the underlying program node in the program tree.
The getTitle() method defines the text which is displayed in the program tree for the node. The
text of the node in the program tree is updated when values are written to the DataModel.
The isDefined() method serves to identify whether the node is completely defined (green) or
still undefined (yellow). Note that a node, which can contain other program nodes (see sec-
tion 8.2. Making the customized Program Nodes available to PolyScope), remains undefined as
long as it has a child that is undefined. The isDefined() method is called when values are writ-
ten to the DataModel to ensure that the program tree reflects the proper state of the program node.
Finally, generateScript(ScriptWriter) is called to add script code to the spot where the underlying
node occurs in the robot program.
As the user interacts with the text input widget, the constructed message is displayed on the
screen in the label with id "messagePreviewLabel". Each Hello World node is defined (green) if
The popup title is the value of the script variable hello_world_popup_title. This variable is ini-
tialized by the script code contributed by the Hello World installation node. Thus, the script
variable serves to pass data from the contributed installation node to the contributed program
node. Another approach to pass information between these two objects is by directly requesting
the installation object through the URCapAPI class. The Hello World program node utilizes this
approach in its updatePopupMessageAndPreview() method.
Furthermore, if the URCap contains an installation node contribution, the data model of the
installation contribution should not be modified from the program node contribution. If such
modifications occur in the call to openView() while a program is running, PolyScope will auto-
matically stop the program every time the URCap program node is encountered in the program
tree, because the installation was changed. It is, however, allowed to read from the data model
in the installation node contribution.
If changes made to properties of nodes in the sub-tree happen outside the scope of a user-
initiated event, these changes will not be recorded on the undo stack.
For example, if two changes are made to the data model and a node with a property set is added
to the sub-tree as a result of the user clicking a button in the user interface, this counts as one
undo action. When a user clicks undo the previous values are restored in the data model as well
as the program tree and a call to the openView() method is made, for an opportunity to display
the new values.
This also means that no values should be cached in member variables, but always retrieved from
the data model, as there is no guarantee that things have not changed. Also keep in mind, that
the user might not select the Command tab of the URCap, so there is no guaranteed call to
openView(). This can be the case when loading a program that has already been setup.
When a contributable program node is created in PolyScope by the user, the data model given
to the method createNode(URCapAPI, DataModel) is empty.
When a program is loaded, the method createNode(URCapAPI, DataModel) is called for each per-
sisted program node contribution to re-create the program tree. In contrast to newly creating
the program node, the data model now contains the data from the persisted node. All modi-
fications to the data model from the program node constructor are ignored. This means that
ideally the program node constructor should not set anything in the data model.
For creating sub-trees a program model can be used. In some of the URCap examples in
Chapter 10. URCap Examples Overview it is demonstrated how a sub-tree can be generated
programmatically. The program model provides the interface TreeNode to create and manipulate
the sub-tree. When a contributable program node is created in PolyScope by the user, the tree
node has no children. The program model can be requested through the URCapAPI interface.
When a program is loaded each program node is deserialized on its own, this includes sub-trees
previously created through the program model. Also now, the tree node requested through
the URCapAPI interface is empty. The program node factory returned by getProgramNodeFactory()
in the ProgramModel interface will return program nodes without any functionality. In particu-
lar, the method createURCapProgramNode(Class<? extends URCapProgramNodeService>) does not call the
createNode(...) method in the specified service. Therefore, modifications are ignored during the
createNode(...) call.
9 Contribution of a Daemon
A daemon can be any executable script or binary file that runs on the control box. The My
Daemon URCap serves as the running example for explaining this functionality and is an exten-
sion of the Hello World example. The My Daemon example offers the same functionality from
the user’s point of view as the Hello World example.
However, the My Daemon URCap performs its tasks through an executable, which acts as a
sort of driver or server. The executable is implemented as Python 2.5 script and C++ binary.
The executables communicate with the Java front-end and URScript executor through XML
encoded Remote Procedure Calls (XML-RPC). Figure 3, page 14, shows the structure of the
My Daemon URCap project.
The /etc/service directory contains links to the URCap daemon executables currently running.
If a daemon executable has a link present but is in fact not running, the ERROR state will be
returned upon querying the daemon’s state. The links to daemon executables follow the lifetime
of the encapsulating URCap and will be removed when the URCap is removed. The initial state
for a daemon is STOPPED, however if it is desired, auto-start can be achieved by calling start() in
the init(DaemonContribution) method right after the daemon has had its resources installed.
20 try {
21 d a e m o n C o n t r i b u t i o n . i ns ta ll R es ou rc e ( new URL ( " file : com / ur / urcap / examples /
mydaemon / impl / daemon / " ) ) ;
22 } catch ( M a l f o r m e d U R L E x c e p t i o n e ) { }
23 }
24
25 @Override
26 public URL getExecutable () {
27 try {
28 // Two equivalent example daemons are available :
29 return new URL ( " file : com / ur / urcap / examples / mydaemon / impl / daemon / hello -
world . py " ) ; // Python executable
30 // return new URL (" file : com / ur / urcap / examples / mydaemon / impl / daemon /
HelloWorld ") ; // C ++ executable
31 } catch ( M a l f o r m e d U R L E x c e p t i o n e ) {
32 return null ;
33 }
34 }
35
36 public D a e m o n C o n t r i b u t i o n getDaemon () {
37 return d a e m o n C o n t r i b u t i o n ;
38 }
39
40 }
Log information with respect to the process handling of the daemon executable are saved
together with the daemon executable (follow the symbolic link of the daemon executable in
Note, that script daemons must include an interpreter directive at the first line to help select
the right program for interpreting the script. For instance, Bash scripts use "#!/bin/bash" and
Python scripts use "#!/usr/bin/env python".
The daemon runs in parallel with PolyScope and can in principle change its state independently.
Therefore, the label below the buttons displays the current run status of the daemon. This label
is updated with a 1 Hz frequency, utilizing the java.util.Timer class. Since the UI update is
initiated from a different thread than the Java AWT thread, the timer task must utilize the
EventQueue.invokeLater functionality. Note, the Timer is added when the My Daemon Installation
screen is opened (see openView()) and removed when the user moves away from the screen (see
closeView()) to conserve computing resources.
Two options are available for Java and URScript to communicate with the daemon:
• TCP/IP sockets can be used to stream data.
• XML encoded Remote Procedure Calls (XML-RPC) can be used for configuration tasks
(e.g. camera calibration) or service execution (e.g. locating the next object).
The advantage of XML-RPC over sockets is that no custom protocol or encoding needs to
be implemented. The URScript XML-RPC implementation supports all URScript data types.
Moreover, a RPC will only return when the function execution has been completed. This is
desirable when the next program step relies on data retrieved from the daemon service. Plain
sockets are on the other hand more efficient for data streaming, since there is no encoding
overhead. Both methods can be complimentary applied and are available for Java ↔ daemon
and URScript ↔ daemon communication.
Listing 9 shows a small URScript example for making a XML-RPC call to a XML-RPC server.
The hello-world.py example daemon (see Listing 21, page 61) can be used as XML-RPC test
server. Simply start the daemon in the My Daemon and run the URScript in a Script node.
The intention of this URScript example is to retrieve a message from the daemon to display
during runtime (similar to the My Daemon program node). The rpc_factory script function
creates a connection to the XML-RPC server in the daemon. The new connection is stored in
the global my_daemon variable and serves as a handle. The next line then requests the XML-RPC
server in the daemon to execute the get_message(...) function with the string argument "Bob"
and return the result. The return value of the RPC call is stored in the mydaemon_message variable
for further processing in the popup(...) script function.
Note, making XML-RPC calls from URScript does not require any additional function stubs or
pre-definitions of the remote function to be executed in URScript. Until the XML-RPC returns
this URScript thread is automatically blocked (i.e. no sync nor Wait is needed). The standard
XML-RPC protocol does not allow void return values and XML-RPC extensions enabling this
are not always compatible.
The My Daemon example also includes a Java XML-RPC client example, see the combination of
the MyDaemonProgramNodeContribution and XMLRPCMyDaemonInterface classes (Listing 19, page 57 and
listing 20, page 59 respectively). Note, the execution of the XML-RPC calls is not on the main
If the first line is not printed directly after installing the SDK, please reboot your PC for the
environment variables to be updated.
The My Daemon URCap comes with a fully functional C++ XML-RPC server example that is
equivalent to the hello-world.py Python daemon. Simply switch the comments in the function
getExecutable() in the MyDaemonDaemonService class (Listing 8, page 27), and recompile to use the
C++ daemon implementation. The popup title should now be appended with "(C++)" instead
of "(Python)" during execution of the URCap.
The C++ daemon directory structure is shown in Figure 6, page 31. For managing the software
construction process of the C++ daemon a tool called Scons is used. The SConstruct file among
other things contains the main configuration, the urtool3 cross-compiler, and libxmlrpc-c inte-
gration. The SConscript files are used to define the compilation targets, e.g. the Hello World
binary.
com.ur.urcap.examples.mydaemon
...
daemon
service
AbyssServer.cpp
AbyssServer.hpp
SConscript
XMLRPCMethods.cpp
XMLRPCMethods.hpp
Data.cpp
Data.hpp
HelloWorld.cpp
HelloWorld.cpp
SConscript
SConstruct
...
This will build a release version of the daemon. Using release=0 will build an executable with
debugging symbols.
The XML-RPC functionality in the C++ daemon relies on the open-source library libxmlrpc-
c (http://xmlrpc-c.sourceforge.net). This library is by default available on the CB3.0/3.1
and CB5.0 control boxes. The service directory contains all relevant XML-RPC code. The
AbyssServer is one of the XML-RPC server implementations supported by libxmlrpc-c. Please
look in the C++ code for more programming hints and links to relevant documentation.
• MyDaemonDaemonService
The MyDaemonInstallationNodeService class has visibility to an instance of the MyDaemonDaemonService
class. This instance is passed in the constructor when a new installation node instance of the
type MyDaemonInstallationNodeContribution is created with the createInstallationNode(...) method.
In this way, the daemon executable can be controlled from the installation node.
• The UI
• Persistence of program and installation files
• Creation and execution of programs
• Program undo/redo functionality
Information:
• Available from:
– URCap API version 1.0.0.
My Daemon is an extension to the Hello World URCap and demonstrates how a Python 2.5 or
C++ daemon can be integrated with the URCap Software Platform. This is useful when
a URCap depends on e.g. a driver or server which is not implemented in Java. Further-
more, the URCap shows how the XML-RPC protocol can be used to communicate with
the daemon from an installation node and in the script code generated by a program node.
Information:
• Available from:
– URCap API version 1.0.0.
– PolyScope version 3.3.0.
• Main API interfaces: DaemonContribution, DaemonService.
Script Function demonstrates how to add functions to the list of available script functions in
the Expression Editor. Script functions often used by end users of a URCap should be
added to this list.
Information:
• Available from:
– URCap API version 1.1.0.
– PolyScope version 3.4.0.
• Main API interfaces: Function, FunctionModel
Pick or Place is a toy example that shows how to make changes to the program tree through
the TreeNode API. The marker interface NonUserInsertable is used for program node con-
tributions that can only be inserted into the program tree by a URCap and not from the
UI of PolyScope by the end user.
Information:
• Available from:
– URCap API version 1.1.0.
– PolyScope version 3.4.0.
• Main API interfaces: ProgramModel, TreeNode, ProgramNodeFactory
Ellipse is a toy example, where a pose is used to define the center point for an ellipse-like
movement. The movement is achieved by inserting a pre-configured MoveP program node
containing pre-defined and named Waypoint nodes. This example demonstrates how to:
• Obtain a pose for the robot position by requesting the end user to define it using the
Move Tab
• Name waypoints
• Request the end user to move the robot to a given target position
Note:
• The functionality of assigning the Waypoint nodes custom names is only available
from URCap API version 1.4.0 (released with PolyScope version 3.7.0/5.1.0)
• Requesting the end user to move the robot to a defined center point is only available
from URCap API version 1.5.0 (released with PolyScope version 3.8.0/5.2.0).
• From URCap API version 1.6.0 (released with PolyScope version 3.9.0/5.3.0) the use
of the deprecated move node config factory (MoveNodeConfigFactory interface) has
been replaced with the equivalent builder and the TCP selection of the MoveP node
is pre-configured.
• Support for allowing the end user to start from and break on child nodes is only
available from URCap API version 1.9.0 (released with PolyScope version 5.6.0).
• From URCap API version 1.11.0 (released with PolyScope 3.14.0/5.9.0) the use of
the deprecated method getUserDefinedRobotPosition(RobotPositionCallback) has been re-
placed with the equivalent getUserDefinedRobotPosition(RobotPositionCallback2) method.
Furthermore, the use of the deprecated factory method for creating fixed position
Waypoint node configurations (createFixedPositionConfig(...) without TCP offset) has
been replaced with the equivalent method taking the TCP offset as parameter as well.
Information:
• Available from:
– URCap API version 1.2.56.
– PolyScope version 3.5.0.
Cycle Counter demonstrates how to work with variables. In this example, the chosen variable
will be incremented each time the program node is executed.
This sample also demonstrates how to allow the end user to use the builtin PolyScope sup-
port for starting from and pausing/breaking on a selected program node in the program
tree. In this case, the end user can start from or break on any child node under the Cycle
Counter (URCap) program node.
Note:
• Support for allowing the end user to start from and break on child nodes is only
available from URCap API version 1.9.0 (released with PolyScope version 5.6.0)
Information:
• Available from:
Idle Time demonstrates how to work with the ProgramNodeVisitor to traverse all program
nodes in a sub-tree. In this example, all Wait nodes will be visited. If a Wait node is
configured to wait for an amount of time, that amount of idle time (in seconds) will accu-
mulate in the selected variable.
Information:
• Available from:
– URCap API version 1.2.56.
– PolyScope version 3.5.0.
• Main API interfaces: ProgramNodeVisitor, WaitNodeConfig
Coordinate Map demonstrates how to capture click or touch coordinates when the user in-
teracts with an image.
Information:
• Available from:
– URCap API version 1.2.56.
– PolyScope version 3.5.0.
• Main API interfaces: TouchEvent
Information:
• Available from:
– URCap API version 1.2.56.
– PolyScope version 3.5.0.
• Main API interfaces: SystemSettings, Localization, Translatable, UnitType, Simpl-
eValueFactory
Node Ordering demonstrates how to define a specific sort order in PolyScope for the program
node contributions from a URCap.
Information:
• Available from:
– URCap API version 1.5.0.
– PolyScope version 3.8.0/5.2.0.
• Main API interfaces: ProgramNodeServiceConfigurable, ProgramNodeConfigura-
The URCap also shows how to use the filler UI element for controlling/grouping the layout
and how to add a custom input validator for detecting errors for enterable user inputs, in
this case an IP address user input (text field).
Information:
• Available from:
– URCap API version 1.7.0.
– PolyScope version 3.11.0/5.5.0.
• Main API interfaces: CustomUserInputConfiguration.
Simple Gripper demonstrates how to create a gripper driver contribution for a basic gripper
that only supports the mandatory ”default” grip and release actions. The URCap uses
digital outputs in the script code generation to trigger grip and release actions.
For further details about gripper driver contributions, please see the separate Gripper
Driver document.
Information:
• Available from:
– URCap API version 1.8.0.
– PolyScope version 3.11.0/5.5.0.
• Main API interfaces: GripperContribution.
Advanced Gripper demonstrates how to create a gripper driver contribution for a more ad-
vanced gripper that supports some of the optional gripper capabilities and controls the
Output Voltage setting of the Tool I/O Interface resource. The URCap shows how to:
• Configure gripper capabilities for, e.g. width, speed, force, vacuum and feedback for
grip and release detection
• Request exclusive control of the Tool I/O Interface resource
• Configure the Output Voltage I/O setting of the Tool I/O Interface
Additional information:
• For details about gripper driver contributions, please see the separate Gripper Driver
document.
• For information about system resource control, please see the separate Resource Con-
trol document.
Note:
• Feedback capabilities of the gripper is only available from URCap API version 1.9.0
(released with PolyScope version 3.12.0/5.6.0)
Information:
• Available from:
– URCap API version 1.8.0.
– PolyScope version 3.11.0/5.5.0.
• Main API interfaces: GripperContribution, GripperCapabilities, SystemConfigu-
ration, GripActionParameters, ReleaseActionParameters, GripperFeedbackCapabili-
ties.
Custom Gripper Setup demonstrates how to create a gripper driver contribution which de-
fines a custom UI in the installation node for setting up the gripper as well as adds a TCP
for the gripper to PolyScope.
For further details about gripper driver contributions, please see the separate Gripper
Driver document.
Information:
• Available from:
– URCap API version 1.8.0.
– PolyScope version 3.11.0/5.5.0.
Dual Zone Gripper demonstrates how to create a gripper driver contribution for a multi-
gripper that supports multiple permanently enabled, individual grippers as well as dynam-
ically adjusts the parameters of a capability (after it has been registered) for all individual
grippers.
In this example, the gripper is a vacuum dual gripper with two independent physical grip-
pers (zones) ”built into” the gripping device. The URCap provides the user the option
to select from three individual zones (grippers), named Zone A, Zone B and Zone A+B.
The user can operate Zone A and Zone B independently of one another or choose to use
both zones at the same by selecting Zone A+B.
A custom UI is defined in the installation node which allows the user to enable the ”Use
Fragile Handling” option. When enabled, this will restrict the maximum vaccum level
available for a grip action. This is achieved by reducing the maximum value and default
value parameters of the registered vacuum capability simultaneously for all zones.
For further details about gripper driver contributions, please see the separate Gripper
Driver document.
Information:
• Available from:
– URCap API version 1.11.0.
– PolyScope version 3.14.0/5.9.0.
• Main API interfaces: GripperContribution, GripperCapabilities, GripperListProvider,
GripperListBuilder, SelectableGripper, SystemConfiguration, TCPConfiguration, Grip-
VacuumCapability
Dynamic Multi-Gripper demonstrates how to create a gripper driver contribution that sup-
ports both a single gripper setup and multi-gripper setup as well as dynamically adjusts
the parameters of a capability (after it has been registered) exclusively for each individual
gripper.
In this example, the gripper driver provides the user the option of choosing between us-
ing a setup with only a single gripper mounted on the robot, or one where two separate,
identical grippers are mounted on the robot at the same time. Each individual gripper
A custom UI is defined in the installation node which allows the user to select how many
gripper are mounted on the robot. The available options are Single and Dual. Selecting
the Single option will disable the secondary gripper, Gripper 2, and only the standard
gripper, Gripper 1, will be available to the user. Choosing the Dual option will enable
both grippers making both of them available to the user. Depending on the option se-
lected, the offset of TCP for the standard gripper (Gripper 1 ) is updated accordingly.
For each individual gripper (Gripper 1 and Gripper 2 ), the user can configure the type of
fingertips attached to the gripper (in the installation node). There are two options, Stan-
dard and Extended. The Standard option is the default whereas the Extended option can
be selected, when fingertips with wide range is attached. Selecting the Extended option
will increase the maximum value and default value parameters of the registered width ca-
pability exclusively for the specific individual gripper (independently of the other gripper).
For further details about gripper driver contributions, please see the separate Gripper
Driver document.
Information:
• Available from:
– URCap API version 1.11.0.
– PolyScope version 3.14.0/5.9.0.
• Main API interfaces: GripperContribution, GripperCapabilities, GripperListProvider,
GripperListBuilder, SelectableGripper, MultiGripperCapability, SystemConfiguration,
TCPConfiguration, WidthCapability
Simple Screwdriver demonstrates how to create a screwdriver driver contribution for a basic
screwdriver that only supports the mandatory start and stop screwdriver operations. The
URCap uses digital outputs in the script code generation to start and stop the screwdriver.
For further details about screwdriver driver contributions, please see the separate Screw-
driver Driver document.
Information:
• Available from:
– URCap API version 1.9.0.
For further details about screwdriver driver contributions, please see the separate Screw-
driver Driver document.
Information:
• Available from:
– URCap API version 1.9.0.
– PolyScope version 5.6.0.
Custom Screwdriver demonstrates how to create a screwdriver contribution driver which de-
fines a custom UI in the Screwdriving installation node for setting up the screwdriver as
well as adds a default TCP for the screwdriver to PolyScope.
Information:
• Available from:
– URCap API version 1.9.0.
– PolyScope version 5.6.0.
This prompts you with a dialog box where you select a group and artifact-id for your new
URCap:
Next you must specify which Universal Robots robot series your URCap is compatible with:
Maybe your URCap requires specific hardware and/or software features, which are only available
on a specific robot series (e.g., e-Series robots only), or the URCap might be compatible with
all robot series. Such considerations should determine your choice here. See section 12.1. Robot
Series Compatibility for more details about robot series compatibility.
Finally, you must also specify the target URCap API version. Choosing an earlier version
of the API will make your URCap compatible with earlier PolyScope versions, but also limit
the functionality accessible through the API. URCap API compatibility is described in section
12.2. API Compatibility.
Pressing the Ok button in the last dialog creates a new Maven project under the sub-folder
Notice that the generated pom.xml file has a section with a set of properties for the new URCap
with meta-data for vendor, contact address, copyright, description, and short license informa-
tion which will be displayed to the user when the URCap is installed in PolyScope. This section
also contains boolean compatibility flag properties specifying the URCap’s compatibility with
the different available robot series. Update the URCap meta-data section with the data relevant
for the new URCap. See Listing 1 for an example of how this section might look.
Should you need to change the version of the URCap API to depend upon after your project
has been setup, this can be done in the pom.xml file in your project. Here you must update
to the desired version in the URCap API dependency under the <dependencies> section of the
pom.xml-file as well as modify the <import-package> element under the maven-bundle-plugin if version
information is present. See listings 11, 12 and 15 for examples of this
12 Compatibility
12.1 Robot Series Compatibility
When developing URCaps you must mandatorily specify which robot series your URCap is com-
patible with, i.e. which robot series your URCap can run on. The purpose is to provide a good
user experience for the URCap installation process by ensuring that end users cannot install
a URCap that is not compatible with their robot and/or Universal Robots offline simulator
software (URSim).
When specifying your URCap’s compatibility, you need to consider, if your URCap depends on
specific hardware and/or software functionality, which is not available on all robot series (e.g.,
only available on e-Series robots), or if the URCap can run on all available robot series. Note
that your URCap’s compatibility must be specified for all robot series available in the line of
robots from Universal Robots (including robot series which your URCap is incompatible with).
These two properties specify the URCap’s compatibility with the CB3 and the e-Series robot
series, respectively. Acceptable values for the properties are either true or false. A value of true
indicates that the URCap can be installed and run on the given robot series, and a value of
false indicates otherwise. Note that both flags must present including the flag for a robot series
which your URCap is not compatible with.
Using the newURCap.sh script mentioned in section 11. Creating new thin Projects using a Maven
Archetype, the compatibility flags are handled automatically for you. On the other hand, if an
existing URCap needs to be updated to support the compatibility flags, you must manually
update your URCap project’s pom.xml file. This will only be necessary, if you used a version of
the URCap SDK lower than 1.12.0 to create the URCap by either using the newURCap.sh script,
or by basing the URCap on a modification of one of the URCap example projects included in
the SDK. The required changes are described in subsection 12.1.3. Updating Existing URCaps.
If the compatibility of your URCap changes, remember to update the value of the URCap’s
compatibility flags accordingly.
Similarly, a URCap’s compatibility flags are examined during startup of PolyScope, if the UR-
Cap was deployed and installed using Maven (described in section 6. Deployment with Maven)
instead of being installed through the PolyScope UI. If the URCap is incompatible, it will not
be allowed to start and an error dialog will be displayed in PolyScope.
The presence of the compatibility flags in a URCap will be verified during startup of PolyScope
(if the URCap was deployed and installed using Maven), and a URCap will not be started, if
any of the flags are missing. Likewise, a URCap with missing compatibility flag(s) will not be
installable through PolyScope. An error dialog will be displayed in PolyScope in these cases.
Note that only URCaps depending on URCap API version 1.12.0 or higher will be prevented
from starting or being installed, if the compatibility flags are not present.
First of all, you must add the two compatibility flag properties, urcap.compatibility.CB3 and
urcap.compatibility.eSeries, to the URCap meta-data section as shown in Listing 13 in section
12.1.1. Compatibility Flags. Remember to specify boolean values corresponding to your UR-
Cap’s compatibility.
Listing 14: Excerpt of the pom.xml file’s Maven bundle plugin configuration with compatiblity
flag support
1 ...
2 < build >
3 < plugins >
4 ...
5 < plugin >
6 < groupId > org . apache . felix </ groupId >
7 < artifactId > maven - bundle - plugin </ artifactId >
8 ...
9 < configuration >
10 < instructions >
11 <! - -******** DO NOT MODIFY THE ENTRIES OF THIS SECTION ******** - - >
12 < Bundle - Category > URCap </ Bundle - Category >
13 ...
14 < URCapCompatibility - CB3 > $ { urcap . compatibility . CB3 } </
URCapCompatibility - CB3 >
15 < URCapCompatibility - eSeries > $ { urcap . compatibility . eSeries } </
URCapCompatibility - eSeries >
16 <! - -*********************************************************** - - >
17 ...
18 < instructions >
19 < configuration >
20 < plugin >
21 ...
22 < plugins >
23 ...
In order for PolyScope to be able to read the specified values for the compatibility flag proper-
ties, the <build> section with configuration of build plugins also needs to be updated. Here the
configuration for the Maven bundle plugin must be updated to contain two new instructions,
<URCapCompatibility-CB3> and <URCapCompatibility-eSeries>, as shown in the excerpt in Listing 14.
For convenience, instead of manually making the changes described above, you can use the
upgradeURCap.sh script to update your pom.xml file.
Below are examples of usage of the script, where /home/myuser/myURCap is the path to the folder
with a URCap project and /home/myuser/sdk is the path to the folder of the URCap SDK.
This will apply the required changes to the URCap project’s pom.xml file inside the folder
/home/myuser/myURCap. Both compatibility flag properties are assigned the boolean value of
true by default (i.e. your URCap is compatible with both the CB3 and e-Series robot series).
Remember to change the value of the compatibility flag properties in the pom.xml file accord-
ingly, if the preassigned default values does not match your URCap’s compatibility.
Alternatively, it is also possible to copy the upgradeURCap.sh script to your URCap project folder
and execute the script from within that folder by typing:
1 $ cd myURCap
2 $ ./ upgradeURCap . sh .
Finally, you have the option of directly specifying the boolean value for the compatibility flags
using the compatibilityCB3 and compatibilityESeries arguments:
1 $ ./ upgradeURCap . sh ../ myURCap c o m p a t i b i l i t y C B 3 = false c o m p a t i b i l i t y E S e r i e s =
true
In this case, you have specified that your URCap is compatible with the e-Series robot series,
but incompatible with the CB3 robot series.
A given version of the API is compatible with a specific version of PolyScope (see table 1 below).
PolyScope will remain backwards compatible with earlier versions of the API. This means that
if you choose to use the newest API, customers using your URCap must be running at least the
It is not a problem if the customer is running a newer (future) version of PolyScope. However,
it is not possible for the customer to use your URCap if he is running an earlier version of
PolyScope than the one the API was released with. A good rule of thumb is thus to choose the
earliest possible version of the API that supports the functionality you wish to use. This will
target the broadest audience.
For instance, if you specify a dependency on the API version 1.1.0, your URCap will only run on
PolyScope version 3.4.0 or newer. If you wish to target the broadest possible audience, you must
use version 1.0.0 of the API and the customers must be running PolyScope version 3.3.0 or newer.
Note that API version 1.12.0 is the final API version supported on CB3 robots. Higher API
versions are only available on e-Series robots.
As an example you could have a dependency on API version 1.5.0 and run it on PolyScope
version 3.7.0/5.1.0 (officially only supporting API version 1.4.0), but in the actual execution
path you can only use types present in API version 1.4.0.
To have the URCap resolve at runtime the pom.xml must have the option ;resolution:=optional
appended to the URCap API entry in the <import-package> section. The full <import-package>
section could look like this:
This will make the URCap start up regardless of what the actual dependency states.
As mentioned you must also structure your code so no code referring unsupported API function-
ality is executed. Also no import or catch clauses referring to unsupported types can be present
in classes that will execute. See listings 16 and 17 for an example of how to structure this. In
listing 16 all code related to unsupported API functionality is located and in 17 a check for
PolyScope version number is performed before creating an instance of the AdvancedFeature class.
If you choose to use this advanced feature, you must test your URCap carefully on all PolyScope
versions you wish to support making sure all code execution paths are tested.
13 Exception Handling
All exceptions thrown and not caught in a URCap will be caught by PolyScope. If this happens
when the end user selects either an installation node or a program node from the URCap, the
UI provided by the URCap will be replaced by a screen displaying information about the error.
In all other cases, a dialog will be shown to the end user.
The error screen and dialog will show that an exception has happened in the URCap along with
meta information about the URCap. It will also contain a section showing the stack trace from
the exception in the URCap code.
14 Troubleshooting
Internally in PolyScope, a URCap is installed as an OSGi bundle with the Apache Felix OSGi
framework. For the purpose of debugging problems, it is possible to inspect various information
about bundles using the Apache Felix command shell.
You can establish a shell connection to the running Apache Felix by opening a TCP connection
on port 6666. Access the Apache Felix shell console by typing:
1 $ nc 127.0.0.1 6666
2
3 Felix Remote Shell Console :
4 ============================
5
6 ->
Note that you need to use the nc command, since the telnet command is not available on the
robot, and 127.0.0.1 because localhost does not work on a robot.
To view a list of installed bundles and their state type the following command:
1 -> ps
2 START LEVEL 1
3 ID State Level Name
4 [ 0] [ Active ] [ 0] System Bundle (5.2.0)
Inside the shell you can type help to see the list of the available commands:
1 -> help
2 uninstall
3 sysprop
4 bundlelevel
5 find
6 version
7 headers
8 refresh
9 start
10 obr
11 inspect
12 ps
13 stop
14 shutdown
15 help
16 update
17 install
18 log
19 cd
20 startlevel
21 resolve
22
23 Use ’ help < command - name > ’ for more information .
24 ->
For example, the headers command can be executed to display different properties of the indi-
vidual installed bundles.
The following example is taken from a small program using the Hello World URCap.
To see the generated script code for an URCap, it is required to save the program. Assuming
the program is called hello.urp, the corresponding script code can be found in hello.script.
The script code of the installation node will be surrounded with begin/end URCap comments
and information about the source of the URCap and its type:
1 ...
2 # begin : URCap Installation Node
3 # Source : Hello World , 1.0.0. SNAPSHOT , Universal Robots
4 # Type : Hello World
5 h e l l o _ w o r l d _ p o p u p _ t i t l e = " Hello World "
6 # end : URCap Installation Node
7 ...
Similarly for program nodes, script code is surrounded with begin/end URCap comments and
information about the source of the URCap and its type as shown below. The program node
generated by PolyScope is the first label after the # begin: URCap Program Node, here $ 2. The
4. Any of the elements <label>, <input>, and <select>, when appearing inside a <form> as a
child of <form>, <div> or <p>.
5. The element <option>, when appearing as a child of <select>.
6. The elements <span>, <em>, <b>, <i>, <br>, and <hr>.
As for standard HTML attributes, only the id and style and src attributes are supported. Data
input widgets correspond to HTML elements as follows:
• Text input field: <input type="text"/>.
• Number input field: <input type="number"/>.
width, height: Serve to specify the size of an element with display attribute different from
inline. The accepted values are numbers, optionally followed by “px”, or percentages
of the respective dimension of the parent element.
padding, padding-top, padding-right, padding-bottom, padding-left:
Used to specify spacing around
an element. The accepted values are numbers, optionally followed by “px”. The at-
tribute padding may take 1, 2 or 4 values. All the other attributes take a single value.
font-size, font-style, font-weight:For modifying the size and type of font used within the
element. Allowed values for font-size are numbers, optionally followed by “px”, or
percentages of the font size of the parent element. Allowed values for font-style are
normal and italic. Allowed values for font-weight are normal and bold.
vertical-align: Sets the vertical aligment of text within a label element. Allowed values for
vertical-align are top, middle and bottom.
text-align: Sets the horizontal aligment of text within a label element. Allowed values for
text-align are left, center and right.
Listing 18: Java class defining functionality for the My Daemon installation node
1 package com . ur . urcap . examples . mydaemon . impl ;
2
3 import com . ur . urcap . api . contribution . D a e m o n C o n t r i b u t i o n ;
4 import com . ur . urcap . api . contribution . I n s t a l l a t i o n N o d e C o n t r i b u t i o n ;
5 import com . ur . urcap . api . domain . data . DataModel ;
6 import com . ur . urcap . api . domain . script . ScriptWriter ;
7 import com . ur . urcap . api . ui . annotation . Input ;
8 import com . ur . urcap . api . ui . annotation . Label ;
9 import com . ur . urcap . api . ui . component . InputButton ;
10 import com . ur . urcap . api . ui . component . InputEvent ;
11 import com . ur . urcap . api . ui . component . In putText Field ;
12 import com . ur . urcap . api . ui . component . La belComp onent ;
13
14 import java . awt . EventQueue ;
15 import java . util . Timer ;
16 import java . util . TimerTask ;
17
18 public class M y D a e m o n I n s t a l l a t i o n N o d e C o n t r i b u t i o n implements
InstallationNodeContribution {
19 private static final String P OPUPTITL E_KEY = " popuptitle " ;
20
21 private static final String XM L RP C_ VA R IA BL E = " my_daemon " ;
60 a p p l y D e s i r e d D a e m o n S t a t u s () ;
61 }
62 }
63
64 @Input ( id = " b t n D i s a b l e D a e m o n " )
65 public void onStopClick ( InputEvent event ) {
66 if ( event . getEventType () == InputEvent . EventType . ON_CHANGE ) {
67 s e t D a e m o n E n a b le d ( false ) ;
68 a p p l y D e s i r e d D a e m o n S t a t u s () ;
69 }
70 }
71
72 @Override
73 public void openView () {
74 e n a b l e D a e m o n B u t t o n . setText ( " Start Daemon " ) ;
75 d i s a b l e D a e m o n B u t t o n . setText ( " Stop daemon " ) ;
76 p op up Ti t le Fi e ld . setText ( getPopupTitle () ) ;
77
78 // UI updates from non - GUI threads must use EventQueue . invokeLater ( or
Swin gUtiliti es . invokeLater )
79 uiTimer = new Timer ( true ) ;
80 uiTimer . schedule ( new TimerTask () {
81 @Override
82 public void run () {
83 EventQueue . invokeLater ( new Runnable () {
84 @Override
85 public void run () {
86 updateUI () ;
186 }
187
188 private void a w a i t D a e m o n R u n n i n g ( long t i m e O u t M i l l i S e c o n d s ) throws
InterruptedException {
189 daemonService . getDaemon () . start () ;
190 long endTime = System . nanoTime () + t i m e O u t M i l l i S e c o n d s * 1000 L * 1000 L ;
191 while ( System . nanoTime () < endTime && ( daemonService . getDaemon () . getState ()
!= D a e m o n C o n t r i b u t i o n . State . RUNNING || ! x m l R p c D a e m o n I n t e r f a c e .
isReachable () ) ) {
192 Thread . sleep (100) ;
193 }
194 }
195
196 private D a e m o n C o n t r i b u t i o n . State getDa emonStat e () {
197 return daemonService . getDaemon () . getState () ;
198 }
199
200 private Boolean is Da e mo nE na b le d () {
201 return model . get ( ENABLED_KEY , true ) ; // This daemon is enabled by default
202 }
203
204 private void s e t D a e m o n E n a b l e d ( Boolean enable ) {
205 model . set ( ENABLED_KEY , enable ) ;
206 }
207
208 public String g e t X M L R P C V a r i a b l e () {
209 return X M LR PC _V A RI AB LE ;
210 }
Listing 19: Java class defining functionality for the My Daemon program node
1 package com . ur . urcap . examples . mydaemon . impl ;
2
3 import com . ur . urcap . api . contribution . P r o g r a m N o d e C o n t r i b u t i o n ;
4 import com . ur . urcap . api . domain . URCapAPI ;
5 import com . ur . urcap . api . domain . data . DataModel ;
6 import com . ur . urcap . api . domain . script . ScriptWriter ;
7 import com . ur . urcap . api . ui . annotation . Input ;
8 import com . ur . urcap . api . ui . annotation . Label ;
9 import com . ur . urcap . api . ui . component . InputEvent ;
10 import com . ur . urcap . api . ui . component . In putTextF ield ;
11 import com . ur . urcap . api . ui . component . La belCompo nent ;
12
13 import java . awt .*;
14 import java . util . Timer ;
15 import java . util . TimerTask ;
16
17 public class M y D a e m o n P r o g r a m N o d e C o n t r i b u t i o n implements
ProgramNodeContribution {
18 private static final String NAME = " name " ;
19
20 private final DataModel model ;
21 private final URCapAPI api ;
22 private Timer uiTimer ;
23
24 public M y D a e m o n P r o g r a m N o d e C o n t r i b u t i o n ( URCapAPI api , DataModel model ) {
25 this . api = api ;
26 this . model = model ;
27 }
28
29 @Input ( id = " yourname " )
30 private Inp utTextF ield nameTextField ;
31
32 @Label ( id = " t i t l e P r e v i e w L a b e l " )
33 private La belCompo nent t i t l e P r e v i e w L a b e l ;
34
35 @Label ( id = " m e s s a g e P r e v i e w L a b e l " )
36 private La belCompo nent m e s s a g e P r e v i e w L a b e l ;
37
38 @Input ( id = " yourname " )
39 public void onInput ( InputEvent event ) {
40 if ( event . getEventType () == InputEvent . EventType . ON_CHANGE ) {
41 setName ( nameTextField . getText () ) ;
42 updatePreview () ;
43 }
44 }
45
46 @Override
47 public void openView () {
48 nameTextField . setText ( getName () ) ;
49
50 // UI updates from non - GUI threads must use EventQueue . invokeLater ( or
Swin gUtiliti es . invokeLater )
51 uiTimer = new Timer ( true ) ;
52 uiTimer . schedule ( new TimerTask () {
53 @Override
54 public void run () {
55 EventQueue . invokeLater ( new Runnable () {
56 @Override
57 public void run () {
32 } catch ( X ml Rp cE x ce pt io n e ) {
33 return false ;
34 }
35 }
36
37 public String getTitle () throws XmlRpcException , U n k n o w n R e s p o n s e E x c e p t i o n {
38 Object result = client . execute ( " get_title " , new ArrayList < String >() ) ;
39 return processString ( result ) ;
40 }
41
42 public String setTitle ( String title ) throws XmlRpcException ,
UnknownResponseException {
43 ArrayList < String > args = new ArrayList < String >() ;
44 args . add ( title ) ;
45 Object result = client . execute ( " set_title " , args ) ;
46 return processString ( result ) ;
47 }
48
49 public String getMessage ( String name ) throws XmlRpcException ,
UnknownResponseException {
50 ArrayList < String > args = new ArrayList < String >() ;
51 args . add ( name ) ;
52 Object result = client . execute ( " get_message " , args ) ;
53 return processString ( result ) ;
54 }
55
56 private boolean pr ocessBoo lean ( Object response ) throws
UnknownResponseException {