271 lines
9.6 KiB
PHP
Executable File
271 lines
9.6 KiB
PHP
Executable File
<?php
|
|
|
|
namespace App\Http\Controllers;
|
|
|
|
use App\Actions\CheckoutRequests\CancelCheckoutRequestAction;
|
|
use App\Actions\CheckoutRequests\CreateCheckoutRequestAction;
|
|
use App\Enums\ActionType;
|
|
use App\Exceptions\AssetNotRequestable;
|
|
use App\Models\Actionlog;
|
|
use App\Models\Asset;
|
|
use App\Models\AssetModel;
|
|
use App\Models\Setting;
|
|
use App\Models\User;
|
|
use App\Notifications\RequestAssetCancelation;
|
|
use App\Notifications\RequestAssetNotification;
|
|
use Illuminate\Auth\Access\AuthorizationException;
|
|
use Illuminate\Http\Request;
|
|
use Illuminate\Http\RedirectResponse;
|
|
use \Illuminate\Contracts\View\View;
|
|
use Exception;
|
|
|
|
/**
|
|
* This controller handles all actions related to the ability for users
|
|
* to view their own assets in the Snipe-IT Asset Management application.
|
|
*
|
|
* @version v1.0
|
|
*/
|
|
class ViewAssetsController extends Controller
|
|
{
|
|
/**
|
|
* Extract custom fields that should be displayed in user view.
|
|
*
|
|
* @param User $user
|
|
* @return array
|
|
*/
|
|
private function extractCustomFields(User $user): array
|
|
{
|
|
$fieldArray = [];
|
|
foreach ($user->assets as $asset) {
|
|
if ($asset->model && $asset->model->fieldset) {
|
|
foreach ($asset->model->fieldset->fields as $field) {
|
|
if ($field->display_in_user_view == '1') {
|
|
$fieldArray[$field->db_column] = $field->name;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return array_unique($fieldArray);
|
|
}
|
|
|
|
/**
|
|
* Get list of users viewable by the current user.
|
|
*
|
|
* @param User $authUser
|
|
* @return \Illuminate\Support\Collection
|
|
*/
|
|
private function getViewableUsers(User $authUser): \Illuminate\Support\Collection
|
|
{
|
|
// SuperAdmin sees all users
|
|
if ($authUser->isSuperUser()) {
|
|
return User::select('id', 'first_name', 'last_name', 'username')
|
|
->where('activated', 1)
|
|
->orderBy('last_name')
|
|
->orderBy('first_name')
|
|
->get();
|
|
}
|
|
|
|
// Regular manager sees only their subordinates + self
|
|
$managedUsers = $authUser->getAllSubordinates();
|
|
|
|
// If user has subordinates, show them with self at beginning
|
|
if ($managedUsers->count() > 0) {
|
|
return collect([$authUser])->merge($managedUsers)
|
|
->sortBy('last_name')
|
|
->sortBy('first_name');
|
|
}
|
|
|
|
// User has no subordinates, only sees themselves
|
|
return collect([$authUser]);
|
|
}
|
|
|
|
/**
|
|
* Get the selected user ID from request or default to current user.
|
|
*
|
|
* @param Request $request
|
|
* @param \Illuminate\Support\Collection $subordinates
|
|
* @param int $defaultUserId
|
|
* @return int
|
|
*/
|
|
private function getSelectedUserId(Request $request, \Illuminate\Support\Collection $subordinates, int $defaultUserId): int
|
|
{
|
|
// If no subordinates or no user_id in request, return default
|
|
if ($subordinates->count() <= 1 || !$request->filled('user_id')) {
|
|
return $defaultUserId;
|
|
}
|
|
|
|
$requestedUserId = (int) $request->input('user_id');
|
|
|
|
// Validate if the requested user is allowed
|
|
if ($subordinates->contains('id', $requestedUserId)) {
|
|
return $requestedUserId;
|
|
}
|
|
|
|
// If invalid ID or not authorized, return default
|
|
return $defaultUserId;
|
|
}
|
|
|
|
/**
|
|
* Show user's assigned assets with optional manager view functionality.
|
|
*
|
|
*/
|
|
public function getIndex(Request $request) : View | RedirectResponse
|
|
{
|
|
$authUser = auth()->user();
|
|
$settings = Setting::getSettings();
|
|
$subordinates = collect();
|
|
$selectedUserId = $authUser->id;
|
|
|
|
// Process manager view if enabled
|
|
if ($settings->manager_view_enabled) {
|
|
$subordinates = $this->getViewableUsers($authUser);
|
|
$selectedUserId = $this->getSelectedUserId($request, $subordinates, $authUser->id);
|
|
}
|
|
|
|
// Load the data for the user to be viewed (either auth user or selected subordinate)
|
|
$userToView = User::with([
|
|
'assets',
|
|
'assets.model',
|
|
'assets.model.fieldset.fields',
|
|
'consumables',
|
|
'accessories',
|
|
'licenses'
|
|
])->find($selectedUserId);
|
|
|
|
// If the user to view couldn't be found (shouldn't happen with proper logic), redirect with error
|
|
if (!$userToView) {
|
|
return redirect()->route('view-assets')->with('error', trans('admin/users/message.user_not_found'));
|
|
}
|
|
|
|
// Process custom fields for the user being viewed
|
|
$fieldArray = $this->extractCustomFields($userToView);
|
|
|
|
// Pass the necessary data to the view
|
|
return view('account/view-assets', [
|
|
'user' => $userToView, // Use 'user' for compatibility with the existing view
|
|
'field_array' => $fieldArray,
|
|
'settings' => $settings,
|
|
'subordinates' => $subordinates,
|
|
'selectedUserId' => $selectedUserId
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Returns view of requestable items for a user.
|
|
*/
|
|
public function getRequestableIndex() : View
|
|
{
|
|
$assets = Asset::with('model', 'defaultLoc', 'location', 'assignedTo', 'requests')->Hardware()->RequestableAssets();
|
|
$models = AssetModel::with([
|
|
'category',
|
|
'requests',
|
|
'assets' => function ($q) {
|
|
$q->where('requestable', 1)
|
|
->whereHas('assetstatus', fn ($s) =>
|
|
$s->where('archived', 0)
|
|
->where(fn ($s) =>
|
|
$s->where('deployable', 1)->orWhere('pending', 1)
|
|
)
|
|
);
|
|
},
|
|
])->RequestableModels()->get();
|
|
|
|
return view('account/requestable-assets', compact('assets', 'models'));
|
|
}
|
|
|
|
public function getRequestItem(Request $request, $itemType, $itemId = null, $cancel_by_admin = false, $requestingUser = null): RedirectResponse
|
|
{
|
|
$item = null;
|
|
$fullItemType = 'App\\Models\\'.studly_case($itemType);
|
|
|
|
if ($itemType == 'asset_model') {
|
|
$itemType = 'model';
|
|
}
|
|
$item = call_user_func([$fullItemType, 'find'], $itemId);
|
|
|
|
$user = auth()->user();
|
|
|
|
$logaction = new Actionlog();
|
|
$logaction->item_id = $data['asset_id'] = $item->id;
|
|
$logaction->item_type = $fullItemType;
|
|
$logaction->created_at = $data['requested_date'] = date('Y-m-d H:i:s');
|
|
|
|
if ($user->location_id) {
|
|
$logaction->location_id = $user->location_id;
|
|
}
|
|
|
|
$logaction->target_id = $data['user_id'] = auth()->id();
|
|
$logaction->target_type = User::class;
|
|
|
|
$data['item_quantity'] = $request->has('request-quantity') ? e($request->input('request-quantity')) : 1;
|
|
$data['requested_by'] = $user->display_name;
|
|
$data['item'] = $item;
|
|
$data['item_type'] = $itemType;
|
|
$data['target'] = auth()->user();
|
|
|
|
if ($fullItemType == Asset::class) {
|
|
$data['item_url'] = route('hardware.show', $item->id);
|
|
} else {
|
|
$data['item_url'] = route("view/${itemType}", $item->id);
|
|
}
|
|
|
|
$settings = Setting::getSettings();
|
|
|
|
if (($item_request = $item->isRequestedBy($user)) || $cancel_by_admin) {
|
|
$item->cancelRequest($requestingUser);
|
|
$data['item_quantity'] = ($item_request) ? $item_request->qty : 1;
|
|
$logaction->logaction(ActionType::RequestCanceled);
|
|
|
|
if (($settings->alert_email != '') && ($settings->alerts_enabled == '1') && (! config('app.lock_passwords'))) {
|
|
$settings->notify(new RequestAssetCancelation($data));
|
|
}
|
|
|
|
return redirect()->back()->with('success')->with('success', trans('admin/hardware/message.requests.canceled'));
|
|
} else {
|
|
$item->request();
|
|
if (($settings->alert_email != '') && ($settings->alerts_enabled == '1') && (! config('app.lock_passwords'))) {
|
|
$logaction->logaction('requested');
|
|
$settings->notify(new RequestAssetNotification($data));
|
|
}
|
|
|
|
return redirect()->route('requestable-assets')->with('success')->with('success', trans('admin/hardware/message.requests.success'));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Process a specific requested asset
|
|
* @param null $assetId
|
|
*/
|
|
public function store(Asset $asset): RedirectResponse
|
|
{
|
|
try {
|
|
CreateCheckoutRequestAction::run($asset, auth()->user());
|
|
return redirect()->route('requestable-assets')->with('success')->with('success', trans('admin/hardware/message.requests.success'));
|
|
} catch (AssetNotRequestable $e) {
|
|
return redirect()->back()->with('error', 'Asset is not requestable');
|
|
} catch (AuthorizationException $e) {
|
|
return redirect()->back()->with('error', trans('admin/hardware/message.requests.error'));
|
|
} catch (Exception $e) {
|
|
report($e);
|
|
return redirect()->back()->with('error', trans('general.something_went_wrong'));
|
|
}
|
|
}
|
|
|
|
public function destroy(Asset $asset): RedirectResponse
|
|
{
|
|
try {
|
|
CancelCheckoutRequestAction::run($asset, auth()->user());
|
|
return redirect()->route('requestable-assets')->with('success')->with('success', trans('admin/hardware/message.requests.canceled'));
|
|
} catch (Exception $e) {
|
|
report($e);
|
|
return redirect()->back()->with('error', trans('general.something_went_wrong'));
|
|
}
|
|
}
|
|
|
|
|
|
public function getRequestedAssets() : View
|
|
{
|
|
return view('account/requested');
|
|
}
|
|
}
|