diff --git a/ajax/processingStatus/create.php b/ajax/processingStatus/create.php
index 8438caa42aab0f0181562e993544160325df038d..0ecb3b3a9cd39f7e33cdee450d88e58fdbcff722 100644
--- a/ajax/processingStatus/create.php
+++ b/ajax/processingStatus/create.php
@@ -1,9 +1,5 @@
 <?php
 
-/**
- * This file contains package_quiqqer_invoice_ajax_processingStatus_create
- */
-
 use QUI\ERP\Accounting\Invoice\ProcessingStatus\Factory;
 
 /**
@@ -11,13 +7,20 @@
  */
 QUI::$Ajax->registerFunction(
     'package_quiqqer_invoice_ajax_processingStatus_create',
-    function ($id, $color, $title) {
+    function ($id, $color, $title, $options) {
+        if (!empty($options)) {
+            $options = \json_decode($options, true);
+        } else {
+            $options = [];
+        }
+
         Factory::getInstance()->createProcessingStatus(
             $id,
             $color,
-            \json_decode($title, true)
+            \json_decode($title, true),
+            $options
         );
     },
-    array('id', 'color', 'title'),
+    ['id', 'color', 'title', 'options'],
     'Permission::checkAdminUser'
 );
diff --git a/ajax/processingStatus/update.php b/ajax/processingStatus/update.php
index 9542cb51b27679a51be0c4fe6dea647d3a2dc375..83ae11bbbdf79a6bb33c25131fceadb8458fde71 100644
--- a/ajax/processingStatus/update.php
+++ b/ajax/processingStatus/update.php
@@ -1,9 +1,5 @@
 <?php
 
-/**
- * This file contains package_quiqqer_invoice_ajax_processingStatus_update
- */
-
 use QUI\ERP\Accounting\Invoice\ProcessingStatus\Handler;
 
 /**
@@ -11,13 +7,20 @@
  */
 QUI::$Ajax->registerFunction(
     'package_quiqqer_invoice_ajax_processingStatus_update',
-    function ($id, $color, $title) {
+    function ($id, $color, $title, $options) {
+        if (!empty($options)) {
+            $options = \json_decode($options, true);
+        } else {
+            $options = [];
+        }
+
         Handler::getInstance()->updateProcessingStatus(
             $id,
             $color,
-            \json_decode($title, true)
+            \json_decode($title, true),
+            $options
         );
     },
-    ['id', 'color', 'title'],
+    ['id', 'color', 'title', 'options'],
     'Permission::checkAdminUser'
 );
diff --git a/bin/backend/classes/Invoices.js b/bin/backend/classes/Invoices.js
index 8804a9790b28d1c227cad8e80c47d4c85648a9cf..22eddc143838bb72b7ffd32aca6e22ebf8f5f0c7 100644
--- a/bin/backend/classes/Invoices.js
+++ b/bin/backend/classes/Invoices.js
@@ -600,6 +600,25 @@ define('package/quiqqer/invoice/bin/backend/classes/Invoices', [
                     showError: false
                 });
             });
+        },
+
+        /**
+         * Set invoice customer files
+         *
+         * @param {Number} invoiceId
+         * @param {String} customerFiles (JSON)
+         * @return {Promise}
+         */
+        setCustomerFiles: function (invoiceId, customerFiles) {
+            return new Promise(function (resolve, reject) {
+                QUIAjax.post('package_quiqqer_invoice_ajax_invoices_setCustomerFiles', resolve, {
+                    'package'    : 'quiqqer/invoice',
+                    invoiceId    : invoiceId,
+                    customerFiles: customerFiles,
+                    onError      : reject,
+                    showError    : false
+                });
+            });
         }
     });
 });
