CorpusController.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. <?php
  2. namespace Drupal\edlp_corpus\Controller;
  3. use Drupal\Core\Controller\ControllerBase;
  4. use Drupal\Core\Language\LanguageInterface;
  5. use Drupal\workflow\Entity\WorkflowManager;
  6. use Drupal\Core\Url;
  7. use Drupal\File\Entity\File;
  8. use Drupal\taxonomy\Entity\Term;
  9. // use Symfony\Component\HttpFoundation\JsonResponse;
  10. use Drupal\Core\Cache\CacheableJsonResponse;
  11. use Drupal\Core\Cache\CacheableMetadata;
  12. use Drupal\core\render\RenderContext;
  13. use Drupal\Core\Ajax\AjaxResponse;
  14. class CorpusController extends ControllerBase {
  15. /**
  16. * Display the markup.
  17. *
  18. * @return array
  19. */
  20. public function contentjson() {
  21. // @see https://www.droptica.com/blog/drupal-8-restjson-integration-simple-javascript-application/
  22. // @see https://www.sitepoint.com/drupal-8-version-entityfieldquery/
  23. // @see https://www.frobiovox.com/posts/2016/03/28/simplify-drupal-8-field-value-calls.html
  24. // @see https://chromatichq.com/blog/dependency-injection-drupal-8-plugins
  25. // Get a node storage object.
  26. // $file_storage = \Drupal::entityManager()->getStorage('node');
  27. $current_langcode = \Drupal::languageManager()->getCurrentLanguage()->getId();
  28. $config = \Drupal::config('system.site');
  29. $query = \Drupal::entityQuery('node')
  30. ->condition('status', 1)
  31. ->condition('type', 'enregistrement');
  32. $nids = $query->execute();
  33. $nodes = entity_load_multiple('node', $nids);
  34. $view_builder = \Drupal::entityTypeManager()->getViewBuilder('node');
  35. $nodes_data = [];
  36. foreach ($nodes as $node) {
  37. // this would be ideal but it's too heavy to load : the whole ajax json goes from 138kb to 1.23Md (even optimized)...
  38. // $node_builder = $view_builder->view($node, 'popup');
  39. // $node = $n->getTranslation($current_langcode);
  40. // remove masqué
  41. $sid = WorkflowManager::getCurrentStateId($node, 'field_workflow');
  42. if($sid != 'corpus_documents_publie') continue;
  43. $entrees = [];
  44. foreach ($node->get('field_entrees')->getValue() as $key => $term) {
  45. $entrees[] = $term['target_id'];
  46. }
  47. // remove if no entries
  48. if(!count($entrees)) continue;
  49. if ($node->hasTranslation($current_langcode)
  50. && !$node->getTranslation($current_langcode)->field_description->isEmpty()) {
  51. $description_values = $node->getTranslation($current_langcode)->get('field_description')->getValue();
  52. }else{
  53. $description_values = $node->get('field_description')->getValue();
  54. }
  55. $description = count($description_values) ? $description_values[0]['value'] : "";
  56. // dpm($description);
  57. $field_son_values = $node->get('field_son')->getValue();
  58. $audio_fid = count($field_son_values) ? $field_son_values[0]['target_id'] : "";
  59. $audio_file = \Drupal\file\Entity\File::load($audio_fid);
  60. $audio_url = null;
  61. // if node don't have a sound file atteched, skip it
  62. if(!$audio_file) continue;
  63. $son_uri = $audio_file->getFileUri();
  64. $audio_url = file_create_url($son_uri);
  65. // has article ?
  66. $article_value = $node->body->getValue();
  67. $has_article = count($article_value);
  68. // if($has_article && $article_value[0]['value'] == "")
  69. // dpm($article_value);
  70. $document_url = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($node, $current_langcode) {
  71. if ($node->hasTranslation($current_langcode)){
  72. return $node->getTranslation($current_langcode)->toUrl()->toString();
  73. }else{
  74. return $node->toUrl()->toString();
  75. }
  76. });
  77. // favoris marker
  78. $nodes_data[] = array(
  79. "nid" => $node->get('nid')->getString(),
  80. "title" => $node->get('title')->getString(),
  81. "description" => $description,
  82. "entrees" => $entrees,
  83. // "son_fid" => $audio_fid,
  84. "audio_url" => $audio_url,
  85. "has_article" => $has_article,
  86. "chutier_action" => 'add',
  87. "document_url" => $document_url,
  88. );
  89. }
  90. $query = \Drupal::entityQuery('taxonomy_term')
  91. ->condition('vid', 'entrees');
  92. $tids = $query->execute();
  93. // $nodes = entity_load_multiple('node', $nids);
  94. foreach ($tids as $tid) {
  95. $entrees[] = $tid;
  96. }
  97. $data = array(
  98. 'date' => time(),
  99. 'site_name' => $config->get('name'),
  100. 'count' => count($nodes),
  101. 'nodes' => $nodes_data,
  102. 'entrees' => $entrees,
  103. 'language' => $current_langcode,
  104. );
  105. // https://spinningcode.org/2017/05/cached-json-responses-in-drupal-8/
  106. // cache is invalidated in edlp_corpus.module by tags
  107. $data['#cache'] = [
  108. 'max-age' => \Drupal\Core\Cache\Cache::PERMANENT,
  109. 'tags' => ['rebuild-corpus-cache'],
  110. 'contexts' => ['languages:language_content'],
  111. ];
  112. // $response = new JsonResponse();
  113. // $response->setData($data);
  114. $response = new CacheableJsonResponse($data);
  115. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($data));
  116. return $response;
  117. return array(
  118. '#markup'=>'Hello Corpus'
  119. );
  120. }
  121. // _ _ _
  122. // | | __ _ __| |_ __| |___ __ ___
  123. // | |__/ _` (_-< _/ _` / _ \/ _(_-<
  124. // |____\__,_/__/\__\__,_\___/\__/__/
  125. private function query() {
  126. $query = \Drupal::entityQuery('node')
  127. ->condition('status', 1)
  128. ->condition('type', 'enregistrement')
  129. ->sort('created', 'DESC')
  130. ->range(0,50);
  131. $nids = $query->execute();
  132. $nodes = entity_load_multiple('node', $nids);
  133. // $current_langcode = \Drupal::languageManager()->getCurrentLanguage()->getId();
  134. $this->lastdocs_nodes = [];
  135. $this->lastdocs_nids = [];
  136. foreach ($nodes as $node) {
  137. // remove masqué
  138. $sid = WorkflowManager::getCurrentStateId($node, 'field_workflow');
  139. if($sid != 'corpus_documents_publie') continue;
  140. $this->lastdocs_nodes[] = $node;
  141. // record an array of nids for corpus map filtering
  142. $this->lastdocs_nids[] = $node->get('nid')->getString();
  143. }
  144. // // record an array of nids for corpus map filtering
  145. // $this->lastdocs_nids = [];
  146. // foreach($nids as $key => $nid){
  147. // $this->lastdocs_nids[] = $nid;
  148. // }
  149. }
  150. private function toRenderable(){
  151. $this->query();
  152. // dpm($this->next_event_node);
  153. return array(
  154. "#theme"=>'edlp_corpus_lastdocs',
  155. '#lastdocs_nodes' => $this->lastdocs_nodes
  156. );
  157. }
  158. /**
  159. * Display lastdocs as a page.
  160. *
  161. * @return renderable array
  162. */
  163. public function lastdocs() {
  164. return $this->toRenderable();
  165. }
  166. /**
  167. * Get lastdocs data as json through ajax.
  168. *
  169. * @return json
  170. */
  171. public function lastdocsjson() {
  172. $renderable = $this->toRenderable();
  173. // $rendered = render($renderable);
  174. // We can't render directly the entity as it throw an exception with cachable data
  175. //http://blog.dcycle.com/blog/2018-01-24/caching-drupal-8-rest-resource/#the-dreaded-leaked-metadata-error
  176. $rendered = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($renderable) {
  177. return render($renderable);
  178. });
  179. $data = [
  180. 'rendered'=> $rendered,
  181. 'title'=>t('Recently uploaded'),
  182. 'documents_lies' => $this->lastdocs_nids,
  183. ];
  184. // translations links
  185. // use Drupal\Core\Url;
  186. // use Drupal\Core\Language\LanguageInterface;
  187. $route_name = 'edlp_corpus.lastdocs';
  188. $links = \Drupal::languageManager()->getLanguageSwitchLinks(LanguageInterface::TYPE_URL, Url::fromRoute($route_name));
  189. if (isset($links->links)) {
  190. $translations_build = [
  191. '#theme' => 'links__language_block',
  192. '#links' => $links->links,
  193. '#attributes' => ['class' => ["language-switcher-{$links->method_id}",],],
  194. '#set_active_class' => TRUE,
  195. ];
  196. $translations_rendered = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($translations_build) {return render($translations_build);});
  197. $data['translations_links'] = $translations_rendered;
  198. }
  199. $data['#cache'] = [
  200. 'max-age' => \Drupal\Core\Cache\Cache::PERMANENT,
  201. 'tags' => ['edlp-lastdocs-cache'],
  202. 'contexts' => [
  203. 'languages:language_content'
  204. ]
  205. ];
  206. // $response = new JsonResponse();
  207. // $response->setData($data);
  208. $response = new CacheableJsonResponse($data);
  209. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($data));
  210. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($renderable));
  211. return $response;
  212. }
  213. // _ _ _ _ ___ _
  214. // /_\ _ _| |_(_)__| |___ ___ |_ _|_ _ __| |_____ __
  215. // / _ \| '_| _| / _| / -_|_-< | || ' \/ _` / -_) \ /
  216. // /_/ \_\_| \__|_\__|_\___/__/ |___|_||_\__,_\___/_\_\
  217. private function articlesQuery() {
  218. $query = \Drupal::entityQuery('node')
  219. ->condition('status', 1)
  220. ->condition('type', 'enregistrement')
  221. ->condition('body', '', "<>")
  222. ->sort('created', 'DESC');
  223. // ->range(0,20);
  224. $nids = $query->execute();
  225. $nodes = entity_load_multiple('node', $nids);
  226. $current_langcode = \Drupal::languageManager()->getCurrentLanguage()->getId();
  227. $this->articles_nodes = [];
  228. $this->articles_nids = [];
  229. foreach ($nodes as $node) {
  230. // remove masqué
  231. $sid = WorkflowManager::getCurrentStateId($node, 'field_workflow');
  232. if($sid != 'corpus_documents_publie') continue;
  233. // TODO: check if article is translated
  234. if ($node->getTranslation($current_langcode)->body->isEmpty()) continue;
  235. $this->articles_nodes[] = $node;
  236. // record an array of nids for corpus map filtering
  237. $this->articles_nids[] = $node->get('nid')->getString();
  238. }
  239. }
  240. private function articlesToRenderable(){
  241. $this->articlesQuery();
  242. // dpm($this->next_event_node);
  243. return array(
  244. "#theme"=>'edlp_corpus_articlesindex',
  245. '#articles_nodes' => $this->articles_nodes
  246. );
  247. }
  248. /**
  249. * Display lastdocs as a page.
  250. *
  251. * @return renderable array
  252. */
  253. public function articlesindex() {
  254. return $this->articlesToRenderable();
  255. }
  256. /**
  257. * Get lastdocs data as json through ajax.
  258. *
  259. * @return json
  260. */
  261. public function articlesindexjson() {
  262. $renderable = $this->articlesToRenderable();
  263. // $rendered = render($renderable);
  264. // We can't render directly the entity as it throw an exception with cachable data
  265. //http://blog.dcycle.com/blog/2018-01-24/caching-drupal-8-rest-resource/#the-dreaded-leaked-metadata-error
  266. $rendered = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($renderable) {
  267. return render($renderable);
  268. });
  269. $data = [
  270. 'rendered'=> $rendered,
  271. 'title'=>'Articles',
  272. 'articles' => $this->articles_nids,
  273. 'documents_lies' => $this->articles_nids,
  274. ];
  275. // translations links
  276. // use Drupal\Core\Url;
  277. // use Drupal\Core\Language\LanguageInterface;
  278. $route_name = 'edlp_corpus.articlesindex';
  279. $links = \Drupal::languageManager()->getLanguageSwitchLinks(LanguageInterface::TYPE_URL, Url::fromRoute($route_name));
  280. if (isset($links->links)) {
  281. $translations_build = [
  282. '#theme' => 'links__language_block',
  283. '#links' => $links->links,
  284. '#attributes' => ['class' => ["language-switcher-{$links->method_id}",],],
  285. '#set_active_class' => TRUE,
  286. ];
  287. $translations_rendered = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($translations_build) {return render($translations_build);});
  288. $data['translations_links'] = $translations_rendered;
  289. }
  290. $data['#cache'] = [
  291. 'max-age' => \Drupal\Core\Cache\Cache::PERMANENT,
  292. 'tags' => ['edlp-articlesindex-cache'],
  293. 'contexts' => [
  294. 'languages:language_content'
  295. ]
  296. ];
  297. // $response = new JsonResponse();
  298. // $response->setData($data);
  299. $response = new CacheableJsonResponse($data);
  300. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($data));
  301. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($renderable));
  302. return $response;
  303. }
  304. // ___ _ _ _ _
  305. // / __|___| | |___ __| |_(_)___ _ _
  306. // | (__/ _ \ | / -_) _| _| / _ \ ' \
  307. // \___\___/_|_\___\__|\__|_\___/_||_|
  308. private function collectionQuery(){
  309. $language = \Drupal::languageManager()->getCurrentLanguage()->getId();
  310. $query = \Drupal::entityQuery('taxonomy_term')
  311. // ->sort('weight', 'DESC')
  312. // ->sort('name', 'DESC')
  313. ->condition('vid', 'entrees');
  314. $tids = $query->execute();
  315. // $terms = entity_load_multiple('taxonomy_term', $tids);
  316. // $terms = \Drupal::entityManager()->getStorage('taxonomy_term')->loadMultiple($t‌​erms);
  317. $terms = Term::loadMultiple($tids);
  318. $ordered_terms = [];
  319. foreach ($terms as $term) {
  320. // remove masqué
  321. $sid = WorkflowManager::getCurrentStateId($term, 'field_workflow');
  322. if($sid == 'generique_masque') continue;
  323. // translate the term
  324. $term = \Drupal::service('entity.repository')->getTranslationFromContext($term, $language);
  325. $name = $term->getName();
  326. $ordered_trans_terms[$name] = $term;
  327. }
  328. ksort($ordered_trans_terms);
  329. $this->entrees_terms = $ordered_trans_terms;
  330. }
  331. private function collectionToRenderable(){
  332. $this->collectionQuery();
  333. return array(
  334. "#theme"=>'edlp_corpus_collection',
  335. '#entrees_terms' => $this->entrees_terms
  336. );
  337. }
  338. public function collection(){
  339. return $this->collectionToRenderable();
  340. }
  341. public function collectionjson(){
  342. $renderable = $this->collectionToRenderable();
  343. // $rendered = render($renderable);
  344. // We can't render directly the entity as it throw an exception with cachable data
  345. //http://blog.dcycle.com/blog/2018-01-24/caching-drupal-8-rest-resource/#the-dreaded-leaked-metadata-error
  346. $rendered = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($renderable) {
  347. return render($renderable);
  348. });
  349. $data = [
  350. 'rendered'=> $rendered,
  351. 'title'=>'Collection'
  352. ];
  353. // translations links
  354. // use Drupal\Core\Url;
  355. // use Drupal\Core\Language\LanguageInterface;
  356. $route_name = 'edlp_corpus.collection';
  357. $links = \Drupal::languageManager()->getLanguageSwitchLinks(LanguageInterface::TYPE_URL, Url::fromRoute($route_name));
  358. if (isset($links->links)) {
  359. $translations_build = [
  360. '#theme' => 'links__language_block',
  361. '#links' => $links->links,
  362. '#attributes' => ['class' => ["language-switcher-{$links->method_id}",],],
  363. '#set_active_class' => TRUE,
  364. ];
  365. $translations_rendered = \Drupal::service('renderer')->executeInRenderContext(new RenderContext(), function () use ($translations_build) {return render($translations_build);});
  366. $data['translations_links'] = $translations_rendered;
  367. }
  368. $data['#cache'] = [
  369. 'max-age' => \Drupal\Core\Cache\Cache::PERMANENT,
  370. 'tags' => ['edlp-articlesindex-cache'],
  371. 'contexts' => [
  372. 'languages:language_content'
  373. ]
  374. ];
  375. // $response = new JsonResponse();
  376. // $response->setData($data);
  377. $response = new CacheableJsonResponse($data);
  378. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($data));
  379. $response->addCacheableDependency(CacheableMetadata::createFromRenderArray($renderable));
  380. return $response;
  381. }
  382. }