mirror of
				https://github.com/alchemy-fr/Phraseanet.git
				synced 2025-10-26 03:13:12 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			400 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			400 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /*
 | |
|  * This file is part of Phraseanet
 | |
|  *
 | |
|  * (c) 2005-2016 Alchemy
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| namespace Alchemy\Phrasea\Controller\Prod;
 | |
| 
 | |
| use Alchemy\Phrasea\Application;
 | |
| use Alchemy\Phrasea\Application\Helper\EntityManagerAware;
 | |
| use Alchemy\Phrasea\Application\Helper\SearchEngineAware;
 | |
| use Alchemy\Phrasea\Controller\Controller;
 | |
| use Alchemy\Phrasea\Controller\RecordsRequest;
 | |
| use Alchemy\Phrasea\Core\Event\Record\DeleteEvent;
 | |
| use Alchemy\Phrasea\Core\Event\Record\RecordEvents;
 | |
| use Alchemy\Phrasea\Core\Event\RecordEdit;
 | |
| use Alchemy\Phrasea\Core\PhraseaEvents;
 | |
| use Alchemy\Phrasea\Model\Entities\BasketElement;
 | |
| use Alchemy\Phrasea\Model\Repositories\BasketElementRepository;
 | |
| use Alchemy\Phrasea\Model\Repositories\StoryWZRepository;
 | |
| use Alchemy\Phrasea\SearchEngine\SearchEngineOptions;
 | |
| use Alchemy\Phrasea\Twig\Fit;
 | |
| use Alchemy\Phrasea\Twig\PhraseanetExtension;
 | |
| use Symfony\Component\EventDispatcher\EventDispatcherInterface;
 | |
| use Symfony\Component\HttpFoundation\Request;
 | |
| use Symfony\Component\HttpFoundation\Response;
 | |
| 
 | |
| class RecordController extends Controller
 | |
| {
 | |
|     use EntityManagerAware;
 | |
|     use SearchEngineAware;
 | |
|     /**
 | |
|      * Get record detailed view
 | |
|      *
 | |
|      * @param Request $request
 | |
|      *
 | |
|      * @return \Symfony\Component\HttpFoundation\JsonResponse
 | |
|      */
 | |
|     public function getRecord(Request $request)
 | |
