Show shopping informations in part details

This commit is contained in:
Jan Böhmer 2019-08-02 12:17:56 +02:00
parent 855eace81d
commit c2b4d100f0
11 changed files with 250 additions and 44 deletions

View file

@ -5,10 +5,12 @@
"php": "^7.1.3", "php": "^7.1.3",
"ext-ctype": "*", "ext-ctype": "*",
"ext-iconv": "*", "ext-iconv": "*",
"ext-mbstring": "*",
"ext-intl": "*", "ext-intl": "*",
"ext-mbstring": "*",
"doctrine/annotations": "^1.6", "doctrine/annotations": "^1.6",
"friendsofsymfony/ckeditor-bundle": "^2.0", "friendsofsymfony/ckeditor-bundle": "^2.0",
"gerardojbaez/money": "^0.3.1",
"ocramius/proxy-manager": "2.1.*",
"omines/datatables-bundle": "^0.2.2", "omines/datatables-bundle": "^0.2.2",
"php-translation/symfony-bundle": "^0.8.1", "php-translation/symfony-bundle": "^0.8.1",
"s9e/text-formatter": "^2.0", "s9e/text-formatter": "^2.0",
@ -35,8 +37,7 @@
"symfony/webpack-encore-bundle": "^1.1", "symfony/webpack-encore-bundle": "^1.1",
"symfony/yaml": "4.3.*", "symfony/yaml": "4.3.*",
"twig/extensions": "^1.5", "twig/extensions": "^1.5",
"webmozart/assert": "^1.4", "webmozart/assert": "^1.4"
"ocramius/proxy-manager": "2.1.*"
}, },
"require-dev": { "require-dev": {
"roave/security-advisories": "dev-master", "roave/security-advisories": "dev-master",

54
composer.lock generated
View file

@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically" "This file is @generated automatically"
], ],
"content-hash": "1a794cf54f952c4bf1bc00d8f952b6a3", "content-hash": "c4b42f65037fd3b81886b8d9c98deb32",
"packages": [ "packages": [
{ {
"name": "doctrine/annotations", "name": "doctrine/annotations",
@ -1397,6 +1397,54 @@
], ],
"time": "2019-04-15T16:29:43+00:00" "time": "2019-04-15T16:29:43+00:00"
}, },
{
"name": "gerardojbaez/money",
"version": "v0.3.1",
"source": {
"type": "git",
"url": "https://github.com/gerardojbaez/money.git",
"reference": "1a29ca19899fad8ae559e9f2c982815ea21f8a6c"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/gerardojbaez/money/zipball/1a29ca19899fad8ae559e9f2c982815ea21f8a6c",
"reference": "1a29ca19899fad8ae559e9f2c982815ea21f8a6c",
"shasum": ""
},
"require": {
"php": ">=5.5.9"
},
"require-dev": {
"phpunit/phpunit": "5.4.*"
},
"type": "library",
"autoload": {
"psr-4": {
"Gerardojbaez\\Money\\": "src/"
},
"files": [
"src/helpers.php"
]
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"authors": [
{
"name": "Gerardo Báez",
"email": "gerardojbaez@gmail.com"
}
],
"description": "A simple and cross-platform alternative to PHP money_format(). 91 currencies supported, including INR.",
"keywords": [
"currency",
"formatter",
"money",
"money_format"
],
"time": "2018-02-24T18:59:00+00:00"
},
{ {
"name": "jdorn/sql-formatter", "name": "jdorn/sql-formatter",
"version": "v1.2.17", "version": "v1.2.17",
@ -7486,8 +7534,8 @@
"php": "^7.1.3", "php": "^7.1.3",
"ext-ctype": "*", "ext-ctype": "*",
"ext-iconv": "*", "ext-iconv": "*",
"ext-mbstring": "*", "ext-intl": "*",
"ext-intl": "*" "ext-mbstring": "*"
}, },
"platform-dev": [] "platform-dev": []
} }

View file

@ -9,6 +9,7 @@ parameters:
partdb_title: 'Part-DB' # The title shown inside of Part-DB (e.g. in the navbar and on homepage) partdb_title: 'Part-DB' # The title shown inside of Part-DB (e.g. in the navbar and on homepage)
banner: '' # The info text shown in the homepage banner: '' # The info text shown in the homepage
use_gravatar: true # Set to false, if no Gravatar images should be used for user profiles. use_gravatar: true # Set to false, if no Gravatar images should be used for user profiles.
default_currency: 'EUR' # The currency that should be used
services: services:
# default configuration for services in *this* file # default configuration for services in *this* file

