837 lines
27 KiB
PHP
837 lines
27 KiB
PHP
<?php
|
|
/*
|
|
* Copyright (C) 2023 Xibo Signage Ltd
|
|
*
|
|
* Xibo - Digital Signage - https://xibosignage.com
|
|
*
|
|
* This file is part of Xibo.
|
|
*
|
|
* Xibo is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* any later version.
|
|
*
|
|
* Xibo 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 Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with Xibo. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
namespace Xibo\Controller;
|
|
|
|
use Slim\Http\Response as Response;
|
|
use Slim\Http\ServerRequest as Request;
|
|
use Xibo\Event\RegionAddedEvent;
|
|
use Xibo\Event\SubPlaylistWidgetsEvent;
|
|
use Xibo\Factory\LayoutFactory;
|
|
use Xibo\Factory\ModuleFactory;
|
|
use Xibo\Factory\RegionFactory;
|
|
use Xibo\Factory\TransitionFactory;
|
|
use Xibo\Factory\WidgetFactory;
|
|
use Xibo\Support\Exception\AccessDeniedException;
|
|
use Xibo\Support\Exception\ControllerNotImplemented;
|
|
use Xibo\Support\Exception\GeneralException;
|
|
use Xibo\Support\Exception\InvalidArgumentException;
|
|
use Xibo\Support\Exception\NotFoundException;
|
|
|
|
/**
|
|
* Class Region
|
|
* @package Xibo\Controller
|
|
*/
|
|
class Region extends Base
|
|
{
|
|
/**
|
|
* @var RegionFactory
|
|
*/
|
|
private $regionFactory;
|
|
|
|
/** @var WidgetFactory */
|
|
private $widgetFactory;
|
|
|
|
/**
|
|
* @var ModuleFactory
|
|
*/
|
|
private $moduleFactory;
|
|
|
|
/**
|
|
* @var LayoutFactory
|
|
*/
|
|
private $layoutFactory;
|
|
|
|
/**
|
|
* @var TransitionFactory
|
|
*/
|
|
private $transitionFactory;
|
|
|
|
/**
|
|
* Set common dependencies.
|
|
* @param RegionFactory $regionFactory
|
|
* @param WidgetFactory $widgetFactory
|
|
* @param TransitionFactory $transitionFactory
|
|
* @param ModuleFactory $moduleFactory
|
|
* @param LayoutFactory $layoutFactory
|
|
*/
|
|
public function __construct(
|
|
$regionFactory,
|
|
$widgetFactory,
|
|
$transitionFactory,
|
|
$moduleFactory,
|
|
$layoutFactory
|
|
) {
|
|
$this->regionFactory = $regionFactory;
|
|
$this->widgetFactory = $widgetFactory;
|
|
$this->transitionFactory = $transitionFactory;
|
|
$this->layoutFactory = $layoutFactory;
|
|
$this->moduleFactory = $moduleFactory;
|
|
}
|
|
|
|
/**
|
|
* Get region by id
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
*/
|
|
public function get(Request $request, Response $response, $id)
|
|
{
|
|
$region = $this->regionFactory->getById($id);
|
|
|
|
if (!$this->getUser()->checkEditable($region)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
$this->getState()->setData([
|
|
'region' => $region,
|
|
'layout' => $this->layoutFactory->getById($region->layoutId),
|
|
'transitions' => $this->transitionData(),
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* Add a region
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws InvalidArgumentException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
* @SWG\Post(
|
|
* path="/region/{id}",
|
|
* operationId="regionAdd",
|
|
* tags={"layout"},
|
|
* summary="Add Region",
|
|
* description="Add a Region to a Layout",
|
|
* @SWG\Parameter(
|
|
* name="id",
|
|
* in="path",
|
|
* description="The Layout ID to add the Region to",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="type",
|
|
* in="formData",
|
|
* description="The type of region this should be, zone, frame, playlist or canvas. Default = frame.",
|
|
* type="string",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="width",
|
|
* in="formData",
|
|
* description="The Width, default 250",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="height",
|
|
* in="formData",
|
|
* description="The Height",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="top",
|
|
* in="formData",
|
|
* description="The Top Coordinate",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="left",
|
|
* in="formData",
|
|
* description="The Left Coordinate",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Response(
|
|
* response=201,
|
|
* description="successful operation",
|
|
* @SWG\Schema(ref="#/definitions/Region"),
|
|
* @SWG\Header(
|
|
* header="Location",
|
|
* description="Location of the new record",
|
|
* type="string"
|
|
* )
|
|
* )
|
|
* )
|
|
*/
|
|
public function add(Request $request, Response $response, $id)
|
|
{
|
|
$layout = $this->layoutFactory->getById($id);
|
|
$sanitizedParams = $this->getSanitizer($request->getParams());
|
|
|
|
if (!$this->getUser()->checkEditable($layout)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
if (!$layout->isChild()) {
|
|
throw new InvalidArgumentException(__('This Layout is not a Draft, please checkout.'), 'layoutId');
|
|
}
|
|
|
|
$layout->load([
|
|
'loadPlaylists' => true,
|
|
'loadTags' => false,
|
|
'loadPermissions' => true,
|
|
'loadCampaigns' => false
|
|
]);
|
|
|
|
// Add a new region
|
|
$region = $this->regionFactory->create(
|
|
$sanitizedParams->getString('type', ['default' => 'frame']),
|
|
$this->getUser()->userId,
|
|
'',
|
|
$sanitizedParams->getInt('width', ['default' => 250]),
|
|
$sanitizedParams->getInt('height', ['default' => 250]),
|
|
$sanitizedParams->getInt('top', ['default' => 50]),
|
|
$sanitizedParams->getInt('left', ['default' => 50]),
|
|
$sanitizedParams->getInt('zIndex', ['default' => 0])
|
|
);
|
|
|
|
$layout->regions[] = $region;
|
|
$layout->save([
|
|
'saveTags' => false
|
|
]);
|
|
|
|
// Dispatch an event to say that we have added a region
|
|
$this->getDispatcher()->dispatch(new RegionAddedEvent($layout, $region), RegionAddedEvent::$NAME);
|
|
|
|
// Return
|
|
$this->getState()->hydrate([
|
|
'httpStatus' => 201,
|
|
'message' => sprintf(__('Added %s'), $region->name),
|
|
'id' => $region->regionId,
|
|
'data' => $region
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws InvalidArgumentException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
* @SWG\Put(
|
|
* path="/region/{id}",
|
|
* operationId="regionEdit",
|
|
* tags={"layout"},
|
|
* summary="Edit Region",
|
|
* description="Edit Region",
|
|
* @SWG\Parameter(
|
|
* name="id",
|
|
* in="path",
|
|
* description="The Region ID to Edit",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="width",
|
|
* in="formData",
|
|
* description="The Width, default 250",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="height",
|
|
* in="formData",
|
|
* description="The Height",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="top",
|
|
* in="formData",
|
|
* description="The Top Coordinate",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="left",
|
|
* in="formData",
|
|
* description="The Left Coordinate",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="zIndex",
|
|
* in="formData",
|
|
* description="The Layer for this Region",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="transitionType",
|
|
* in="formData",
|
|
* description="The Transition Type. Must be a valid transition code as returned by /transition",
|
|
* type="string",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="transitionDuration",
|
|
* in="formData",
|
|
* description="The transition duration in milliseconds if required by the transition type",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="transitionDirection",
|
|
* in="formData",
|
|
* description="The transition direction if required by the transition type.",
|
|
* type="string",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="loop",
|
|
* in="formData",
|
|
* description="Flag indicating whether this region should loop if there is only 1 media item in the timeline",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Response(
|
|
* response=200,
|
|
* description="successful operation",
|
|
* @SWG\Schema(ref="#/definitions/Region")
|
|
* )
|
|
* )
|
|
*/
|
|
public function edit(Request $request, Response $response, $id)
|
|
{
|
|
$region = $this->regionFactory->getById($id);
|
|
$sanitizedParams = $this->getSanitizer($request->getParams());
|
|
|
|
if (!$this->getUser()->checkEditable($region)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
// Check that this Regions Layout is in an editable state
|
|
$layout = $this->layoutFactory->getById($region->layoutId);
|
|
|
|
if (!$layout->isChild()) {
|
|
throw new InvalidArgumentException(__('This Layout is not a Draft, please checkout.'), 'layoutId');
|
|
}
|
|
|
|
// Load before we save
|
|
$region->load();
|
|
|
|
$region->name = $sanitizedParams->getString('name');
|
|
$region->width = $sanitizedParams->getDouble('width');
|
|
$region->height = $sanitizedParams->getDouble('height');
|
|
$region->top = $sanitizedParams->getDouble('top', ['default' => 0]);
|
|
$region->left = $sanitizedParams->getDouble('left', ['default' => 0]);
|
|
$region->zIndex = $sanitizedParams->getInt('zIndex');
|
|
$region->type = $sanitizedParams->getString('type');
|
|
$region->syncKey = $sanitizedParams->getString('syncKey', ['defaultOnEmptyString' => true]);
|
|
|
|
// Loop
|
|
$region->setOptionValue('loop', $sanitizedParams->getCheckbox('loop'));
|
|
|
|
// Transitions
|
|
$region->setOptionValue('transitionType', $sanitizedParams->getString('transitionType'));
|
|
$region->setOptionValue('transitionDuration', $sanitizedParams->getInt('transitionDuration'));
|
|
$region->setOptionValue('transitionDirection', $sanitizedParams->getString('transitionDirection'));
|
|
|
|
// Save
|
|
$region->save();
|
|
|
|
// Mark the layout as needing rebuild
|
|
$layout->load(\Xibo\Entity\Layout::$loadOptionsMinimum);
|
|
|
|
$saveOptions = \Xibo\Entity\Layout::$saveOptionsMinimum;
|
|
$saveOptions['setBuildRequired'] = true;
|
|
|
|
$layout->save($saveOptions);
|
|
|
|
// Return
|
|
$this->getState()->hydrate([
|
|
'message' => sprintf(__('Edited %s'), $region->name),
|
|
'id' => $region->regionId,
|
|
'data' => $region
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* Delete a region
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws InvalidArgumentException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
* @SWG\Delete(
|
|
* path="/region/{regionId}",
|
|
* operationId="regionDelete",
|
|
* tags={"layout"},
|
|
* summary="Region Delete",
|
|
* description="Delete an existing region",
|
|
* @SWG\Parameter(
|
|
* name="regionId",
|
|
* in="path",
|
|
* description="The Region ID to Delete",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Response(
|
|
* response=204,
|
|
* description="successful operation"
|
|
* )
|
|
* )
|
|
*/
|
|
public function delete(Request $request, Response $response, $id)
|
|
{
|
|
$region = $this->regionFactory->getById($id);
|
|
|
|
if (!$this->getUser()->checkDeleteable($region)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
// Check that this Regions Layout is in an editable state
|
|
$layout = $this->layoutFactory->getById($region->layoutId);
|
|
|
|
if (!$layout->isChild())
|
|
throw new InvalidArgumentException(__('This Layout is not a Draft, please checkout.'), 'layoutId');
|
|
|
|
$region->delete();
|
|
|
|
// Return
|
|
$this->getState()->hydrate([
|
|
'httpStatus' => 204,
|
|
'message' => sprintf(__('Deleted %s'), $region->name)
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* Update Positions
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws InvalidArgumentException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
* @SWG\Put(
|
|
* path="/region/position/all/{layoutId}",
|
|
* operationId="regionPositionAll",
|
|
* tags={"layout"},
|
|
* summary="Position Regions",
|
|
* description="Position all regions for a Layout",
|
|
* @SWG\Parameter(
|
|
* name="layoutId",
|
|
* in="path",
|
|
* description="The Layout ID",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="regions",
|
|
* in="formData",
|
|
* description="Array of regions and their new positions. Each array element should be json encoded and have regionId, top, left, width and height.",
|
|
* type="array",
|
|
* required=true,
|
|
* @SWG\Items(
|
|
* type="string"
|
|
* )
|
|
* ),
|
|
* @SWG\Response(
|
|
* response=200,
|
|
* description="successful operation",
|
|
* @SWG\Schema(ref="#/definitions/Layout")
|
|
* )
|
|
* )
|
|
*/
|
|
function positionAll(Request $request, Response $response, $id)
|
|
{
|
|
// Create the layout
|
|
$layout = $this->layoutFactory->loadById($id);
|
|
|
|
if (!$this->getUser()->checkEditable($layout)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
// Check that this Layout is a Draft
|
|
if (!$layout->isChild()) {
|
|
throw new InvalidArgumentException(__('This Layout is not a Draft, please checkout.'), 'layoutId');
|
|
}
|
|
|
|
// Pull in the regions and convert them to stdObjects
|
|
$regions = $request->getParam('regions', null);
|
|
|
|
if ($regions == null) {
|
|
throw new InvalidArgumentException(__('No regions present'));
|
|
}
|
|
$regions = json_decode($regions);
|
|
|
|
// Go through each region and update the region in the layout we have
|
|
foreach ($regions as $newCoordinates) {
|
|
// TODO attempt to sanitize?
|
|
// Check that the properties we are expecting do actually exist
|
|
if (!property_exists($newCoordinates, 'regionid'))
|
|
throw new InvalidArgumentException(__('Missing regionid property'));
|
|
|
|
if (!property_exists($newCoordinates, 'top'))
|
|
throw new InvalidArgumentException(__('Missing top property'));
|
|
|
|
if (!property_exists($newCoordinates, 'left'))
|
|
throw new InvalidArgumentException(__('Missing left property'));
|
|
|
|
if (!property_exists($newCoordinates, 'width'))
|
|
throw new InvalidArgumentException(__('Missing width property'));
|
|
|
|
if (!property_exists($newCoordinates, 'height'))
|
|
throw new InvalidArgumentException(__('Missing height property'));
|
|
|
|
$regionId = $newCoordinates->regionid;
|
|
|
|
// Load the region
|
|
$region = $layout->getRegion($regionId);
|
|
|
|
// Check Permissions
|
|
if (!$this->getUser()->checkEditable($region)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
// New coordinates
|
|
$region->top = $newCoordinates->top;
|
|
$region->left = $newCoordinates->left;
|
|
$region->width = $newCoordinates->width;
|
|
$region->height = $newCoordinates->height;
|
|
$region->zIndex = $newCoordinates->zIndex;
|
|
$this->getLog()->debug('Set ' . $region);
|
|
}
|
|
|
|
// Mark the layout as having changed
|
|
$layout->status = 0;
|
|
$layout->save();
|
|
|
|
// Return
|
|
$this->getState()->hydrate([
|
|
'message' => sprintf(__('Edited %s'), $layout->layout),
|
|
'id' => $layout->layoutId,
|
|
'data' => $layout
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* Represents the Preview inside the Layout Designer
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws \Twig\Error\LoaderError
|
|
* @throws \Twig\Error\RuntimeError
|
|
* @throws \Twig\Error\SyntaxError
|
|
* @throws \Xibo\Support\Exception\ControllerNotImplemented
|
|
* @throws \Xibo\Support\Exception\GeneralException
|
|
*/
|
|
public function preview(Request $request, Response $response, $id)
|
|
{
|
|
$sanitizedQuery = $this->getSanitizer($request->getParams());
|
|
|
|
$widgetId = $sanitizedQuery->getInt('widgetId', ['default' => null]);
|
|
$seq = $sanitizedQuery->getInt('seq', ['default' => 1]);
|
|
|
|
// Load our region
|
|
try {
|
|
$region = $this->regionFactory->getById($id);
|
|
$region->load();
|
|
|
|
// What type of region are we?
|
|
$additionalContexts = [];
|
|
if ($region->type === 'canvas' || $region->type === 'playlist') {
|
|
$this->getLog()->debug('preview: canvas or playlist region');
|
|
|
|
// Get the first playlist we can find
|
|
$playlist = $region->getPlaylist()->setModuleFactory($this->moduleFactory);
|
|
|
|
// Expand this Playlist out to its individual Widgets
|
|
$widgets = $playlist->expandWidgets();
|
|
|
|
$countWidgets = count($widgets);
|
|
|
|
// Select the widget at the required sequence
|
|
$widget = $playlist->getWidgetAt($seq, $widgets);
|
|
$widget->load();
|
|
} else {
|
|
$this->getLog()->debug('preview: single widget');
|
|
|
|
// Assume we're a frame, single Widget Requested
|
|
$widget = $this->widgetFactory->getById($widgetId);
|
|
$widget->load();
|
|
|
|
if ($widget->type === 'subplaylist') {
|
|
// Get the sub-playlist widgets
|
|
$event = new SubPlaylistWidgetsEvent($widget, $widget->tempId);
|
|
$this->getDispatcher()->dispatch($event, SubPlaylistWidgetsEvent::$NAME);
|
|
$additionalContexts['countSubPlaylistWidgets'] = count($event->getWidgets());
|
|
}
|
|
|
|
$countWidgets = 1;
|
|
}
|
|
|
|
$this->getLog()->debug('There are ' . $countWidgets . ' widgets.');
|
|
|
|
// Output a preview
|
|
$module = $this->moduleFactory->getByType($widget->type);
|
|
$this->getState()->html = $this->moduleFactory
|
|
->createWidgetHtmlRenderer()
|
|
->preview(
|
|
$module,
|
|
$region,
|
|
$widget,
|
|
$sanitizedQuery,
|
|
$this->urlFor(
|
|
$request,
|
|
'library.download',
|
|
[
|
|
'regionId' => $region->regionId,
|
|
'id' => $widget->getPrimaryMedia()[0] ?? null
|
|
]
|
|
) . '?preview=1',
|
|
$additionalContexts
|
|
);
|
|
$this->getState()->extra['countOfWidgets'] = $countWidgets;
|
|
$this->getState()->extra['empty'] = false;
|
|
} catch (NotFoundException) {
|
|
$this->getState()->extra['empty'] = true;
|
|
$this->getState()->extra['text'] = __('Empty Playlist');
|
|
} catch (InvalidArgumentException $e) {
|
|
$this->getState()->extra['empty'] = true;
|
|
$this->getState()->extra['text'] = __('Please correct the error with this Widget');
|
|
}
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
private function transitionData()
|
|
{
|
|
return [
|
|
'in' => $this->transitionFactory->getEnabledByType('in'),
|
|
'out' => $this->transitionFactory->getEnabledByType('out'),
|
|
'compassPoints' => array(
|
|
array('id' => 'N', 'name' => __('North')),
|
|
array('id' => 'NE', 'name' => __('North East')),
|
|
array('id' => 'E', 'name' => __('East')),
|
|
array('id' => 'SE', 'name' => __('South East')),
|
|
array('id' => 'S', 'name' => __('South')),
|
|
array('id' => 'SW', 'name' => __('South West')),
|
|
array('id' => 'W', 'name' => __('West')),
|
|
array('id' => 'NW', 'name' => __('North West'))
|
|
)
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Add a drawer
|
|
|
|
* @SWG\Post(
|
|
* path="/region/drawer/{id}",
|
|
* operationId="regionDrawerAdd",
|
|
* tags={"layout"},
|
|
* summary="Add drawer Region",
|
|
* description="Add a drawer Region to a Layout",
|
|
* @SWG\Parameter(
|
|
* name="id",
|
|
* in="path",
|
|
* description="The Layout ID to add the Region to",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Response(
|
|
* response=201,
|
|
* description="successful operation",
|
|
* @SWG\Schema(ref="#/definitions/Region"),
|
|
* @SWG\Header(
|
|
* header="Location",
|
|
* description="Location of the new record",
|
|
* type="string"
|
|
* )
|
|
* )
|
|
* )
|
|
*
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws InvalidArgumentException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
*/
|
|
public function addDrawer(Request $request, Response $response, $id) :Response
|
|
{
|
|
$layout = $this->layoutFactory->getById($id);
|
|
if (!$this->getUser()->checkEditable($layout)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
if (!$layout->isChild()) {
|
|
throw new InvalidArgumentException(__('This Layout is not a Draft, please checkout.'), 'layoutId');
|
|
}
|
|
|
|
$layout->load([
|
|
'loadPlaylists' => true,
|
|
'loadTags' => false,
|
|
'loadPermissions' => true,
|
|
'loadCampaigns' => false
|
|
]);
|
|
|
|
// Add a new region
|
|
// we default to layout width/height/0/0
|
|
$drawer = $this->regionFactory->create(
|
|
'drawer',
|
|
$this->getUser()->userId,
|
|
$layout->layout . '-' . (count($layout->regions) + 1 . ' - drawer'),
|
|
$layout->width,
|
|
$layout->height,
|
|
0,
|
|
0,
|
|
0,
|
|
1
|
|
);
|
|
|
|
$layout->drawers[] = $drawer;
|
|
$layout->save([
|
|
'saveTags' => false
|
|
]);
|
|
|
|
// Return
|
|
$this->getState()->hydrate([
|
|
'httpStatus' => 201,
|
|
'message' => sprintf(__('Added drawer %s'), $drawer->name),
|
|
'id' => $drawer->regionId,
|
|
'data' => $drawer
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
|
|
/**
|
|
* @param Request $request
|
|
* @param Response $response
|
|
* @param $id
|
|
* @return \Psr\Http\Message\ResponseInterface|Response
|
|
* @throws AccessDeniedException
|
|
* @throws GeneralException
|
|
* @throws InvalidArgumentException
|
|
* @throws NotFoundException
|
|
* @throws ControllerNotImplemented
|
|
*
|
|
* @SWG\Put(
|
|
* path="/region/drawer/{id}",
|
|
* operationId="regionDrawerSave",
|
|
* tags={"layout"},
|
|
* summary="Save Drawer",
|
|
* description="Save Drawer",
|
|
* @SWG\Parameter(
|
|
* name="id",
|
|
* in="path",
|
|
* description="The Drawer ID to Save",
|
|
* type="integer",
|
|
* required=true
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="width",
|
|
* in="formData",
|
|
* description="The Width, default 250",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Parameter(
|
|
* name="height",
|
|
* in="formData",
|
|
* description="The Height",
|
|
* type="integer",
|
|
* required=false
|
|
* ),
|
|
* @SWG\Response(
|
|
* response=200,
|
|
* description="successful operation",
|
|
* @SWG\Schema(ref="#/definitions/Region")
|
|
* )
|
|
* )
|
|
*/
|
|
public function saveDrawer(Request $request, Response $response, $id)
|
|
{
|
|
$region = $this->regionFactory->getById($id);
|
|
$sanitizedParams = $this->getSanitizer($request->getParams());
|
|
|
|
if (!$this->getUser()->checkEditable($region)) {
|
|
throw new AccessDeniedException();
|
|
}
|
|
|
|
// Check that this Regions Layout is in an editable state
|
|
$layout = $this->layoutFactory->getById($region->layoutId);
|
|
|
|
if (!$layout->isChild()) {
|
|
throw new InvalidArgumentException(__('This Layout is not a Draft, please checkout.'), 'layoutId');
|
|
}
|
|
|
|
// Save
|
|
$region->load();
|
|
$region->width = $sanitizedParams->getDouble('width', ['default' => $layout->width]);
|
|
$region->height = $sanitizedParams->getDouble('height', ['default' => $layout->height]);
|
|
$region->save();
|
|
|
|
// Return
|
|
$this->getState()->hydrate([
|
|
'message' => sprintf(__('Edited Drawer %s'), $region->name),
|
|
'id' => $region->regionId,
|
|
'data' => $region
|
|
]);
|
|
|
|
return $this->render($request, $response);
|
|
}
|
|
}
|