| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // {{{ License
 | 
					
						
							|  |  |  | // This file is part of GNU social - https://www.gnu.org/software/social
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // GNU social is free software: you can redistribute it and/or modify
 | 
					
						
							|  |  |  | // it under the terms of the GNU Affero General Public License as published by
 | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or
 | 
					
						
							|  |  |  | // (at your option) any later version.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // GNU social is distributed in the hope that it will be useful,
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
					
						
							|  |  |  | // GNU Affero General Public License for more details.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU Affero General Public License
 | 
					
						
							|  |  |  | // along with GNU social.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  | // }}}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							| 
									
										
										
										
											2021-07-21 16:42:33 +00:00
										 |  |  |  * Convert a Form from our declarative to Symfony's representation | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @package  GNUsocial | 
					
						
							|  |  |  |  * @category Wrapper | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-02-19 23:29:43 +00:00
										 |  |  |  * @author    Hugo Sales <hugo@hsal.es> | 
					
						
							|  |  |  |  * @copyright 2020-2021 Free Software Foundation, Inc http://www.fsf.org | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  |  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace App\Core; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  | use App\Core\DB\DB; | 
					
						
							|  |  |  | use App\Util\Formatting; | 
					
						
							|  |  |  | use Symfony\Component\Form\Extension\Core\Type\SubmitType; | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  | use Symfony\Component\Form\Form as SymfForm; | 
					
						
							|  |  |  | use Symfony\Component\Form\FormFactoryInterface; | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  | use Symfony\Component\HttpFoundation\Request; | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-21 16:42:33 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * This class converts our own form representation to Symfony's | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Example: | 
					
						
							|  |  |  |  * ``` | 
					
						
							|  |  |  |  * $form = Form::create([ | 
					
						
							|  |  |  |  * ['content',     TextareaType::class, ['label' => ' ', 'data' => '', 'attr' => ['placeholder' => _m($placeholder_string[$rand_key])]]], | 
					
						
							|  |  |  |  * ['attachments', FileType::class,     ['label' => ' ', 'data' => null, 'multiple' => true, 'required' => false]], | 
					
						
							|  |  |  |  * ['visibility',  ChoiceType::class,   ['label' => _m('Visibility:'), 'expanded' => true, 'data' => 'public', 'choices' => [_m('Public') => 'public', _m('Instance') => 'instance', _m('Private') => 'private']]], | 
					
						
							|  |  |  |  * ['to',          ChoiceType::class,   ['label' => _m('To:'), 'multiple' => true, 'expanded' => true, 'choices' => $to_tags]], | 
					
						
							|  |  |  |  * ['post',        SubmitType::class,   ['label' => _m('Post')]], | 
					
						
							|  |  |  |  * ]); | 
					
						
							|  |  |  |  * ``` | 
					
						
							|  |  |  |  * turns into | 
					
						
							|  |  |  |  * ``` | 
					
						
							|  |  |  |  * \Symfony\Component\Form\Form { | 
					
						
							|  |  |  |  * config: Symfony\Component\Form\FormBuilder { ... } | 
					
						
							|  |  |  |  * ... | 
					
						
							|  |  |  |  * children: Symfony\Component\Form\Util\OrderedHashMap { | 
					
						
							|  |  |  |  * elements: array:5 [ | 
					
						
							|  |  |  |  * "content" => Symfony\Component\Form\Form { ... } | 
					
						
							|  |  |  |  * "attachments" => Symfony\Component\Form\Form { ... } | 
					
						
							|  |  |  |  * "visibility" => Symfony\Component\Form\Form { ... } | 
					
						
							|  |  |  |  * "to" => Symfony\Component\Form\Form { ... } | 
					
						
							|  |  |  |  * "post" => Symfony\Component\Form\SubmitButton { ... } | 
					
						
							|  |  |  |  * ] | 
					
						
							|  |  |  |  * ... | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ... | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ``` | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  | abstract class Form | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     private static ?FormFactoryInterface $form_factory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public static function setFactory($ff): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         self::$form_factory = $ff; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Create a form with the given associative array $form as fields | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  |     public static function create(array $form, | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                                   ?object $target = null, | 
					
						
							|  |  |  |                                   array $extra_data = [], | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  |                                   string $type = 'Symfony\Component\Form\Extension\Core\Type\FormType', | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                                   array $builder_options = []): SymfForm | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-09-04 19:35:41 +00:00
										 |  |  |         $name = $form[array_key_last($form)][0]; | 
					
						
							|  |  |  |         $fb   = self::$form_factory->createNamedBuilder($name, $type, array_merge($builder_options, ['translation_domain' => false])); | 
					
						
							| 
									
										
										
										
											2021-07-21 16:42:33 +00:00
										 |  |  |         foreach ($form as [$key, $class, $options]) { | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |             if ($target != null && empty($options['data']) && (strstr($key, 'password') == false) && $class != SubmitType::class) { | 
					
						
							|  |  |  |                 if (isset($extra_data[$key])) { | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |                     // @codeCoverageIgnoreStart
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                     $options['data'] = $extra_data[$key]; | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |                 // @codeCoverageIgnoreEnd
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                 } else { | 
					
						
							|  |  |  |                     $method = 'get' . ucfirst(Formatting::snakeCaseToCamelCase($key)); | 
					
						
							|  |  |  |                     if (method_exists($target, $method)) { | 
					
						
							|  |  |  |                         $options['data'] = $target->{$method}(); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             unset($options['hide']); | 
					
						
							|  |  |  |             if (isset($options['transformer'])) { | 
					
						
							|  |  |  |                 $transformer = $options['transformer']; | 
					
						
							|  |  |  |                 unset($options['transformer']); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             $fb->add($key, $class, $options); | 
					
						
							|  |  |  |             if (isset($transformer)) { | 
					
						
							|  |  |  |                 $fb->get($key)->addModelTransformer(new $transformer()); | 
					
						
							|  |  |  |                 unset($transformer); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         return $fb->getForm(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-25 17:50:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Whether the given $field of $form has the `required` property | 
					
						
							|  |  |  |      * set, defaults to true | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-07-25 17:50:23 +00:00
										 |  |  |     public static function isRequired(array $form, string $field): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $form[$field][2]['required'] ?? true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 19:47:15 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Handle the full life cycle of a form. Creates it with @see | 
					
						
							|  |  |  |      * self::create and inserts the submitted values into the database | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |     public static function handle(array $form_definition, Request $request, ?object $target, array $extra_args = [], ?callable $extra_step = null, array $create_args = [], SymfForm $testing_only_form = null) | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |         $form = $testing_only_form ?? self::create($form_definition, $target, ...$create_args); | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $form->handleRequest($request); | 
					
						
							|  |  |  |         if ($form->isSubmitted() && $form->isValid()) { | 
					
						
							|  |  |  |             $data = $form->getData(); | 
					
						
							|  |  |  |             if ($target == null) { | 
					
						
							|  |  |  |                 return $data; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             unset($data['translation_domain'], $data['save']); | 
					
						
							|  |  |  |             foreach ($data as $key => $val) { | 
					
						
							|  |  |  |                 $method = 'set' . ucfirst(Formatting::snakeCaseToCamelCase($key)); | 
					
						
							|  |  |  |                 if (method_exists($target, $method)) { | 
					
						
							|  |  |  |                     if (isset($extra_args[$key])) { | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |                         // @codeCoverageIgnoreStart
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                         $target->{$method}($val, $extra_args[$key]); | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |                     // @codeCoverageIgnoreEnd
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                     } else { | 
					
						
							|  |  |  |                         $target->{$method}($val); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (isset($extra_step)) { | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |                 // @codeCoverageIgnoreStart
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |                 $extra_step($data, $extra_args); | 
					
						
							| 
									
										
										
										
											2021-07-22 12:31:19 +00:00
										 |  |  |                 // @codeCoverageIgnoreEnd
 | 
					
						
							| 
									
										
										
										
											2020-07-27 03:40:09 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |             DB::flush(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $form; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-23 20:27:02 +00:00
										 |  |  | } |