View file

@ -33,6 +33,8 @@ declare(strict_types=1);
namespace App\Entity; namespace App\Entity;
use Doctrine\ORM\Mapping as ORM; use Doctrine\ORM\Mapping as ORM;
use Doctrine\ORM\PersistentCollection;
use Exception;
/** /**
* Class Orderdetail. * Class Orderdetail.
@ -79,6 +81,12 @@ class Orderdetail extends DBElement
*/ */
protected $supplier_product_url; protected $supplier_product_url;
/**
* @var \DateTime The date when this element was created.
* @ORM\Column(type="datetimetz", name="datetime_added")
*/
protected $addedDate;
/** /**
* Returns the ID as an string, defined by the element class. * Returns the ID as an string, defined by the element class.
* This should have a form like P000014, for a part with ID 14. * This should have a form like P000014, for a part with ID 14.
@ -142,6 +150,17 @@ class Orderdetail extends DBElement
return (bool) $this->obsolete; return (bool) $this->obsolete;
} }
/**
* Returns the date/time when the element was created.
* Returns null if the element was not yet saved to DB yet.
*
* @return \DateTime|null The creation time of the part.
*/
public function getAddedDate(): ?\DateTime
{
return $this->addedDate;
}
/** /**
* Get the link to the website of the article on the suppliers website. * Get the link to the website of the article on the suppliers website.
* *
@ -152,8 +171,8 @@ class Orderdetail extends DBElement
*/ */
public function getSupplierProductUrl(bool $no_automatic_url = false): string public function getSupplierProductUrl(bool $no_automatic_url = false): string
{ {
if ($no_automatic_url || '' !== $this->supplierpartnr) { if ($no_automatic_url || '' !== $this->supplier_product_url) {
return $this->supplierpartnr; return $this->supplier_product_url;
} }
return $this->getSupplier()->getAutoProductUrl($this->supplierpartnr); // maybe an automatic url is available... return $this->getSupplier()->getAutoProductUrl($this->supplierpartnr); // maybe an automatic url is available...
@ -162,12 +181,12 @@ class Orderdetail extends DBElement
/** /**
* Get all pricedetails. * Get all pricedetails.
* *
* @return Pricedetails[] all pricedetails as a one-dimensional array of Pricedetails objects, * @return Pricedetail[] all pricedetails as a one-dimensional array of Pricedetails objects,
* sorted by minimum discount quantity * sorted by minimum discount quantity
* *
* @throws Exception if there was an error * @throws Exception if there was an error
*/ */
public function getPricedetails(): array public function getPricedetails(): PersistentCollection
{ {
return $this->pricedetails; return $this->pricedetails;
} }

View file

@ -92,7 +92,7 @@ class Part extends AttachmentContainingDBElement
protected $master_picture_attachment; protected $master_picture_attachment;
/** /**
* @var * @var Orderdetail[]
* @ORM\OneToMany(targetEntity="Orderdetail", mappedBy="part") * @ORM\OneToMany(targetEntity="Orderdetail", mappedBy="part")
* *
* @ColumnSecurity(prefix="orderdetails", type="object") * @ColumnSecurity(prefix="orderdetails", type="object")
@ -499,7 +499,7 @@ class Part extends AttachmentContainingDBElement
* *
* @param bool $hide_obsolete If true, obsolete orderdetails will NOT be returned * @param bool $hide_obsolete If true, obsolete orderdetails will NOT be returned
* *
* @return Orderdetails[] * all orderdetails as a one-dimensional array of Orderdetails objects * @return Orderdetail[] * all orderdetails as a one-dimensional array of Orderdetails objects
* (empty array if there are no ones) * (empty array if there are no ones)
* * the array is sorted by the suppliers names / minimum order quantity * * the array is sorted by the suppliers names / minimum order quantity
* *

View file

@ -95,31 +95,27 @@ class Pricedetail extends DBElement
return $this->orderdetail; return $this->orderdetail;
} }
public function getPrice() : float
{
return (float) $this->price;
}
/** /**
* Get the price. * Get the price for a single unit.
* *
* @param bool $as_money_string * if true, this method returns a money string incl. currency
* * if false, this method returns the price as float
* @param int $multiplier The returned price (float or string) will be multiplied * @param int $multiplier The returned price (float or string) will be multiplied
* with this multiplier. * with this multiplier.
* *
* You will get the price for $multiplier parts. If you want the price which is stored * You will get the price for $multiplier parts. If you want the price which is stored
* in the database, you have to pass the "price_related_quantity" count as $multiplier. * in the database, you have to pass the "price_related_quantity" count as $multiplier.
* *
* @return float the price as a float number (if "$as_money_string == false") * @return float the price as a float number
* @return string the price as a string incl. currency (if "$as_money_string == true")
*
* @see floatToMoneyString()
*/ */
public function getPrice(bool $as_money_string = false, int $multiplier = 1) public function getPricePerUnit(int $multiplier = 1) : float
{ {
$price = ($this->price * $multiplier) / $this->price_related_quantity; $price = ($this->price * $multiplier) / $this->price_related_quantity;
if ($as_money_string) {
throw new \Exception('Not implemented yet...');
//return floatToMoneyString($price);
}
return $price; return $price;
} }

View file

@ -0,0 +1,59 @@
<?php
/**
*
* part-db version 0.1
* Copyright (C) 2005 Christoph Lechner
* http://www.cl-projects.de/
*
* part-db version 0.2+
* Copyright (C) 2009 K. Jacobs and others (see authors.php)
* http://code.google.com/p/part-db/
*
* Part-DB Version 0.4+
* Copyright (C) 2016 - 2019 Jan Böhmer
* https://github.com/jbtronics
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*
*/
namespace App\Services;
use Gerardojbaez\Money\Money;
use Symfony\Component\DependencyInjection\ParameterBag\ContainerBagInterface;
class MoneyFormatter
{
private $params;
public function __construct(ContainerBagInterface $params)
{
$this->params = $params;
}
public function format($amount, string $currency = "") : string
{
if ($currency === "") {
$currency = $this->params->get("default_currency");
}
$money = new Money($amount, $currency);
return $money->format();
}
}

View file

@ -32,6 +32,7 @@ namespace App\Twig;
use App\Entity\Attachment; use App\Entity\Attachment;
use App\Entity\DBElement; use App\Entity\DBElement;
use App\Services\EntityURLGenerator; use App\Services\EntityURLGenerator;
use App\Services\MoneyFormatter;
use App\Services\TreeBuilder; use App\Services\TreeBuilder;
use Symfony\Component\Cache\Adapter\AdapterInterface; use Symfony\Component\Cache\Adapter\AdapterInterface;
use Symfony\Component\Serializer\SerializerInterface; use Symfony\Component\Serializer\SerializerInterface;
@ -47,22 +48,26 @@ class AppExtension extends AbstractExtension
protected $cache; protected $cache;
protected $serializer; protected $serializer;
protected $treeBuilder; protected $treeBuilder;
protected $moneyFormatter;
public function __construct(EntityURLGenerator $entityURLGenerator, AdapterInterface $cache, public function __construct(EntityURLGenerator $entityURLGenerator, AdapterInterface $cache,
SerializerInterface $serializer, TreeBuilder $treeBuilder) SerializerInterface $serializer, TreeBuilder $treeBuilder,
MoneyFormatter $moneyFormatter)
{ {
$this->entityURLGenerator = $entityURLGenerator; $this->entityURLGenerator = $entityURLGenerator;
$this->cache = $cache; $this->cache = $cache;
$this->serializer = $serializer; $this->serializer = $serializer;
$this->treeBuilder = $treeBuilder; $this->treeBuilder = $treeBuilder;
$this->moneyFormatter = $moneyFormatter;
} }
public function getFilters() public function getFilters()
{ {
return [ return [
new TwigFilter('entityURL', [$this, 'generateEntityURL']), new TwigFilter('entityURL', [$this, 'generateEntityURL']),
new TwigFilter('bbCode', [$this, 'parseBBCode'], ['pre_escape' => 'html', 'is_safe' => ['html']]), new TwigFilter('bbCode', [$this, 'parseBBCode'], ['pre_escape' => 'html', 'is_safe' => ['html']]),
]; new TwigFilter('moneyFormat', [$this, 'formatCurrency'])
];
} }
public function getTests() public function getTests()
@ -107,4 +112,9 @@ class AppExtension extends AbstractExtension
return $item->get(); return $item->get();
} }
public function formatCurrency($amount, $currency = "")
{
return $this->moneyFormatter->format($amount, $currency);
}
} }

