AddDependencyCallsCompilerPass.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. <?php
  2. /*
  3. * This file is part of the Sonata project.
  4. *
  5. * (c) Thomas Rabaix <thomas.rabaix@sonata-project.org>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Sonata\AdminBundle\DependencyInjection\Compiler;
  11. use Symfony\Component\DependencyInjection\Definition;
  12. use Symfony\Component\DependencyInjection\ContainerBuilder;
  13. use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
  14. use Symfony\Component\DependencyInjection\Reference;
  15. use Symfony\Component\DependencyInjection\ContainerInterface;
  16. use Symfony\Component\Config\Definition\Exception\InvalidConfigurationException;
  17. /**
  18. * Add all dependencies to the Admin class, this avoid to write too many lines
  19. * in the configuration files.
  20. *
  21. * @author Thomas Rabaix <thomas.rabaix@sonata-project.org>
  22. */
  23. class AddDependencyCallsCompilerPass implements CompilerPassInterface
  24. {
  25. /**
  26. * {@inheritDoc}
  27. */
  28. public function process(ContainerBuilder $container)
  29. {
  30. $groupDefaults = $admins = $classes = array();
  31. $pool = $container->getDefinition('sonata.admin.pool');
  32. foreach ($container->findTaggedServiceIds('sonata.admin') as $id => $tags) {
  33. foreach ($tags as $attributes) {
  34. $definition = $container->getDefinition($id);
  35. $arguments = $definition->getArguments();
  36. if (strlen($arguments[0]) == 0) {
  37. $definition->replaceArgument(0, $id);
  38. }
  39. if (strlen($arguments[2]) == 0) {
  40. $definition->replaceArgument(2, 'SonataAdminBundle:CRUD');
  41. }
  42. $this->applyConfigurationFromAttribute($definition, $attributes);
  43. $this->applyDefaults($container, $id, $attributes);
  44. $arguments = $definition->getArguments();
  45. $admins[] = $id;
  46. $classes[$arguments[1]] = $id;
  47. $showInDashBord = (boolean)(isset($attributes['show_in_dashboard']) ? $attributes['show_in_dashboard'] : true);
  48. if (!$showInDashBord) {
  49. continue;
  50. }
  51. $groupName = isset($attributes['group']) ? $attributes['group'] : 'default';
  52. $labelCatalogue = isset($attributes['label_catalogue']) ? $attributes['label_catalogue'] : 'SonataAdminBundle';
  53. if (!isset($groupDefaults[$groupName])) {
  54. $groupDefaults[$groupName] = array(
  55. 'label' => $groupName,
  56. 'label_catalogue' => $labelCatalogue
  57. );
  58. }
  59. $groupDefaults[$groupName]['items'][] = $id;
  60. }
  61. }
  62. $dashboardGroupsSettings = $container->getParameter('sonata.admin.configuration.dashboard_groups');
  63. if (!empty($dashboardGroupsSettings)) {
  64. $groups = $dashboardGroupsSettings;
  65. foreach ($dashboardGroupsSettings as $groupName => $group) {
  66. if (!isset($groupDefaults[$groupName])) {
  67. $groupDefaults[$groupName] = array(
  68. 'items' => array(),
  69. 'label' => $groupName
  70. );
  71. }
  72. if (empty($group['items'])) {
  73. $groups[$groupName]['items'] = $groupDefaults[$groupName]['items'];
  74. }
  75. if (empty($group['label'])) {
  76. $groups[$groupName]['label'] = $groupDefaults[$groupName]['label'];
  77. }
  78. if (empty($group['label_catalogue'])) {
  79. $groups[$groupName]['label_catalogue'] = 'SonataAdminBundle';
  80. }
  81. if (!empty($group['item_adds'])) {
  82. $group['items'] = array_merge($groupDefaults[$groupName]['items'], $group['item_adds']);
  83. }
  84. }
  85. } else {
  86. $groups = $groupDefaults;
  87. }
  88. $pool->addMethodCall('setAdminServiceIds', array($admins));
  89. $pool->addMethodCall('setAdminGroups', array($groups));
  90. $pool->addMethodCall('setAdminClasses', array($classes));
  91. $routeLoader = $container->getDefinition('sonata.admin.route_loader');
  92. $routeLoader->replaceArgument(1, $admins);
  93. }
  94. /**
  95. * This method read the attribute keys and configure admin class to use the related dependency
  96. *
  97. * @param \Symfony\Component\DependencyInjection\Definition $definition
  98. * @param array $attributes
  99. */
  100. public function applyConfigurationFromAttribute(Definition $definition, array $attributes) {
  101. $keys = array(
  102. 'model_manager',
  103. 'form_contractor',
  104. 'show_builder',
  105. 'list_builder',
  106. 'datagrid_builder',
  107. 'translator',
  108. 'configuration_pool',
  109. 'router',
  110. 'validator',
  111. 'security_handler',
  112. 'menu_factory',
  113. 'route_builder',
  114. 'label_translator_strategy',
  115. );
  116. foreach ($keys as $key) {
  117. $method = 'set'.$this->camelize($key);
  118. if (!isset($attributes[$key]) || $definition->hasMethodCall($method)) {
  119. continue;
  120. }
  121. $definition->addMethodCall($method, array(new Reference($attributes[$key])));
  122. }
  123. }
  124. /**
  125. * Apply the default values required by the AdminInterface to the Admin service definition
  126. *
  127. * @param \Symfony\Component\DependencyInjection\ContainerBuilder $container
  128. * @param string $serviceId
  129. * @param array $attributes
  130. * @return \Symfony\Component\DependencyInjection\Definition
  131. */
  132. public function applyDefaults(ContainerBuilder $container, $serviceId, array $attributes = array())
  133. {
  134. $definition = $container->getDefinition($serviceId);
  135. $settings = $container->getParameter('sonata.admin.configuration.admin_services');
  136. $definition->setScope(ContainerInterface::SCOPE_PROTOTYPE);
  137. $manager_type = $attributes['manager_type'];
  138. $addServices = isset($settings[$serviceId]) ? $settings[$serviceId] : array();
  139. $defaultAddServices = array(
  140. 'model_manager' => sprintf('sonata.admin.manager.%s', $manager_type),
  141. 'form_contractor' => sprintf('sonata.admin.builder.%s_form', $manager_type),
  142. 'show_builder' => sprintf('sonata.admin.builder.%s_show', $manager_type),
  143. 'list_builder' => sprintf('sonata.admin.builder.%s_list', $manager_type),
  144. 'datagrid_builder' => sprintf('sonata.admin.builder.%s_datagrid', $manager_type),
  145. 'translator' => 'translator',
  146. 'configuration_pool' => 'sonata.admin.pool',
  147. 'route_generator' => 'sonata.admin.route.default_generator',
  148. 'validator' => 'validator',
  149. 'security_handler' => 'sonata.admin.security.handler',
  150. 'menu_factory' => 'knp_menu.factory',
  151. 'route_builder' => 'sonata.admin.route.path_info',
  152. 'label_translator_strategy' => 'sonata.admin.label.strategy.native'
  153. );
  154. $definition->addMethodCall('setManagerType', array($manager_type));
  155. foreach ($defaultAddServices as $attr => $addServiceId) {
  156. $method = 'set'.$this->camelize($attr);
  157. if (isset($addServices[$attr]) || !$definition->hasMethodCall($method)) {
  158. $definition->addMethodCall($method, array(new Reference(isset($addServices[$attr]) ? $addServices[$attr] : $addServiceId)));
  159. }
  160. }
  161. if (isset($service['label'])) {
  162. $label = $service['label'];
  163. } elseif (isset($attributes['label'])) {
  164. $label = $attributes['label'];
  165. } else {
  166. $label = '-';
  167. }
  168. $definition->addMethodCall('setLabel', array($label));
  169. $this->fixTemplates($container, $definition);
  170. if ($container->hasParameter('sonata.admin.configuration.security.information') && !$definition->hasMethodCall('setSecurityInformation')) {
  171. $definition->addMethodCall('setSecurityInformation', array('%sonata.admin.configuration.security.information%'));
  172. }
  173. $definition->addMethodCall('initialize');
  174. return $definition;
  175. }
  176. /**
  177. * @param \Symfony\Component\DependencyInjection\ContainerBuilder $container
  178. * @param \Symfony\Component\DependencyInjection\Definition $definition
  179. * @return void
  180. */
  181. public function fixTemplates(ContainerBuilder $container, Definition $definition)
  182. {
  183. $definedTemplates = $container->getParameter('sonata.admin.configuration.templates');
  184. $methods = array();
  185. $pos = 0;
  186. foreach ($definition->getMethodCalls() as $method) {
  187. if ($method[0] == 'setTemplates') {
  188. $definedTemplates = array_merge($definedTemplates, $method[1][0]);
  189. continue;
  190. }
  191. if ($method[0] == 'setTemplate') {
  192. $definedTemplates[$method[1][0]] = $method[1][1];
  193. continue;
  194. }
  195. $methods[$pos] = $method;
  196. $pos++;
  197. }
  198. $definition->setMethodCalls($methods);
  199. // make sure the default templates are defined
  200. $definedTemplates = array_merge(array(
  201. 'user_block' => 'SonataAdminBundle:Core:user_block.html.twig',
  202. 'layout' => 'SonataAdminBundle::standard_layout.html.twig',
  203. 'ajax' => 'SonataAdminBundle::ajax_layout.html.twig',
  204. 'dashboard' => 'SonataAdminBundle:Core:dashboard.html.twig',
  205. 'list' => 'SonataAdminBundle:CRUD:list.html.twig',
  206. 'show' => 'SonataAdminBundle:CRUD:show.html.twig',
  207. 'edit' => 'SonataAdminBundle:CRUD:edit.html.twig',
  208. 'history' => 'SonataAdminBundle:CRUD:history.html.twig',
  209. 'history_revision' => 'SonataAdminBundle:CRUD:history_revision.html.twig',
  210. 'action' => 'SonataAdminBundle:CRUD:action.html.twig',
  211. 'short_object_description' => 'SonataAdminBundle:Helper:short-object-description.html.twig',
  212. ), $definedTemplates);
  213. $definition->addMethodCall('setTemplates', array($definedTemplates));
  214. }
  215. /**
  216. * method taken from PropertyPath
  217. *
  218. * @param $property
  219. * @return mixed
  220. */
  221. protected function camelize($property)
  222. {
  223. return preg_replace(array('/(^|_)+(.)/e', '/\.(.)/e'), array("strtoupper('\\2')", "'_'.strtoupper('\\1')"), $property);
  224. }
  225. }