ver. 0.276: ShopOrder migration, Integrations cleanup, global admin search
This commit is contained in:
@@ -68,6 +68,159 @@ class SettingsController
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Globalna wyszukiwarka admin (produkty + zamowienia) - AJAX.
|
||||
*/
|
||||
public function globalSearchAjax(): void
|
||||
{
|
||||
global $mdb;
|
||||
|
||||
$phrase = trim((string)\S::get('q'));
|
||||
if ($phrase === '' || mb_strlen($phrase) < 2) {
|
||||
echo json_encode([
|
||||
'status' => 'ok',
|
||||
'items' => [],
|
||||
]);
|
||||
exit;
|
||||
}
|
||||
|
||||
$phrase = mb_substr($phrase, 0, 120);
|
||||
$phraseNormalized = preg_replace('/\s+/', ' ', $phrase);
|
||||
$phraseNormalized = trim((string)$phraseNormalized);
|
||||
$like = '%' . $phrase . '%';
|
||||
$likeNormalized = '%' . $phraseNormalized . '%';
|
||||
|
||||
$items = [];
|
||||
$defaultLang = (string)\front\factory\Languages::default_language();
|
||||
|
||||
try {
|
||||
$productStmt = $mdb->query(
|
||||
'SELECT '
|
||||
. 'p.id, p.ean, p.sku, p.parent_id, psl.name '
|
||||
. 'FROM pp_shop_products AS p '
|
||||
. 'LEFT JOIN pp_shop_products_langs AS psl ON psl.product_id = p.id AND psl.lang_id = :lang_id '
|
||||
. 'WHERE '
|
||||
. '(p.ean LIKE :q1 OR p.sku LIKE :q2 OR psl.name LIKE :q3) '
|
||||
. 'AND p.archive != 1 '
|
||||
. 'ORDER BY p.id DESC '
|
||||
. 'LIMIT 15',
|
||||
[
|
||||
':lang_id' => $defaultLang,
|
||||
':q1' => $like,
|
||||
':q2' => $like,
|
||||
':q3' => $like,
|
||||
]
|
||||
);
|
||||
} catch (\Throwable $e) {
|
||||
$productStmt = false;
|
||||
}
|
||||
|
||||
$productRows = $productStmt ? $productStmt->fetchAll() : [];
|
||||
if (is_array($productRows)) {
|
||||
foreach ($productRows as $row) {
|
||||
$productId = (int)($row['id'] ?? 0);
|
||||
if ($productId <= 0) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$name = trim((string)($row['name'] ?? ''));
|
||||
if ($name === '') {
|
||||
$name = 'Produkt #' . $productId;
|
||||
}
|
||||
|
||||
$meta = [];
|
||||
$sku = trim((string)($row['sku'] ?? ''));
|
||||
$ean = trim((string)($row['ean'] ?? ''));
|
||||
if ($sku !== '') {
|
||||
$meta[] = 'SKU: ' . $sku;
|
||||
}
|
||||
if ($ean !== '') {
|
||||
$meta[] = 'EAN: ' . $ean;
|
||||
}
|
||||
|
||||
$items[] = [
|
||||
'type' => 'product',
|
||||
'title' => $name,
|
||||
'subtitle' => implode(' | ', $meta),
|
||||
'url' => '/admin/shop_product/product_edit/id=' . $productId,
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
try {
|
||||
$orderStmt = $mdb->query(
|
||||
'SELECT '
|
||||
. 'id, number, client_name, client_surname, client_email, client_phone '
|
||||
. 'FROM pp_shop_orders '
|
||||
. 'WHERE '
|
||||
. '('
|
||||
. 'number LIKE :q1 '
|
||||
. 'OR client_email LIKE :q2 '
|
||||
. 'OR client_name LIKE :q3 '
|
||||
. 'OR client_surname LIKE :q4 '
|
||||
. 'OR client_phone LIKE :q5 '
|
||||
. "OR CONCAT_WS(' ', TRIM(client_name), TRIM(client_surname)) LIKE :q6 "
|
||||
. "OR CONCAT_WS(' ', TRIM(client_surname), TRIM(client_name)) LIKE :q7 "
|
||||
. ') '
|
||||
. 'ORDER BY id DESC '
|
||||
. 'LIMIT 15',
|
||||
[
|
||||
':q1' => $like,
|
||||
':q2' => $like,
|
||||
':q3' => $like,
|
||||
':q4' => $like,
|
||||
':q5' => $like,
|
||||
':q6' => $likeNormalized,
|
||||
':q7' => $likeNormalized,
|
||||
]
|
||||
);
|
||||
} catch (\Throwable $e) {
|
||||
$orderStmt = false;
|
||||
}
|
||||
|
||||
$orderRows = $orderStmt ? $orderStmt->fetchAll() : [];
|
||||
if (is_array($orderRows)) {
|
||||
foreach ($orderRows as $row) {
|
||||
$orderId = (int)($row['id'] ?? 0);
|
||||
if ($orderId <= 0) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$orderNumber = trim((string)($row['number'] ?? ''));
|
||||
$clientName = trim((string)($row['client_name'] ?? ''));
|
||||
$clientSurname = trim((string)($row['client_surname'] ?? ''));
|
||||
$clientEmail = trim((string)($row['client_email'] ?? ''));
|
||||
$clientPhone = trim((string)($row['client_phone'] ?? ''));
|
||||
|
||||
$title = $orderNumber !== '' ? 'Zamówienie ' . $orderNumber : 'Zamówienie #' . $orderId;
|
||||
$subtitleParts = [];
|
||||
$fullName = trim($clientName . ' ' . $clientSurname);
|
||||
if ($fullName !== '') {
|
||||
$subtitleParts[] = $fullName;
|
||||
}
|
||||
if ($clientEmail !== '') {
|
||||
$subtitleParts[] = $clientEmail;
|
||||
}
|
||||
if ($clientPhone !== '') {
|
||||
$subtitleParts[] = $clientPhone;
|
||||
}
|
||||
|
||||
$items[] = [
|
||||
'type' => 'order',
|
||||
'title' => $title,
|
||||
'subtitle' => implode(' | ', $subtitleParts),
|
||||
'url' => '/admin/shop_order/order_details/order_id=' . $orderId,
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
echo json_encode([
|
||||
'status' => 'ok',
|
||||
'items' => array_slice($items, 0, 20),
|
||||
]);
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Zapis ustawien (AJAX).
|
||||
*/
|
||||
|
||||
323
autoload/admin/Controllers/ShopOrderController.php
Normal file
323
autoload/admin/Controllers/ShopOrderController.php
Normal file
@@ -0,0 +1,323 @@
|
||||
<?php
|
||||
namespace admin\Controllers;
|
||||
|
||||
use Domain\Order\OrderAdminService;
|
||||
use admin\ViewModels\Common\PaginatedTableViewModel;
|
||||
|
||||
class ShopOrderController
|
||||
{
|
||||
private OrderAdminService $service;
|
||||
|
||||
public function __construct(OrderAdminService $service)
|
||||
{
|
||||
$this->service = $service;
|
||||
}
|
||||
|
||||
public function list(): string
|
||||
{
|
||||
return $this->view_list();
|
||||
}
|
||||
|
||||
public function view_list(): string
|
||||
{
|
||||
$sortableColumns = [
|
||||
'number',
|
||||
'date_order',
|
||||
'status',
|
||||
'summary',
|
||||
'client',
|
||||
'order_email',
|
||||
'client_phone',
|
||||
'transport',
|
||||
'payment_method',
|
||||
'total_orders',
|
||||
'paid',
|
||||
];
|
||||
|
||||
$statusOptions = ['' => '- status -'];
|
||||
foreach ($this->service->statuses() as $statusId => $statusName) {
|
||||
$statusOptions[(string)$statusId] = (string)$statusName;
|
||||
}
|
||||
|
||||
$filterDefinitions = [
|
||||
['key' => 'number', 'label' => 'Nr zamówienia', 'type' => 'text'],
|
||||
['key' => 'date_from', 'label' => 'Data od', 'type' => 'date'],
|
||||
['key' => 'date_to', 'label' => 'Data do', 'type' => 'date'],
|
||||
['key' => 'status', 'label' => 'Status', 'type' => 'select', 'options' => $statusOptions],
|
||||
['key' => 'client', 'label' => 'Klient', 'type' => 'text'],
|
||||
['key' => 'address', 'label' => 'Adres', 'type' => 'text'],
|
||||
['key' => 'order_email', 'label' => 'Email', 'type' => 'text'],
|
||||
['key' => 'client_phone', 'label' => 'Telefon', 'type' => 'text'],
|
||||
['key' => 'transport', 'label' => 'Dostawa', 'type' => 'text'],
|
||||
['key' => 'payment_method', 'label' => 'Płatność', 'type' => 'text'],
|
||||
];
|
||||
|
||||
$listRequest = \admin\Support\TableListRequestFactory::fromRequest(
|
||||
$filterDefinitions,
|
||||
$sortableColumns,
|
||||
'date_order'
|
||||
);
|
||||
|
||||
$result = $this->service->listForAdmin(
|
||||
$listRequest['filters'],
|
||||
$listRequest['sortColumn'],
|
||||
$listRequest['sortDir'],
|
||||
$listRequest['page'],
|
||||
$listRequest['perPage']
|
||||
);
|
||||
|
||||
$statusesMap = $this->service->statuses();
|
||||
$rows = [];
|
||||
$lp = ($listRequest['page'] - 1) * $listRequest['perPage'] + 1;
|
||||
|
||||
foreach ($result['items'] as $item) {
|
||||
$orderId = (int)($item['id'] ?? 0);
|
||||
$orderNumber = (string)($item['number'] ?? '');
|
||||
$statusId = (int)($item['status'] ?? 0);
|
||||
$statusLabel = (string)($statusesMap[$statusId] ?? ('Status #' . $statusId));
|
||||
|
||||
$rows[] = [
|
||||
'lp' => $lp++ . '.',
|
||||
'date_order' => $this->formatDateTime((string)($item['date_order'] ?? '')),
|
||||
'number' => '<a href="/admin/shop_order/order_details/order_id=' . $orderId . '">' . htmlspecialchars($orderNumber, ENT_QUOTES, 'UTF-8') . '</a>',
|
||||
'paid' => ((int)($item['paid'] ?? 0) === 1)
|
||||
? '<i class="fa fa-check text-success"></i>'
|
||||
: '<i class="fa fa-times text-dark"></i>',
|
||||
'status' => htmlspecialchars($statusLabel, ENT_QUOTES, 'UTF-8'),
|
||||
'summary' => number_format((float)($item['summary'] ?? 0), 2, '.', ' ') . ' zł',
|
||||
'client' => htmlspecialchars((string)($item['client'] ?? ''), ENT_QUOTES, 'UTF-8') . ' | zamówienia: <strong>' . (int)($item['total_orders'] ?? 0) . '</strong>',
|
||||
'address' => (string)($item['address'] ?? ''),
|
||||
'order_email' => (string)($item['order_email'] ?? ''),
|
||||
'client_phone' => (string)($item['client_phone'] ?? ''),
|
||||
'transport' => (string)($item['transport'] ?? ''),
|
||||
'payment_method' => (string)($item['payment_method'] ?? ''),
|
||||
'_actions' => [
|
||||
[
|
||||
'label' => 'Szczegóły',
|
||||
'url' => '/admin/shop_order/order_details/order_id=' . $orderId,
|
||||
'class' => 'btn btn-xs btn-primary',
|
||||
],
|
||||
[
|
||||
'label' => 'Usuń',
|
||||
'url' => '/admin/shop_order/order_delete/id=' . $orderId,
|
||||
'class' => 'btn btn-xs btn-danger',
|
||||
'confirm' => 'Na pewno chcesz usunąć wybrane zamówienie?',
|
||||
'confirm_ok' => 'Usuń',
|
||||
'confirm_cancel' => 'Anuluj',
|
||||
],
|
||||
],
|
||||
];
|
||||
}
|
||||
|
||||
$total = (int)$result['total'];
|
||||
$totalPages = max(1, (int)ceil($total / $listRequest['perPage']));
|
||||
|
||||
$viewModel = new PaginatedTableViewModel(
|
||||
[
|
||||
['key' => 'lp', 'label' => 'Lp.', 'class' => 'text-center', 'sortable' => false],
|
||||
['key' => 'date_order', 'sort_key' => 'date_order', 'label' => 'Data dodania', 'class' => 'text-center', 'sortable' => true],
|
||||
['key' => 'number', 'sort_key' => 'number', 'label' => 'Nr zamówienia', 'class' => 'text-center', 'sortable' => true, 'raw' => true],
|
||||
['key' => 'paid', 'sort_key' => 'paid', 'label' => '', 'class' => 'text-center', 'sortable' => true, 'raw' => true],
|
||||
['key' => 'status', 'sort_key' => 'status', 'label' => 'Status', 'sortable' => true, 'raw' => true],
|
||||
['key' => 'summary', 'sort_key' => 'summary', 'label' => 'Wartość', 'class' => 'text-right align-middle', 'sortable' => true],
|
||||
['key' => 'client', 'sort_key' => 'client', 'label' => 'Klient', 'sortable' => true, 'raw' => true],
|
||||
['key' => 'address', 'label' => 'Adres', 'sortable' => false],
|
||||
['key' => 'order_email', 'sort_key' => 'order_email', 'label' => 'Email', 'sortable' => true],
|
||||
['key' => 'client_phone', 'sort_key' => 'client_phone', 'label' => 'Telefon', 'sortable' => true],
|
||||
['key' => 'transport', 'sort_key' => 'transport', 'label' => 'Dostawa', 'sortable' => true],
|
||||
['key' => 'payment_method', 'sort_key' => 'payment_method', 'label' => 'Płatność', 'sortable' => true],
|
||||
],
|
||||
$rows,
|
||||
$listRequest['viewFilters'],
|
||||
[
|
||||
'column' => $listRequest['sortColumn'],
|
||||
'dir' => $listRequest['sortDir'],
|
||||
],
|
||||
[
|
||||
'page' => $listRequest['page'],
|
||||
'per_page' => $listRequest['perPage'],
|
||||
'total' => $total,
|
||||
'total_pages' => $totalPages,
|
||||
],
|
||||
array_merge($listRequest['queryFilters'], [
|
||||
'sort' => $listRequest['sortColumn'],
|
||||
'dir' => $listRequest['sortDir'],
|
||||
'per_page' => $listRequest['perPage'],
|
||||
]),
|
||||
$listRequest['perPageOptions'],
|
||||
$sortableColumns,
|
||||
'/admin/shop_order/list/',
|
||||
'Brak danych w tabeli.'
|
||||
);
|
||||
|
||||
return \Tpl::view('shop-order/orders-list', [
|
||||
'viewModel' => $viewModel,
|
||||
]);
|
||||
}
|
||||
|
||||
public function details(): string
|
||||
{
|
||||
return $this->order_details();
|
||||
}
|
||||
|
||||
public function order_details(): string
|
||||
{
|
||||
$orderId = (int)\S::get('order_id');
|
||||
$order = $this->service->details($orderId);
|
||||
|
||||
$coupon = null;
|
||||
if (!empty($order) && !empty($order['coupon_id'])) {
|
||||
$coupon = new \shop\Coupon((int)$order['coupon_id']);
|
||||
}
|
||||
|
||||
return \Tpl::view('shop-order/order-details', [
|
||||
'order' => $order,
|
||||
'coupon' => $coupon,
|
||||
'order_statuses' => $this->service->statuses(),
|
||||
'next_order_id' => $this->service->nextOrderId($orderId),
|
||||
'prev_order_id' => $this->service->prevOrderId($orderId),
|
||||
]);
|
||||
}
|
||||
|
||||
public function edit(): string
|
||||
{
|
||||
return $this->order_edit();
|
||||
}
|
||||
|
||||
public function order_edit(): string
|
||||
{
|
||||
$orderId = (int)\S::get('order_id');
|
||||
|
||||
return \Tpl::view('shop-order/order-edit', [
|
||||
'order' => $this->service->details($orderId),
|
||||
'order_statuses' => $this->service->statuses(),
|
||||
'transport' => \shop\Transport::transport_list(),
|
||||
'payment_methods' => \shop\PaymentMethod::method_list(),
|
||||
]);
|
||||
}
|
||||
|
||||
public function save(): void
|
||||
{
|
||||
$this->order_save();
|
||||
}
|
||||
|
||||
public function order_save(): void
|
||||
{
|
||||
$saved = $this->service->saveOrderByAdmin([
|
||||
'order_id' => (int)\S::get('order_id'),
|
||||
'client_name' => (string)\S::get('client_name'),
|
||||
'client_surname' => (string)\S::get('client_surname'),
|
||||
'client_street' => (string)\S::get('client_street'),
|
||||
'client_postal_code' => (string)\S::get('client_postal_code'),
|
||||
'client_city' => (string)\S::get('client_city'),
|
||||
'client_email' => (string)\S::get('client_email'),
|
||||
'firm_name' => (string)\S::get('firm_name'),
|
||||
'firm_street' => (string)\S::get('firm_street'),
|
||||
'firm_postal_code' => (string)\S::get('firm_postal_code'),
|
||||
'firm_city' => (string)\S::get('firm_city'),
|
||||
'firm_nip' => (string)\S::get('firm_nip'),
|
||||
'transport_id' => (int)\S::get('transport_id'),
|
||||
'inpost_paczkomat' => (string)\S::get('inpost_paczkomat'),
|
||||
'payment_method_id' => (int)\S::get('payment_method_id'),
|
||||
]);
|
||||
|
||||
if ($saved) {
|
||||
\S::alert('Zamówienie zostało zapisane.');
|
||||
}
|
||||
|
||||
header('Location: /admin/shop_order/order_details/order_id=' . (int)\S::get('order_id'));
|
||||
exit;
|
||||
}
|
||||
|
||||
public function notes_save(): void
|
||||
{
|
||||
$this->service->saveNotes((int)\S::get('order_id'), (string)\S::get('notes'));
|
||||
}
|
||||
|
||||
public function order_status_change(): void
|
||||
{
|
||||
$response = $this->service->changeStatus(
|
||||
(int)\S::get('order_id'),
|
||||
(int)\S::get('status'),
|
||||
(string)\S::get('email') === 'true'
|
||||
);
|
||||
|
||||
echo json_encode($response);
|
||||
exit;
|
||||
}
|
||||
|
||||
public function order_resend_confirmation_email(): void
|
||||
{
|
||||
$response = $this->service->resendConfirmationEmail((int)\S::get('order_id'));
|
||||
|
||||
echo json_encode(['result' => $response]);
|
||||
exit;
|
||||
}
|
||||
|
||||
public function set_order_as_unpaid(): void
|
||||
{
|
||||
$orderId = (int)\S::get('order_id');
|
||||
$this->service->setOrderAsUnpaid($orderId);
|
||||
|
||||
header('Location: /admin/shop_order/order_details/order_id=' . $orderId);
|
||||
exit;
|
||||
}
|
||||
|
||||
public function set_order_as_paid(): void
|
||||
{
|
||||
$orderId = (int)\S::get('order_id');
|
||||
$this->service->setOrderAsPaid($orderId, (int)\S::get('send_mail') === 1);
|
||||
|
||||
header('Location: /admin/shop_order/order_details/order_id=' . $orderId);
|
||||
exit;
|
||||
}
|
||||
|
||||
public function send_order_to_apilo(): void
|
||||
{
|
||||
$orderId = (int)\S::get('order_id');
|
||||
|
||||
if ($this->service->sendOrderToApilo($orderId)) {
|
||||
\S::alert('Zamówienie zostanie wysłane ponownie do apilo.com');
|
||||
} else {
|
||||
\S::alert('Wystąpił błąd podczas wysyłania zamówienia do apilo.com');
|
||||
}
|
||||
|
||||
header('Location: /admin/shop_order/order_details/order_id=' . $orderId);
|
||||
exit;
|
||||
}
|
||||
|
||||
public function toggle_trustmate_send(): void
|
||||
{
|
||||
echo json_encode($this->service->toggleTrustmateSend((int)\S::get('order_id')));
|
||||
exit;
|
||||
}
|
||||
|
||||
public function delete(): void
|
||||
{
|
||||
$this->order_delete();
|
||||
}
|
||||
|
||||
public function order_delete(): void
|
||||
{
|
||||
if ($this->service->deleteOrder((int)\S::get('id'))) {
|
||||
\S::alert('Zamówienie zostało usunięte');
|
||||
}
|
||||
|
||||
header('Location: /admin/shop_order/list/');
|
||||
exit;
|
||||
}
|
||||
|
||||
private function formatDateTime(string $value): string
|
||||
{
|
||||
if ($value === '') {
|
||||
return '';
|
||||
}
|
||||
|
||||
$ts = strtotime($value);
|
||||
if ($ts === false) {
|
||||
return $value;
|
||||
}
|
||||
|
||||
return date('Y-m-d H:i', $ts);
|
||||
}
|
||||
}
|
||||
@@ -2,6 +2,7 @@
|
||||
namespace admin\Controllers;
|
||||
|
||||
use Domain\PaymentMethod\PaymentMethodRepository;
|
||||
use Domain\Integrations\IntegrationsRepository;
|
||||
use admin\ViewModels\Common\PaginatedTableViewModel;
|
||||
use admin\ViewModels\Forms\FormAction;
|
||||
use admin\ViewModels\Forms\FormEditViewModel;
|
||||
@@ -240,7 +241,10 @@ class ShopPaymentMethodController
|
||||
|
||||
private function getApiloPaymentTypes(): array
|
||||
{
|
||||
$rawSetting = \admin\factory\Integrations::apilo_settings('payment-types-list');
|
||||
global $mdb;
|
||||
|
||||
$integrationsRepository = new IntegrationsRepository( $mdb );
|
||||
$rawSetting = $integrationsRepository -> getSetting( 'apilo', 'payment-types-list' );
|
||||
$raw = null;
|
||||
|
||||
if (is_array($rawSetting)) {
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
namespace admin\Controllers;
|
||||
|
||||
use Domain\ShopStatus\ShopStatusRepository;
|
||||
use Domain\Integrations\IntegrationsRepository;
|
||||
use admin\ViewModels\Common\PaginatedTableViewModel;
|
||||
use admin\ViewModels\Forms\FormAction;
|
||||
use admin\ViewModels\Forms\FormEditViewModel;
|
||||
@@ -246,8 +247,12 @@ class ShopStatusesController
|
||||
|
||||
private function getApiloStatusList(): array
|
||||
{
|
||||
global $mdb;
|
||||
|
||||
$integrationsRepository = new IntegrationsRepository( $mdb );
|
||||
|
||||
$list = [];
|
||||
$raw = @unserialize(\admin\factory\Integrations::apilo_settings('status-types-list'));
|
||||
$raw = @unserialize( $integrationsRepository -> getSetting( 'apilo', 'status-types-list' ) );
|
||||
if (is_array($raw)) {
|
||||
foreach ($raw as $apiloStatus) {
|
||||
if (isset($apiloStatus['id'], $apiloStatus['name'])) {
|
||||
|
||||
@@ -3,6 +3,7 @@ namespace admin\Controllers;
|
||||
|
||||
use Domain\Transport\TransportRepository;
|
||||
use Domain\PaymentMethod\PaymentMethodRepository;
|
||||
use Domain\Integrations\IntegrationsRepository;
|
||||
use admin\ViewModels\Common\PaginatedTableViewModel;
|
||||
use admin\ViewModels\Forms\FormAction;
|
||||
use admin\ViewModels\Forms\FormEditViewModel;
|
||||
@@ -314,7 +315,10 @@ class ShopTransportController
|
||||
|
||||
private function getApiloCarrierAccounts(): array
|
||||
{
|
||||
$rawSetting = \admin\factory\Integrations::apilo_settings('carrier-account-list');
|
||||
global $mdb;
|
||||
|
||||
$integrationsRepository = new IntegrationsRepository( $mdb );
|
||||
$rawSetting = $integrationsRepository -> getSetting( 'apilo', 'carrier-account-list' );
|
||||
$raw = null;
|
||||
|
||||
if (is_array($rawSetting)) {
|
||||
|
||||
Reference in New Issue
Block a user