mirror of
https://github.com/opensourcepos/opensourcepos.git
synced 2025-12-25 18:47:53 -05:00
* Improve code style and PSR-12 compliance - refactored code formatting to adhere to PSR-12 guidelines - standardized coding conventions across the codebase - added missing framework files and reverted markup changes - reformatted arrays for enhanced readability - updated language files for consistent styling and clarity - minor miscellaneous improvements
356 lines
14 KiB
PHP
356 lines
14 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use CodeIgniter\Database\ResultInterface;
|
|
use CodeIgniter\Model;
|
|
use Config\OSPOS;
|
|
use ReflectionException;
|
|
|
|
/**
|
|
* Receiving class
|
|
*/
|
|
class Receiving extends Model
|
|
{
|
|
protected $table = 'receivings';
|
|
protected $primaryKey = 'receiving_id';
|
|
protected $useAutoIncrement = true;
|
|
protected $useSoftDeletes = false;
|
|
protected $allowedFields = [
|
|
'receiving_time',
|
|
'supplier_id',
|
|
'employee_id',
|
|
'comment',
|
|
'receiving_id',
|
|
'payment_type',
|
|
'reference'
|
|
];
|
|
|
|
/**
|
|
* @param int $receiving_id
|
|
* @return ResultInterface
|
|
*/
|
|
public function get_info(int $receiving_id): ResultInterface
|
|
{
|
|
$builder = $this->db->table('receivings');
|
|
$builder->join('people', 'people.person_id = receivings.supplier_id', 'LEFT');
|
|
$builder->join('suppliers', 'suppliers.person_id = receivings.supplier_id', 'LEFT');
|
|
$builder->where('receiving_id', $receiving_id);
|
|
|
|
return $builder->get();
|
|
}
|
|
|
|
/**
|
|
* @param string $reference
|
|
* @return ResultInterface
|
|
*/
|
|
public function get_receiving_by_reference(string $reference): ResultInterface
|
|
{
|
|
$builder = $this->db->table('receivings');
|
|
$builder->where('reference', $reference);
|
|
|
|
return $builder->get();
|
|
}
|
|
|
|
/**
|
|
* @param string $receipt_receiving_id
|
|
* @return bool
|
|
*/
|
|
public function is_valid_receipt(string $receipt_receiving_id): bool // TODO: maybe receipt_receiving_id should be an array rather than a space delimited string
|
|
{
|
|
if (!empty($receipt_receiving_id)) {
|
|
// RECV #
|
|
$pieces = explode(' ', $receipt_receiving_id);
|
|
|
|
if (count($pieces) == 2 && preg_match('/(RECV|KIT)/', $pieces[0])) {
|
|
return $this->exists($pieces[1]);
|
|
} else {
|
|
return $this->get_receiving_by_reference($receipt_receiving_id)->getNumRows() > 0;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @param int $receiving_id
|
|
* @return bool
|
|
*/
|
|
public function exists(int $receiving_id): bool
|
|
{
|
|
$builder = $this->db->table('receivings');
|
|
$builder->where('receiving_id', $receiving_id);
|
|
|
|
return ($builder->get()->getNumRows() == 1);
|
|
}
|
|
|
|
/**
|
|
* @param $receiving_id
|
|
* @param $receiving_data
|
|
* @return bool
|
|
*/
|
|
public function update($receiving_id = null, $receiving_data = null): bool
|
|
{
|
|
$builder = $this->db->table('receivings');
|
|
$builder->where('receiving_id', $receiving_id);
|
|
|
|
return $builder->update($receiving_data);
|
|
}
|
|
|
|
/**
|
|
* @throws ReflectionException
|
|
*/
|
|
public function save_value(array $items, int $supplier_id, int $employee_id, string $comment, string $reference, ?string $payment_type, int $receiving_id = NEW_ENTRY): int // TODO: $receiving_id gets overwritten before it's evaluated. It doesn't make sense to pass this here.
|
|
{
|
|
$attribute = model(Attribute::class);
|
|
$inventory = model('Inventory');
|
|
$item = model(Item::class);
|
|
$item_quantity = model(Item_quantity::class);
|
|
$supplier = model(Supplier::class);
|
|
|
|
if (count($items) == 0) {
|
|
return -1; // TODO: Replace -1 with a constant
|
|
}
|
|
|
|
$receivings_data = [
|
|
'receiving_time' => date('Y-m-d H:i:s'),
|
|
'supplier_id' => $supplier->exists($supplier_id) ? $supplier_id : null,
|
|
'employee_id' => $employee_id,
|
|
'payment_type' => $payment_type,
|
|
'comment' => $comment,
|
|
'reference' => $reference
|
|
];
|
|
|
|
// Run these queries as a transaction, we want to make sure we do all or nothing
|
|
$this->db->transStart();
|
|
|
|
$builder = $this->db->table('receivings');
|
|
$builder->insert($receivings_data);
|
|
$receiving_id = $this->db->insertID();
|
|
|
|
$builder = $this->db->table('receivings_items');
|
|
|
|
foreach ($items as $line => $item_data) {
|
|
$config = config(OSPOS::class)->settings;
|
|
$cur_item_info = $item->get_info($item_data['item_id']);
|
|
|
|
$receivings_items_data = [
|
|
'receiving_id' => $receiving_id,
|
|
'item_id' => $item_data['item_id'],
|
|
'line' => $item_data['line'],
|
|
'description' => $item_data['description'],
|
|
'serialnumber' => $item_data['serialnumber'],
|
|
'quantity_purchased' => $item_data['quantity'],
|
|
'receiving_quantity' => $item_data['receiving_quantity'],
|
|
'discount' => $item_data['discount'],
|
|
'discount_type' => $item_data['discount_type'],
|
|
'item_cost_price' => $cur_item_info->cost_price,
|
|
'item_unit_price' => $item_data['price'],
|
|
'item_location' => $item_data['item_location']
|
|
];
|
|
|
|
$builder->insert($receivings_items_data);
|
|
|
|
$items_received = $item_data['receiving_quantity'] != 0 ? $item_data['quantity'] * $item_data['receiving_quantity'] : $item_data['quantity'];
|
|
|
|
// Update cost price, if changed AND is set in config as wanted
|
|
if ($cur_item_info->cost_price != $item_data['price'] && $config['receiving_calculate_average_price']) {
|
|
$item->change_cost_price($item_data['item_id'], $items_received, $item_data['price'], $cur_item_info->cost_price);
|
|
}
|
|
|
|
// Update stock quantity
|
|
$item_quantity_value = $item_quantity->get_item_quantity($item_data['item_id'], $item_data['item_location']);
|
|
$item_quantity->save_value(
|
|
[
|
|
'quantity' => $item_quantity_value->quantity + $items_received,
|
|
'item_id' => $item_data['item_id'],
|
|
'location_id' => $item_data['item_location']
|
|
],
|
|
$item_data['item_id'],
|
|
$item_data['item_location']
|
|
);
|
|
|
|
$recv_remarks = 'RECV ' . $receiving_id;
|
|
$inv_data = [
|
|
'trans_date' => date('Y-m-d H:i:s'),
|
|
'trans_items' => $item_data['item_id'],
|
|
'trans_user' => $employee_id,
|
|
'trans_location' => $item_data['item_location'],
|
|
'trans_comment' => $recv_remarks,
|
|
'trans_inventory' => $items_received
|
|
];
|
|
|
|
$inventory->insert($inv_data, false);
|
|
$attribute->copy_attribute_links($item_data['item_id'], 'receiving_id', $receiving_id);
|
|
}
|
|
|
|
$this->db->transComplete();
|
|
|
|
return $this->db->transStatus() ? $receiving_id : -1;
|
|
}
|
|
|
|
|
|
/**
|
|
* @throws ReflectionException
|
|
*/
|
|
public function delete_list(array $receiving_ids, int $employee_id, bool $update_inventory = true): bool
|
|
{
|
|
$success = true;
|
|
|
|
// Start a transaction to assure data integrity
|
|
$this->db->transStart();
|
|
|
|
foreach ($receiving_ids as $receiving_id) {
|
|
$success &= $this->delete_value($receiving_id, $employee_id, $update_inventory);
|
|
}
|
|
|
|
// Execute transaction
|
|
$this->db->transComplete();
|
|
|
|
$success &= $this->db->transStatus();
|
|
|
|
return $success;
|
|
}
|
|
|
|
/**
|
|
* @throws ReflectionException
|
|
*/
|
|
public function delete_value(int $receiving_id, int $employee_id, bool $update_inventory = true): bool
|
|
{
|
|
// Start a transaction to assure data integrity
|
|
$this->db->transStart();
|
|
|
|
if ($update_inventory) {
|
|
// TODO: defect, not all item deletions will be undone? get array with all the items involved in the sale to update the inventory tracking
|
|
$items = $this->get_receiving_items($receiving_id)->getResultArray();
|
|
|
|
$inventory = model('Inventory');
|
|
$item_quantity = model(Item_quantity::class);
|
|
|
|
foreach ($items as $item) {
|
|
// Create query to update inventory tracking
|
|
$inv_data = [
|
|
'trans_date' => date('Y-m-d H:i:s'),
|
|
'trans_items' => $item['item_id'],
|
|
'trans_user' => $employee_id,
|
|
'trans_comment' => 'Deleting receiving ' . $receiving_id,
|
|
'trans_location' => $item['item_location'],
|
|
'trans_inventory' => $item['quantity_purchased'] * (-$item['receiving_quantity'])
|
|
];
|
|
// Update inventory
|
|
$inventory->insert($inv_data, false);
|
|
|
|
// Update quantities
|
|
$item_quantity->change_quantity($item['item_id'], $item['item_location'], $item['quantity_purchased'] * (-$item['receiving_quantity']));
|
|
}
|
|
}
|
|
|
|
// Delete all items
|
|
$builder = $this->db->table('receivings_items');
|
|
$builder->delete(['receiving_id' => $receiving_id]);
|
|
|
|
// Delete sale itself
|
|
$builder = $this->db->table('receivings');
|
|
$builder->delete(['receiving_id' => $receiving_id]);
|
|
|
|
// Execute transaction
|
|
$this->db->transComplete();
|
|
|
|
return $this->db->transStatus();
|
|
}
|
|
|
|
/**
|
|
* @param int $receiving_id
|
|
* @return ResultInterface
|
|
*/
|
|
public function get_receiving_items(int $receiving_id): ResultInterface
|
|
{
|
|
$builder = $this->db->table('receivings_items');
|
|
$builder->where('receiving_id', $receiving_id);
|
|
|
|
return $builder->get();
|
|
}
|
|
|
|
/**
|
|
* @param int $receiving_id
|
|
* @return object
|
|
*/
|
|
public function get_supplier(int $receiving_id): object
|
|
{
|
|
$builder = $this->db->table('receivings');
|
|
$builder->where('receiving_id', $receiving_id);
|
|
|
|
$supplier = model(Supplier::class);
|
|
return $supplier->get_info($builder->get()->getRow()->supplier_id);
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
public function get_payment_options(): array
|
|
{
|
|
return [
|
|
lang('Sales.cash') => lang('Sales.cash'),
|
|
lang('Sales.check') => lang('Sales.check'),
|
|
lang('Sales.debit') => lang('Sales.debit'),
|
|
lang('Sales.credit') => lang('Sales.credit'),
|
|
lang('Sales.due') => lang('Sales.due')
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Create a temp table that allows us to do easy report/receiving queries
|
|
*/
|
|
public function create_temp_table(array $inputs): void
|
|
{
|
|
$config = config(OSPOS::class)->settings;
|
|
$db_prefix = $this->db->getPrefix();
|
|
|
|
if (empty($inputs['receiving_id'])) {
|
|
$where = empty($config['date_or_time_format'])
|
|
? 'DATE(`receiving_time`) BETWEEN ' . $this->db->escape($inputs['start_date']) . ' AND ' . $this->db->escape($inputs['end_date'])
|
|
: 'receiving_time BETWEEN ' . $this->db->escape(rawurldecode($inputs['start_date'])) . ' AND ' . $this->db->escape(rawurldecode($inputs['end_date']));
|
|
} else {
|
|
$where = 'receivings_items.receiving_id = ' . $this->db->escape($inputs['receiving_id']);
|
|
}
|
|
|
|
$builder = $this->db->table('receivings_items');
|
|
$builder->select([
|
|
'MAX(DATE(`receiving_time`)) AS receiving_date',
|
|
'MAX(`receiving_time`) AS receiving_time',
|
|
'receivings_items.receiving_id AS receiving_id',
|
|
'MAX(`comment`) AS comment',
|
|
'MAX(`item_location`) AS item_location',
|
|
'MAX(`reference`) AS reference',
|
|
'MAX(`payment_type`) AS payment_type',
|
|
'MAX(`employee_id`) AS employee_id',
|
|
'items.item_id AS item_id',
|
|
'MAX(`' . $db_prefix . 'receivings`.`supplier_id`) AS supplier_id',
|
|
'MAX(`quantity_purchased`) AS quantity_purchased',
|
|
'MAX(`' . $db_prefix . 'receivings_items`.`receiving_quantity`) AS item_receiving_quantity',
|
|
'MAX(`item_cost_price`) AS item_cost_price',
|
|
'MAX(`item_unit_price`) AS item_unit_price',
|
|
'MAX(`discount`) AS discount',
|
|
'MAX(`discount_type`) AS discount_type',
|
|
'receivings_items.line AS line',
|
|
'MAX(`serialnumber`) AS serialnumber',
|
|
'MAX(`' . $db_prefix . 'receivings_items`.`description`) AS description',
|
|
'MAX(CASE WHEN `' . $db_prefix . 'receivings_items`.`discount_type` = ' . PERCENT . ' THEN `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` - `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` * `discount` / 100 ELSE `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` - `discount` END) AS subtotal',
|
|
'MAX(CASE WHEN `' . $db_prefix . 'receivings_items`.`discount_type` = ' . PERCENT . ' THEN `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` - `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` * `discount` / 100 ELSE `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` - `discount` END) AS total',
|
|
'MAX((CASE WHEN `' . $db_prefix . 'receivings_items`.`discount_type` = ' . PERCENT . ' THEN `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` - `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` * `discount` / 100 ELSE `item_unit_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` - discount END) - (`item_cost_price` * `quantity_purchased`)) AS profit',
|
|
'MAX(`item_cost_price` * `quantity_purchased` * `' . $db_prefix . 'receivings_items`.`receiving_quantity` ) AS cost'
|
|
]);
|
|
$builder->join('receivings', 'receivings_items.receiving_id = receivings.receiving_id', 'inner');
|
|
$builder->join('items', 'receivings_items.item_id = items.item_id', 'inner');
|
|
$builder->where($where);
|
|
$builder->groupBy(['receivings_items.receiving_id', 'items.item_id', 'receivings_items.line']);
|
|
$selectQuery = $builder->getCompiledSelect();
|
|
|
|
// QueryBuilder does not support creating temporary tables.
|
|
$sql = 'CREATE TEMPORARY TABLE IF NOT EXISTS ' . $this->db->prefixTable('receivings_items_temp') .
|
|
' (INDEX(receiving_date), INDEX(receiving_time), INDEX(receiving_id)) AS (' . $selectQuery . ')';
|
|
|
|
$this->db->query($sql);
|
|
}
|
|
}
|