diff --git a/docs/_data/menu.yml b/docs/_data/menu.yml
index 570a836bc0dd386361c9b796b7f1da5cb85f12e7..8ec421f3d7addaa39cbc0d1f84b5998af49fecfd 100644
--- a/docs/_data/menu.yml
+++ b/docs/_data/menu.yml
@@ -6,5 +6,7 @@ Options: /options.html
 Operands: /operands.html
 Commands: /commands.html
 Help Text: /help.html
+Localization: /localization.html
+Validation: /validation.html
 Example: /example.html
 Development: /development.html
diff --git a/docs/help.md b/docs/help.md
index 8322585f61a7201458bf42ce2fb40f223fad18e3..e1ff564ad6d145193abe0a322c3a7ba63c5039e6 100644
--- a/docs/help.md
+++ b/docs/help.md
@@ -19,38 +19,13 @@ The method `HelpInterface::render(GetOpt, array)` receives the `GetOpt` object f
 
 ### Localization
 
-By default, `GetOpt` displays standard help text in English. This can be customized in several ways.
-
-#### Switching to an existing language
-
-`GetOpt` comes bundled with help texts translated in 
-[several languages](https://github.com/getopt-php/getopt-php/tree/master/resources/localization).
-These are located under `vendor/ulrichsg/getopt-php/resources/localization/<lang>.php`.
-
-You can switch to one of these languages by calling `GetOpt\GetOpt::setHelpLang($language)`  
-
-```php
-<?php
-$getopt = new \GetOpt\GetOpt();
-$getopt->setHelpLang('de');                         
-```
-
-Translations for additional languages are welcome; if you would like to contribute, please 
-[submit a pull request](https://github.com/getopt-php/getopt-php/compare).
-
-#### Switching to a custom language
-
-It is also possible to use a custom language file by specifying its path; the script must return an array in the
-same format as the bundled language files. 
-
-```php
-<?php
-$getopt = new \GetOpt\GetOpt();
-$getopt->setHelpLang(__DIR__ . '/path/to/cn.php');
-```
+By default, `GetOpt` displays standard help text in English. Have a look at [Localization](localization.md) for more
+information about localization.
 
 #### Override the localization
 
+> **Deprecated** note that this method is deprecated for messages. Please use localization files instead.
+
 The `GetOpt\Help` class can be used to define the standard help text, with the `setTexts(array $texts)` method. 
 The provided array overwrites the existing localization:
 
@@ -108,6 +83,19 @@ $getopt->getHelp()
     ->setCommandsTemplate('path/to/my/commandsTemplate.php');
 ```
 
+### Tables
+
+Options, operands and commands are shown in a table with two columns. The columns are divided with two spaces and the
+description (second column) will automatically break after the last space that fits into the terminal. The number of
+columns that fit into the terminal is determined in the following sequence:
+
+1. a constant `COLUMNS`,
+2. an environment variable `COLUMNS`,
+3. the result from `tput cols` command
+4. value `90`
+
+This is limited by the setting `GetOpt\Help::MAX_WIDTH` (default: `120`).
+
 ### The Parts of Help
 
 In the following sections, you will find a complete description of the three parts the Help is split into, and what
@@ -124,21 +112,25 @@ command has to be given, where the options should be entered and the name and or
    `Usage: path/to/app make:config [options] <file>`
  - No commands, options and operands defined and strict operands:  
    `Usage: path/to/app`
+   
+#### Operands
 
-#### Options
+When one of the operands has a description the table of operands will be shown. The table can be suppressed by passing
+the option `HIDE_OPERANDS` with a falsy value (e. g. `false`).
 
-Options are shown in a table with the options (including argument) in the left column, and
-the description of each option in the right column.
+The table shows the operands name in the left column and the description on the right column what might look like this:
 
-Long descriptions automatically break after the last space that fits into the
-terminal's width. The number of columns is determined in the following sequence:
+```
+Operands:
+  <source>    The source that should be copied.
+  [<target>]  The target where source should be copied to. If target is ommitted the file 
+              will be copied to the current working directory.
+```
 
-1. a constant `COLUMNS`,
-2. an environment variable `COLUMNS`,
-3. the result from `tput cols` command
-4. value `90`
+#### Options
 
-This is limited by the setting `GetOpt\Help::MAX_WIDTH` (default: `120`).
+Options are shown in a table with the options (including argument) in the left column, and
+the description of each option in the right column.
 
 In the end it might look something like this:
 
diff --git a/docs/localization.md b/docs/localization.md
new file mode 100644
index 0000000000000000000000000000000000000000..4779e10ac1d2864c2ffcc2d20d7254cfcb4ab173
--- /dev/null
+++ b/docs/localization.md
@@ -0,0 +1,34 @@
+---
+layout: default
+title: Localization
+permalink: /localization.html
+---
+# {{ page.title }}
+
+Texts shown to a user are localized using the `Translator` which is a simple key-value translation class using php files
+for a translation map. These files can be found in [resources/localization](https://github.com/getopt-php/getopt-php/tree/master/resources/localization)
+and contributions are highly appreciated.
+
+By default the Translator is using the English translations. You can change to a predefined translation by calling
+`GetOpt::setLang($language)` where `$language` is a two-letter iso code of the language.
+
+```php
+<?php
+\GetOpt\GetOpt::setLang('de');
+```
+
+## Custom language files
+
+You can also use custom language files by passing a path to your language file. The script must return an array in the
+same format as the bundled language files.
+
+```php
+<?php
+\GetOpt\GetOpt::setLang(__DIR__ . '/path/to/cn.php');
+```
+
+Translations missing in a language file are translated with the English translation table.
+
+## Errors messages
+
+The messages of exceptions thrown because of user input are also translated using this translator. 
diff --git a/docs/operands.md b/docs/operands.md
index ddc0ec2949e8a92028326937d6f6d24f8d8ffb86..4b9704d22064816d10186cddb8dc4318d6dc1ebd 100644
--- a/docs/operands.md
+++ b/docs/operands.md
@@ -66,21 +66,15 @@ you are planning such things you should consider using `->getOperand('operand1')
 
 ### Validation
 
-Again: it is the same functionality as for validating options. It follows a small example. See 
-[Options Validation](options.html#validation) for more details.
+You can validate the argument of an operand using the `->setValidation($callable)`. To learn more about validation
+please refer to the section [Validation](validation.md) of this handbook.
 
-```php
-<?php
-$getopt = new \GetOpt\GetOpt();
-$getopt->addOperands([
-    \GetOpt\Operand::create('file', \GetOpt\Operand::REQUIRED)
-        ->setValidation('is_readable'),
-    \GetOpt\Operand::create('destination', \GetOpt\Operand::MULTIPLE)
-        ->setValidation(function ($value) {
-            return file_exists($value) && is_dir($value) && is_writeable($value); 
-        }),
-]);
-```
+### Description
+
+Since version 3.2 you can also set the description of operands with `->setDescription($description)`. When one of the 
+operands has a description the table of operands will be shown in the help.
+
+> **Note:** all operands will be shown even if they don't have a description to show the order of operands.
 
 ## Working With Operands
 
diff --git a/docs/options.md b/docs/options.md
index b4df566ec1f4baf4ad29bb3aa1b2a47bcaedf8de..179d03a1994913785abfdc2bf62d59bc728e4c81 100644
--- a/docs/options.md
+++ b/docs/options.md
@@ -275,54 +275,8 @@ var_dump($getopt->getOption('domain')); // ['example.com', 'example.org']
 
 ### Validation
 
-This library does not come with a bunch of validators that you can use and extend. Instead you provide a callable or
-closure that has to return a truthy value if the value is valid (further called the validator).
-
-The validator gets the value as first and only parameter. For a lot of php standard functions this is enough (eg. 
-`is_numeric`). The value will always be a string or null. Here comes an example that shows how to check that it has
-a valid json value:
-
-```php
-<?php
-$getopt = new \GetOpt\GetOpt([
-    \GetOpt\Option::create(null, 'data', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
-        ->setValidation(function ($value) {
-            return $value === 'null' || json_decode($value) !== null;
-        })
-]);
-```
-
-```console
-$ php program.php --data null
-$ php program.php --data []
-$ php program.php --data '{"a":"alpha"}'
-$ php program.php --data invalid
-```
-
-#### Advanced Validation
-
-The validator is also executed if the option mode is `NO_ARGUMENT`. This way we can also check other circumstances
-inside our application as well as the current status of options.
-
-A use case for this could be to define exclusive options (which is also the reason because it was asked in a feature
-request). Let's say our program has the options `alpha` and `omega` but when you define `alpha` the `omega` option is
-forbidden and vise versa:
-
-```php
-<?php
-$getopt = new \GetOpt\GetOpt();
-
-$getopt->addOptions([
-    \GetOpt\Option::create(null, 'alpha', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
-        ->setValidation(function () use ($getopt) {
-            return !$getopt->getOption('omega');
-        }),
-    \GetOpt\Option::create(null, 'omega', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
-        ->setValidation(function () use ($getopt) {
-            return !$getopt->getOption('alpha');
-        }),
-]);
-```
+You can validate the argument of an option using the `->setValidation($callable)`. To learn more about validation
+please refer to the section [Validation](validation.md) of this handbook.
 
 ## Allow Custom Options
 
diff --git a/docs/validation.md b/docs/validation.md
new file mode 100644
index 0000000000000000000000000000000000000000..576c0db1bc218f1d357cfa68f538b48454b41a0c
--- /dev/null
+++ b/docs/validation.md
@@ -0,0 +1,98 @@
+---
+layout: default
+title: Validation
+permalink: /validation.html
+---
+# {{ page.title }}
+
+This library does not come with a bunch of validators that you can use and extend. Instead you provide a callable or
+closure that has to return a truthy value if the value is valid (further called the validator).
+
+The validator gets the value as first and only parameter. For a lot of php standard functions this is enough (eg. 
+`is_numeric`). The value will always be a string or null. Here comes an example that shows how to check that it has
+a valid json value:
+
+```php
+<?php
+$getopt = new \GetOpt\GetOpt([
+    \GetOpt\Option::create(null, 'data', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
+        ->setValidation(function ($value) {
+            return $value === 'null' || json_decode($value) !== null;
+        })
+]);
+```
+
+```console
+$ php program.php --data null
+$ php program.php --data []
+$ php program.php --data '{"a":"alpha"}'
+$ php program.php --data invalid
+```
+
+## Validation Message
+
+Since version 3.2 GetOpt supports custom validation messages. `Option::setValidation()` and `Operand::setValidation` now
+allow to pass a second parameter the validation message. This message gets passed to `sprintf($message, $desc, $value)`
+where `$desc` is the description of the validated object (e. g. `Option 'data'`) and `$value` the original value that
+was not valid.
+
+The validation message can also be a callable that is then called with `$object` and `$value`. To get the description
+you can then use `$object->describe()`.
+
+```php
+<?php
+\GetOpt\Option::create('n', 'count', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
+    ->setValidation(function ($value) {
+        return is_numeric($value) && (int)$value == round((double)$value);
+    }, 'Count has to be an integer');
+\GetOpt\Operand::create('pretend')
+    ->setValidation('is_resource', function(\GetOpt\Operand $operand, $value) {
+        return $value . ' is invalid for ' . $operand->getName();
+    });
+```
+
+## Reusable Validator
+
+You can also create functions or classes that can be used as validator and reuse them.
+
+```php
+<?php
+class InArrayValidator {
+    protected $array;
+    public function __construct(array $array) {
+        $this->array = $array;
+    }
+    public function __invoke($value) {
+        return in_array($value, $this->array);
+    }
+}
+
+\GetOpt\Operand::create('type')->setValidation(
+    new InArrayValidator(['file', 'f', 'directory', 'd', 'link', 'l'])
+);
+```
+
+## Advanced Validation
+
+The validator is also executed if the option mode is `NO_ARGUMENT`. This way we can also check other circumstances
+inside our application as well as the current status of options.
+
+A use case for this could be to define exclusive options (which is also the reason because it was asked in a feature
+request). Let's say our program has the options `alpha` and `omega` but when you define `alpha` the `omega` option is
+forbidden and vise versa:
+
+```php
+<?php
+$getopt = new \GetOpt\GetOpt();
+
+$getopt->addOptions([
+    \GetOpt\Option::create(null, 'alpha', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
+        ->setValidation(function () use ($getopt) {
+            return !$getopt->getOption('omega');
+        }),
+    \GetOpt\Option::create(null, 'omega', \GetOpt\GetOpt::REQUIRED_ARGUMENT)
+        ->setValidation(function () use ($getopt) {
+            return !$getopt->getOption('alpha');
+        }),
+]);
+```
diff --git a/resources/localization/de.php b/resources/localization/de.php
index e10f5a67f11bd245850ed354909a6806db5fbf4c..e36965d19b8af4ab54e28b1172b2a1e86f5e8311 100644
--- a/resources/localization/de.php
+++ b/resources/localization/de.php
@@ -12,7 +12,7 @@ return [
     'no-more-operands' => 'Unerwarteter Operand %s',
     'operand-missing' => 'Operand %s muss angegeben werden',
     'option-argument-missing' => 'Option \'%s\' erwartet einen Wert',
-    'option-value-invalid' => 'Der Wert für Option \'%s\' ist ungültig',
+    'value-invalid' => 'Der Wert für %s ist ungültig',
     'option' => 'Option',
     'operand' => 'Operand',
     'argument' => 'Argument',
diff --git a/resources/localization/en.php b/resources/localization/en.php
index 89bc65e22ee37631fabd3cec3d4b6d9417de4870..e23a5fd1ba26d7154491d044547271dad718d8fe 100644
--- a/resources/localization/en.php
+++ b/resources/localization/en.php
@@ -12,5 +12,5 @@ return [
     'no-more-operands' => 'No more operands expected - got %s',
     'operand-missing' => 'Operand %s is required',
     'option-argument-missing' => 'Option \'%s\' must have a value',
-    'option-value-invalid' => 'Option \'%s\' has an invalid value',
+    'value-invalid' => '%s has an invalid value',
 ];
diff --git a/resources/localization/fr.php b/resources/localization/fr.php
index e696580774e58b6a1ae7f0ca9d60a18f9dad6946..2df830b06bd9002a31eec94222f2308da546a66d 100644
--- a/resources/localization/fr.php
+++ b/resources/localization/fr.php
@@ -12,7 +12,7 @@ return [
     'no-more-operands' => 'Argument inattendu - %s',
     'operand-missing' => 'L\'argument %s est obligatoire',
     'option-argument-missing' => 'L\'option \'%s\' doit avoir une valeur',
-    'option-value-invalid' => 'La valeur de l\'option \'%s\' est invalide',
+    'value-invalid' => 'La valeur de %s est invalide',
     'option' => 'option', // optimization: otherwise we would have to load the fallback
     'argument' => 'argument',
     'operand' => 'argument',
diff --git a/src/Argument.php b/src/Argument.php
index 84af1aa683656071f5939bc25003240490b32b3b..607b97f0204a322d6f8e61dba34ac791e30f9d5b 100644
--- a/src/Argument.php
+++ b/src/Argument.php
@@ -104,7 +104,7 @@ class Argument implements Describable
         }
 
         return ucfirst(sprintf(
-            $this->validationMessage ?: '%s has an invalid value',
+            $this->validationMessage ?: GetOpt::translate('value-invalid'),
             $this->describe(),
             $value
         ));