mirror of
https://github.com/snipe/snipe-it.git
synced 2024-12-26 22:19:41 -08:00
ac63642224
* Add manufacturer to licenses. Shows in table and edit. Need to improve manufacturer view to show lists beyond assets still. * Remove extra closing tags, formatting * Work on making the manufacturer view show more options. Need to figure out how to change the table dynamically. * Cleanup formatting and fix a few weirdities in hardware/view.blade.php * Standardize on two-space tabs in this file, as it seems the most * common. * Fix a few places where we call number_format without guaranteeing the * item is a number and not a string. * Show a "No Results" message on components page if there are no * components. * Show table of licenses on manufacturer view page. This reworks the ManufacturersController::getDataView method to delegate the view to a sub method (currently assets or licenses, but plan to extend to consumables/accessories/components as well). We then put tabs at the top of the view to show multiple tables. This just duplicates the table layout from licenses/index.blade, but I wonder if theres a way to centralize that code, maybe through partials, over time.. The only known missing part of manufacturers for licenses would be adding it to the importer, but the license importer should probably migrate to object importer before doing too much more... * Add manufacturer to accessory. * Add consumables to the manufacturer view page.
524 lines
21 KiB
PHP
Executable file
524 lines
21 KiB
PHP
Executable file
<?php
|
|
namespace App\Http\Controllers;
|
|
|
|
use App\Models\Company;
|
|
use App\Models\Manufacturer;
|
|
use App\Models\Setting;
|
|
use Auth;
|
|
use Gate;
|
|
use Input;
|
|
use Lang;
|
|
use Redirect;
|
|
use Str;
|
|
use View;
|
|
|
|
/**
|
|
* This controller handles all actions related to Manufacturers for
|
|
* the Snipe-IT Asset Management application.
|
|
*
|
|
* @version v1.0
|
|
*/
|
|
class ManufacturersController extends Controller
|
|
{
|
|
/**
|
|
* Returns a view that invokes the ajax tables which actually contains
|
|
* the content for the manufacturers listing, which is generated in getDatatable.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::getDatatable() method that generates the JSON response
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getIndex()
|
|
{
|
|
// Show the page
|
|
return View::make('manufacturers/index', compact('manufacturers'));
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns a view that displays a form to create a new manufacturer.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::postCreate()
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getCreate()
|
|
{
|
|
return View::make('manufacturers/edit')->with('manufacturer', new Manufacturer);
|
|
}
|
|
|
|
|
|
/**
|
|
* Validates and stores the data for a new manufacturer.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::postCreate()
|
|
* @since [v1.0]
|
|
* @return Redirect
|
|
*/
|
|
public function postCreate()
|
|
{
|
|
$manufacturer = new Manufacturer;
|
|
$manufacturer->name = e(Input::get('name'));
|
|
$manufacturer->user_id = Auth::user()->id;
|
|
|
|
if ($manufacturer->save()) {
|
|
return redirect()->to("admin/settings/manufacturers")->with('success', trans('admin/manufacturers/message.create.success'));
|
|
}
|
|
|
|
return redirect()->back()->withInput()->withErrors($manufacturer->getErrors());
|
|
|
|
}
|
|
|
|
/**
|
|
* Returns a view that displays a form to edit a manufacturer.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::postEdit()
|
|
* @param int $manufacturerId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getEdit($manufacturerId = null)
|
|
{
|
|
// Check if the manufacturer exists
|
|
if (is_null($manufacturer = Manufacturer::find($manufacturerId))) {
|
|
// Redirect to the manufacturer page
|
|
return redirect()->to('admin/settings/manufacturers')->with('error', trans('admin/manufacturers/message.does_not_exist'));
|
|
}
|
|
|
|
// Show the page
|
|
return View::make('manufacturers/edit', compact('manufacturer'));
|
|
}
|
|
|
|
|
|
/**
|
|
* Validates and stores the updated manufacturer data.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::getEdit()
|
|
* @param int $manufacturerId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function postEdit($manufacturerId = null)
|
|
{
|
|
// Check if the manufacturer exists
|
|
if (is_null($manufacturer = Manufacturer::find($manufacturerId))) {
|
|
// Redirect to the manufacturer page
|
|
return redirect()->to('admin/settings/manufacturers')->with('error', trans('admin/manufacturers/message.does_not_exist'));
|
|
}
|
|
|
|
// Save the data
|
|
$manufacturer->name = e(Input::get('name'));
|
|
|
|
// Was it created?
|
|
if ($manufacturer->save()) {
|
|
// Redirect to the new manufacturer page
|
|
return redirect()->to("admin/settings/manufacturers")->with('success', trans('admin/manufacturers/message.update.success'));
|
|
}
|
|
|
|
return redirect()->back()->withInput()->withErrors($manufacturer->getErrors());
|
|
|
|
|
|
}
|
|
|
|
/**
|
|
* Deletes a manufacturer.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @param int $manufacturerId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getDelete($manufacturerId)
|
|
{
|
|
// Check if the manufacturer exists
|
|
if (is_null($manufacturer = Manufacturer::find($manufacturerId))) {
|
|
// Redirect to the manufacturers page
|
|
return redirect()->to('admin/settings/manufacturers')->with('error', trans('admin/manufacturers/message.not_found'));
|
|
}
|
|
|
|
if ($manufacturer->has_models() > 0) {
|
|
|
|
// Redirect to the asset management page
|
|
return redirect()->to('admin/settings/manufacturers')->with('error', trans('admin/manufacturers/message.assoc_users'));
|
|
} else {
|
|
|
|
// Delete the manufacturer
|
|
$manufacturer->delete();
|
|
|
|
// Redirect to the manufacturers management page
|
|
return redirect()->to('admin/settings/manufacturers')->with('success', trans('admin/manufacturers/message.delete.success'));
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* Returns a view that invokes the ajax tables which actually contains
|
|
* the content for the manufacturers detail listing, which is generated in getDatatable.
|
|
* This data contains a listing of all assets that belong to that manufacturer.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::getDataView()
|
|
* @param int $manufacturerId
|
|
* @since [v1.0]
|
|
* @return View
|
|
*/
|
|
public function getView($manufacturerId = null)
|
|
{
|
|
$manufacturer = Manufacturer::find($manufacturerId);
|
|
|
|
if (isset($manufacturer->id)) {
|
|
return View::make('manufacturers/view', compact('manufacturer'));
|
|
} else {
|
|
// Prepare the error message
|
|
$error = trans('admin/manufacturers/message.does_not_exist', compact('id'));
|
|
|
|
// Redirect to the user management page
|
|
return redirect()->route('manufacturers')->with('error', $error);
|
|
}
|
|
|
|
|
|
}
|
|
|
|
/**
|
|
* Generates the JSON used to display the manufacturer listings.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::getIndex()
|
|
* @since [v1.0]
|
|
* @return String JSON
|
|
*/
|
|
public function getDatatable()
|
|
{
|
|
$manufacturers = Manufacturer::select(array('id','name'))->with('assets')
|
|
->whereNull('deleted_at');
|
|
|
|
if (Input::has('search')) {
|
|
$manufacturers = $manufacturers->TextSearch(e(Input::get('search')));
|
|
}
|
|
|
|
if (Input::has('offset')) {
|
|
$offset = e(Input::get('offset'));
|
|
} else {
|
|
$offset = 0;
|
|
}
|
|
|
|
if (Input::has('limit')) {
|
|
$limit = e(Input::get('limit'));
|
|
} else {
|
|
$limit = 50;
|
|
}
|
|
|
|
$allowed_columns = ['id','name'];
|
|
$order = Input::get('order') === 'asc' ? 'asc' : 'desc';
|
|
$sort = in_array(Input::get('sort'), $allowed_columns) ? Input::get('sort') : 'created_at';
|
|
|
|
$manufacturers->orderBy($sort, $order);
|
|
|
|
$manufacturersCount = $manufacturers->count();
|
|
$manufacturers = $manufacturers->skip($offset)->take($limit)->get();
|
|
|
|
$rows = array();
|
|
|
|
foreach ($manufacturers as $manufacturer) {
|
|
$actions = '<a href="'.route('update/manufacturer', $manufacturer->id).'" class="btn btn-warning btn-sm" style="margin-right:5px;"><i class="fa fa-pencil icon-white"></i></a><a data-html="false" class="btn delete-asset btn-danger btn-sm" data-toggle="modal" href="'.route('delete/manufacturer', $manufacturer->id).'" data-content="'.trans('admin/manufacturers/message.delete.confirm').'" data-title="'.trans('general.delete').' '.htmlspecialchars($manufacturer->name).'?" onClick="return false;"><i class="fa fa-trash icon-white"></i></a>';
|
|
|
|
$rows[] = array(
|
|
'id' => $manufacturer->id,
|
|
'name' => (string)link_to('admin/settings/manufacturers/'.$manufacturer->id.'/view', e($manufacturer->name)),
|
|
'assets' => $manufacturer->assets->count(),
|
|
'actions' => $actions
|
|
);
|
|
}
|
|
|
|
$data = array('total' => $manufacturersCount, 'rows' => $rows);
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
|
|
/**
|
|
* Generates the JSON used to display the manufacturer detail.
|
|
* This JSON returns data on all of the assets with the specified
|
|
* manufacturer ID number.
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @see ManufacturersController::getView()
|
|
* @param int $manufacturerId
|
|
* @since [v1.0]
|
|
* @return String JSON
|
|
*/
|
|
public function getDataView($manufacturerId, $itemtype = null)
|
|
{
|
|
$manufacturer = Manufacturer::with('assets.company')->find($manufacturerId);
|
|
|
|
switch ($itemtype) {
|
|
case "assets":
|
|
return $this->getDataAssetsView($manufacturer);
|
|
case "licenses":
|
|
return $this->getDataLicensesView($manufacturer);
|
|
case "accessories":
|
|
return $this->getDataAccessoriesView($manufacturer);
|
|
case "consumables":
|
|
return $this->getDataConsumablesView($manufacturer);
|
|
}
|
|
|
|
throw new Exception("We shouldn't be here");
|
|
|
|
}
|
|
|
|
protected function getDataAssetsView(Manufacturer $manufacturer)
|
|
{
|
|
$manufacturer_assets = $manufacturer->assets;
|
|
|
|
if (Input::has('search')) {
|
|
$manufacturer_assets = $manufacturer_assets->TextSearch(e(Input::get('search')));
|
|
}
|
|
|
|
if (Input::has('offset')) {
|
|
$offset = e(Input::get('offset'));
|
|
} else {
|
|
$offset = 0;
|
|
}
|
|
|
|
if (Input::has('limit')) {
|
|
$limit = e(Input::get('limit'));
|
|
} else {
|
|
$limit = 50;
|
|
}
|
|
|
|
$order = Input::get('order') === 'asc' ? 'asc' : 'desc';
|
|
|
|
$allowed_columns = ['id','name','serial','asset_tag'];
|
|
$sort = in_array(Input::get('sort'), $allowed_columns) ? Input::get('sort') : 'created_at';
|
|
$count = $manufacturer_assets->count();
|
|
|
|
$rows = array();
|
|
|
|
foreach ($manufacturer_assets as $asset) {
|
|
|
|
$actions = '';
|
|
if ($asset->deleted_at=='') {
|
|
$actions = '<div style=" white-space: nowrap;"><a href="'.route('clone/hardware', $asset->id).'" class="btn btn-info btn-sm" title="Clone asset"><i class="fa fa-files-o"></i></a> <a href="'.route('update/hardware', $asset->id).'" class="btn btn-warning btn-sm"><i class="fa fa-pencil icon-white"></i></a> <a data-html="false" class="btn delete-asset btn-danger btn-sm" data-toggle="modal" href="'.route('delete/hardware', $asset->id).'" data-content="'.trans('admin/hardware/message.delete.confirm').'" data-title="'.trans('general.delete').' '.htmlspecialchars($asset->asset_tag).'?" onClick="return false;"><i class="fa fa-trash icon-white"></i></a></div>';
|
|
} elseif ($asset->deleted_at!='') {
|
|
$actions = '<a href="'.route('restore/hardware', $asset->id).'" class="btn btn-warning btn-sm"><i class="fa fa-recycle icon-white"></i></a>';
|
|
}
|
|
|
|
if ($asset->availableForCheckout()) {
|
|
if (Gate::allows('assets.checkout')) {
|
|
$inout = '<a href="'.route('checkout/hardware', $asset->id).'" class="btn btn-info btn-sm">'.trans('general.checkout').'</a>';
|
|
}
|
|
} else {
|
|
if (Gate::allows('assets.checkin')) {
|
|
$inout = '<a href="'.route('checkin/hardware', $asset->id).'" class="btn btn-primary btn-sm">'.trans('general.checkin').'</a>';
|
|
}
|
|
}
|
|
|
|
$rows[] = array(
|
|
'id' => $asset->id,
|
|
'name' => (string)link_to('/hardware/'.$asset->id.'/view', e($asset->showAssetName())),
|
|
'model' => e($asset->model->name),
|
|
'asset_tag' => e($asset->asset_tag),
|
|
'serial' => e($asset->serial),
|
|
'assigned_to' => ($asset->assigneduser) ? (string)link_to('/admin/users/'.$asset->assigneduser->id.'/view', e($asset->assigneduser->fullName())): '',
|
|
'actions' => $actions,
|
|
'companyName' => e(Company::getName($asset)),
|
|
);
|
|
|
|
if (isset($inout)) {
|
|
$row['change'] = $inout;
|
|
}
|
|
}
|
|
|
|
$data = array('total' => $count, 'rows' => $rows);
|
|
return $data;
|
|
}
|
|
|
|
protected function getDataLicensesView(Manufacturer $manufacturer)
|
|
{
|
|
$licenses = $manufacturer->licenses;
|
|
|
|
if (Input::has('search')) {
|
|
$licenses = $licenses->TextSearch(Input::get('search'));
|
|
}
|
|
|
|
$licenseCount = $licenses->count();
|
|
|
|
$rows = array();
|
|
|
|
foreach ($licenses as $license) {
|
|
$actions = '<span style="white-space: nowrap;">';
|
|
|
|
if (Gate::allows('licenses.checkout')) {
|
|
$actions .= '<a href="' . route('freecheckout/license', $license->id)
|
|
. '" class="btn btn-primary btn-sm' . (($license->remaincount() > 0) ? '' : ' disabled') . '" style="margin-right:5px;">' . trans('general.checkout') . '</a> ';
|
|
}
|
|
|
|
if (Gate::allows('licenses.create')) {
|
|
$actions .= '<a href="' . route('clone/license', $license->id)
|
|
. '" class="btn btn-info btn-sm" style="margin-right:5px;" title="Clone asset"><i class="fa fa-files-o"></i></a>';
|
|
}
|
|
if (Gate::allows('licenses.edit')) {
|
|
$actions .= '<a href="' . route('update/license', $license->id)
|
|
. '" class="btn btn-warning btn-sm" style="margin-right:5px;"><i class="fa fa-pencil icon-white"></i></a>';
|
|
}
|
|
if (Gate::allows('licenses.delete')) {
|
|
$actions .= '<a data-html="false" class="btn delete-asset btn-danger btn-sm" data-toggle="modal" href="'
|
|
. route('delete/license', $license->id)
|
|
. '" data-content="' . trans('admin/licenses/message.delete.confirm') . '" data-title="' . trans('general.delete') . ' ' . htmlspecialchars($license->name) . '?" onClick="return false;"><i class="fa fa-trash icon-white"></i></a>';
|
|
}
|
|
$actions .='</span>';
|
|
|
|
$rows[] = array(
|
|
'id' => $license->id,
|
|
'name' => (string) link_to('/admin/licenses/'.$license->id.'/view', $license->name),
|
|
'serial' => (string) link_to('/admin/licenses/'.$license->id.'/view', mb_strimwidth($license->serial, 0, 50, "...")),
|
|
'totalSeats' => $license->totalSeatsByLicenseID(),
|
|
'remaining' => $license->remaincount(),
|
|
'license_name' => e($license->license_name),
|
|
'license_email' => e($license->license_email),
|
|
'purchase_date' => ($license->purchase_date) ? $license->purchase_date : '',
|
|
'expiration_date' => ($license->expiration_date) ? $license->expiration_date : '',
|
|
'purchase_cost' => ($license->purchase_cost) ? number_format($license->purchase_cost, 2) : '',
|
|
'purchase_order' => ($license->purchase_order) ? e($license->purchase_order) : '',
|
|
'order_number' => ($license->order_number) ? e($license->order_number) : '',
|
|
'notes' => ($license->notes) ? e($license->notes) : '',
|
|
'actions' => $actions,
|
|
'companyName' => is_null($license->company) ? '' : e($license->company->name),
|
|
'manufacturer' => $license->manufacturer ? (string) link_to('/admin/settings/manufacturers/'.$license->manufacturer_id.'/view', $license->manufacturer->name) : ''
|
|
);
|
|
}
|
|
|
|
$data = array('total' => $licenseCount, 'rows' => $rows);
|
|
|
|
return $data;
|
|
}
|
|
|
|
public function getDataAccessoriesView(Manufacturer $manufacturer)
|
|
{
|
|
$accessories = $manufacturer->accessories;
|
|
|
|
if (Input::has('search')) {
|
|
$accessories = $accessories->TextSearch(e(Input::get('search')));
|
|
}
|
|
|
|
if (Input::has('limit')) {
|
|
$limit = e(Input::get('limit'));
|
|
} else {
|
|
$limit = 50;
|
|
}
|
|
|
|
$accessCount = $accessories->count();
|
|
|
|
$rows = array();
|
|
|
|
foreach ($accessories as $accessory) {
|
|
|
|
$actions = '<nobr>';
|
|
if (Gate::allows('accessories.checkout')) {
|
|
$actions .= '<a href="' . route('checkout/accessory',
|
|
$accessory->id) . '" style="margin-right:5px;" class="btn btn-info btn-sm" ' . (($accessory->numRemaining() > 0) ? '' : ' disabled') . '>' . trans('general.checkout') . '</a>';
|
|
}
|
|
if (Gate::allows('accessories.edit')) {
|
|
$actions .= '<a href="' . route('update/accessory',
|
|
$accessory->id) . '" class="btn btn-warning btn-sm" style="margin-right:5px;"><i class="fa fa-pencil icon-white"></i></a>';
|
|
}
|
|
if (Gate::allows('accessories.delete')) {
|
|
$actions .= '<a data-html="false" class="btn delete-asset btn-danger btn-sm" data-toggle="modal" href="' . route('delete/accessory',
|
|
$accessory->id) . '" data-content="' . trans('admin/accessories/message.delete.confirm') . '" data-title="' . trans('general.delete') . ' ' . htmlspecialchars($accessory->name) . '?" onClick="return false;"><i class="fa fa-trash icon-white"></i></a>';
|
|
}
|
|
$actions .= '</nobr>';
|
|
$company = $accessory->company;
|
|
|
|
$rows[] = array(
|
|
'name' => '<a href="'.url('admin/accessories/'.$accessory->id).'/view">'. $accessory->name.'</a>',
|
|
'category' => ($accessory->category) ? (string)link_to('admin/settings/categories/'.$accessory->category->id.'/view', $accessory->category->name) : '',
|
|
'qty' => e($accessory->qty),
|
|
'order_number' => e($accessory->order_number),
|
|
'min_amt' => e($accessory->min_amt),
|
|
'location' => ($accessory->location) ? e($accessory->location->name): '',
|
|
'purchase_date' => e($accessory->purchase_date),
|
|
'purchase_cost' => number_format($accessory->purchase_cost, 2),
|
|
'numRemaining' => $accessory->numRemaining(),
|
|
'actions' => $actions,
|
|
'companyName' => is_null($company) ? '' : e($company->name),
|
|
'manufacturer' => $accessory->manufacturer ? (string) link_to('/admin/settings/manufacturers/'.$accessory->manufacturer_id.'/view', $accessory->manufacturer->name) : ''
|
|
|
|
);
|
|
}
|
|
|
|
$data = array('total'=>$accessCount, 'rows'=>$rows);
|
|
|
|
return $data;
|
|
}
|
|
|
|
public function getDataConsumablesView($manufacturer)
|
|
{
|
|
$consumables = $manufacturer->consumables;
|
|
|
|
if (Input::has('search')) {
|
|
$consumables = $consumables->TextSearch(e(Input::get('search')));
|
|
}
|
|
|
|
if (Input::has('limit')) {
|
|
$limit = e(Input::get('limit'));
|
|
} else {
|
|
$limit = 50;
|
|
}
|
|
|
|
$consumCount = $consumables->count();
|
|
|
|
$rows = array();
|
|
|
|
foreach ($consumables as $consumable) {
|
|
$actions = '<nobr>';
|
|
if (Gate::allows('consumables.checkout')) {
|
|
$actions .= '<a href="' . route('checkout/consumable',
|
|
$consumable->id) . '" style="margin-right:5px;" class="btn btn-info btn-sm" ' . (($consumable->numRemaining() > 0) ? '' : ' disabled') . '>' . trans('general.checkout') . '</a>';
|
|
}
|
|
|
|
if (Gate::allows('consumables.edit')) {
|
|
$actions .= '<a href="' . route('update/consumable',
|
|
$consumable->id) . '" class="btn btn-warning btn-sm" style="margin-right:5px;"><i class="fa fa-pencil icon-white"></i></a>';
|
|
}
|
|
if (Gate::allows('consumables.delete')) {
|
|
$actions .= '<a data-html="false" class="btn delete-asset btn-danger btn-sm" data-toggle="modal" href="' . route('delete/consumable',
|
|
$consumable->id) . '" data-content="' . trans('admin/consumables/message.delete.confirm') . '" data-title="' . trans('general.delete') . ' ' . htmlspecialchars($consumable->name) . '?" onClick="return false;"><i class="fa fa-trash icon-white"></i></a>';
|
|
}
|
|
|
|
$actions .='</nobr>';
|
|
|
|
$company = $consumable->company;
|
|
|
|
$rows[] = array(
|
|
'id' => $consumable->id,
|
|
'name' => (string)link_to('admin/consumables/'.$consumable->id.'/view', e($consumable->name)),
|
|
'location' => ($consumable->location) ? e($consumable->location->name) : '',
|
|
'min_amt' => e($consumable->min_amt),
|
|
'qty' => e($consumable->qty),
|
|
'manufacturer' => ($consumable->manufacturer) ? (string) link_to('/admin/settings/manufacturers/'.$consumable->manufacturer_id.'/view', $consumable->manufacturer->name): '',
|
|
'model_no' => e($consumable->model_no),
|
|
'item_no' => e($consumable->item_no),
|
|
'category' => ($consumable->category) ? (string) link_to('/admin/settings/categories/'.$consumable->category_id.'/view', $consumable->category->name) : 'Missing category',
|
|
'order_number' => e($consumable->order_number),
|
|
'purchase_date' => e($consumable->purchase_date),
|
|
'purchase_cost' => ($consumable->purchase_cost!='') ? number_format($consumable->purchase_cost, 2): '' ,
|
|
'numRemaining' => $consumable->numRemaining(),
|
|
'actions' => $actions,
|
|
'companyName' => is_null($company) ? '' : e($company->name),
|
|
);
|
|
}
|
|
|
|
$data = array('total' => $consumCount, 'rows' => $rows);
|
|
|
|
return $data;
|
|
}
|
|
}
|