FormContractor.php 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. <?php
  2. /*
  3. * This file is part of the Sonata package.
  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\Builder\ORM;
  11. use Sonata\AdminBundle\Admin\AdminInterface;
  12. use Sonata\AdminBundle\Admin\FieldDescriptionInterface;
  13. use Sonata\AdminBundle\Admin\ORM\FieldDescription;
  14. use Sonata\AdminBundle\Model\ModelManagerInterface;
  15. use Sonata\AdminBundle\Builder\FormContractorInterface;
  16. use Sonata\AdminBundle\Form\Type\AdminType;
  17. use Sonata\AdminBundle\Form\Type\ModelType;
  18. use Sonata\AdminBundle\Admin\NoValueException;
  19. use Symfony\Component\Form\FormBuilder;
  20. use Symfony\Component\Form\FormInterface;
  21. use Symfony\Component\Form\FormFactoryInterface;
  22. use Doctrine\ORM\Mapping\ClassMetadataInfo;
  23. class FormContractor implements FormContractorInterface
  24. {
  25. protected $fieldFactory;
  26. public function __construct(FormFactoryInterface $formFactory)
  27. {
  28. $this->formFactory = $formFactory;
  29. }
  30. /**
  31. * Returns the field associated to a FieldDescriptionInterface
  32. * ie : build the embedded form from the related AdminInterface instance
  33. *
  34. * @throws RuntimeException
  35. * @param \Symfony\Component\Form\FormBuilder $formBuilder
  36. * @param \Sonata\AdminBundle\Admin\FieldDescriptionInterface $fieldDescription
  37. * @param null $fieldName
  38. * @return FieldGroup
  39. */
  40. protected function defineChildFormBuilder(FormBuilder $formBuilder, FieldDescriptionInterface $fieldDescription, $fieldName = null)
  41. {
  42. // $fieldName = $fieldName ?: $fieldDescription->getFieldName();
  43. //
  44. // $associatedAdmin = $fieldDescription->getAssociationAdmin();
  45. //
  46. // if (!$associatedAdmin) {
  47. // throw new \RuntimeException(sprintf('inline mode for field `%s` required an Admin definition', $fieldName));
  48. // }
  49. //
  50. // // retrieve the related object
  51. // $childBuilder = $formBuilder->create($fieldName, 'sonata_type_admin', array(
  52. // 'field_description' => $fieldDescription
  53. // ));
  54. //
  55. // $formBuilder->add($childBuilder);
  56. //
  57. // $associatedAdmin->defineFormBuilder($childBuilder);
  58. }
  59. /**
  60. * Returns an OneToOne associated field
  61. *
  62. * @param \Sonata\AdminBundle\Admin\FieldDescriptionInterface $fieldDescription
  63. * @param string
  64. * @return array();
  65. */
  66. protected function getOneToOneFieldOptions($type, FieldDescriptionInterface $fieldDescription)
  67. {
  68. $options = array();
  69. if (!$fieldDescription->hasAssociationAdmin()) {
  70. return $options;
  71. }
  72. // tweak the widget depend on the edit mode
  73. if ($fieldDescription->getOption('edit') == 'inline') {
  74. return $options;
  75. }
  76. $options['class'] = $fieldDescription->getTargetEntity();
  77. $options['data_class'] = $fieldDescription->getTargetEntity();
  78. $options['model_manager'] = $fieldDescription->getAdmin()->getModelManager();
  79. if ($fieldDescription->getOption('edit') == 'list') {
  80. $options['parent'] = 'text';
  81. }
  82. return $options;
  83. }
  84. /**
  85. * Returns the OneToMany associated field
  86. *
  87. * @param \Symfony\Component\Form\FormBuilder $formBuilder
  88. * @param \Sonata\AdminBundle\Admin\FieldDescriptionInterface $fieldDescription
  89. * @return array
  90. */
  91. protected function getOneToManyFieldOptions($type, FieldDescriptionInterface $fieldDescription)
  92. {
  93. $options = array();
  94. if (!$fieldDescription->hasAssociationAdmin()) {
  95. return $options;
  96. }
  97. if ($fieldDescription->getOption('edit') == 'inline') {
  98. // create a collection type with the generated prototype
  99. $options['type'] = 'sonata_type_admin';
  100. $options['modifiable'] = true;
  101. $options['type_options'] = array(
  102. 'sonata_field_description' => $fieldDescription,
  103. 'data_class' => $fieldDescription->getAssociationAdmin()->getClass()
  104. );
  105. return $options;
  106. }
  107. return $this->getManyToManyFieldOptions($type, $fieldDescription);
  108. }
  109. /**
  110. * @param \Symfony\Component\Form\FormBuilder $formBuilder
  111. * @param string
  112. * @return array
  113. */
  114. protected function getManyToManyFieldOptions($type, FieldDescriptionInterface $fieldDescription)
  115. {
  116. $options = array();
  117. if (!$fieldDescription->hasAssociationAdmin()) {
  118. return array();
  119. }
  120. if ($type == 'sonata_type_model') {
  121. $options['class'] = $fieldDescription->getTargetEntity();
  122. $options['multiple'] = true;
  123. $options['parent'] = 'choice';
  124. $options['model_manager'] = $fieldDescription->getAdmin()->getModelManager();
  125. }
  126. return $options;
  127. }
  128. /**
  129. * The method defines the correct default settings for the provided FieldDescription
  130. *
  131. * @param \Sonata\AdminBundle\Admin\AdminInterface $admin
  132. * @param \Sonata\AdminBundle\Admin\FieldDescriptionInterface $fieldDescription
  133. * @param array $options
  134. * @return void
  135. */
  136. public function fixFieldDescription(AdminInterface $admin, FieldDescriptionInterface $fieldDescription, array $options = array())
  137. {
  138. $fieldDescription->mergeOptions($options);
  139. if ($admin->getModelManager()->hasMetadata($admin->getClass())) {
  140. $metadata = $admin->getModelManager()->getMetadata($admin->getClass());
  141. // set the default field mapping
  142. if (isset($metadata->fieldMappings[$fieldDescription->getName()])) {
  143. $fieldDescription->setFieldMapping($metadata->fieldMappings[$fieldDescription->getName()]);
  144. }
  145. // set the default association mapping
  146. if (isset($metadata->associationMappings[$fieldDescription->getName()])) {
  147. $fieldDescription->setAssociationMapping($metadata->associationMappings[$fieldDescription->getName()]);
  148. }
  149. }
  150. if (!$fieldDescription->getType()) {
  151. throw new \RuntimeException(sprintf('Please define a type for field `%s` in `%s`', $fieldDescription->getName(), get_class($admin)));
  152. }
  153. $fieldDescription->setAdmin($admin);
  154. $fieldDescription->setOption('edit', $fieldDescription->getOption('edit', 'standard'));
  155. if (in_array($fieldDescription->getMappingType(), array(ClassMetadataInfo::ONE_TO_MANY, ClassMetadataInfo::MANY_TO_MANY, ClassMetadataInfo::MANY_TO_ONE, ClassMetadataInfo::ONE_TO_ONE ))) {
  156. $admin->attachAdminClass($fieldDescription);
  157. }
  158. }
  159. public function getFormFactory()
  160. {
  161. return $this->formFactory;
  162. }
  163. /**
  164. * @param string $name
  165. * @param array $options
  166. * @return \Symfony\Component\Form\FormBuilder
  167. */
  168. public function getFormBuilder($name, array $options = array())
  169. {
  170. return $this->getFormFactory()->createNamedBuilder('form', $name, null, $options);
  171. }
  172. public function getDefaultOptions($type, FieldDescriptionInterface $fieldDescription, array $options = array())
  173. {
  174. $options['sonata_field_description'] = $fieldDescription;
  175. if (!is_string($type)) {
  176. return $options;
  177. }
  178. // only add default options to Admin Bundle widget ...
  179. $types = array('sonata_type_model', 'sonata_type_admin', 'sonata_type_collection');
  180. if (!in_array($type, $types)) {
  181. return $options;
  182. }
  183. $fieldOptions = array();
  184. switch ($fieldDescription->getMappingType()) {
  185. case ClassMetadataInfo::ONE_TO_MANY:
  186. $fieldOptions = $this->getOneToManyFieldOptions($type, $fieldDescription);
  187. break;
  188. case ClassMetadataInfo::MANY_TO_MANY:
  189. $fieldOptions = $this->getManyToManyFieldOptions($type, $fieldDescription);
  190. break;
  191. case ClassMetadataInfo::MANY_TO_ONE:
  192. case ClassMetadataInfo::ONE_TO_ONE:
  193. $fieldOptions = $this->getOneToOneFieldOptions($type, $fieldDescription);
  194. break;
  195. }
  196. return array_merge($fieldOptions, $options);
  197. }
  198. }