diff --git a/bin/backend/classes/ProcessingStatus.js b/bin/backend/classes/ProcessingStatus.js
index ff49fcaf1e689b6c45dab355be9cf33c1c210d89..a05e27c2dca9954ce7a15322c4c40d1814bcf1c0 100644
--- a/bin/backend/classes/ProcessingStatus.js
+++ b/bin/backend/classes/ProcessingStatus.js
@@ -54,9 +54,12 @@ define('package/quiqqer/invoice/bin/backend/classes/ProcessingStatus', [
          * @param {String|Number} id - Processing Status ID
          * @param {String} color
          * @param {Object} title - {de: '', en: ''}
+         * @param {Object} [Options]
          * @return {Promise}
          */
-        createProcessingStatus: function (id, color, title) {
+        createProcessingStatus: function (id, color, title, Options) {
+            Options = Options || {};
+
             return new Promise(function (resolve, reject) {
                 QUIAjax.post('package_quiqqer_invoice_ajax_processingStatus_create', function (result) {
                     require([
@@ -71,6 +74,7 @@ define('package/quiqqer/invoice/bin/backend/classes/ProcessingStatus', [
                     id       : id,
                     color    : color,
                     title    : JSON.encode(title),
+                    options  : JSON.encode(Options),
                     onError  : reject
                 });
             });
@@ -122,15 +126,19 @@ define('package/quiqqer/invoice/bin/backend/classes/ProcessingStatus', [
          * @param {String|Number} id - Processing Status ID
          * @param {String} color
          * @param {Object} title - {de: '', en: ''}
+         * @param {Object} [Options]
          * @return {Promise}
          */
-        updateProcessingStatus: function (id, color, title) {
+        updateProcessingStatus: function (id, color, title, Options) {
+            Options = Options || {};
+
             return new Promise(function (resolve, reject) {
                 QUIAjax.post('package_quiqqer_invoice_ajax_processingStatus_update', resolve, {
                     'package': 'quiqqer/invoice',
                     id       : id,
                     color    : color,
                     title    : JSON.encode(title),
+                    options  : JSON.encode(Options),
                     onError  : reject
                 });
             });
diff --git a/bin/backend/controls/settings/ProcessingStatus.html b/bin/backend/controls/settings/ProcessingStatus.html
index 6c9a2963eecbdf2796a47577727e785f37b84f2d..99805571fdce2f1eee2bbf5954039f6c58852151 100644
--- a/bin/backend/controls/settings/ProcessingStatus.html
+++ b/bin/backend/controls/settings/ProcessingStatus.html
@@ -4,8 +4,8 @@
         <tr>
             <td>
                 <label class="field-container">
-                    <span class="field-container-item" title="">
-                        Title
+                    <span class="field-container-item" title="{{labelTitle}}">
+                        {{labelTitle}}
                     </span>
                     <input name="title" data-qui="controls/lang/InputMultiLang"/>
                 </label>
@@ -14,8 +14,8 @@
         <tr>
             <td>
                 <label class="field-container">
-                    <span class="field-container-item" title="">
-                        Status Nr.
+                    <span class="field-container-item" title="{{labelId}}">
+                        {{labelId}}
                     </span>
                     <input type="number" name="id" min="0" class="field-container-field"/>
                 </label>
@@ -24,13 +24,28 @@
         <tr>
             <td>
                 <label class="field-container">
-                    <span class="field-container-item" title="">
-                        Farbe
+                    <span class="field-container-item" title="{{labelColor}}">
+                        {{labelColor}}
                     </span>
                     <input type="color" name="color" class="field-container-field"/>
                 </label>
             </td>
         </tr>
+        <tr>
+            <td>
+                <label class="field-container">
+                    <span class="field-container-item" title="">
+                        {{labelPreventInvoicePosting}}
+                    </span>
+                    <span class="field-container-field">
+                        <input type="checkbox" name="preventInvoicePosting"/>
+                    </span>
+                </label>
+                <div class="field-container-item-desc">
+                    {{{descPreventInvoicePosting}}}
+                </div>
+            </td>
+        </tr>
         </tbody>
     </table>
 </form>
\ No newline at end of file
diff --git a/bin/backend/controls/settings/ProcessingStatus.js b/bin/backend/controls/settings/ProcessingStatus.js
index cac29b3720a89de99dbc21336eebe2565451da4e..00faee2aeaa8a7901dbbb476b822726be59c9da5 100644
--- a/bin/backend/controls/settings/ProcessingStatus.js
+++ b/bin/backend/controls/settings/ProcessingStatus.js
@@ -56,13 +56,19 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
 
             ProcessingStatus.getList().then(function (result) {
                 for (var i = 0, len = result.data.length; i < len; i++) {
-                    result.data[i].colorNode = new Element('span', {
-                        html   : result.data[i].color,
+                    const Status = result.data[i];
+
+                    Status.colorNode = new Element('span', {
+                        html   : Status.color,
                         'class': 'quiqqer-invoice-processing-status-color',
                         styles : {
-                            backgroundColor: result.data[i].color
+                            backgroundColor: Status.color
                         }
                     });
+
+                    Status.preventInvoicePosting = new Element('span', {
+                        'class': Status.options.preventInvoicePosting ? 'fa fa-check' : 'fa fa-close'
+                    });
                 }
 
                 self.$Grid.setData(result);
@@ -141,6 +147,11 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
                     dataIndex: 'title',
                     dataType : 'integer',
                     width    : 200
+                }, {
+                    header   : QUILocale.get(lg, 'processingStatus.grid.preventInvoicePosting'),
+                    dataIndex: 'preventInvoicePosting',
+                    dataType : 'node',
+                    width    : 120
                 }]
             });
 
@@ -189,7 +200,7 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
             new QUIConfirm({
                 icon     : 'fa fa-plus',
                 title    : QUILocale.get(lg, 'dialog.processingStatus.create.title'),
-                maxHeight: 400,
+                maxHeight: 500,
                 maxWidth : 600,
                 autoclose: false,
                 events   : {
@@ -199,7 +210,14 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
                         Win.Loader.show();
 
                         Content.addClass('quiqqer-invoice-processing-status-window');
-                        Content.set('html', Mustache.render(template));
+
+                        Content.set('html', Mustache.render(template, {
+                            labelTitle                : QUILocale.get(lg, 'processingStatus.tpl.labelTitle'),
+                            labelId                   : QUILocale.get(lg, 'processingStatus.tpl.labelId'),
+                            labelColor                : QUILocale.get(lg, 'processingStatus.tpl.labelColor'),
+                            labelPreventInvoicePosting: QUILocale.get(lg, 'processingStatus.tpl.labelPreventInvoicePosting'),
+                            descPreventInvoicePosting : QUILocale.get(lg, 'processingStatus.tpl.descPreventInvoicePosting')
+                        }));
 
                         var Form = Content.getElement('form');
 
@@ -229,10 +247,16 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
                             } catch (e) {
                             }
 
+                            // Options
+                            const Options = {
+                                preventInvoicePosting: data.preventInvoicePosting
+                            };
+
                             ProcessingStatus.createProcessingStatus(
                                 data.id,
                                 data.color,
-                                title
+                                title,
+                                Options
                             ).then(function () {
                                 return Win.close();
                             }).then(function () {
@@ -264,7 +288,7 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
             new QUIConfirm({
                 icon     : 'fa fa-edit',
                 title    : QUILocale.get(lg, 'dialog.processingStatus.edit.title'),
-                maxHeight: 400,
+                maxHeight: 500,
                 maxWidth : 600,
                 autoclose: false,
                 ok_button: {
@@ -278,7 +302,13 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
                         Win.Loader.show();
 
                         Content.addClass('quiqqer-invoice-processing-status-window');
-                        Content.set('html', Mustache.render(template));
+                        Content.set('html', Mustache.render(template, {
+                            labelTitle                : QUILocale.get(lg, 'processingStatus.tpl.labelTitle'),
+                            labelId                   : QUILocale.get(lg, 'processingStatus.tpl.labelId'),
+                            labelColor                : QUILocale.get(lg, 'processingStatus.tpl.labelColor'),
+                            labelPreventInvoicePosting: QUILocale.get(lg, 'processingStatus.tpl.labelPreventInvoicePosting'),
+                            descPreventInvoicePosting : QUILocale.get(lg, 'processingStatus.tpl.descPreventInvoicePosting')
+                        }));
 
                         var Form = Content.getElement('form');
 
@@ -287,6 +317,9 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
                             Form.elements.color.value = details.color;
                             Form.elements.title.value = JSON.encode(details.title);
 
+                            // Options
+                            Form.elements.preventInvoicePosting.checked = details.options.preventInvoicePosting;
+
                             return QUI.parse(Content);
                         }).then(function () {
                             Win.Loader.hide();
@@ -310,11 +343,16 @@ define('package/quiqqer/invoice/bin/backend/controls/settings/ProcessingStatus',
                             } catch (e) {
                             }
 
+                            // Options
+                            const Options = {
+                                preventInvoicePosting: data.preventInvoicePosting
+                            };
 
                             ProcessingStatus.updateProcessingStatus(
                                 data.id,
                                 data.color,
-                                title
+                                title,
+                                Options
                             ).then(function () {
                                 return Win.close();
                             }).then(function () {
diff --git a/locale.xml b/locale.xml
index f0aa9e9453d06f437b314a7af866c6b9e13ecb3a..772604821347de0329845ff550f5f04e93ce0b70 100644
--- a/locale.xml
+++ b/locale.xml
@@ -1225,6 +1225,10 @@
             <de><![CDATA[Farbe]]></de>
             <en><![CDATA[Color]]></en>
         </locale>
+        <locale name="processingStatus.grid.preventInvoicePosting">
+            <de><![CDATA[verhindert Buchung]]></de>
+            <en><![CDATA[prevents posting]]></en>
+        </locale>
 
         <locale name="dialog.processingStatus.delete.title">
             <de><![CDATA[Rechnungsstatus entfernen]]></de>
@@ -1248,6 +1252,26 @@
             The invoice status is not recoverable.
             ]]></en>
         </locale>
+        <locale name="processingStatus.tpl.labelTitle">
+            <de><![CDATA[Titel]]></de>
+            <en><![CDATA[Title]]></en>
+        </locale>
+        <locale name="processingStatus.tpl.labelId">
+            <de><![CDATA[Status-Nr.]]></de>
+            <en><![CDATA[Status no.]]></en>
+        </locale>
+        <locale name="processingStatus.tpl.labelColor">
+            <de><![CDATA[Farbe]]></de>
+            <en><![CDATA[Color]]></en>
+        </locale>
+        <locale name="processingStatus.tpl.labelPreventInvoicePosting">
+            <de><![CDATA[Buchen von Rechnungsentwürfen verhindern]]></de>
+            <en><![CDATA[Prevents posting of invoice drafts]]></en>
+        </locale>
+        <locale name="processingStatus.tpl.descPreventInvoicePosting">
+            <de><![CDATA[Rechnungsentwürfe mit diesem Status können <b>nicht gebucht</b> werden, solange sie diesen Status innehaben. Sinnvoll für Status, die eine manuelle Überprüfung einer Rechnung erfordern.]]></de>
+            <en><![CDATA[Draft invoices with this status can <b>not be posted</b> as long as they have this status. Useful for statuses that require manual review of an invoice.]]></en>
+        </locale>
 
         <locale name="dialog.processingStatus.create.title">
             <de><![CDATA[Rechnungsstatus hinzufügen]]></de>
@@ -1846,6 +1870,11 @@
             <de><![CDATA[Die Rechnung besitzt keine Bezahlmethode.]]></de>
             <en><![CDATA[The invoice has no payment method.]]></en>
         </locale>
+        <locale name="exception.invoice.verification.status_prevents_posting">
+            <de><![CDATA[Der aktuelle Status der Rechnung verhindert eine Buchung. Bitte prüfe die Rechnung.]]></de>
+            <en><![CDATA[The current status of the invoice prevents posting. Please check the invoice.]]></en>
+        </locale>
+
         <locale name="exception.invoice.not.found" html="true">
             <de><![CDATA[Die gewünschte Rechnung wurde leider nicht gefunden.]]></de>
             <en><![CDATA[The desired invoice was unfortunately not found.]]></en>
diff --git a/src/QUI/ERP/Accounting/Invoice/EventHandler.php b/src/QUI/ERP/Accounting/Invoice/EventHandler.php
index 9db5ea6a84ec217663b0bcc62454e9c5255f2e46..561ccb951eae55a90247f9035f5cb3b77b235f64 100644
--- a/src/QUI/ERP/Accounting/Invoice/EventHandler.php
+++ b/src/QUI/ERP/Accounting/Invoice/EventHandler.php
@@ -295,6 +295,10 @@ public static function onQuiqqerErpOutputSendMailBefore(
         $customerFiles = $Invoice->getCustomerFiles();
 
         foreach ($customerFiles as $entry) {
+            if (empty($entry['options']['attachToEmail'])) {
+                continue;
+            }
+
             $file = QUI\ERP\Customer\CustomerFiles::getFileByHash($Invoice->getCustomer()->getId(), $entry['hash']);
 
             if ($file) {
diff --git a/src/QUI/ERP/Accounting/Invoice/Invoice.php b/src/QUI/ERP/Accounting/Invoice/Invoice.php
index e421e949273d2cd7b4795655c89a399aaa0c1bac..cf646a29946749819d05bb980b132a0fc06a35c5 100644
--- a/src/QUI/ERP/Accounting/Invoice/Invoice.php
+++ b/src/QUI/ERP/Accounting/Invoice/Invoice.php
@@ -7,6 +7,7 @@
 namespace QUI\ERP\Accounting\Invoice;
 
 use QUI;
+use QUI\ERP\Customer\CustomerFiles;
 use QUI\Permissions\Permission;
 use QUI\ERP\Money\Price;
 use QUI\ERP\Accounting\ArticleListUnique;
@@ -45,10 +46,10 @@ class Invoice extends QUI\QDOM
     //    const PAYMENT_STATUS_STORNO = 3; // Alias for cancel
     //    const PAYMENT_STATUS_CREATE_CREDIT = 5;
 
-    const DUNNING_LEVEL_OPEN = 0; // No Dunning -> Keine Mahnung
-    const DUNNING_LEVEL_REMIND = 1; // Payment reminding -> Zahlungserinnerung
-    const DUNNING_LEVEL_DUNNING = 2; // Dunning -> Erste Mahnung
-    const DUNNING_LEVEL_DUNNING2 = 3; // Second dunning -> Zweite Mahnung
+    const DUNNING_LEVEL_OPEN       = 0; // No Dunning -> Keine Mahnung
+    const DUNNING_LEVEL_REMIND     = 1; // Payment reminding -> Zahlungserinnerung
+    const DUNNING_LEVEL_DUNNING    = 2; // Dunning -> Erste Mahnung
+    const DUNNING_LEVEL_DUNNING2   = 3; // Second dunning -> Zweite Mahnung
     const DUNNING_LEVEL_COLLECTION = 4; // Collection -> Inkasso
 
     /**
@@ -1504,4 +1505,110 @@ public function getData(string $key)
     }
 
     //endregion
+
+    // region Attached customer files
+
+    /**
+     * Add a customer file to this invoice
+     *
+     * @param string $fileHash - SHA256 hash of the file basename
+     * @param array $options (optional) - File options; see $defaultOptions in code for what's possible
+     *
+     * @throws Exception
+     * @throws QUI\Exception
+     */
+    public function addCustomerFile(string $fileHash, array $options = [])
+    {
+        $Customer = $this->getCustomer();
+
+        if (empty($Customer)) {
+            throw new Exception(
+                QUI::getLocale()->get('quiqqer/invoice', 'exception.Invoice.addCustomerFile.no_customer')
+            );
+        }
+
+        $file = CustomerFiles::getFileByHash($Customer->getId(), $fileHash);
+
+        if (empty($file)) {
+            throw new Exception(
+                QUI::getLocale()->get('quiqqer/invoice', 'exception.Invoice.addCustomerFile.file_not_found')
+            );
+        }
+
+        $defaultOptions = [
+            'attachToEmail' => false
+        ];
+
+        // set default options
+        foreach ($defaultOptions as $k => $v) {
+            if (!isset($options[$k])) {
+                $options[$k] = $v;
+            }
+        }
+
+        // cleanup
+        foreach ($options as $k => $v) {
+            if (!isset($defaultOptions[$k])) {
+                unset($options[$k]);
+            }
+        }
+
+        $fileEntry = [
+            'hash'    => $fileHash,
+            'options' => $options
+        ];
+
+        $customerFiles   = $this->getCustomerFiles();
+        $customerFiles[] = $fileEntry;
+
+        $this->data['customer_files'] = $customerFiles;
+
+        QUI::getDataBase()->update(
+            Handler::getInstance()->invoiceTable(),
+            [
+                'data' => \json_encode($this->data)
+            ],
+            [
+                'id' => $this->getCleanId()
+            ]
+        );
+    }
+
+    /**
+     * Clear customer files
+     *
+     * @return void
+     */
+    public function clearCustomerFiles(): void
+    {
+        $this->data['customer_files'] = [];
+
+        QUI::getDataBase()->update(
+            Handler::getInstance()->invoiceTable(),
+            [
+                'data' => \json_encode($this->data)
+            ],
+            [
+                'id' => $this->getCleanId()
+            ]
+        );
+    }
+
+    /**
+     * Get customer files that are attached to this invoice.
+     *
+     * @return array - Contains file hash and file options
+     */
+    public function getCustomerFiles(): array
+    {
+        $customerFiles = $this->getData('customer_files');
+
+        if (empty($customerFiles)) {
+            return [];
+        }
+
+        return $customerFiles;
+    }
+
+    // endregion
 }
diff --git a/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Factory.php b/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Factory.php
index 8ff5639629389426d90f1714c557147b1ff96ded..bbe6e0e2ed6472a70d74f63444434c175c27c726 100644
--- a/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Factory.php
+++ b/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Factory.php
@@ -22,10 +22,11 @@ class Factory extends QUI\Utils\Singleton
      * @param string|integer $id - processing ID
      * @param string $color - color of the status
      * @param array $title - title
+     * @param array $options (optional) - Status options
      * @throws Exception|QUI\Exception
      * @todo permissions
      */
-    public function createProcessingStatus($id, $color, array $title)
+    public function createProcessingStatus($id, $color, array $title, array $options = [])
     {
         $list = Handler::getInstance()->getList();
         $id   = (int)$id;
@@ -42,7 +43,11 @@ public function createProcessingStatus($id, $color, array $title)
         $Package = QUI::getPackage('quiqqer/invoice');
         $Config  = $Package->getConfig();
 
-        $Config->setValue('processing_status', $id, $color);
+        $Config->setValue('processing_status', $id, \json_encode([
+            'color'   => $color,
+            'options' => $options
+        ]));
+
         $Config->save();
 
         // translations
diff --git a/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Handler.php b/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Handler.php
index 9e2c0c235c3a77af21b40de4b7c4065dd1f9b10e..8e5fe44e54c7c1427c783a10388ac449b948a8ab 100644
--- a/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Handler.php
+++ b/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Handler.php
@@ -18,6 +18,11 @@
  */
 class Handler extends QUI\Utils\Singleton
 {
+    /**
+     * Status options
+     */
+    const STATUS_OPTION_PREVENT_INVOICE_POSTING = 'preventInvoicePosting';
+
     /**
      * @var array
      */
@@ -64,9 +69,9 @@ public function getProcessingStatusList()
         $list   = $this->getList();
         $result = [];
 
-        foreach ($list as $entry => $color) {
+        foreach ($list as $statusId => $v) {
             try {
-                $result[] = $this->getProcessingStatus($entry);
+                $result[] = $this->getProcessingStatus($statusId);
             } catch (Exception $Exception) {
             }
         }
@@ -123,13 +128,14 @@ public function deleteProcessingStatus($id)
      * @param int|string $id
      * @param int|string $color
      * @param array $title
+     * @param array $options (optional)
      *
      * @throws Exception
      * @throws QUI\Exception
      *
      * @todo permissions
      */
-    public function updateProcessingStatus($id, $color, array $title)
+    public function updateProcessingStatus($id, $color, array $title, array $options = [])
     {
         $Status = $this->getProcessingStatus($id);
 
@@ -158,12 +164,15 @@ public function updateProcessingStatus($id, $color, array $title)
 
         QUI\Translator::publish('quiqqer/invoice');
 
-
         // update config
         $Package = QUI::getPackage('quiqqer/invoice');
         $Config  = $Package->getConfig();
 
-        $Config->setValue('processing_status', $Status->getId(), $color);
+        $Config->setValue('processing_status', $Status->getId(), \json_encode([
+            'color'   => $color,
+            'options' => $options
+        ]));
+
         $Config->save();
     }
 }
diff --git a/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Status.php b/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Status.php
index 400b68f64d7d08890d501eedef28e9e57c7a81fc..c27587133f479b2583fb53325b897337883cd1b3 100644
--- a/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Status.php
+++ b/src/QUI/ERP/Accounting/Invoice/ProcessingStatus/Status.php
@@ -25,6 +25,15 @@ class Status
      */
     protected $color;
 
+    /**
+     * Default options.
+     *
+     * @var array
+     */
+    protected array $options = [
+        Handler::STATUS_OPTION_PREVENT_INVOICE_POSTING => false
+    ];
+
     /**
      * Status constructor.
      *
@@ -42,8 +51,21 @@ public function __construct($id)
             ]);
         }
 
-        $this->id    = (int)$id;
-        $this->color = $list[$id];
+        $this->id = (int)$id;
+        $data     = $list[$id];
+
+        // Fallback for old data structure
+        if (\mb_strpos($data, '#') === 0) {
+            $this->color = $data;
+        } else {
+            $data = \json_decode($data, true);
+
+            $this->color = $data['color'];
+
+            foreach ($data['options'] as $k => $v) {
+                $this->setOption($k, $v);
+            }
+        }
     }
 
     //region Getter
@@ -85,6 +107,48 @@ public function getColor()
 
     //endregion
 
+    // region Options
+
+    /**
+     * Set status option
+     *
+     * @param string $key - See $this->options for available options
+     * @param $value
+     */
+    public function setOption(string $key, $value)
+    {
+        if (\array_key_exists($key, $this->options)) {
+            $this->options[$key] = $value;
+        }
+    }
+
+    /**
+     * Get status option
+     *
+     * @param string $key
+     * @return mixed|null - Option value or NULL if option does not exist
+     */
+    public function getOption(string $key)
+    {
+        if (\array_key_exists($key, $this->options)) {
+            return $this->options[$key];
+        }
+
+        return null;
+    }
+
+    /**
+     * Get all status options
+     *
+     * @return array
+     */
+    public function getOptions(): array
+    {
+        return $this->options;
+    }
+
+    // endregion
+
     /**
      * Status as array
      *
@@ -110,9 +174,10 @@ public function toArray($Locale = null)
         }
 
         return [
-            'id'    => $this->getId(),
-            'title' => $title,
-            'color' => $this->getColor()
+            'id'      => $this->getId(),
+            'title'   => $title,
+            'color'   => $this->getColor(),
+            'options' => $this->getOptions()
         ];
     }
 }
diff --git a/src/QUI/ERP/Accounting/Invoice/RestApi/Provider.php b/src/QUI/ERP/Accounting/Invoice/RestApi/Provider.php
index 6bd1c028549a7adeed22a71828cecd14fba1f9df..1d127d9e59585c12879d85a22ba82b4d16146725 100644
--- a/src/QUI/ERP/Accounting/Invoice/RestApi/Provider.php
+++ b/src/QUI/ERP/Accounting/Invoice/RestApi/Provider.php
@@ -11,6 +11,7 @@
 use QUI\REST\Utils\RequestUtils;
 use QUI\ERP\Accounting\Invoice\Factory as InvoiceFactory;
 use QUI\ERP\Currency\Handler as CurrencyHandler;
+use QUI\ERP\Accounting\Invoice\ProcessingStatus\Handler as ProcessingStatuses;
 
 /**
  * Class Provider
@@ -94,7 +95,9 @@ public function createInvoice(RequestInterface $Request, ResponseInterface $Resp
             'currency'       => false,
             'payment_method' => false,
 
-            'additional_invoice_text' => false
+            'additional_invoice_text' => false,
+            'files'                   => false,
+            'processing_status'       => false,
         ];
 
         // Remove unknown fields
@@ -148,9 +151,9 @@ public function createInvoice(RequestInterface $Request, ResponseInterface $Resp
         $InvoiceDraft = $Factory->createInvoice($SystemUser);
 
         // Customer
-        if (!empty($invoiceData['customer_no'])) {
-            $User = false;
+        $User = false;
 
+        if (!empty($invoiceData['customer_no'])) {
             try {
                 $User = QUI\ERP\Customer\Customers::getInstance()->getCustomerByCustomerNo($invoiceData['customer_no']);
                 $InvoiceDraft->setAttribute('customer_id', $User->getId());
@@ -272,6 +275,61 @@ public function createInvoice(RequestInterface $Request, ResponseInterface $Resp
             $InvoiceDraft->addArticle($Article);
         }
 
+        // Files
+        if ($User && !empty($invoiceData['files'])) {
+            $fileDir = QUI::getPackage('quiqqer/invoice')->getVarDir().'/uploads/'.$InvoiceDraft->getId();
+            QUI\Utils\System\File::mkdir($fileDir);
+
+            $fileCounter = 0;
+
+            foreach ($invoiceData['files'] as $file) {
+                if (!\is_array($file) || !isset($file['name']) || !isset($file['content'])) {
+                    continue;
+                }
+
+                // Write file data to file
+                $localFile = $fileDir;
+
+                if (!empty($file['name'])) {
+                    $localFile .= $file['name'];
+                } else {
+                    $localFile .= 'file_'.++$fileCounter;
+                }
+
+                \file_put_contents($localFile, \hex2bin($file['content']));
+
+                $fileInfo = \pathinfo($localFile);
+                $fileHash = \hash('sha256', $fileInfo['basename']);
+
+                try {
+                    QUI\ERP\Customer\CustomerFiles::addFileToCustomer($User->getId(), $localFile);
+
+                    $fileOptions = [];
+
+                    if (!empty($file['options'])) {
+                        $fileOptions = $file['options'];
+                    }
+
+                    $InvoiceDraft->addCustomerFile($fileHash, $fileOptions);
+                } catch (\Exception $Exception) {
+                    QUI\System\Log::writeException($Exception);
+                }
+            }
+        }
+
+        // Processing status
+        if (!empty($invoiceData['processing_status'])) {
+            $statusId      = (int)$invoiceData['processing_status'];
+            $StatusHandler = ProcessingStatuses::getInstance();
+
+            try {
+                $StatusHandler->getProcessingStatus($statusId);
+                $InvoiceDraft->setAttribute('processing_status', $statusId);
+            } catch (\Exception $Exception) {
+                QUI\System\Log::writeException($Exception);
+            }
+        }
+
         try {
             $InvoiceDraft->update($SystemUser);
         } catch (\Exception $Exception) {
diff --git a/src/QUI/ERP/Accounting/Invoice/Utils/Invoice.php b/src/QUI/ERP/Accounting/Invoice/Utils/Invoice.php
index 74efd01ff454ad135970d8ebf15b958d7a4d7b17..2dbd8bcae954fb6fd93980857aad6b834cb16025 100644
--- a/src/QUI/ERP/Accounting/Invoice/Utils/Invoice.php
+++ b/src/QUI/ERP/Accounting/Invoice/Utils/Invoice.php
@@ -10,6 +10,7 @@
 use QUI\ERP\Accounting\Invoice\Exception;
 use QUI\ERP\Accounting\Invoice\InvoiceTemporary;
 use QUI\ERP\Accounting\Payments\Transactions\Transaction;
+use QUI\ERP\Accounting\Invoice\ProcessingStatus\Handler as ProcessingStatuses;
 
 /**
  * Class Invoice
@@ -104,7 +105,7 @@ public static function getMissingAttributes(InvoiceTemporary $Invoice): array
             self::getMissingAddressFields($Invoice)
         );
 
-        //articles
+        // articles
         if (!$Articles->count()) {
             $missing[] = 'article';
         }
@@ -117,6 +118,23 @@ public static function getMissingAttributes(InvoiceTemporary $Invoice): array
             $missing[] = 'payment';
         }
 
+        // Status that prevents posting
+        $statusId = $Invoice->getAttribute('processing_status');
+
+        if (!empty($statusId)) {
+            try {
+                $Status = ProcessingStatuses::getInstance()->getProcessingStatus(
+                    $statusId
+                );
+
+                if ($Status->getOption(ProcessingStatuses::STATUS_OPTION_PREVENT_INVOICE_POSTING)) {
+                    $missing[] = 'status_prevents_posting';
+                }
+            } catch (\Exception $Exception) {
+                QUI\System\Log::writeException($Exception);
+            }
+        }
+
         // api
         QUI::getEvents()->fireEvent('onQuiqqerInvoiceMissingAttributes', [$Invoice, &$missing]);
 
@@ -269,6 +287,9 @@ public static function getMissingAttributeMessage(string $missingAttribute): str
 
             case 'invoice_address_country':
                 return $Locale->get($lg, 'exception.invoice.verification.country');
+
+            case 'status_prevents_posting':
+                return $Locale->get($lg, 'exception.invoice.verification.status_prevents_posting');
         }
 
         $message = false;