View file

@ -108,6 +108,9 @@
"./config/packages/fos_ckeditor.yaml" "./config/packages/fos_ckeditor.yaml"
] ]
}, },
"gerardojbaez/money": {
"version": "v0.3.1"
},
"jdorn/sql-formatter": { "jdorn/sql-formatter": {
"version": "v1.2.17" "version": "v1.2.17"
}, },

View file

@ -0,0 +1,67 @@
<div class="table-responsive">
<table class="table table-striped table-header table-hover">
<thead>
<tr>
<th>{% trans %}part.supplier.name{% endtrans %}</th>
<th>{% trans %}part.supplier.partnr{% endtrans %}</th>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
{% for order in part.orderdetails %}
<tr class="{% if order.obsolete %}table-danger{% endif %}">
<td>{{ order.supplier.name }}</td>
<td>{% if order.supplierProductUrl is not empty %}
<a href="{{ order.supplierProductUrl }}" target="_blank" class="link-external">{{ order.supplierPartNr }}</a>
{% else %}
{{ order.supplierPartNr }}
{% endif %}
</td>
<td>
{% if order.pricedetails is not empty %}
<table class="table table-bordered table-sm table-striped table-hover">
<thead class="thead-dark">
<tr>
<th>{% trans %}part.order.minamount{% endtrans %}</th>
<th>{% trans %}part.order.price{% endtrans %}</th>
<th>{% trans %}part.order.single_price{% endtrans %}</th>
</tr>
</thead>
<tbody>
{% for detail in order.pricedetails %}
<tr>
<td>
{{ detail.MinDiscountQuantity }}
</td>
<td>
{{ detail.Price | moneyFormat }} <i>{% trans %}part.order.price_per{% endtrans %}</i> {{ detail.PriceRelatedQuantity }}
</td>
<td>
{{ detail.PricePerUnit | moneyFormat}}
</td>
</tr>
{% endfor %}
</tbody>
</table>
{% endif %}
</td>
<td> {# Action for order information #}
<div class="btn-group">
<button type="button" class="btn btn-info btn-sm dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" data-boundary="window">
Action
</button>
<div class="dropdown-menu">
<span class="text-muted dropdown-item-text" ><i class="fas fa-lightbulb fa-fw"></i> <b>ID:</b> {{ order.iD }}</span>
<span class="text-muted dropdown-item-text" ><i class="fas fa-history fa-fw"></i> <b>{% trans %}createdAt{% endtrans %}:</b> {{ order.addedDate | localizeddate("short")}}</span>
<div class="dropdown-divider"></div>
<a class="dropdown-item" href="#"><i class="fas fa-edit fa-fw"></i> {% trans %}edit.caption_short{% endtrans %}</a>
<a class="dropdown-item" href="#"><i class="fas fa-trash fa-fw"></i> {% trans %}delete.caption{% endtrans %}</a>
</div>
</div>
</td>
</tr>
{% endfor %}
</tbody>
</table>
</div>

View file

@ -112,9 +112,11 @@
<div class="tab-pane fade {% if part.comment is empty %} show active{% endif %}" id="attachments" role="tabpanel" aria-labelledby="profile-tab"> <div class="tab-pane fade {% if part.comment is empty %} show active{% endif %}" id="attachments" role="tabpanel" aria-labelledby="profile-tab">
Test Test
</div> </div>
<div class="tab-pane fade" id="suppliers" role="tabpanel" aria-labelledby="profile-tab">
<div class="tab-pane fade" id="suppliers" role="tabpanel" aria-labelledby="profile-tab">
{% include "Parts/_order_infos.html.twig" %}
</div> </div>
<div class="tab-pane fade" id="history" role="tabpanel" aria-labelledby="profile-tab"> <div class="tab-pane fade" id="history" role="tabpanel" aria-labelledby="profile-tab">
TODO TODO
</div> </div>
@ -128,22 +130,22 @@
{% if is_granted('create', part) %} {% if is_granted('create', part) %}
<br> <br>
<div class="btn-group mt-2"> <div class="btn-group mt-2">
<a class="btn btn-primary" href="{{ part|entityURL('clone') }}"> <a class="btn btn-primary" href="{{ part|entityURL('clone') }}">
<i class="fas fa-clone"></i> <i class="fas fa-clone"></i>
{% trans %}part.clone.btn{% endtrans %} {% trans %}part.clone.btn{% endtrans %}
</a>
<button type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">
<span class="caret"></span>
</button>
<div class="dropdown-menu" role="menu">
<a class="dropdown-item" href="{{ part|entityURL('create') }}">
<i class="fas fa-plus-square"></i>
{% trans %}part.create.btn{% endtrans %}
</a> </a>
<button type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown">
<span class="caret"></span>
</button>
<div class="dropdown-menu" role="menu">
<a class="dropdown-item" href="{{ part|entityURL('create') }}">
<i class="fas fa-plus-square"></i>
{% trans %}part.create.btn{% endtrans %}
</a>
</div>
</div> </div>
</div>
{% endif %} {% endif %}
</div> </div>