Files
librenms-librenms/app/Http/Controllers/Select/SelectController.php
Tony Murray f78edf4d11 New Map Menu (#15969)
* Map Menu
New top level Map menu.
Ability to group custom maps.
Ajax Select controller improvements

* Fix style
2024-04-25 15:32:33 -05:00

165 lines
4.9 KiB
PHP

<?php
/**
* SelectController.php
*
* Controller class for select2 ajax controllers.
*
* 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 3 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, see <https://www.gnu.org/licenses/>.
*
* @link https://www.librenms.org
*
* @copyright 2018 Tony Murray
* @author Tony Murray <murraytony@gmail.com>
*/
namespace App\Http\Controllers\Select;
use App\Http\Controllers\PaginatedAjaxController;
use Illuminate\Contracts\Pagination\Paginator;
use Illuminate\Database\Eloquent\Builder as EloquentBuilder;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Query\Builder;
use Illuminate\Http\Request;
use Illuminate\Support\Collection;
abstract class SelectController extends PaginatedAjaxController
{
protected ?string $idField = null;
protected ?string $textField = null;
final protected function baseRules()
{
return [
'limit' => 'int',
'page' => 'int',
'term' => 'nullable|string',
];
}
/**
* The default method called by the route handler
*
* @param \Illuminate\Http\Request $request
* @return \Illuminate\Http\JsonResponse
*/
public function __invoke(Request $request)
{
$this->validate($request, $this->rules());
$limit = $request->get('limit', 50);
$query = $this->baseQuery($request);
if ($this->idField && $this->textField) {
$query->select([$this->idField, $this->textField]);
}
$this->filterById($query, $request->get('id'));
$this->filter($request, $query, $this->filterFields($request));
$this->search($request->get('term'), $query, $this->searchFields($request));
$this->sort($request, $query);
$paginator = $query->simplePaginate($limit);
return $this->formatResponse($paginator);
}
/**
* @param Paginator|Collection $paginator
* @return \Illuminate\Http\JsonResponse
*/
protected function formatResponse($paginator)
{
$results = collect($paginator->items())->map([$this, 'formatItem']);
// prepend the initial item, unless filtered out
if ($this->canPrependFirstItem(request())) {
$results->prepend($this->prependItem());
}
return response()->json([
'results' => $results,
'pagination' => ['more' => $paginator->hasMorePages()],
]);
}
/**
* Default item formatting, should supply at least id and text keys
* Check select2 docs.
* Default implementation uses primary key and the first value in the model
* If only one value is in the model attributes, that is the id and text.
*
* @param Model $model
* @return array
*/
public function formatItem($model)
{
if ($this->idField && $this->textField) {
return [
'id' => $model->getAttribute($this->idField),
'text' => $model->getAttribute($this->textField),
];
}
// guess
$attributes = collect($model->getAttributes());
return [
'id' => $attributes->count() == 1 ? $attributes->first() : $model->getKey(),
'text' => $attributes->forget($model->getKeyName())->first(),
];
}
protected function prependItem(): ?array
{
return null;
}
protected function canPrependFirstItem(Request $request): bool
{
$item = $this->prependItem();
if (empty($item)) {
return false;
}
if ($request->page > 1) {
return false;
}
if ($request->has('id') && $request->id != $item['id']) { // purposely loose comparison
return false;
}
if ($request->has('term') && ! str_contains(strtolower($item['text']), strtolower($request->term))) {
return false;
}
return true;
}
protected function filterById(EloquentBuilder|Builder $query, ?string $id): EloquentBuilder|Builder
{
if ($id) {
// multiple
if (str_contains($id, ',')) {
$keys = explode(',', $id);
return $this->idField ? $query->whereIn($this->idField, $keys) : $query->whereKey($keys);
}
// use id field if given
return $this->idField ? $query->where($this->idField, $id) : $query->whereKey($id);
}
return $query;
}
}