From 4504be9cda82ea4aeac1eb2a055c38786169e58c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Patrick=20M=C3=BCller?= <p.mueller@pcsg.de> Date: Mon, 7 Jun 2021 16:28:07 +0200 Subject: [PATCH] temp commit --- ajax/create.php | 63 +++++- bin/backend/controls/Manager.Create.html | 185 ++++++++++++++---- bin/backend/controls/Manager.css | 31 ++- bin/backend/controls/Manager.js | 179 +++++++++-------- composer.json | 3 +- database.xml | 4 + events.xml | 8 + locale.xml | 69 ++++++- package.xml | 4 + products.xml | 17 ++ src/QUI/ERP/Coupons/Events.php | 183 +++++++++++++++++ .../Coupons/Products/CouponProductType.php | 47 +++++ src/QUI/ERP/Coupons/Products/Handler.php | 143 ++++++++++++++ 13 files changed, 802 insertions(+), 134 deletions(-) create mode 100644 products.xml create mode 100644 src/QUI/ERP/Coupons/Products/CouponProductType.php create mode 100644 src/QUI/ERP/Coupons/Products/Handler.php diff --git a/ajax/create.php b/ajax/create.php index 8058856..8ef89fb 100644 --- a/ajax/create.php +++ b/ajax/create.php @@ -1,11 +1,9 @@ <?php -/** - * This file contains package_quiqqer_coupons_ajax_create - */ - use QUI\ERP\Coupons\Handler; use QUI\Utils\Security\Orthos; +use QUI\ERP\Coupons\CouponCodeException; +use QUI\ERP\Discount\Handler as DiscountsHandler; /** * Create new CouponCode(s) @@ -27,16 +25,63 @@ function ($attributes) { unset($attributes['amount']); } - $discountIds = []; + // Check required fields + $requiredFields = [ + 'discountAmount' + ]; + + foreach ($requiredFields as $field) { + if (!isset($attributes[$field])) { + throw new CouponCodeException([ + 'quiqqer/coupons', + 'exception.ajax.create.missing_field' + ]); + } + } + + // Create discount + switch ($attributes['discountType']) { + case 'percentage': + $discountType = DiscountsHandler::DISCOUNT_TYPE_PERCENT; + break; - if (!empty($attributes['discountIds'])) { - $discountIds = \explode(',', $attributes['discountIds']); + default: + $discountType = DiscountsHandler::DISCOUNT_TYPE_CURRENCY; } + $Discounts = DiscountsHandler::getInstance(); + $NewDiscount = $Discounts->createChild([ + 'active' => 1, + 'discount' => (float)$attributes['discountAmount'], + 'discount_type' => $discountType, + ]); + + $L = QUI::getLocale(); + for ($i = 0; $i < $amount; $i++) { - $couponCodes[] = Handler::createCouponCode($discountIds, $attributes); + $NewCouponCode = Handler::createCouponCode([$NewDiscount->getId()], $attributes); + + if ($i === 0) { + \QUI\Translator::update( + 'quiqqer/discount', + 'discount.'.$NewDiscount->getId().'.title', + 'quiqqer/discount', + [ + 'de' => $L->getByLang('de', 'quiqqer/coupons', 'Discount.default_title', [ + 'couponCode' => $NewCouponCode->getCode() + ]), + 'en' => $L->getByLang('en', 'quiqqer/coupons', 'Discount.default_title', [ + 'couponCode' => $NewCouponCode->getCode() + ]) + ] + ); + + \QUI\Translator::publish('quiqqer/discount'); + } + + $couponCodes[] = $NewCouponCode; } - } catch (\QUI\ERP\Coupons\CouponCodeException $Exception) { + } catch (CouponCodeException $Exception) { QUI::getMessagesHandler()->addError( QUI::getLocale()->get( 'quiqqer/coupons', diff --git a/bin/backend/controls/Manager.Create.html b/bin/backend/controls/Manager.Create.html index 54ba00f..c45b833 100644 --- a/bin/backend/controls/Manager.Create.html +++ b/bin/backend/controls/Manager.Create.html @@ -1,43 +1,152 @@ <div class="quiqqer-coupons-manager-create"> <form> - <label> - <span>{{labelDiscount}}</span> - <input type="hidden" name="discountIds" data-qui="package/quiqqer/discount/bin/controls/Select"/> - </label> - <div class="quiqqer-coupons-manager-create-settings-btn"></div> - <div class="quiqqer-coupons-manager-create-settings"> - <label> - <span>{{labelTitle}}</span> - <input type="text" name="title"/> - </label> - <label> - <span>{{labelCode}}</span> - <input type="text" name="code"/> - </label> - <label> - <span>{{labelReusable}}</span> - <select name="maxUsages"> - <option value="oncePerUser" selected>{{labelReusableOncePerUser}}</option> - <option value="once">{{labelReusableOnce}}</option> - <option value="unlimited">{{labelReusableUnlimited}}</option> - </select> - </label> - <label> - <span>{{labelUsers}}</span> - <input type="hidden" name="userIds" data-qui="controls/users/Select"/> - </label> - <label> - <span>{{labelGroups}}</span> - <input type="hidden" name="groupIds" data-qui="controls/groups/Select"/> - </label> - <label> - <span>{{labelDate}}</span> - <input type="date" placeholder="YYYY-MM-DD HH:MM" name="validUntilDate"/> - </label> - <label> - <span>{{labelAmount}}</span> - <input type="number" name="amount" min="1" value="1"/> - </label> + <div class="quiqqer-coupons-manager-create-basic"> + <h2> + {{headerBasics}} + </h2> + <table class="data-table data-table-flexbox"> + <tbody> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelTitle}}</span> + <input type="text" class="field-container-field" name="title"/> + </label> + <div class="field-container-item-desc"> + {{{descTitle}}} + </div> + </td> + </tr> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelCode}}</span> + <input type="text" class="field-container-field" name="code"/> + </label> + <div class="field-container-item-desc"> + {{{descCode}}} + </div> + </td> + </tr> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelAmount}}</span> + <input type="number" name="amount" class="field-container-field" min="1" value="1"/> + </label> + </td> + </tr> + </tbody> + </table> + </div> + <div class="quiqqer-coupons-manager-create-advanced"> + <div class="quiqqer-coupons-manager-create-advanced-discount"> + <h2> + {{headerDiscount}} + </h2> + <table class="data-table data-table-flexbox"> + <tbody> + {{#isCreate}} + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelDiscountAmount}}</span> + <input type="number" name="discountAmount" class="field-container-field" required/> + </label> + </td> + </tr> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelDiscountType}}</span> + <select class="field-container-field" name="discountType"> + <option value="flat" selected>{{labelDiscountTypeFlat}}</option> + <option value="percentage">{{labelDiscountTypePercentage}}</option> + </select> + </label> + </td> + </tr> + {{/isCreate}} + {{^isCreate}} + <tr class="quiqqer-coupons-manager-create-advanced-discount-select"> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelDiscount}}</span> + <input type="hidden" + class="field-container-field" + name="discountIds" + data-qui="package/quiqqer/discount/bin/controls/Select" + data-qui-options-multiple="0" + data-qui-options-max="1" + required + /> + </label> + <div class="field-container-item-desc quiqqer-coupons-manager-create-advanced-discount-buttons"> + + </div> + </td> + </tr> + {{/isCreate}} + </tbody> + </table> + </div> + <div class="quiqqer-coupons-manager-create-advanced-restrictions"> + <h2> + {{headerRestrictions}} + </h2> + <table class="data-table data-table-flexbox"> + <tbody> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelReusable}}</span> + <select class="field-container-field" name="maxUsages"> + <option value="oncePerUser" selected>{{labelReusableOncePerUser}}</option> + <option value="once">{{labelReusableOnce}}</option> + <option value="unlimited">{{labelReusableUnlimited}}</option> + </select> + </label> + </td> + </tr> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelDate}}</span> + <input type="date" + name="validUntilDate" + placeholder="YYYY-MM-DD HH:MM" + class="field-container-field" + /> + </label> + </td> + </tr> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelUsers}}</span> + <input type="hidden" + name="userIds" + class="field-container-field" + data-qui="controls/users/Select" + /> + </label> + </td> + </tr> + <tr> + <td> + <label class="field-container"> + <span class="field-container-item">{{labelGroups}}</span> + <input type="hidden" + name="groupIds" + class="field-container-field" + data-qui="controls/groups/Select" + /> + </label> + </td> + </tr> + </tbody> + </table> + </div> </div> </form> </div> \ No newline at end of file diff --git a/bin/backend/controls/Manager.css b/bin/backend/controls/Manager.css index 7399dd2..61aa78d 100644 --- a/bin/backend/controls/Manager.css +++ b/bin/backend/controls/Manager.css @@ -23,11 +23,6 @@ margin-bottom: 15px; } -.quiqqer-coupons-manager-create label > span { - float: left; - margin-bottom: 5px; -} - .quiqqer-coupons-manager-create-sendmail span, .quiqqer-coupons-manager-create-sendmail input { float: right !important; @@ -42,7 +37,7 @@ } .quiqqer-coupons-manager-create-settings { - display: none; + } .quiqqer-coupons-manager-create-settings-btn { @@ -88,4 +83,28 @@ .quiqqer-coupons-manager-usages-tbl-user { width: 200px; +} + +.quiqqer-coupons-manager-create h2 { + font-size: 18px; + margin-bottom: 10px; +} + +.quiqqer-coupons-manager-create-advanced { + align-content: stretch; + display: flex; + flex-direction: row; + float: left; + justify-content: space-between; + width: 100%; +} + +.quiqqer-coupons-manager-create-advanced-discount { + padding-right: 10px; + width: 50%; +} + +.quiqqer-coupons-manager-create-advanced-restrictions { + padding-left: 10px; + width: 50%; } \ No newline at end of file diff --git a/bin/backend/controls/Manager.js b/bin/backend/controls/Manager.js index 13d3c8c..57b4b97 100644 --- a/bin/backend/controls/Manager.js +++ b/bin/backend/controls/Manager.js @@ -331,13 +331,15 @@ define('package/quiqqer/coupons/bin/backend/controls/Manager', [ * @param {Object} [CouponData] - If omitted create new CouponCode */ $showDetails: function (CouponData) { - var self = this; + let Form; + let DiscountSelect; CouponData = CouponData || false; - var FuncSubmit = function () { - var Content = Popup.getContent(); - var Form = Content.getElement('form'); + var FuncSubmit = () => { + if (!Form.reportValidity()) { + return; + } Popup.Loader.show(); @@ -348,21 +350,48 @@ define('package/quiqqer/coupons/bin/backend/controls/Manager', [ return; } - self.refresh(); + this.refresh(); Popup.close(); }); return; } - CouponCodes.create(QUIFormUtils.getFormData(Form)).then(function (couponCodeId) { + CouponCodes.create(QUIFormUtils.getFormData(Form)).then((couponCodeId) => { if (!couponCodeId) { Popup.Loader.hide(); return; } - self.refresh(); - Popup.close(); + require([ + 'package/quiqqer/translator/bin/classes/Translator' + ], (Translator) => { + new Translator().publish(lg).then(() => { + this.refresh(); + Popup.close(); + }).then(() => { + this.refresh(); + Popup.close(); + }); + }); + }).catch(() => { + Popup.Loader.hide(); + }); + }; + + var openDiscount = () => { + Popup.Loader.show(); + + require([ + 'package/quiqqer/discount/bin/controls/Discounts', + 'utils/Panels' + ], function (DiscountsManagerPanel, PanelUtils) { + const Panel = new DiscountsManagerPanel(); + + PanelUtils.openPanelInTasks(Panel).then((PanelOpened) => { + PanelOpened.editChild(parseInt(DiscountSelect.getValue())); + Popup.close(); + }); }); }; @@ -374,22 +403,20 @@ define('package/quiqqer/coupons/bin/backend/controls/Manager', [ title : CouponData ? QUILocale.get(lg, 'controls.manager.details.popup.title_edit', {code: CouponData.code}) : QUILocale.get(lg, 'controls.manager.details.popup.title_new'), - maxHeight : 335, - maxWidth : 450, + maxHeight : 1000, + maxWidth : 1000, events : { - onOpen: function () { - var Content = Popup.getContent(); - var Form = Content.getElement('form'); - var SubmitBtn = Popup.getButton('submit'); + onOpen: (Win) => { + const Content = Popup.getContent(); + Form = Content.getElement('form'); Form.addEvent('submit', function (event) { event.stop(); FuncSubmit(); }); - var Settings = Content.getElement('.quiqqer-coupons-manager-create-settings'); - var Amount = Content.getElement('input[name="amount"]'); - var Code = Content.getElement('input[name="code"]'); + const Amount = Content.getElement('input[name="amount"]'); + const Code = Content.getElement('input[name="code"]'); Code.addEvent('keyup', function () { if (Code.value !== '') { @@ -400,97 +427,93 @@ define('package/quiqqer/coupons/bin/backend/controls/Manager', [ } }); - new QUIButton({ - textimage: 'fa fa-cogs', - text : QUILocale.get(lg, 'controls.Manager.create.settings_btn.show'), - events : { - onClick: function (Btn) { - if (Settings.getStyle('display') === 'none') { - Settings.setStyle('display', 'block'); - - Btn.setAttribute( - 'text', - QUILocale.get(lg, 'controls.Manager.create.settings_btn.hide') - ); - - Content.getElement('input[name="title"]').focus(); - - Popup.setAttribute('maxHeight', 850); - Popup.resize(); - } else { - Settings.setStyle('display', 'none'); - - Btn.setAttribute( - 'text', - QUILocale.get(lg, 'controls.Manager.create.settings_btn.show') - ); - - Popup.setAttribute('maxHeight', 335); - Popup.resize(); - } - } - } - }).inject(Content.getElement( - '.quiqqer-coupons-manager-create-settings-btn' - )); + let EditDiscountBtn = false; if (CouponData) { CouponData.discountIds = CouponData.discountIds.join(','); QUIFormUtils.setDataToForm(CouponData, Form); + + EditDiscountBtn = new QUIButton({ + 'class' : 'optional', + textimage: 'fa fa-percent', + text : QUILocale.get(lg, 'controls.manager.details.popup.btn.open_discount'), + title : QUILocale.get(lg, 'controls.manager.details.popup.btn.open_discount'), + events : { + onClick: openDiscount + }, + styles : { + float: 'right' + } + }).inject( + Content.getElement('.quiqqer-coupons-manager-create-advanced-discount-buttons') + ); } - Popup.Loader.show(); + Win.Loader.show(); + + QUI.parse(Content).then(() => { + Win.Loader.hide(); - QUI.parse(Content).then(function () { - var DiscountSelect = QUI.Controls.getById( + if (!CouponData) { + return; + } + + DiscountSelect = QUI.Controls.getById( Content.getElement('input[name="discountIds"]').get('data-quiid') ); DiscountSelect.addEvents({ - onChange: function () { - if (DiscountSelect.getValue() === '') { - SubmitBtn.disable(); - } else { - SubmitBtn.enable(); - } + onAddItem : () => { + EditDiscountBtn.enable(); + }, + onRemoveItem: () => { + EditDiscountBtn.disable(); } }); - - Popup.Loader.hide(); }); } }, closeButton: true, content : Mustache.render(templateCreate, { - labelTitle : QUILocale.get(lg, lgPrefix + 'labelTitle'), - labelCode : QUILocale.get(lg, lgPrefix + 'labelCode'), - labelUsers : QUILocale.get(lg, lgPrefix + 'labelUsers'), - labelGroups : QUILocale.get(lg, lgPrefix + 'labelGroups'), - labelDate : QUILocale.get(lg, lgPrefix + 'labelDate'), - labelAmount : QUILocale.get(lg, lgPrefix + 'labelAmount'), - labelReusable : QUILocale.get(lg, lgPrefix + 'labelReusable'), - labelReusableOncePerUser: QUILocale.get(lg, lgPrefix + 'labelReusableOncePerUser'), - labelReusableOnce : QUILocale.get(lg, lgPrefix + 'labelReusableOnce'), - labelReusableUnlimited : QUILocale.get(lg, lgPrefix + 'labelReusableUnlimited'), - labelDiscount : QUILocale.get(lg, lgPrefix + 'labelDiscount') + labelTitle : QUILocale.get(lg, lgPrefix + 'labelTitle'), + labelCode : QUILocale.get(lg, lgPrefix + 'labelCode'), + labelUsers : QUILocale.get(lg, lgPrefix + 'labelUsers'), + labelGroups : QUILocale.get(lg, lgPrefix + 'labelGroups'), + labelDate : QUILocale.get(lg, lgPrefix + 'labelDate'), + labelAmount : QUILocale.get(lg, lgPrefix + 'labelAmount'), + labelReusable : QUILocale.get(lg, lgPrefix + 'labelReusable'), + labelReusableOncePerUser : QUILocale.get(lg, lgPrefix + 'labelReusableOncePerUser'), + labelReusableOnce : QUILocale.get(lg, lgPrefix + 'labelReusableOnce'), + labelReusableUnlimited : QUILocale.get(lg, lgPrefix + 'labelReusableUnlimited'), + labelDiscount : QUILocale.get(lg, lgPrefix + 'labelDiscount'), + headerBasics : QUILocale.get(lg, lgPrefix + 'headerBasics'), + headerDiscount : QUILocale.get(lg, lgPrefix + 'headerDiscount'), + headerRestrictions : QUILocale.get(lg, lgPrefix + 'headerRestrictions'), + descCode : QUILocale.get(lg, lgPrefix + 'descCode'), + descTitle : QUILocale.get(lg, lgPrefix + 'descTitle'), + labelDiscountAmount : QUILocale.get(lg, lgPrefix + 'labelDiscountAmount'), + labelDiscountType : QUILocale.get(lg, lgPrefix + 'labelDiscountType'), + labelDiscountTypeFlat : QUILocale.get(lg, lgPrefix + 'labelDiscountTypeFlat'), + labelDiscountTypePercentage: QUILocale.get(lg, lgPrefix + 'labelDiscountTypePercentage'), + + isCreate: !CouponData }) }); Popup.open(); Popup.addButton(new QUIButton({ - name : 'submit', - disabled: true, - text : CouponData ? + name : 'submit', + text : CouponData ? QUILocale.get(lg, 'controls.manager.details.popup.btn.confirm_text_edit') : QUILocale.get(lg, 'controls.manager.details.popup.btn.confirm_text_new'), - alt : CouponData ? + alt : CouponData ? QUILocale.get(lg, 'controls.manager.details.popup.btn.confirm_edit') : QUILocale.get(lg, 'controls.manager.details.popup.btn.confirm_new'), - title : CouponData ? + title : CouponData ? QUILocale.get(lg, 'controls.manager.details.popup.btn.confirm_edit') : QUILocale.get(lg, 'controls.manager.details.popup.btn.confirm_new'), - events : { + events: { onClick: FuncSubmit } })); diff --git a/composer.json b/composer.json index 24a15dd..9a9e1f9 100644 --- a/composer.json +++ b/composer.json @@ -16,7 +16,8 @@ }, "require": { "quiqqer\/quiqqer": "^1.2|*@dev", - "quiqqer\/discount": "^1|*@dev" + "quiqqer\/discount": "^1|*@dev", + "quiqqer/products": "^1.3|*@dev" }, "autoload": { "psr-4": { diff --git a/database.xml b/database.xml index 7d721f3..717ebb7 100644 --- a/database.xml +++ b/database.xml @@ -2,6 +2,10 @@ <database> <global> + <table name="products"> + <field type="BIGINT(20) NULL DEFAULT NULL">couponDiscountId</field> + </table> + <table name="quiqqer_coupons"> <field type="BIGINT(20) NOT NULL AUTO_INCREMENT PRIMARY KEY">id</field> <field type="MEDIUMTEXT NULL">discountIds</field> diff --git a/events.xml b/events.xml index 40d75ed..0af8a32 100644 --- a/events.xml +++ b/events.xml @@ -1,5 +1,13 @@ <?xml version="1.0" encoding="UTF-8"?> <events> + <event on="onPackageSetup-quiqqer/coupons" + fire="\QUI\ERP\Coupons\Events::onPackageSetup" + /> + + <event on="onQuiqqerProductsProductCreate" + fire="\QUI\ERP\Coupons\Events::onQuiqqerProductsProductCreate" + /> + <event on="onQuiqqer::order::orderProcessBasketEnd" fire="\QUI\ERP\Coupons\Events::templateOrderProcessBasketEnd" /> diff --git a/locale.xml b/locale.xml index bdffa68..5ac287b 100644 --- a/locale.xml +++ b/locale.xml @@ -80,9 +80,34 @@ <de><![CDATA[Gutschein-Code: [code]]]></de> <en><![CDATA[Coupon code: [code]]]></en> </locale> + + <locale name="fieldCategory.quiqqer-coupons-fields"> + <de><![CDATA[Gutschein-Einstellungen]]></de> + <en><![CDATA[Coupn code settings]]></en> + </locale> + </groups> <groups name="quiqqer/coupons" datatype="php"> + + <locale name="Discount.default_title"> + <de><![CDATA[Coupon-Code [couponCode]]]></de> + <en><![CDATA[Coupon code [couponCode]]]></en> + </locale> + <locale name="Discount.default_title.product"> + <de><![CDATA[Produkt "[productTitle]" (#[productId])]]></de> + <en><![CDATA[Product "[productTitle]" (#[productId])]]></en> + </locale> + + <locale name="exception.ajax.create.missing_field"> + <de><![CDATA[Bitte fülle alle Pflichtfelder aus.]]></de> + <en><![CDATA[Please fill out all required fields.]]></en> + </locale> + <locale name="product_type.CouponProductType.title"> + <de><![CDATA[Gutschein]]></de> + <en><![CDATA[Coupon]]></en> + </locale> + <!-- Ajax --> <locale name="message.ajax.general_error"> <de> @@ -181,6 +206,10 @@ <de><![CDATA[Verwendungszweck (optional)]]></de> <en><![CDATA[Usage (optional)]]></en> </locale> + <locale name="controls.manager.create.template.descTitle"> + <de><![CDATA[Dient der <b>internen</b> Beschreibung des Verwendungszwecks für diesen Gutschein-Code. Kunden sehen diese Beschreibung <b>nicht</b>.]]></de> + <en><![CDATA[Used for <b>internal</b> description of the purpose for this coupon code. Customers will <b>not</b> see this description.]]></en> + </locale> <locale name="controls.manager.create.template.labelAmount"> <de><![CDATA[Anzahl generierte Gutschein-Codes]]></de> <en><![CDATA[Amount of generated coupon codes]]></en> @@ -194,8 +223,12 @@ <en><![CDATA[Restrict to the following groups (optional)]]></en> </locale> <locale name="controls.manager.create.template.labelCode"> - <de><![CDATA[Gutschein-Code (optional - wird automatisch generiert, wenn Feld leergelassen wird)]]></de> - <en><![CDATA[Coupon code (optional - is generated automatically if left empty)]]></en> + <de><![CDATA[Gutschein-Code (optional)]]></de> + <en><![CDATA[Coupon code (optional)]]></en> + </locale> + <locale name="controls.manager.create.template.descCode"> + <de><![CDATA[Wird dieses Feld leergelassen, wird automatisch ein zufälliger neuer Gutschein-Code generiert.]]></de> + <en><![CDATA[If this field is left blank, a random new coupon code will be generated automatically.]]></en> </locale> <locale name="controls.manager.create.template.labelReusable"> <de><![CDATA[Erlaubte Verwendung]]></de> @@ -209,6 +242,22 @@ <de><![CDATA[Insgesamt nur einmal einlösbar]]></de> <en><![CDATA[Only redeemable once in total]]></en> </locale> + <locale name="controls.manager.create.template.labelDiscountAmount"> + <de><![CDATA[Rabatt-Betrag]]></de> + <en><![CDATA[Discount amount]]></en> + </locale> + <locale name="controls.manager.create.template.labelDiscountType"> + <de><![CDATA[Rabatt-Typ]]></de> + <en><![CDATA[Discount type]]></en> + </locale> + <locale name="controls.manager.create.template.labelDiscountTypeFlat"> + <de><![CDATA[Festbetrag]]></de> + <en><![CDATA[Fixed amount]]></en> + </locale> + <locale name="controls.manager.create.template.labelDiscountTypePercentage"> + <de><![CDATA[Prozentual]]></de> + <en><![CDATA[Percentage]]></en> + </locale> <locale name="controls.manager.create.template.labelReusableUnlimited"> <de><![CDATA[Unbegrenzt einlösbar]]></de> <en><![CDATA[Unlimited redeemable]]></en> @@ -217,6 +266,18 @@ <de><![CDATA[Verknüpfter Rabatt]]></de> <en><![CDATA[Linked discount]]></en> </locale> + <locale name="controls.manager.create.template.headerBasics"> + <de><![CDATA[Grundeinstellungen]]></de> + <en><![CDATA[Basic settings]]></en> + </locale> + <locale name="controls.manager.create.template.headerDiscount"> + <de><![CDATA[Ermäßigung]]></de> + <en><![CDATA[Discount]]></en> + </locale> + <locale name="controls.manager.create.template.headerRestrictions"> + <de><![CDATA[Nutzungseinschränkungen]]></de> + <en><![CDATA[Usage restrictions]]></en> + </locale> <locale name="controls.manager.details.popup.title_new"> <de><![CDATA[Neuen Gutschein-Code erstellen]]></de> <en><![CDATA[Create new Coupon code]]></en> @@ -241,6 +302,10 @@ <de><![CDATA[Hier klicken, um die Änderungen am Gutschein-Code zu speichern]]></de> <en><![CDATA[Click here to save the changes made to the Coupon code]]></en> </locale> + <locale name="controls.manager.details.popup.btn.open_discount"> + <de><![CDATA[Rabatt konfigurieren]]></de> + <en><![CDATA[Configure discount]]></en> + </locale> <locale name="controls.manager.tbl.btn.create"> <de><![CDATA[Erstellen]]></de> <en><![CDATA[Create]]></en> diff --git a/package.xml b/package.xml index fd311af..95e98ac 100644 --- a/package.xml +++ b/package.xml @@ -23,5 +23,9 @@ <license><![CDATA[GPL-3.0+]]></license> </copyright> + <provider> + <productType src="\QUI\ERP\Coupons\Products\CouponProductType"/> + </provider> + </package> </quiqqer> \ No newline at end of file diff --git a/products.xml b/products.xml new file mode 100644 index 0000000..e21d978 --- /dev/null +++ b/products.xml @@ -0,0 +1,17 @@ +<quiqqer> + <products> + <fieldCategories> + <fieldCategory name="quiqqer-coupons-fields"> + <title> + <locale group="quiqqer/coupons" var="fieldCategory.quiqqer-coupons-fields"/> + </title> + <icon>fa fa-credit-card</icon> + <fields> + <field>670</field> + <field>671</field> + <field>672</field> + </fields> + </fieldCategory> + </fieldCategories> + </products> +</quiqqer> \ No newline at end of file diff --git a/src/QUI/ERP/Coupons/Events.php b/src/QUI/ERP/Coupons/Events.php index 655b98b..cb1b43e 100644 --- a/src/QUI/ERP/Coupons/Events.php +++ b/src/QUI/ERP/Coupons/Events.php @@ -3,11 +3,17 @@ namespace QUI\ERP\Coupons; use QUI; +use QUI\ERP\Order\AbstractOrder; +use QUI\ERP\Products\Handler\Fields; +use QUI\ERP\Products\Interfaces\ProductInterface; use Quiqqer\Engine\Collector; use QUI\ERP\Order\Basket\Basket; use QUI\ERP\Order\Basket\BasketGuest; use QUI\ERP\Coupons\Handler as CouponsHandler; use QUI\ERP\Discount\EventHandling as DiscountEvents; +use QUI\ERP\Coupons\Products\CouponProductType; +use QUI\ERP\Coupons\Products\Handler as CouponProductsHandler; +use QUI\ERP\Products\Handler\Products; /** * Class Events @@ -16,6 +22,21 @@ */ class Events { + /** + * quiqqer/quiqqer: onPackageSetup + * + * @param QUI\Package\Package $Package + * @return void + */ + public static function onPackageSetup(QUI\Package\Package $Package) + { + try { + self::createProductFields(); + } catch (\Exception $Exception) { + QUI\System\Log::writeException($Exception); + } + } + /** * Template event quiqqer/order: onQuiqqer::order::orderProcessBasketEnd * @@ -390,4 +411,166 @@ protected static function addCouponToOrder($Order, $coupon) } catch (\Exception $Exception) { } } + + /** + * Create all fixed product fields that quiqqer/stock-management provides + * + * @return void + * @throws QUI\Exception + */ + protected static function createProductFields() + { + $fields = [ + CouponProductType::PRODUCT_FIELD_ID_TRANSFERABLE => [ + 'title' => [ + 'de' => 'Gutschein-Code ist übertragbar', + 'en' => 'Coupon code is transferable' + ], + 'type' => Fields::TYPE_BOOL, + 'public' => false, + 'standard' => false + ], + CouponProductType::PRODUCT_FIELD_ID_GENERATE_PDF => [ + 'title' => [ + 'de' => 'Gutschein-Code als PDF bereitstellen', + 'en' => 'Provice coupon code as PDF' + ], + 'type' => Fields::TYPE_BOOL, + 'public' => false, + 'standard' => false + ], + CouponProductType::PRODUCT_FIELD_ID_COUPON_AMOUNT => [ + 'title' => [ + 'de' => 'Gutschein-Code Wert', + 'en' => 'Coupon code amount' + ], + 'type' => Fields::TYPE_FLOAT, + 'public' => false, + 'standard' => false + ] + ]; + + $fieldsCreated = false; + + foreach ($fields as $fieldId => $field) { + try { + Fields::getField($fieldId); + continue; + } catch (\Exception $Exception) { + // Field does not exist -> create it + } + + try { + Fields::createField([ + 'id' => $fieldId, + 'type' => $field['type'], + 'titles' => $field['title'], + 'workingtitles' => $field['title'], + 'systemField' => 0, + 'standardField' => !empty($field['standard']) ? 1 : 0, + 'publicField' => !empty($field['public']) ? 1 : 0, + 'options' => !empty($field['options']) ? $field['options'] : null + ]); + } catch (\Exception $Exception) { + QUI\System\Log::writeException($Exception); + continue; + } + + $fieldsCreated = true; + } + + if ($fieldsCreated) { + QUI\Translator::publish('quiqqer/products'); + } + } + + /** + * Assign plan product fields to a product + * + * @param ProductInterface $Product + * @return void + */ + public static function onQuiqqerProductsProductCreate(ProductInterface $Product) + { + if (!($Product instanceof CouponProductType)) { + return; + } + + $fields = [ + CouponProductType::PRODUCT_FIELD_ID_TRANSFERABLE => true, + CouponProductType::PRODUCT_FIELD_ID_GENERATE_PDF => true, + CouponProductType::PRODUCT_FIELD_ID_COUPON_AMOUNT => null + ]; + + foreach ($fields as $fieldId => $value) { + try { + $Field = Fields::getField($fieldId); + $Field->setValue($value); + + $Product->addOwnField($Field); + } catch (\Exception $Exception) { + QUI\System\Log::writeException($Exception); + } + } + + try { + $Product->update(QUI::getUsers()->getSystemUser()); + } catch (\Exception $Exception) { + QUI\System\Log::writeException($Exception); + } + } + + /** + * quiqqer/products: onQuiqqerProductsProductSave + * + * Create or update a discount that is associated with a product + * + * @param QUI\ERP\Products\Product\Product $Product + */ + public static function onQuiqqerProductsProductSave(QUI\ERP\Products\Product\Product $Product) + { + if ($Product instanceof CouponProductType) { + try { + CouponProductsHandler::updateProductDiscount($Product); + } catch (\Exception $Exception) { + QUI\System\Log::writeException($Exception); + } + } + } + + /** + * quiqqer/order: onQuiqqerOrderCreated + * + * Parse coupon attributes from order and create coupon codes for the buyer. + * + * @param AbstractOrder $Order + * @return void + */ + public static function onQuiqqerOrderCreated(AbstractOrder $Order) + { + /** @var QUI\ERP\Accounting\Article $Article */ + foreach ($Order->getArticles() as $Article) { + try { + // Do not parse coupon codes / discounts + if (empty($Article->getId()) || !\is_numeric($Article->getId())) { + continue; + } + + $Product = Products::getProduct($Article->getId()); + + // Only parse download products + if (!($Product->getType() instanceof CouponProductType)) { + continue; + } + + CouponProductsHandler::createCouponCodeFromProduct($Product); + } catch (\Exception $Exception) { + if ($Exception->getCode() === 404) { + QUI\System\Log::writeDebugException($Exception); + } else { + QUI\System\Log::writeException($Exception); + } + } + } + } } diff --git a/src/QUI/ERP/Coupons/Products/CouponProductType.php b/src/QUI/ERP/Coupons/Products/CouponProductType.php new file mode 100644 index 0000000..88aa1eb --- /dev/null +++ b/src/QUI/ERP/Coupons/Products/CouponProductType.php @@ -0,0 +1,47 @@ +<?php + +namespace QUI\ERP\Coupons\Products; + +use QUI; +use QUI\ERP\Products\Product\Types\DigitalProduct; + +/** + * Class DigitalProduct + * + * Represents a non-physical product that does not require shipping. + */ +class CouponProductType extends DigitalProduct +{ + /** + * Special fields for coupon products + */ + const PRODUCT_FIELD_ID_TRANSFERABLE = 670; + const PRODUCT_FIELD_ID_GENERATE_PDF = 671; + const PRODUCT_FIELD_ID_COUPON_AMOUNT = 672; + + /** + * @param QUI\Locale $Locale + * @return string + */ + public static function getTypeTitle($Locale = null) + { + if ($Locale === null) { + $Locale = QUI::getLocale(); + } + + return $Locale->get('quiqqer/coupons', 'product_type.CouponProductType.title'); + } + + /** + * @param QUI\Locale $Locale + * @return string + */ + public static function getTypeDescription($Locale = null) + { + if ($Locale === null) { + $Locale = QUI::getLocale(); + } + + return $Locale->get('quiqqer/coupons', 'product_type.CouponProductType.description'); + } +} diff --git a/src/QUI/ERP/Coupons/Products/Handler.php b/src/QUI/ERP/Coupons/Products/Handler.php new file mode 100644 index 0000000..9787329 --- /dev/null +++ b/src/QUI/ERP/Coupons/Products/Handler.php @@ -0,0 +1,143 @@ +<?php + +namespace QUI\ERP\Coupons\Products; + +use QUI; +use QUI\ERP\Order\AbstractOrder; +use QUI\ERP\Products\Product\Product; +use QUI\ERP\Discount\Discount; +use QUI\ERP\Discount\Handler as DiscountHandler; + +/** + * Class Handler + * + * Handles coupon code generation from orders / products. + */ +class Handler +{ + /** + * Create coupon codes from an order + * + * @param Product $Product + */ + public static function createCouponCodeFromProduct(Product $Product) + { + + } + + /** + * Create and/or updates the discount that is associated with a coupon code product. + * + * @param Product $Product + * @return void + * + * @throws QUI\Exception + */ + public static function updateProductDiscount(Product $Product): void + { + $Discount = self::getProductDiscount($Product); + + if (!$Discount) { + $Discount = self::createProductDiscount($Product); + } + + $discountAmount = $Product->getFieldValue(CouponProductType::PRODUCT_FIELD_ID_COUPON_AMOUNT); + + if (empty($discountAmount)) { + $discountAmount = $Product->getPrice()->getValue(); + } + + $Discount->setAttributes([ + 'discount' => $discountAmount + ]); + + $Discount->update(); + } + + /** + * Creates a new discount for all coupon codes generated by the product. + * + * @param Product $Product + * @return Discount + * + * @throws QUI\Exception + */ + protected static function createProductDiscount(Product $Product): Discount + { + $Handler = DiscountHandler::getInstance(); + + $discountAmount = $Product->getFieldValue(CouponProductType::PRODUCT_FIELD_ID_COUPON_AMOUNT); + + if (empty($discountAmount)) { + $discountAmount = $Product->getPrice()->getValue(); + } + + /** @var Discount $NewDiscount */ + $NewDiscount = $Handler->createChild([ + 'active' => 1, + 'discount' => $discountAmount, + 'discount_type' => DiscountHandler::DISCOUNT_TYPE_CURRENCY + ]); + + $L = new QUI\Locale(); + $discountTitle = []; + + foreach (QUI::availableLanguages() as $lang) { + $L->setCurrent($lang); + + $discountTitle[$lang] = $L->get('quiqqer/coupons', 'Discount.default_title.product', [ + 'productId' => $Product->getId(), + 'productTitle' => $Product->getTitle($L) + ]); + } + + \QUI\Translator::update( + 'quiqqer/discount', + 'discount.'.$NewDiscount->getId().'.title', + 'quiqqer/discount', + $discountTitle + ); + + \QUI\Translator::publish('quiqqer/discount'); + + QUI::getDataBase()->update( + QUI\ERP\Products\Utils\Tables::getProductTableName(), + [ + 'couponDiscountId' => $NewDiscount->getId() + ], + [ + 'id' => $Product->getId() + ] + ); + + return $NewDiscount; + } + + /** + * Get the discount that is associated with a coupon code product. + * + * @param Product $Product + * @return Discount|false - Discount or false if no discount created yet + * + * @throws QUI\Exception + */ + public static function getProductDiscount(Product $Product) + { + $result = QUI::getDataBase()->fetch([ + 'select' => ['couponDiscountId'], + 'from' => QUI\ERP\Products\Utils\Tables::getProductTableName(), + 'where' => [ + 'id' => $Product->getId() + ] + ]); + + if (empty($result)) { + return false; + } + + /** @var Discount $Discount */ + $Discount = DiscountHandler::getInstance()->getChild($result[0]['couponDiscountId']); + + return $Discount; + } +} -- GitLab