Views.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. <?php
  2. namespace Drupal\views;
  3. /**
  4. * Static service container wrapper for views.
  5. */
  6. class Views {
  7. /**
  8. * The translation manager.
  9. *
  10. * @var \Drupal\Core\StringTranslation\TranslationInterface
  11. */
  12. protected static $translationManager;
  13. /**
  14. * A static cache for handler types data.
  15. *
  16. * @var array
  17. */
  18. protected static $handlerTypes;
  19. /**
  20. * A list of all available views plugin types.
  21. *
  22. * @var array
  23. */
  24. protected static $plugins = [
  25. 'access' => 'plugin',
  26. 'area' => 'handler',
  27. 'argument' => 'handler',
  28. 'argument_default' => 'plugin',
  29. 'argument_validator' => 'plugin',
  30. 'cache' => 'plugin',
  31. 'display_extender' => 'plugin',
  32. 'display' => 'plugin',
  33. 'exposed_form' => 'plugin',
  34. 'field' => 'handler',
  35. 'filter' => 'handler',
  36. 'join' => 'plugin',
  37. 'pager' => 'plugin',
  38. 'query' => 'plugin',
  39. 'relationship' => 'handler',
  40. 'row' => 'plugin',
  41. 'sort' => 'handler',
  42. 'style' => 'plugin',
  43. 'wizard' => 'plugin',
  44. ];
  45. /**
  46. * Returns the views data service.
  47. *
  48. * @return \Drupal\views\ViewsData
  49. * Returns a views data cache object.
  50. */
  51. public static function viewsData() {
  52. return \Drupal::service('views.views_data');
  53. }
  54. /**
  55. * Returns the views data helper service.
  56. *
  57. * @return \Drupal\views\ViewsDataHelper
  58. * Returns a views data helper object.
  59. */
  60. public static function viewsDataHelper() {
  61. return \Drupal::service('views.views_data_helper');
  62. }
  63. /**
  64. * Returns the view executable factory service.
  65. *
  66. * @return \Drupal\views\ViewExecutableFactory
  67. * Returns a views executable factory.
  68. */
  69. public static function executableFactory() {
  70. return \Drupal::service('views.executable');
  71. }
  72. /**
  73. * Returns the view analyzer.
  74. *
  75. * @return \Drupal\views\Analyzer
  76. * Returns a view analyzer object.
  77. */
  78. public static function analyzer() {
  79. return \Drupal::service('views.analyzer');
  80. }
  81. /**
  82. * Returns the plugin manager for a certain views plugin type.
  83. *
  84. * @param string $type
  85. * The plugin type, for example filter.
  86. *
  87. * @return \Drupal\views\Plugin\ViewsPluginManager
  88. */
  89. public static function pluginManager($type) {
  90. return \Drupal::service('plugin.manager.views.' . $type);
  91. }
  92. /**
  93. * Returns the plugin manager for a certain views handler type.
  94. *
  95. * @return \Drupal\views\Plugin\ViewsHandlerManager
  96. */
  97. public static function handlerManager($type) {
  98. return \Drupal::service('plugin.manager.views.' . $type);
  99. }
  100. /**
  101. * Loads a view from configuration and returns its executable object.
  102. *
  103. * @param string $id
  104. * The view ID to load.
  105. *
  106. * @return \Drupal\views\ViewExecutable|null
  107. * A view executable instance or NULL if the view does not exist.
  108. */
  109. public static function getView($id) {
  110. $view = \Drupal::entityTypeManager()->getStorage('view')->load($id);
  111. if ($view) {
  112. return static::executableFactory()->get($view);
  113. }
  114. return NULL;
  115. }
  116. /**
  117. * Fetches a list of all base tables available
  118. *
  119. * @param string $type
  120. * Either 'display', 'style' or 'row'.
  121. * @param string $key
  122. * For style plugins, this is an optional type to restrict to. May be
  123. * 'normal', 'summary', 'feed' or others based on the needs of the display.
  124. * @param array $base
  125. * An array of possible base tables.
  126. *
  127. * @return
  128. * A keyed array of in the form of 'base_table' => 'Description'.
  129. */
  130. public static function fetchPluginNames($type, $key = NULL, array $base = []) {
  131. $definitions = static::pluginManager($type)->getDefinitions();
  132. $plugins = [];
  133. foreach ($definitions as $id => $plugin) {
  134. // Skip plugins that don't conform to our key, if they have one.
  135. if ($key && isset($plugin['display_types']) && !in_array($key, $plugin['display_types'])) {
  136. continue;
  137. }
  138. if (empty($plugin['no_ui']) && (empty($base) || empty($plugin['base']) || array_intersect($base, $plugin['base']))) {
  139. $plugins[$id] = $plugin['title'];
  140. }
  141. }
  142. if (!empty($plugins)) {
  143. asort($plugins);
  144. return $plugins;
  145. }
  146. return $plugins;
  147. }
  148. /**
  149. * Gets all the views plugin definitions.
  150. *
  151. * @return array
  152. * An array of plugin definitions for all types.
  153. */
  154. public static function getPluginDefinitions() {
  155. $plugins = [];
  156. foreach (ViewExecutable::getPluginTypes() as $plugin_type) {
  157. $plugins[$plugin_type] = static::pluginManager($plugin_type)->getDefinitions();
  158. }
  159. return $plugins;
  160. }
  161. /**
  162. * Gets enabled display extenders.
  163. */
  164. public static function getEnabledDisplayExtenders() {
  165. $enabled = array_filter((array) \Drupal::config('views.settings')->get('display_extenders'));
  166. return array_combine($enabled, $enabled);
  167. }
  168. /**
  169. * Return a list of all view IDs and display IDs that have a particular
  170. * setting in their display's plugin settings.
  171. *
  172. * @param string $type
  173. * A flag from the display plugin definitions (e.g, 'uses_menu_links').
  174. *
  175. * @return array
  176. * A list of arrays containing the $view_id and $display_id.
  177. * @code
  178. * array(
  179. * array($view_id, $display_id),
  180. * array($view_id, $display_id),
  181. * );
  182. * @endcode
  183. */
  184. public static function getApplicableViews($type) {
  185. // Get all display plugins which provides the type.
  186. $display_plugins = static::pluginManager('display')->getDefinitions();
  187. $plugin_ids = [];
  188. foreach ($display_plugins as $id => $definition) {
  189. if (!empty($definition[$type])) {
  190. $plugin_ids[$id] = $id;
  191. }
  192. }
  193. $entity_ids = \Drupal::entityQuery('view')
  194. ->condition('status', TRUE)
  195. ->condition("display.*.display_plugin", $plugin_ids, 'IN')
  196. ->execute();
  197. $result = [];
  198. foreach (\Drupal::entityTypeManager()->getStorage('view')->loadMultiple($entity_ids) as $view) {
  199. // Check each display to see if it meets the criteria and is enabled.
  200. foreach ($view->get('display') as $id => $display) {
  201. // If the key doesn't exist, enabled is assumed.
  202. $enabled = !empty($display['display_options']['enabled']) || !array_key_exists('enabled', $display['display_options']);
  203. if ($enabled && in_array($display['display_plugin'], $plugin_ids)) {
  204. $result[] = [$view->id(), $id];
  205. }
  206. }
  207. }
  208. return $result;
  209. }
  210. /**
  211. * Returns an array of all views as fully loaded $view objects.
  212. *
  213. * @return \Drupal\views\Entity\View[]
  214. * An array of loaded view entities.
  215. */
  216. public static function getAllViews() {
  217. return \Drupal::entityTypeManager()->getStorage('view')->loadMultiple();
  218. }
  219. /**
  220. * Returns an array of all enabled views.
  221. *
  222. * @return \Drupal\views\Entity\View[]
  223. * An array of loaded enabled view entities.
  224. */
  225. public static function getEnabledViews() {
  226. $query = \Drupal::entityQuery('view')
  227. ->condition('status', TRUE)
  228. ->execute();
  229. return \Drupal::entityTypeManager()->getStorage('view')->loadMultiple($query);
  230. }
  231. /**
  232. * Returns an array of all disabled views.
  233. *
  234. * @return \Drupal\views\Entity\View[]
  235. * An array of loaded disabled view entities.
  236. */
  237. public static function getDisabledViews() {
  238. $query = \Drupal::entityQuery('view')
  239. ->condition('status', FALSE)
  240. ->execute();
  241. return \Drupal::entityTypeManager()->getStorage('view')->loadMultiple($query);
  242. }
  243. /**
  244. * Returns an array of view as options array, that can be used by select,
  245. * checkboxes and radios as #options.
  246. *
  247. * @param bool $views_only
  248. * If TRUE, only return views, not displays.
  249. * @param string $filter
  250. * Filters the views on status. Can either be 'all' (default), 'enabled' or
  251. * 'disabled'
  252. * @param mixed $exclude_view
  253. * View or current display to exclude.
  254. * Either a:
  255. * - views object (containing $exclude_view->storage->name and $exclude_view->current_display)
  256. * - views name as string: e.g. my_view
  257. * - views name and display id (separated by ':'): e.g. my_view:default
  258. * @param bool $optgroup
  259. * If TRUE, returns an array with optgroups for each view (will be ignored for
  260. * $views_only = TRUE). Can be used by select
  261. * @param bool $sort
  262. * If TRUE, the list of views is sorted ascending.
  263. *
  264. * @return array
  265. * An associative array for use in select.
  266. * - key: view name and display id separated by ':', or the view name only.
  267. */
  268. public static function getViewsAsOptions($views_only = FALSE, $filter = 'all', $exclude_view = NULL, $optgroup = FALSE, $sort = FALSE) {
  269. // Filter the big views array.
  270. switch ($filter) {
  271. case 'all':
  272. case 'disabled':
  273. case 'enabled':
  274. $filter = ucfirst($filter);
  275. $views = call_user_func("static::get{$filter}Views");
  276. break;
  277. default:
  278. return [];
  279. }
  280. // Prepare exclude view strings for comparison.
  281. if (empty($exclude_view)) {
  282. $exclude_view_name = '';
  283. $exclude_view_display = '';
  284. }
  285. elseif (is_object($exclude_view)) {
  286. $exclude_view_name = $exclude_view->storage->id();
  287. $exclude_view_display = $exclude_view->current_display;
  288. }
  289. else {
  290. // Append a ':' to the $exclude_view string so we always have more than one
  291. // item to explode.
  292. list($exclude_view_name, $exclude_view_display) = explode(':', "$exclude_view:");
  293. }
  294. $options = [];
  295. foreach ($views as $view) {
  296. $id = $view->id();
  297. // Return only views.
  298. if ($views_only && $id != $exclude_view_name) {
  299. $options[$id] = $view->label();
  300. }
  301. // Return views with display ids.
  302. else {
  303. foreach ($view->get('display') as $display_id => $display) {
  304. if (!($id == $exclude_view_name && $display_id == $exclude_view_display)) {
  305. if ($optgroup) {
  306. $options[$id][$id . ':' . $display['id']] = t('@view : @display', ['@view' => $id, '@display' => $display['id']]);
  307. }
  308. else {
  309. $options[$id . ':' . $display['id']] = t('View: @view - Display: @display', ['@view' => $id, '@display' => $display['id']]);
  310. }
  311. }
  312. }
  313. }
  314. }
  315. if ($sort) {
  316. ksort($options);
  317. }
  318. return $options;
  319. }
  320. /**
  321. * Returns a list of plugins and metadata about them.
  322. *
  323. * @return array
  324. * An array keyed by PLUGIN_TYPE:PLUGIN_NAME, like 'display:page' or
  325. * 'pager:full', containing an array with the following keys:
  326. * - title: The plugin's title.
  327. * - type: The plugin type.
  328. * - module: The module providing the plugin.
  329. * - views: An array of enabled Views that are currently using this plugin,
  330. * keyed by machine name.
  331. */
  332. public static function pluginList() {
  333. $plugin_data = static::getPluginDefinitions();
  334. $plugins = [];
  335. foreach (static::getEnabledViews() as $view) {
  336. foreach ($view->get('display') as $display) {
  337. foreach ($plugin_data as $type => $info) {
  338. if ($type == 'display' && isset($display['display_plugin'])) {
  339. $name = $display['display_plugin'];
  340. }
  341. elseif (isset($display['display_options']["{$type}_plugin"])) {
  342. $name = $display['display_options']["{$type}_plugin"];
  343. }
  344. elseif (isset($display['display_options'][$type]['type'])) {
  345. $name = $display['display_options'][$type]['type'];
  346. }
  347. else {
  348. continue;
  349. }
  350. // Key first by the plugin type, then the name.
  351. $key = $type . ':' . $name;
  352. // Add info for this plugin.
  353. if (!isset($plugins[$key])) {
  354. $plugins[$key] = [
  355. 'type' => $type,
  356. 'title' => $info[$name]['title'],
  357. 'provider' => $info[$name]['provider'],
  358. 'views' => [],
  359. ];
  360. }
  361. // Add this view to the list for this plugin.
  362. $plugins[$key]['views'][$view->id()] = $view->id();
  363. }
  364. }
  365. }
  366. return $plugins;
  367. }
  368. /**
  369. * Provide a list of views handler types used in a view, with some information
  370. * about them.
  371. *
  372. * @return array
  373. * An array of associative arrays containing:
  374. * - title: The title of the handler type.
  375. * - ltitle: The lowercase title of the handler type.
  376. * - stitle: A singular title of the handler type.
  377. * - lstitle: A singular lowercase title of the handler type.
  378. * - plural: Plural version of the handler type.
  379. * - (optional) type: The actual internal used handler type. This key is
  380. * just used for header,footer,empty to link to the internal type: area.
  381. */
  382. public static function getHandlerTypes() {
  383. // Statically cache this so translation only occurs once per request for all
  384. // of these values.
  385. if (!isset(static::$handlerTypes)) {
  386. static::$handlerTypes = [
  387. 'field' => [
  388. // title
  389. 'title' => static::t('Fields'),
  390. // Lowercase title for mid-sentence.
  391. 'ltitle' => static::t('fields'),
  392. // Singular title.
  393. 'stitle' => static::t('Field'),
  394. // Singular lowercase title for mid sentence
  395. 'lstitle' => static::t('field'),
  396. 'plural' => 'fields',
  397. ],
  398. 'argument' => [
  399. 'title' => static::t('Contextual filters'),
  400. 'ltitle' => static::t('contextual filters'),
  401. 'stitle' => static::t('Contextual filter'),
  402. 'lstitle' => static::t('contextual filter'),
  403. 'plural' => 'arguments',
  404. ],
  405. 'sort' => [
  406. 'title' => static::t('Sort criteria'),
  407. 'ltitle' => static::t('sort criteria'),
  408. 'stitle' => static::t('Sort criterion'),
  409. 'lstitle' => static::t('sort criterion'),
  410. 'plural' => 'sorts',
  411. ],
  412. 'filter' => [
  413. 'title' => static::t('Filter criteria'),
  414. 'ltitle' => static::t('filter criteria'),
  415. 'stitle' => static::t('Filter criterion'),
  416. 'lstitle' => static::t('filter criterion'),
  417. 'plural' => 'filters',
  418. ],
  419. 'relationship' => [
  420. 'title' => static::t('Relationships'),
  421. 'ltitle' => static::t('relationships'),
  422. 'stitle' => static::t('Relationship'),
  423. 'lstitle' => static::t('Relationship'),
  424. 'plural' => 'relationships',
  425. ],
  426. 'header' => [
  427. 'title' => static::t('Header'),
  428. 'ltitle' => static::t('header'),
  429. 'stitle' => static::t('Header'),
  430. 'lstitle' => static::t('Header'),
  431. 'plural' => 'header',
  432. 'type' => 'area',
  433. ],
  434. 'footer' => [
  435. 'title' => static::t('Footer'),
  436. 'ltitle' => static::t('footer'),
  437. 'stitle' => static::t('Footer'),
  438. 'lstitle' => static::t('Footer'),
  439. 'plural' => 'footer',
  440. 'type' => 'area',
  441. ],
  442. 'empty' => [
  443. 'title' => static::t('No results behavior'),
  444. 'ltitle' => static::t('no results behavior'),
  445. 'stitle' => static::t('No results behavior'),
  446. 'lstitle' => static::t('No results behavior'),
  447. 'plural' => 'empty',
  448. 'type' => 'area',
  449. ],
  450. ];
  451. }
  452. return static::$handlerTypes;
  453. }
  454. /**
  455. * Returns a list of plugin types.
  456. *
  457. * @param string $type
  458. * (optional) filter the list of plugins by type. Available options are
  459. * 'plugin' or 'handler'.
  460. *
  461. * @return array
  462. * An array of plugin types.
  463. */
  464. public static function getPluginTypes($type = NULL) {
  465. if ($type === NULL) {
  466. return array_keys(static::$plugins);
  467. }
  468. if (!in_array($type, ['plugin', 'handler'])) {
  469. throw new \Exception('Invalid plugin type used. Valid types are "plugin" or "handler".');
  470. }
  471. return array_keys(array_filter(static::$plugins, function ($plugin_type) use ($type) {
  472. return $plugin_type == $type;
  473. }));
  474. }
  475. /**
  476. * Translates a string to the current language or to a given language.
  477. *
  478. * See the t() documentation for details.
  479. */
  480. protected static function t($string, array $args = [], array $options = []) {
  481. if (empty(static::$translationManager)) {
  482. static::$translationManager = \Drupal::service('string_translation');
  483. }
  484. return static::$translationManager->translate($string, $args, $options);
  485. }
  486. }