浏览代码

Checkout-Funktion

- Zahlung mit Karte oder bar
- Summe für beide Varianten

- Druckfunktion für Linux
Robert Bedner 2 年之前
父节点
当前提交
c227df0325

+ 1 - 0
.gitignore

@@ -0,0 +1 @@
+api/config.php

+ 4 - 0
api/config-default.php

@@ -0,0 +1,4 @@
+<?php
+define('DB_CONN_STRING', "mysql:host=localhost;dbname=kasse");
+define('DB_USER', "root");
+define('DB_PASSWORD', "");

+ 47 - 14
api/controllers/RequestController.php

@@ -1,4 +1,6 @@
 <?php
+require_once(dirname(__FILE__)."/../config.php");
+
 if(!function_exists("array_column"))
 {
 	function array_column($array,$column_name)
@@ -7,6 +9,12 @@ if(!function_exists("array_column"))
 	}
 }
 
+if(!function_exists("printer_write"))
+{
+	function printer_write($printer, $text) {
+		shell_exec("echo \"" . $text . "\" > " . $printer);
+	}
+}
 
 class RequestController
 {
@@ -19,7 +27,7 @@ class RequestController
 	{
         date_default_timezone_set('Europe/Berlin');
 		$this->mdate = date('Y-m-d H:i:s', strtotime("now"));
-		$this->dbh = new PDO("mysql:host=localhost;dbname=kasse", "root", "gc01mysql");
+		$this->dbh = new PDO(DB_CONN_STRING, DB_USER, DB_PASSWORD);
 		$this->dbh->setAttribute(PDO::ATTR_CASE, PDO::CASE_LOWER);
         $q = $this->dbh->query("SELECT * FROM projects			 				  
                                 WHERE active = '1' 
@@ -34,14 +42,18 @@ class RequestController
 	function bootstrap ($data)
 	{
 		$result = array();
+		$action = (isset($_REQUEST['a'])) ? $_REQUEST['a'] : "";
 
-		switch ($_REQUEST['a']) {
+		switch ($action) {
 			case 'pos_token':
 				$result = $this->posToken($data);
 				break;
 			case 'pos_invoices':
 				$result = $this->posInvoices($data);
 				break;
+			case 'pos_checkout':
+				$result = $this->posCheckout($data);
+				break;
             case 'pos_print':
                 $result = $this->posPrint($data);
                 break;
@@ -60,6 +72,9 @@ class RequestController
             case 'sellers':
 				$result = $this->sellers($data);
 				break;
+			default:
+				$result = array("No action defined!");
+				break;
 		}
 
 		$this->setJsonHeader();
@@ -158,7 +173,9 @@ class RequestController
 		$q = $this->dbh->query("SELECT * FROM invoice_header WHERE invoice_number = '{$data['invoice_number']}' LIMIT 1");
 		$result = $q->fetch(PDO::FETCH_ASSOC);
 		if ($result) {
-			$q = $this->dbh->query("SELECT * FROM invoice_details WHERE invoice_number = '{$data['invoice_number']}' ORDER BY line_number DESC ");
+			$q = $this->dbh->query("SELECT * FROM invoice_details 
+			                        WHERE invoice_number = '{$data['invoice_number']}' 
+									ORDER BY line_number DESC ");
 			$result['details'] = $q->fetchAll(PDO::FETCH_ASSOC);
 		}
 		return $result;
@@ -167,7 +184,9 @@ class RequestController
 	private function posInfos ($data)
 	{
 		$result = array('items' => array(), 'invoices' => array());
-		$q = $this->dbh->query("SELECT item FROM invoice_details WHERE project_id = '{$this->project_id}' AND item NOT IN ('000-000','123-456') AND cancelled = 0 ORDER BY item ");
+		$q = $this->dbh->query("SELECT item FROM invoice_details 
+		                        WHERE project_id = '{$this->project_id}' AND item NOT IN ('000-000','123-456') AND cancelled = 0 
+								ORDER BY item ");
 		if ($q) {
 			$r = $q->fetchAll(PDO::FETCH_ASSOC);
 			$result['items'] = array_column($r, 'item');
@@ -175,11 +194,13 @@ class RequestController
 
 		if ($data['pos_id'] == 0) {
             $q = $this->dbh->query("SELECT * FROM `view_invoice_header` a
-								WHERE a.project_id = '{$this->project_id}' ORDER BY a.invoice_number ");
+								    WHERE a.project_id = '{$this->project_id}' 
+									ORDER BY a.invoice_number ");
 		    
         } else {
             $q = $this->dbh->query("SELECT * FROM `view_invoice_header` a
-								WHERE a.project_id = '{$this->project_id}' AND a.pos_id = '{$data['pos_id']}' ORDER BY a.invoice_number ");
+								    WHERE a.project_id = '{$this->project_id}' AND a.pos_id = '{$data['pos_id']}' 
+									ORDER BY a.invoice_number ");
         }
 
 		if ($q) {
@@ -195,11 +216,13 @@ class RequestController
 
 		if ($data['pos_id'] == 0) {
             $q = $this->dbh->query("SELECT * FROM `view_invoice_header` a
-								WHERE a.project_id = '{$this->project_id}' ORDER BY a.invoice_number ");
+								    WHERE a.project_id = '{$this->project_id}' 
+									ORDER BY a.invoice_number ");
 		    
         } else {
             $q = $this->dbh->query("SELECT * FROM `view_invoice_header` a
-								WHERE a.project_id = '{$this->project_id}' AND a.pos_id = '{$data['pos_id']}' ORDER BY a.invoice_number ");
+								    WHERE a.project_id = '{$this->project_id}' AND a.pos_id = '{$data['pos_id']}' 
+								    ORDER BY a.invoice_number ");
         }
 
 		if ($q) {
@@ -208,17 +231,27 @@ class RequestController
 				
 		return $result;
 	}
+
+	private function posCheckout ($data)
+    {
+        $result = $data['invoice'];
+        $this->dbh->exec("UPDATE invoice_header 
+					      SET checkout = '{$result['checkout']}'
+						  WHERE invoice_number = '{$result['invoice_number']}' ");
+		return $result;
+	}
     
     private function posPrint ($data)
     {
         $result = $this->posInvoices($data);
-        $this->dbh->exec("UPDATE invoice_header SET
-									printed = '1'
-							   WHERE invoice_number = '{$result['invoice_number']}' ");
+        $this->dbh->exec("UPDATE invoice_header 
+						  SET printed = '1'
+						  WHERE invoice_number = '{$result['invoice_number']}' ");
 
         // print options
-        $ptr = printer_open("POS-58");
-        printer_set_option($ptr, PRINTER_MODE, "raw");
+        // $ptr = printer_open("POS-58");
+        // printer_set_option($ptr, PRINTER_MODE, "raw");
+		$ptr = "/dev/usb/lp0";
 
         printer_write($ptr, $this->project['print_header']);
         printer_write($ptr, str_repeat("\r\n", 2));
@@ -238,7 +271,7 @@ class RequestController
 
         printer_write($ptr, $this->project['print_footer']);
         printer_write($ptr, str_repeat("\r\n", 7));
-        printer_close($ptr);
+        // printer_close($ptr);
 
         return $result;
     }

+ 1 - 0
index.html

@@ -38,6 +38,7 @@
 
 <script src="scripts/app.js"></script>
 <script src="scripts/controllers/main.js"></script>
+<script src="scripts/controllers/checkout.js"></script>
 <script src="scripts/controllers/dashboard.js"></script>
 <script src="scripts/controllers/invoice.js"></script>
 <script src="scripts/controllers/pos.js"></script>

+ 5 - 3
queries/create.sql

@@ -3,7 +3,7 @@
 -- http://www.phpmyadmin.net
 --
 -- Host: localhost
--- Erstellungszeit: 22. Jun 2022 um 20:12
+-- Erstellungszeit: 15. Jul 2022 um 17:41
 -- Server Version: 5.5.16
 -- PHP-Version: 5.3.8
 
@@ -55,6 +55,7 @@ CREATE TABLE IF NOT EXISTS `invoice_header` (
   `line_count` int(11) NOT NULL,
   `total` decimal(6,2) NOT NULL,
   `paid` decimal(6,2) NOT NULL,
+  `checkout` tinyint(4) NOT NULL DEFAULT '0',
   `printed` tinyint(4) NOT NULL DEFAULT '0',
   `mdate` datetime NOT NULL,
   `cdate` datetime NOT NULL,
@@ -113,7 +114,7 @@ CREATE TABLE IF NOT EXISTS `projects` (
   `mdate` datetime NOT NULL,
   `cdate` datetime NOT NULL,
   PRIMARY KEY (`project_id`)
-) ENGINE=InnoDB  DEFAULT CHARSET=latin1 AUTO_INCREMENT=8 ;
+) ENGINE=InnoDB  DEFAULT CHARSET=latin1 AUTO_INCREMENT=9 ;
 
 -- --------------------------------------------------------
 
@@ -170,6 +171,7 @@ CREATE TABLE IF NOT EXISTS `view_invoice_header` (
 ,`invoice_date` datetime
 ,`cashier` varchar(30)
 ,`paid` decimal(6,2)
+,`checkout` tinyint(4)
 ,`printed` tinyint(4)
 ,`line_count` bigint(21)
 ,`total` decimal(28,2)
@@ -203,7 +205,7 @@ CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW
 --
 DROP TABLE IF EXISTS `view_invoice_header`;
 
-CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW `view_invoice_header` AS select `a`.`invoice_number` AS `invoice_number`,`a`.`project_id` AS `project_id`,`a`.`pos_id` AS `pos_id`,`a`.`invoice_date` AS `invoice_date`,`a`.`cashier` AS `cashier`,`a`.`paid` AS `paid`,`a`.`printed` AS `printed`,`b`.`line_count` AS `line_count`,`b`.`total` AS `total`,`a`.`mdate` AS `mdate`,`a`.`cdate` AS `cdate` from (`invoice_header` `a` left join `view_totals` `b` on((`a`.`invoice_number` = `b`.`invoice_number`)));
+CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`localhost` SQL SECURITY DEFINER VIEW `view_invoice_header` AS select `a`.`invoice_number` AS `invoice_number`,`a`.`project_id` AS `project_id`,`a`.`pos_id` AS `pos_id`,`a`.`invoice_date` AS `invoice_date`,`a`.`cashier` AS `cashier`,`a`.`paid` AS `paid`,`a`.`checkout` AS `checkout`,`a`.`printed` AS `printed`,`b`.`line_count` AS `line_count`,`b`.`total` AS `total`,`a`.`mdate` AS `mdate`,`a`.`cdate` AS `cdate` from (`invoice_header` `a` left join `view_totals` `b` on((`a`.`invoice_number` = `b`.`invoice_number`)));
 
 -- --------------------------------------------------------
 

+ 2 - 1
queries/view_invoice_header.sql

@@ -6,7 +6,8 @@ select
 	`a`.`pos_id` AS `pos_id`,
 	`a`.`invoice_date` AS `invoice_date`,
 	`a`.`cashier` AS `cashier`,
-	`a`.`paid` AS `paid`,
+	`a`.`paid` AS `paid`,	
+	`a`.`checkout` AS `checkout`,
 	`a`.`printed` AS `printed`,
 	`b`.`line_count` AS `line_count`,
 	`b`.`total` AS `total`,

+ 4 - 0
scripts/app.js

@@ -145,6 +145,10 @@ angular.module('pos', ['ngRoute'])
                 templateUrl: 'templates/main.html',
                 controller: 'MainCtrl'
             })
+            .when('/checkout', {
+                templateUrl: 'templates/checkout.html',
+                controller: 'CheckoutCtrl'
+            })
             .when('/dashboard', {
                 templateUrl: 'templates/dashboard.html',
                 controller: 'DashboardCtrl'

+ 419 - 0
scripts/controllers/checkout.js

@@ -0,0 +1,419 @@
+'use strict';
+
+angular.module('pos')
+    .controller('CheckoutCtrl', function ($scope, $routeParams, $http, $timeout, $interval) {
+        var webservice = 'api/?a=pos';
+
+        $scope.postParams = {
+            'project_id': "5",
+            'pos_id': 0,
+            'cashier': window.localStorage.getItem("cashier"),
+            'token': window.localStorage.getItem("token"),
+            'active': 1,
+            'disabled': 0
+        };
+        var postParams = $scope.postParams;
+
+        $scope.posList = _.range(0, 9);
+        
+        $scope.filterInvoices = function (flag) {
+            if (flag == -1) {
+                return _.filter($scope.invoices, function (e) { return e.checkout != 0; });
+            }
+            return _.filter($scope.invoices, function (e) { return e.checkout == flag; });
+        };        
+        
+        $scope.lineTemplate = function () {
+            if ($scope.currentInvoice.last_line_number === undefined) {
+                $scope.currentInvoice.last_line_number = parseInt($scope.currentInvoice.details[0].line_number, 10);
+            }
+            $scope.currentInvoice.last_line_number += 1;
+            return {
+                'line_number': $scope.currentInvoice.last_line_number,
+                'item': "",
+                'seller_id': "",
+                'item_id': "",
+                'price': 0
+            };
+        };
+
+        $scope.invoiceTemplate = function () {
+            return {
+                    'invoice_number': "",
+                    'project_id': $scope.postParams.project_id,
+                    'pos_id': $scope.postParams.pos_id,
+                    'invoice_date': moment(),
+                    'cashier': $scope.postParams.cashier,
+                    'last_line_number': 1,
+                    'line_count': 0,
+                    'valid': false,
+                    'total': 0.0,
+                    'paid': 0.0,
+                    'checkout': 0,
+                    'printed': 0,
+                    'details': [
+                        {
+                            'line_number': 1,
+                            'item': "",
+                            'seller_id': "",
+                            'item_id': "",
+                            'price': 0.0
+                        }
+                    ]
+                };
+        };
+
+        var successSound = new Audio("sounds/success.wav");
+        var errorSound = new Audio("sounds/error.wav");
+
+		$scope.warning = 0;
+		$scope.warningInfo = [
+			"",
+			"Dieses Etikett wurde bereits gescannt",
+			"Etiketten-Nummer ungültig",
+			"Preis ungültig"
+		];
+		
+        $scope.currentLine = 0;
+        $scope.currentLineType = '.item';
+        $scope.currentInvoice = $scope.invoiceTemplate();
+        $scope.showCurrentInvoice = false;
+
+        $scope.offlineInvoices = JSON.parse(window.localStorage.getItem("offlineInvoices"));
+        $scope.invoices = [];
+        $scope.soldItems = [];
+
+		$scope.total = function (pos) {
+            return _.reduce($scope.filterInvoices(pos), function (sum, i) { return sum + parseFloat(i.total); }, 0);
+        };
+
+        $scope.invoiceLineCount = function () {
+            return $scope.currentInvoice.details.length;
+        };
+
+        $scope.invoiceTotal = function () {
+            return _.reduce($scope.currentInvoice.details, function (sum, d) { return sum + parseFloat(d.price); }, 0);
+        };
+
+        $scope.addItem = function () {
+            $scope.currentInvoice.details.unshift($scope.lineTemplate());
+        };
+
+        $scope.removeItem = function (item) {
+            $scope.currentInvoice.details = _.without($scope.currentInvoice.details, item);
+        };
+
+        $scope.currentChange = function () {
+            if ($scope.currentInvoice.paid === 0.0
+                || $scope.currentInvoice.paid == ""
+                || $scope.currentInvoice.paid == Number.NaN) {
+                return 0.0;
+            }
+
+            $scope.currentInvoice.total = $scope.invoiceTotal();
+            return $scope.currentInvoice.paid - $scope.currentInvoice.total;
+        };
+
+        $scope.getInvoiceByNumber = function (invoiceNumber) {
+            return _.find($scope.invoices, function (e) { return e.invoice_number === invoiceNumber; });
+        };
+
+        $scope.editInvoice = function (invoiceNumber) {
+			
+            $scope.currentInvoice = $scope.getInvoiceByNumber(invoiceNumber);
+            $scope.selectItem(0);
+        };
+
+        $scope.jumpEvent = function (event) {
+            var key = String.fromCharCode(event.which);
+
+            switch (key) {
+                case '*':
+                    $scope.selectPrice($scope.currentLine+1);
+                    return false;
+                case '/':
+                    $('#paid').select();
+                    $scope.validate();
+                    return false;
+                case '+':
+                    $scope.selectItem(0);
+                    return false;
+            }
+            return true;
+        };
+        $("body").on('keypress', $scope.jumpEvent);
+
+
+        $scope.lineNumberComparator = function (v1, v2) {
+            return (parseInt(v1, 10) < parseInt(v2, 10)) ? -1 : 1;
+        };
+
+        $scope.keyPressModal = function (e) {
+            if (e.which === 13) {
+                e.preventDefault();
+                $(".modal").modal("hide");
+                $scope.selectItem();
+            }
+            return true;
+        };
+
+        $scope.showWarning = function (num) {
+            $scope.warning = num;
+            $('#warningModal').modal('show');
+            errorSound.play();
+        };
+
+        $scope.selectItem = function (index, itemType) {
+            if (index !== undefined) {
+                $scope.currentLine = index;
+            }
+            if (itemType !== undefined) {
+                $scope.currentLineType = itemType;
+            } else {
+                $scope.currentLineType = '.item';
+            }
+
+            $timeout(function () {
+                $($scope.currentLineType)[$scope.currentLine].select();
+            }, 300, true);
+        };
+
+        $scope.selectPrice = function (index) {
+            $scope.selectItem(index, '.price');
+        };
+
+        $scope.currentSoldItems = function (index) {
+            var list = _.pluck($scope.currentInvoice.details, 'item');
+            var needle = list[index];
+            list[index] = "";
+            for (var i = 0; i < list.length; i++) {
+                if (list[i] === needle) {
+                    return i;
+                }
+            }
+            return index;
+        };
+
+        $scope.keyPressItem = function (e, i) {
+            if (e.which === 13) {
+                e.preventDefault();
+
+                var currentItem = $scope.currentInvoice.details[i];
+                if ($scope.currentLine !== i && currentItem.item === "") {
+                    return true;
+                }
+
+                if (!currentItem.item.match(/^\d{3}-\d{3}$/gi)) {
+                    $scope.showWarning(2);
+				} else if ($scope.currentInvoice.invoice_number == 0 && _.contains($scope.soldItems, currentItem.item)) {
+                    $scope.showWarning(1);
+                } else if ($scope.currentSoldItems(i) !== i) {
+                    errorSound.play();
+                    $scope.selectPrice($scope.currentSoldItems(i));
+                    currentItem.item = "";
+                    return false;
+                } else {
+                    successSound.play();
+                    $scope.selectPrice(i);
+					return true;
+                }
+
+				$scope.selectItem(i);
+                return false;
+            }
+            var key = String.fromCharCode(e.which);
+            if (key == '/') {
+                e.preventDefault();
+            }
+
+            return true;
+        };
+
+        $scope.keyPressPrice = function (e, i) {
+            if (e.which === 13) {
+                var currentPrice = $scope.currentInvoice.details[i].price;
+
+                if (!currentPrice.toString().match(/^\d+(\.\d\d?)?$/gi) || currentPrice <= 0) {
+                    errorSound.play();
+                    e.preventDefault();
+                    $scope.selectPrice(i);
+                    return false;
+                }
+
+                if ($scope.currentInvoice.details[0].item !== "") {
+                    $scope.addItem();
+                }
+
+                successSound.play();
+                $scope.selectItem(0);
+                return false;
+            }
+            var key = String.fromCharCode(e.which);
+            if (key == '/') {
+                e.preventDefault();
+            }
+            return true;
+        };
+
+        $scope.keyPressPaid = function (e) {
+            $scope.validate();
+
+            if (e.which === 13) {
+                $scope.checkOut();
+            }
+        };
+
+
+        $scope.validate = function () {
+            var i = 0, item;
+            for (i = $scope.currentInvoice.details.length - 1; i >= 0; i--) {
+                if ($scope.currentInvoice.details[i].price == 0 && $scope.currentInvoice.details[i].item === "") {
+                    $scope.removeItem($scope.currentInvoice.details[i]);
+                    break;
+                }
+
+                item = $scope.currentInvoice.details[i];
+                if (!item.price.toString().match(/^\d+(\.\d\d?)?$/gi) || item.price <= 0) {
+                    $scope.selectPrice(i);
+                    errorSound.play();
+                    $scope.currentInvoice.valid = false;
+                    return false;
+                }
+                if (item.item === "" || !item.item.match(/^\d{3}-\d{3}$/gi)
+                    || ($scope.currentInvoice.invoice_number == 0 && _.contains($scope.soldItems, item.item))) {
+                    $scope.selectItem(i);
+                    errorSound.play();
+                    $scope.currentInvoice.valid = false;
+                    return false;
+                }
+
+                item.seller_id = item.item.substring(0, 3);
+                item.item_id = item.item.substring(4, 7);
+            }
+            if ($scope.currentInvoice.details.length === 0) {
+                $scope.addItem();
+                $scope.selectItem(0);
+                errorSound.play();
+                $scope.currentInvoice.valid = false;
+                return false;
+            }
+
+            $scope.currentInvoice.cashier = $scope.postParams.cashier;
+            $scope.currentInvoice.line_count = $scope.currentInvoice.details.length;
+            $scope.currentInvoice.invoice_date = moment();
+            $scope.currentInvoice.valid = true;
+            return true;
+        };
+
+        $scope.checkOut = function () {
+            if (isNaN($scope.currentInvoice.paid) || $scope.currentInvoice.paid == '' || $scope.currentInvoice.paid == 0) {
+                $scope.currentInvoice.paid = $scope.currentInvoice.total;
+            }
+
+            $scope.saveInvoice($scope.currentInvoice);
+            window.localStorage.setItem("cashier", $scope.postParams.cashier);
+            $scope.currentInvoice = $scope.invoiceTemplate();
+            $scope.selectItem(0);
+        };
+
+
+
+        if ($scope.offlineInvoices === null) {
+            $scope.offlineInvoices = [];
+        }
+
+        if ($scope.postParams.token === null) {
+            $scope.postParams.token = 0;
+        }
+
+        $scope.setActive = function () {
+            $scope.postParams.active = 1;
+            $scope.refreshStatus();
+        };
+
+        $scope.setInactive = function () {
+            $scope.postParams.active = 0;
+            $scope.refreshStatus();
+        };
+
+        $scope.closePos = function () {
+            $scope.postParams.disabled = 1;
+            $scope.refreshStatus();
+        };
+
+        $scope.refreshStatus = function () {
+            $http.post(webservice + '_token', postParams).then(function (response) {
+                $scope.postParams.token = response.data.token;
+                window.localStorage.setItem("token", $scope.postParams.token);
+                $scope.postParams.active = response.data.active;
+                $scope.postParams.disabled = response.data.disabled;
+            });
+        };
+
+		$scope.saveInvoice = function (invoice) {
+            $http.post(webservice + '_invoices', { 'params': $scope.postParams, 'invoice': invoice })
+				.then(function (response) {
+                    if (response.data.invoice_number !== undefined) {
+                        $scope.offlineInvoices = _.without($scope.offlineInvoices, invoice);
+                        $scope.invoices[$scope.invoices.length] = response.data;
+                    } else if (!_.contains($scope.offlineInvoices, invoice)) {
+                        $scope.offlineInvoices.push(invoice);
+                    }
+                    window.localStorage.setItem("offlineInvoices", JSON.stringify($scope.offlineInvoices));
+				});			
+		};
+		
+		$scope.loadInvoice = function (invoiceNumber) {
+            $http.post(webservice + '_invoices', { 'params': $scope.postParams, 'invoice_number': invoiceNumber })
+				.then(function (response) {
+                    if (response.data.invoice_number !== undefined) {
+                        $scope.currentInvoice = response.data;
+                        $scope.showCurrentInvoice = true;
+                        $scope.selectItem(0);
+                    } 
+				});			
+		};
+
+        $scope.printInvoice = function (invoiceNumber) {
+            $http.post(webservice + '_print', { 'params': $scope.postParams, 'invoice_number': invoiceNumber })
+                .then(function (response) {
+                    if (response.data.invoice_number !== undefined) {
+                        var invoice = $scope.getInvoiceByNumber(invoiceNumber);
+                        invoice.printed = 1;
+                        $scope.currentInvoice = null;
+                    }
+                });
+        };
+
+        $scope.setCheckout = function (state) {
+            $scope.currentInvoice.checkout = state;
+        };
+
+
+        $scope.performCheckout = function (invoice) {
+            $http.post(webservice + '_checkout', { 'params': $scope.postParams, 'invoice': invoice })
+                .then(function (response) {
+                    if (response.data.invoice_number === undefined) {
+                        // undo
+                        invoice.checkout = 0;
+                    }
+                    $scope.refresh();
+                });
+        };
+
+		$scope.refresh = function () {
+            for (var i = 0; i < $scope.offlineInvoices.length; i++) {
+                $timeout($scope.saveInvoice($scope.offlineInvoices[i]), 3000 * i);
+            }
+
+            $http.post(webservice + '_infos', $scope.postParams)
+				.then(function (response) {
+					//$scope.soldItems = response.data.items;
+                    $scope.invoices = response.data.invoices;
+				});
+		};
+
+        $interval($scope.refresh, 60 * 1000);
+        $scope.refreshStatus();
+        $scope.refresh();
+    });

+ 82 - 0
styles/main.css

@@ -137,4 +137,86 @@ input.ng-pristine {
 .table > tbody > tr > th
 {
     vertical-align: middle;
+}
+
+.pos1 {
+    color: blue;
+}
+
+.pos1-bg {
+    color: white;
+    background-color: blue;
+}
+
+.pos2 {
+    color: yellow;
+}
+
+.pos2-bg {
+    color: black;
+    background-color: yellow;
+}
+
+.pos3 {
+    color: red;
+}
+
+.pos3-bg {
+    color: white;
+    background-color: red;
+}
+
+.pos4 {
+    color: green;
+}
+
+.pos4-bg {
+    color: white;
+    background-color: green;
+}
+
+.pos5 {
+    color: white;
+}
+
+.pos5-bg {
+    color: black;
+    background-color: white;
+}
+
+.pos6 {
+    color: turquoise;
+}
+
+.pos6-bg {
+    color: black;
+    background-color: turquoise;
+    ;
+}
+
+.pos7 {
+    color: orange;
+}
+
+.pos7-bg {
+    color: black;
+    background-color: orange;
+}
+
+.pos8 {
+    color: darkviolet;
+}
+
+.pos8-bg {
+    color: black;
+    background-color: darkviolet;
+}
+
+.pos9 {
+    color: pink;
+}
+
+.pos9-bg {
+    color: black;
+    background-color: pink;
 }

+ 187 - 0
templates/checkout.html

@@ -0,0 +1,187 @@
+<div class="row ">
+
+    <div class="col-md-4">
+        <h3>Offene Rechnungen</h3>
+        <table class="table table-striped">
+            <tr>
+                <th class="normal">Datum</th>
+                <th class="normal">Kasse</th>
+                <th class="normal">Anz. Art.</th>
+                <th class="normal text-right">Gesamt-Preis</th>
+                <th></th>
+            </tr>
+
+            <tr ng-repeat="invoice in filterInvoices(0)" >
+                <td class="normal">{{invoice.invoice_date|datum}}</td>
+                <td>
+                    <span class="label label-default pos{{invoice.pos_id}}-bg">{{invoice.pos_id}}</span>
+                </td>
+                <td class="text-right">{{invoice.line_count}}</td>
+                <td class="text-right">{{invoice.total|euro}}</td>
+                <td>
+                    <button type="button" class="btn btn-default" data-toggle="modal" data-target="#invoiceModal" ng-click="loadInvoice(invoice.invoice_number)">
+                        <i class="glyphicon glyphicon-shopping-cart"></i>
+                    </button>
+                </td>
+            </tr>
+        </table>
+        <br/><br/>
+
+        <h3>Bezahlte Rechnungen</h3>
+        <table class="table table-striped">
+            <tr>
+                <th></th>
+                <th></th>
+                <th class="normal">Datum</th>
+                <th class="normal">Kasse</th>
+                <th class="normal">Anz. Art.</th>
+                <th class="normal text-right">Gesamt-Preis</th>
+                <th class="normal text-right">Gegeben<br/>Rückgeld</th>
+            </tr>
+
+            <tr ng-repeat="invoice in filterInvoices(-1)" >
+                <td><button type="button" class="btn" data-toggle="modal" data-target="#invoiceModal" ng-click="loadInvoice(invoice.invoice_number)"><i class="glyphicon glyphicon-pencil"></i></button></td>
+                <td>
+                    <button type="button" class="btn btn-warning" ng-click="printInvoice(invoice.invoice_number)" ng-show="invoice.printed==0"><i class="glyphicon glyphicon-print"></i></button>
+                    <button type="button" class="btn btn-success" ng-click="printInvoice(invoice.invoice_number)" ng-show="invoice.printed==1"><i class="glyphicon glyphicon-check"></i></button>
+                </td>
+                <td class="normal">{{invoice.invoice_date|datum}}</td>
+                <td>
+                    <span class="label label-default pos{{invoice.pos_id}}-bg">{{invoice.pos_id}}</span>
+                </td>
+                <td class="text-right">{{invoice.line_count}}</td>
+                <td class="text-right">{{invoice.total|euro}}</td>
+                <td class="text-right" ng-show="invoice.checkout==2">
+                    <i class="glyphicon glyphicon-credit-card"></i>
+                </td>
+                <td class="normal text-right" ng-show="invoice.checkout==1">{{invoice.paid|euro}}<br/>{{(invoice.paid-invoice.total)|euro}}</td>
+            </tr>
+            <tr>
+                <td></td>
+                <td></td>
+                <th class="normal">Kassenstand</th>
+                <td></td>
+                <td></td>
+                <th class="text-right">{{total(1)|euro}}</th>
+                <td class="text-right"></td>
+            </tr>
+            <tr>
+                <td></td>
+                <td></td>
+                <th class="normal">Konto</th>
+                <td></td>
+                <td></td>
+                <th class="text-right">{{total(2)|euro}}</th>
+                <td class="text-right"></td>
+            </tr>
+        </table>
+
+    </div>
+
+</div>
+
+<div class="modal fade" tabindex="-1" role="dialog" aria-labelledby="inactiveLabel" id="invoiceModal">
+    <div class="modal-dialog modal-lg" role="document">
+        <div class="modal-content">
+            <div class="modal-header">
+                <h4 class="modal-title" id="inactiveLabel">Bezahlvorgang</h4>
+            </div>
+            <div class="modal-body">
+
+        <table class="table table-striped">
+            <tr class="warning">
+                <th class="col-md-1"># {{currentInvoice.invoice_number}}</th>
+                <td class="col-md-2" colspan="3">
+                    {{currentInvoice.details.length}} Artikel
+                </td>
+                <th class="col-md-1 text-right big">
+                    {{invoiceTotal()|euro}} €
+                </th>
+            </tr>
+            <tr ng-show="currentInvoice.checkout==0">
+                <td colspan="3"></td>
+                <td>
+                    <button type="button" class="btn btn-lg btn-primary" ng-click="setCheckout(2)">
+                        <i class="glyphicon glyphicon-credit-card"></i> Karte
+                    </button>
+                </td>
+                <td>
+                    <button type="button" class="btn btn-lg btn-info" ng-click="setCheckout(1)">
+                        <i class="glyphicon glyphicon-piggy-bank"></i> Bar
+                    </button>
+                </td>
+            </tr>
+
+            <tr ng-show="currentInvoice.checkout==1">
+                <td><i class="glyphicon glyphicon-download-alt"></i></td>
+                <td colspan="3">
+                    Von Kunde erhalten (/)
+                </td>
+                <td class="text-right">
+                    <input id="paid" class="form-control text-right" type="text" ng-model="currentInvoice.paid" data-price/>
+                </td>
+            </tr>
+            <tr ng-show="currentInvoice.checkout==1">
+                <td><i class="glyphicon glyphicon-log-out"></i></td>
+                <th colspan="3">
+                    Rückgeld
+                </th>
+                <td class="text-right">
+                    {{currentChange()|euro}} €
+                </td>
+            </tr>
+
+            <tr ng-show="currentInvoice.checkout!=0">
+                <td><i class="glyphicon glyphicon-print"></i></td>
+                <td colspan="3">
+                    Drucken?
+                </td>
+                <td class="text-right">
+                    <button type="button" class="btn btn-warning" ng-click="printInvoice(currentInvoice.invoice_number)" ng-show="currentInvoice.printed==0"><i class="glyphicon glyphicon-print"></i></button>
+                    <button type="button" class="btn btn-success" ng-click="printInvoice(currentInvoice.invoice_number)" ng-show="currentInvoice.printed==1"><i class="glyphicon glyphicon-check"></i></button>
+                </td>
+            </tr>
+            <tr ng-show="currentInvoice.checkout!=0">
+                <td><i class="glyphicon glyphicon-download-alt"></i></td>
+                <td colspan="3">
+                    Zahlung erfolgreich?
+                </td>
+                <td class="text-right">
+                    <button type="button" class="btn btn-lg btn-success" data-toggle="modal" data-target="#invoiceModal" ng-click="performCheckout(currentInvoice)">
+                        <i class="glyphicon glyphicon-ok"></i> Ja 
+                    </button>
+                    <button type="button" class="btn btn-lg btn-danger" ng-click="currentInvoice.checkout=0">
+                        <i class="glyphicon glyphicon-remove"></i> Nein 
+                    </button>
+                </td>
+            </tr>            
+
+
+            <tr>
+                <th></th>
+                <th class="text-right">Pos</th>
+                <th>Artikel</th>
+                <th class="text-right">Preis</th>
+                <th></th>
+            </tr>
+            <tr ng-repeat="b in currentInvoice.details">
+                <td></td>
+                <td class="normal text-right">
+                    {{currentInvoice.details.length-$index}}
+                </td>
+                <td class="normal">
+                    {{b.item}}
+                </td>
+                <td class="normal text-right">
+                    {{b.price}}
+                </td>
+                <td></td>
+            </tr>
+
+        </table>
+                
+                
+            </div>
+        </div>
+    </div>
+</div>

+ 1 - 1
templates/pos.html

@@ -54,7 +54,7 @@
 
     <div class="col-md-5">
         <table class="table">
-            <tr>
+            <tr class="pos{{postParams.pos_id}}-bg">
                 <td style="width: 180px; font-size: 30pt; text-align: center;">
                     Kasse Nr. {{postParams.pos_id}}
                 </td>