|     {
 | |
|         if (!$request->isXmlHttpRequest()) {
 | |
|             $this->app->abort(400);
 | |
|         }
 | |
| 
 | |
|         $searchEngine = $options = null;
 | |
|         $train = '';
 | |
| 
 | |
|         if ('' === $env = strtoupper($request->get('env', ''))) {
 | |
|             $this->app->abort(400, '`env` parameter is missing');
 | |
|         }
 | |
| 
 | |
|         // Use $request->get as HTTP method can be POST or GET
 | |
|         if ('RESULT' == $env = strtoupper($request->get('env', ''))) {
 | |
|             try {
 | |
|                 $options = SearchEngineOptions::hydrate($this->app, $request->get('options_serial'));
 | |
|                 $searchEngine = $this->getSearchEngine();
 | |
|             } catch (\Exception $e) {
 | |
|                 $this->app->abort(400, 'Search-engine options are not valid or missing');
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $pos = (int) $request->get('pos', 0);
 | |
|         $query = $request->get('query', '');
 | |
|         $reloadTrain = !! $request->get('roll', false);
 | |
| 
 | |
|         $record = new \record_preview(
 | |
|             $this->app,
 | |
|             $env,
 | |
|             $pos < 0 ? 0 : $pos,
 | |
|             $request->get('cont', ''),
 | |
|             $searchEngine,
 | |
|             $query,
 | |
|             $options
 | |
|         );
 | |
| 
 | |
|         $currentRecord = $this->getContainerResult($record);
 | |
| 
 | |
|         if ($record->is_from_reg()) {
 | |
|             $train = $this->render('prod/preview/reg_train.html.twig', ['record' => $record]);
 | |
|         } else if ($record->is_from_basket() && $reloadTrain) {
 | |
|             $train = $this->render('prod/preview/basket_train.html.twig', ['record' => $record]);
 | |
|         } else if ($record->is_from_feed()) {
 | |
|             $train = $this->render('prod/preview/feed_train.html.twig', ['record' => $record]);
 | |
|         }
 | |
| 
 | |
|         $recordCaptions = [];
 | |
|         foreach ($record->get_caption()->get_fields(null, true) as $field) {
 | |
|             // get field's values
 | |
|             $recordCaptions[$field->get_name()] = $field->get_serialized_values();
 | |
|         }
 | |
|         $recordCaptions["technicalInfo"] = $record->getPositionFromTechnicalInfos();
 | |
| 
 | |
|         //  escape record title before rendering
 | |
|         $recordTitle = explode("</span>", $record->get_title());
 | |
|         if (count($recordTitle) >1) {
 | |
|             $recordTitle[1] = htmlspecialchars($recordTitle[1]);
 | |
|             $recordTitle = implode("</span>", $recordTitle);
 | |
|         } else {
 | |
|             $recordTitle = htmlspecialchars($record->get_title());
 | |
|         }
 | |
| 
 | |
|         return $this->app->json([
 | |
|             "desc"            => $this->render('prod/preview/caption.html.twig', [
 | |
|                 'record'        => $record,
 | |
|                 'highlight'     => $query,
 | |
|                 'searchEngine'  => $searchEngine,
 | |
|                 'searchOptions' => $options,
 | |
|             ]),
 | |
|             "recordCaptions"  => $recordCaptions,
 | |
|             "html_preview"    => $this->render('common/preview.html.twig', [
 | |
|                 'record'        => $record
 | |
|             ]),
 | |
|             "others"          => $this->render('prod/preview/appears_in.html.twig', [
 | |
|                 'parents'       => $record->get_grouping_parents(),
 | |
|                 'baskets'       => $record->get_container_baskets($this->getEntityManager(), $this->getAuthenticatedUser()),
 | |
|             ]),
 | |
|             "current"         => $train,
 | |
|             "record"          => $currentRecord,
 | |
|             "history"         => $this->render('prod/preview/short_history.html.twig', [
 | |
|                 'record'        => $record,
 | |
|             ]),
 | |
|             "popularity"      => $this->render('prod/preview/popularity.html.twig', [
 | |
|                 'record'        => $record,
 | |
|             ]),
 | |
|             "tools"           => $this->render('prod/preview/tools.html.twig', [
 | |
|                 'record'        => $record,
 | |
|             ]),
 | |
|             "pos"             => $record->getNumber(),
 | |
|             "title"           => $recordTitle,
 | |
|             "databox_name"    => $record->getDatabox()->get_dbname(),
 | |
|             "collection_name" => $record->getCollection()->get_name(),
 | |
|             "collection_logo" => $record->getCollection()->getLogo($record->getBaseId(), $this->app),
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param \record_preview $recordContainer
 | |
|      * @return array
 | |
|      */
 | |
|     private function getContainerResult(\record_preview $recordContainer)
 | |
|     {
 | |
|         /* @var $recordPreview \media_subdef */
 | |
|         $helpers = new PhraseanetExtension($this->app);
 | |
| 
 | |
|         $recordData = [
 | |
|           'databoxId' => $recordContainer->getBaseId(),
 | |
|           'id' => $recordContainer->getId(),
 | |
|           'isGroup' => $recordContainer->isStory(),
 | |
|           'url' => (string)$helpers->getThumbnailUrl($recordContainer),
 | |
|         ];
 | |
|         $userHaveAccess = $this->app->getAclForUser($this->getAuthenticatedUser())->has_access_to_subdef($recordContainer, 'preview');
 | |
|         if ($userHaveAccess) {
 | |
|             $recordPreview = $recordContainer->get_preview();
 | |
|         } else {
 | |
|             $recordPreview = $recordContainer->get_thumbnail();
 | |
|         }
 | |
| 
 | |
|         $recordData['preview'] = [
 | |
|           'width' => $recordPreview->get_width(),
 | |
|           'height' => $recordPreview->get_height(),
 | |
|           'url' => $this->app->url('alchemy_embed_view', [
 | |
|             'url' => (string)($this->getAuthenticatedUser() ? $recordPreview->get_url() : $recordPreview->get_permalink()->get_url()),
 | |
|             'autoplay' => false
 | |
|           ])
 | |
|         ];
 | |
| 
 | |
|         return $recordData;
 | |
|     }
 | |
| 
 | |
|     public function getRecordById($sbasId, $recordId)
 | |
|     {
 | |
|         $record = new \record_adapter($this->app, $sbasId, $recordId);
 | |
|         return $this->app->json([
 | |
|             "html_preview"  => $this->render('common/preview.html.twig', [
 | |
|                 'record'        => $record
 | |
|             ]),
 | |
|             "desc"  => $this->render('common/caption.html.twig', [
 | |
|                 'record'        => $record,
 | |
|                 'view'          => 'preview'
 | |
|             ])
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      *  Delete a record or a list of records
 | |
|      *
 | |
|      * @param  Request $request
 | |
|      * @return Response
 | |
|      */
 | |
|     public function doDeleteRecords(Request $request)
 | |
|     {
 | |
|         $flatten = (bool)($request->request->get('del_children')) ? RecordsRequest::FLATTEN_YES_PRESERVE_STORIES : RecordsRequest::FLATTEN_NO;
 | |
|         $records = RecordsRequest::fromRequest(
 | |
|             $this->app,
 | |
|             $request,
 | |
|             $flatten,
 | |
|             [\ACL::CANDELETERECORD]
 | |
|         );
 | |
| 
 | |
|         $basketElementsRepository = $this->getBasketElementRepository();
 | |
|         $StoryWZRepository = $this->getStoryWorkZoneRepository();
 | |
| 
 | |
|         $deleted = [];
 | |
| 
 | |
|         /** @var \collection[] $trashCollectionsBySbasId */
 | |
|         $trashCollectionsBySbasId = [];
 | |
| 
 | |
|         $manager = $this->getEntityManager();
 | |
| 
 | |
|         /** @var \record_adapter $record */
 | |
|         foreach ($records as $record) {
 | |
|             try {
 | |
|                 $basketElements = $basketElementsRepository->findElementsByRecord($record);
 | |
| 
 | |
|                 foreach ($basketElements as $element) {
 | |
|                     $manager->remove($element);
 | |
|                     $deleted[] = $element->getRecord($this->app)->getId();
 | |
|                 }
 | |
| 
 | |
|                 $attachedStories = $StoryWZRepository->findByRecord($this->app, $record);
 | |
| 
 | |
|                 foreach ($attachedStories as $attachedStory) {
 | |
|                     $manager->remove($attachedStory);
 | |
|                 }
 | |
| 
 | |
|                 foreach ($record->get_grouping_parents() as $story) {
 | |
|                     $this->getEventDispatcher()->dispatch(PhraseaEvents::RECORD_EDIT, new RecordEdit($story));
 | |
|                 }
 | |
| 
 | |
|                 $sbasId = $record->getDatabox()->get_sbas_id();
 | |
|                 if (!array_key_exists($sbasId, $trashCollectionsBySbasId)) {
 | |
|                     $trashCollectionsBySbasId[$sbasId] = $record->getDatabox()->getTrashCollection();
 | |
|                 }
 | |
|                 $deleted[] = $record->getId();
 | |
|                 if ($trashCollectionsBySbasId[$sbasId] !== null) {
 | |
|                     if($record->getCollection()->get_coll_id() == $trashCollectionsBySbasId[$sbasId]->get_coll_id()) {
 | |
|                         // record is already in trash so delete it
 | |
|                         $this->getEventDispatcher()->dispatch(RecordEvents::DELETE, new DeleteEvent($record));
 | |
|                     } else {
 | |
|                         // move to trash collection
 | |
|                         $record->move_to_collection($trashCollectionsBySbasId[$sbasId], $this->getApplicationBox());
 | |
|                         // disable permalinks
 | |
|                         foreach($record->get_subdefs() as $subdef) {
 | |
|                             if( ($pl = $subdef->get_permalink()) ) {
 | |
|                                     $pl->set_is_activated(false);
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 } else {
 | |
|                     // no trash collection, delete
 | |
|                     $this->getEventDispatcher()->dispatch(RecordEvents::DELETE, new DeleteEvent($record));
 | |
|                 }
 | |
|             } catch (\Exception $e) {
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $manager->flush();
 | |
| 
 | |
|         return $this->app->json($deleted);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return BasketElementRepository
 | |
|      */
 | |
|     private function getBasketElementRepository()
 | |
|     {
 | |
|         return $this->app['repo.basket-elements'];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return StoryWZRepository
 | |
|      */
 | |
|     private function getStoryWorkZoneRepository()
 | |
|     {
 | |
|         return $this->app['repo.story-wz'];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      *  Delete a record or a list of records
 | |
|      *
 | |
|      * @param  Request $request
 | |
|      * @return string   html
 | |
|      */
 | |
|     public function whatCanIDelete(Request $request)
 | |
|     {
 | |
|         $viewParms = [];
 | |
| 
 | |
|         // pre-count records that would be trashed/deleted when the "deleted children" will be un-checked
 | |
| 
 | |
|         $records = RecordsRequest::fromRequest(
 | |
|             $this->app,
 | |
|             $request,
 | |
|             RecordsRequest::FLATTEN_NO,
 | |
|             [\ACL::CANDELETERECORD]
 | |
|         );
 | |
| 
 | |
|         $filteredRecords = $this->filterRecordToDelete($records);
 | |
| 
 | |
|         $viewParms['parents_only'] = [
 | |
|             'records'        => $records,
 | |
|             'trashableCount' => count($filteredRecords['trash']),
 | |
|             'deletableCount' => count($filteredRecords['delete'])
 | |
|         ];
 | |
| 
 | |
|         // pre-count records that would be trashed/deleted when the "deleted children" will be checked
 | |
|         //
 | |
|         $records = RecordsRequest::fromRequest(
 | |
|             $this->app,
 | |
|             $request,
 | |
|             RecordsRequest::FLATTEN_YES_PRESERVE_STORIES,
 | |
|             [\ACL::CANDELETERECORD]
 | |
|         );
 | |
|         $filteredRecords = $this->filterRecordToDelete($records);
 | |
|         $viewParms['with_children'] = [
 | |
|             'records'        => $records,
 | |
|             'trashableCount' => count($filteredRecords['trash']),
 | |
|             'deletableCount' => count($filteredRecords['delete'])
 | |
|         ];
 | |
| 
 | |
|         return $this->render(
 | |
|             'prod/actions/delete_records_confirm.html.twig',
 | |
|             $viewParms
 | |
|         );
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * classifies records in two groups (does NOT delete anything)
 | |
|      * - 'trash'  : the record can go to trash because the db has a "_TRASH_" coll, and the record is not already into it
 | |
|      * - 'delete' : the record would be deleted because the db has no trash, or the record is already trashed
 | |
|      *
 | |
|      * @param RecordsRequest $records
 | |
|      * @return array
 | |
|      */
 | |
|     private function filterRecordToDelete(RecordsRequest $records)
 | |
|     {
 | |
|         $ret = [
 | |
|             'trash'  => [],
 | |
|             'delete' => []
 | |
|         ];
 | |
| 
 | |
|         $trashCollectionsBySbasId = [];
 | |
|         foreach ($records as $record) {
 | |
|             /** @var \record_adapter $record */
 | |
|             $sbasId = $record->getDatabox()->get_sbas_id();
 | |
|             if (!array_key_exists($sbasId, $trashCollectionsBySbasId)) {
 | |
|                 $trashCollectionsBySbasId[$sbasId] = $record->getDatabox()->getTrashCollection();
 | |
|             }
 | |
|             if ($trashCollectionsBySbasId[$sbasId] !== null) {
 | |
|                 if ($record->getCollection()->get_coll_id() == $trashCollectionsBySbasId[$sbasId]->get_coll_id()) {
 | |
|                     // record is already in trash
 | |
|                     $ret['delete'][] = $record;
 | |
|                 }
 | |
|                 else {
 | |
|                     // will be moved to trash
 | |
|                     $ret['trash'][] = $record;
 | |
|                 }
 | |
|             }
 | |
|             else {
 | |
|                 // trash does not exist
 | |
|                 $ret['delete'][] = $record;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $ret;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      *  Renew url list of records
 | |
|      *
 | |
|      * @param Request $request
 | |
|      *
 | |
|      * @return \Symfony\Component\HttpFoundation\JsonResponse
 | |
|      * @throws \Alchemy\Phrasea\Cache\Exception
 | |
|      */
 | |
|     public function renewUrl(Request $request)
 | |
|     {
 | |
|         $records = RecordsRequest::fromRequest($this->app, $request, !!$request->request->get('renew_children_url'));
 | |
| 
 | |
|         $renewed = [];
 | |
|         foreach ($records as $record) {
 | |
|             $renewed[$record->getId()] = (string) $record->get_preview()->renew_url();
 | |
|         };
 | |
| 
 | |
|         return $this->app->json($renewed);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return EventDispatcherInterface
 | |
|      */
 | |
|     private function getEventDispatcher()
 | |
|     {
 | |
|         return $this->app['dispatcher'];
 | |
|     }
 | |
| }
 | 
