`. To allow for breaking up a long list of radio buttons or checkboxes (with `"orientation": "vertical"`) into logical sections, a set of special CSS _classes_ are available named `tw-space-above-N` and `tw-space-below-N` for numbers 1 to 5. Applying one of these classes to an element in the `htmlLabel` (e.g. `
Label`) will add a corresponding amount of space above or below that option in the annotation UI.
+
+
+
+```json
+[
+ {
+ "name": "sentiment",
+ "type": "radio",
+ "title": "Sentiment",
+ "orientation": "vertical",
+ "options": [
+ {
+ "value": "positive",
+ "htmlLabel": "Positive"
+ },
+ {
+ "value": "neutral",
+ "htmlLabel": "Neutral"
+ },
+ {
+ "value": "negative",
+ "htmlLabel": "Negative"
+ },
+ {
+ "value": "unknown",
+ "htmlLabel": "Cannot be determined"
+ }
+ ]
+ }
+]
+```
+
+
+
+### Selector input
+
+
+
+```json
+[
+ {
+ "name": "mylabel",
+ "type": "selector",
+ "optional": true, //Optional - Set if validation is not required
+ "options": [ // The options that the user is able to select from
+ {"value": "value1", "label": "Text to show user 1"},
+ {"value": "value2", "label": "Text to show user 2"},
+ {"value": "value3", "label": "Text to show user 3"}
+ ],
+ "title": "Title string", //Optional
+ "description": "Description string", //Optional
+ "valSuccess": "Success message when the field is validated", //Optional
+ "valError": "Error message when the field fails validation" //Optional
+ }
+]
+```
+
+
+
+### Optional help text
+
+Optionally, radio buttons and checkboxes can be given help text to provide additional per-choice context or information to help annotators.
+
+
+
+
+```json
+[
+ {
+ "name": "mylabel",
+ "type": "radio",
+ "optional": true, //Optional - Set if validation is not required
+ "orientation": "vertical", //Optional - default is "horizontal"
+ "options": [ // The options that the user is able to select from
+ {"value": "value1", "label": "Text to show user 1", "helptext": "Additional help text for option 1"},
+ {"value": "value2", "label": "Text to show user 2", "helptext": "Additional help text for option 2"},
+ {"value": "value3", "label": "Text to show user 3"}
+ ],
+ "title": "Title string", //Optional
+ "description": "Description string", //Optional
+ "valSuccess": "Success message when the field is validated", //Optional
+ "valError": "Error message when the field fails validation" //Optional
+ }
+]
+```
+
+
+
+### Alternative way to provide options for radio, checkbox and selector
+
+A dictionary (key value pairs) and also be provided to the `options` field of the radio, checkbox and selector widgets
+but note that the ordering of the options are **not guaranteed** as javascript does not sort dictionaries by
+the order in which keys are added. Note that additional help texts for radio buttons and checkboxes are not supported using this syntax.
+
+
+
+```json
+[
+ {
+ "name": "mylabel",
+ "type": "radio",
+ "optional": true, //Optional - Set if validation is not required
+ "options": { // The options can be specified as a dictionary, ordering is not guaranteed
+ "value1": "Text to show user 1",
+ "value2": "Text to show user 2",
+ "value3": "Text to show user 3"
+ },
+ "title": "Title string", //Optional
+ "description": "Description string", //Optional
+ "valSuccess": "Success message when the field is validated", //Optional
+ "valError": "Error message when the field fails validation" //Optional
+ }
+]
+```
+
+
+
+### Dynamic options for radio, checkbox and selector
+
+All the examples above have a "static" list of available options for the radio, checkbox and selector widgets, where the complete options list is enumerated in the project configuration and every document offers the same set of options. However it is also possible to take some or all of the options from the _document_ data rather than the _configuration_ data. For example:
+
+
+
+**Project configuration**
+
+```json
+[
+ {
+ "name": "uri",
+ "type": "radio",
+ "title": "Select the most appropriate URI",
+ "options":[
+ {"fromDocument": "candidates"},
+ {"value": "none", "label": "None of the above"},
+ {"value": "unknown", "label": "Cannot be determined without more context"}
+ ]
+ }
+]
+```
+
+**Document**
+
+```json
+{
+ "text": "President Bush visited the air base yesterday...",
+ "candidates": [
+ {
+ "value": "http://dbpedia.org/resource/George_W._Bush",
+ "label": "George W. Bush (Jnr)"
+ },
+ {
+ "value": "http://dbpedia.org/resource/George_H._W._Bush",
+ "label": "George H. W. Bush (Snr)"
+ }
+ ]
+}
+```
+
+
+
+`"fromDocument"` is a dot-separated property path leading to the location within each document where the additional options can be found, for example `"fromDocument":"candidates"` looks for a top-level property named `candidates` in each document, `"fromDocument": "options.custom"` would look for a property named `options` which is itself an object with a property named `custom`. The target property in the document may be in any of the following forms:
+
+- an array _of objects_, each with `value` and `label` (and optionally `helptext`) properties, exactly as in the static configuration format - this is the format used in the example above
+- an array _of strings_, where the same string will be used as both the value and the label for that option
+- an arbitrary ["dictionary"](#options-as-dict) object mapping values to labels
+- a _single string_, which is parsed into a list of options
+
+The "single string" alternative is designed to be easier to use when [importing documents](documents_annotations_management.md#importing-documents) from CSV files. It allows you to provide any number of options in a _single_ CSV column value. Within the column the options are separated by semicolons, and each option is of the form `value=label`. Whitespace around the delimiters is ignored, both between options and between the value and label of a single option. For example given CSV document data of
+
+| text | options |
+|-----------------|---------------------------------------------------|
+| Favourite fruit | `apple=Apples; orange = Oranges; kiwi=Kiwi fruit` |
+
+a `{"fromDocument": "options"}` configuration would produce the equivalent of
+
+```json
+[
+ {"value": "apple", "label": "Apples"},
+ {"value": "orange", "label": "Oranges"},
+ {"value": "kiwi", "label": "Kiwi fruit"}
+]
+```
+
+If your values or labels may need to contain the default separator characters `;` or `=` you can select different separators by adding extra properties to the configuration:
+
+```json
+{"fromDocument": "options", "separator": "~~", "valueLabelSeparator": "::"}
+```
+
+| text | options |
+|-----------------|------------------------------------------------------|
+| Favourite fruit | `apple::Apples ~~ orange::Oranges ~~ kiwi::Kiwi fruit` |
+
+The separators can be more than one character, and you can set `"valueLabelSeparator":""` to disable label splitting altogether and just use the value as its own label.
+
+### Mixing static and dynamic options
+
+Static and `fromDocument` options may be freely interspersed in any order, so you can have a fully-dynamic set of options by specifying _only_ a `fromDocument` entry with no static options, or you can have static options that are listed first followed by dynamic options, or dynamic options first followed by static, etc.
+
+### Conditional components
+
+By default all components listed in the project configuration will be shown for all documents. However this is not always appropriate, for example you may have some components that are only relevant to certain documents, or only relevant for particular combinations of values in _other_ components. To allow for these kinds of scenarios any component can have a field named `if` specifying the conditions under which that component should be shown.
+
+The `if` field is an _expression_ that is able to refer to fields in both the current _document_ being annotated and the current state of the other annotation components. The expression language is largely based on a subset of the standard JavaScript expression syntax but with a few additional syntax elements to ease working with array data and regular expressions.
+
+The following simple example shows how you might implement an "Other (please specify)" pattern, where the user can select from a list of choices but also has the option to supply their own answer if none of the choices are appropriate. The free text field is only shown if the user selects the "other" choice.
+
+
+
+**Project configuration**
+
+```json
+[
+ {
+ "name": "uri",
+ "type": "radio",
+ "title": "Select the most appropriate URI",
+ "options":[
+ {"fromDocument": "candidates"},
+ {"value": "other", "label": "Other"}
+ ]
+ },
+ {
+ "name": "otherValue",
+ "type": "text",
+ "title": "Please specify another value",
+ "if": "annotation.uri == 'other'",
+ "regex": "^(https?|urn):",
+ "valError": "Please specify a URI (starting http:, https: or urn:)"
+ }
+]
+```
+
+**Document**
+
+```json
+{
+ "text": "President Bush visited the air base yesterday...",
+ "candidates": [
+ {
+ "value": "http://dbpedia.org/resource/George_W._Bush",
+ "label": "George W. Bush (Jnr)"
+ },
+ {
+ "value": "http://dbpedia.org/resource/George_H._W._Bush",
+ "label": "George H. W. Bush (Snr)"
+ }
+ ]
+}
+```
+
+
+Note that validation rules (such as `optional`, `minSelected` or `regex`) are not applied to components that are hidden by an `if` expression - hidden components will never be included in the annotation output, even if they would be considered "required" had they been visible.
+
+Components can also be made conditional on properties of the _document_, or a combination of the document and the annotation values, for example
+
+
+
+**Project configuration**
+
+```json
+[
+ {
+ "name": "htmldisplay",
+ "type": "html",
+ "text": "{{{text}}}"
+ },
+ {
+ "name": "sentiment",
+ "type": "radio",
+ "title": "Sentiment",
+ "description": "Please select a sentiment of the text above.",
+ "options": [
+ {"value": "negative", "label": "Negative"},
+ {"value": "neutral", "label": "Neutral"},
+ {"value": "positive", "label": "Positive"}
+ ]
+ },
+ {
+ "name": "reason",
+ "type": "text",
+ "title": "Why do you disagree with the suggested value?",
+ "if": "annotation.sentiment !== document.preanno.sentiment"
+ }
+]
+```
+
+**Documents**
+
+```json
+[
+ {
+ "text": "I love the thing!",
+ "preanno": { "sentiment": "positive" }
+ },
+ {
+ "text": "I hate the thing!",
+ "preanno": { "sentiment": "negative" }
+ },
+ {
+ "text": "The thing is ok, I guess...",
+ "preanno": { "sentiment": "neutral" }
+ }
+]
+```
+
+
+
+The full list of supported constructions is as follows:
+
+- the `annotation` variable refers to the current state of the annotation components for this document
+ - the current value of a particular component can be accessed as `annotation.componentName` or `annotation['component name']` - the brackets version will always work, the dot version works if the component's `name` is a valid JavaScript identifier
+ - if a component has not been set since the form was last cleared the value may be `null` or `undefined` - the expression should be written to cope with both
+ - the value of a `text`, `textarea`, `radio` or `selector` component will be a single string (or null/undefined), the value of a `checkbox` component will be an _array_ of strings since more than one value may be selected. If no value is selected the array may be null, undefined or empty, the expression must be prepared to handle any of these
+- the `document` variable refers to the current document that is being annotated
+ - again properties of the document can be accessed as `document.propertyName` or `document['property name']`
+ - continue the same pattern for nested properties e.g. `document.scores.label1`
+ - individual elements of array properties can be accessed by zero-based index (e.g. `document.options[0]`)
+- various comparison operators are available:
+ - `==` and `!=` (equal and not-equal)
+ - `<`, `<=`, `>=`, `>` (less-than, less-or-equal, greater-or-equal, greater-than)
+ - these operators follow JavaScript rules, which are not always intuitive. Generally if both arguments are strings then they will be compared by lexicographic order, but if either argument is a number then the other one will also be converted to a number before comparing. So if the `score` component is set to the value "10" (a string of two digits) then `annotation.score < 5` would be _false_ (10 is converted to number and compared to 5) but `annotation.score < '5'` would be _true_ (the string "10" sorts before the string "5")
+ - `in` checks for the presence of an item in an array or a key in an object
+ - e.g. `'other' in annotation.someCheckbox` checks if the `other` option has been ticked in a checkbox component (whose value is an array)
+ - this is different from normal JavaScript rules, where `i in myArray` checks for the presence of an array _index_ rather than an array _item_
+- other operators
+ - `+` (concatenate strings, or add numbers)
+ - if either argument is a string then both sides are converted to strings and concatenated together
+ - otherwise both sides are treated as numbers and added
+ - `-`, `*`, `/`, `%` (subtraction, multiplication, division and remainder)
+ - `&&`, `||` (boolean AND and OR)
+ - `!` (prefix boolean NOT, e.g. `!annotation.selected` is true if `selected` is false/null/undefined and false otherwise)
+ - conditional operator `expr ? valueIfTrue : valueIfFalse` (exactly as in JavaScript, first evaluates the test `expr`, then either the `valueIfTrue` or `valueIfFalse` depending on the outcome of the test)
+- `value =~ /regex/` tests whether the given string value contains any matches for the given [regular expression](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_expressions#writing_a_regular_expression_pattern)
+ - use `^` and/or `$` to anchor the match to the start and/or end of the value, for example `annotation.example =~ /^a/i` checks whether the `example` annotation value _starts with_ "a" or "A" (the `/i` flag makes the expression case-insensitive)
+ - since the project configuration is entered as JSON, any backslash characters within the regex must be doubled to escape them from the JSON parser, i.e. `"if": "annotation.option =~ /\\s/"` would check if `option` contains any space characters (for which the regular expression literal is `/\s/`)
+- _Quantifier_ expressions let you check whether `any` or `all` of the items in an array or key/value pairs in an object match a predicate expression. The general form is `any(x in expr, predicate)` or `all(x in expr, predicate)`, where `expr` is an expression that resolves to an array or object value, `x` is a new identifier, and `predicate` is the expression to test each item against. The `predicate` expression can refer to the `x` identifier
+ - `any(option in annotation.someCheckbox, option > 3)`
+ - `all(e in document.scores, e.value < 0.7)` (assuming `scores` is an object mapping labels to scores, e.g. `{"scores": {"positive": 0.5, "negative": 0.3}}`)
+ - when testing a predicate against an _object_ each entry has `.key` and `.value` properties giving the key and value of the current entry
+ - on a null, undefined or empty array/object, `any` will return _false_ (since there are no items that pass the test) and `all` will return _true_ (since there are no items that _fail_ the test)
+ - the predicate is optional - `any(arrayExpression)` resolves to `true` if any item in the array has a value that JavaScript considers to be "truthy", i.e. anything other than the number 0, the empty string, null or undefined. So `any(annotation.myCheckbox)` is a convenient way to check whether _at least one_ option has been selected in a `checkbox` component.
+
+If the `if` expression for a particular component is _syntactically invalid_ (missing operands, mis-matched brackets, etc.) then the condition will be ignored and the component will always be displayed as though it did not have an `if` expression at all. Conversely, if the expression is valid but an error occurs while _evaluating_ it, this will be treated the same as if the expression returned `false`, and the associated component will not be displayed. The behaviour is this way around as the most common reason for errors during evaluation is attempting to refer to annotation components that have not yet been filled in - if this is not appropriate in your use case you must account for the possibility within your expression. For example, suppose `confidence` is a `radio` or `selector` component with values ranging from 1 to 5, then another component that declares
+
+```
+"if": "annotation.confidence && annotation.confidence < 4"`
+```
+
+will hide this component if `confidence` is unset, displaying it only if `confidence` is set to a value less than 4, whereas
+
+```
+"if": "!annotation.confidence || annotation.confidence < 4"
+```
+
+will hide this component only if `confidence` is actually _set_ to a value of 4 or greater - it will _show_ this component if `confidence` is unset. Either approach may be correct depending on your project's requirements.
+
+To assist managers in authoring project configurations with `if` conditions, the "preview" mode on the project configuration page will display details of any errors that occur when parsing the expressions, or when evaluating them against the **Document input preview** data. You are encouraged to test your expressions thoroughly against a variety of inputs to ensure they behave as intended, before opening your project to annotators.
+
+
diff --git a/docs/versioned/2.3.0/manageradminguide/project_management.md b/docs/versioned/2.3.0/manageradminguide/project_management.md
new file mode 100644
index 00000000..f1fd0cfe
--- /dev/null
+++ b/docs/versioned/2.3.0/manageradminguide/project_management.md
@@ -0,0 +1,38 @@
+# Annotation Project Management
+
+## Project Listing
+Clicking on the `Projects` link in the top navigation bar takes you to a contains a list of existing
+projects. The project names are shown along with their summaries. Clicking on a project name will
+take you to the project management page.
+
+
+## Project Management Page
+
+The project management page contains all the functionalities to manage an annotation project. The page
+is composed of three main tabs:
+
+* [Configuration](project_config.md) - Configure project settings including what annotations are captured.
+* [Documents & Annotation](documents_annotations_management.md) - Manage documents and annotations. Upload documents, see contents of a document's annotations and import/export documents.
+* [Annotators](annotators_management.md) - Manage the recruitment of annotators.
+
+::: warning
+
+Annotators can only be recruited to an annotation project after it has been configured and documents
+are uploaded to the project.
+
+:::
+
+
+## Project status icons
+In the **Project listing** and **Project management page**, icon badges are used to provide a quick overview of the project's status:
+
+*
1 - Number of completed annotations in the project.
+*
1 - Number of rejected annotations in the project.
+*
1 - Number of timed out annotations in the project.
+*
1 - Number of aborted annotations in the project.
+*
1 - Number of pending annotations in the project.
+*
2/60 - Number of occupied annotation tasks over number of total tasks in the project.
+*
20/5/10 - Number of documents, training documents and test documents in the project.
+*
1 - Number of annotators recruited in the project. Annotators are removed from the project when they have completed all annotation tasks in their quota.
+
+
diff --git a/package.json b/package.json
index 940daf3b..ff85aa5b 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "gate-teamware",
- "version": "2.2.0",
+ "version": "2.3.0",
"description": "A service for collaborative document annotation.",
"main": "index.js",
"scripts": {