Afform - Rename submitFile param from entityName to modelName
[civicrm-core.git] / ext / afform / core / afform.php
CommitLineData
66aa0f5e
TO
1<?php
2
3require_once 'afform.civix.php';
4use CRM_Afform_ExtensionUtil as E;
5
bc3b7c5b
TO
6/**
7 * Filter the content of $params to only have supported afform fields.
8 *
9 * @param array $params
10 * @return array
11 */
12function _afform_fields_filter($params) {
5591cfbf 13 $result = [];
67d666c6 14 $fields = \Civi\Api4\Afform::getfields(FALSE)->setAction('create')->execute()->indexBy('name');
50868e8d
CW
15 foreach ($fields as $fieldName => $field) {
16 if (isset($params[$fieldName])) {
17 $result[$fieldName] = $params[$fieldName];
d1ec770c 18
e38db494
CW
19 if ($field['data_type'] === 'Boolean' && !is_bool($params[$fieldName])) {
20 $result[$fieldName] = CRM_Utils_String::strtobool($params[$fieldName]);
d1ec770c
TO
21 }
22 }
bc3b7c5b
TO
23 }
24 return $result;
25}
26
8f4a0ee9 27/**
5591cfbf 28 * @param \Symfony\Component\DependencyInjection\ContainerBuilder $container
8f4a0ee9
TO
29 */
30function afform_civicrm_container($container) {
77dccccb 31 $container->addResource(new \Symfony\Component\Config\Resource\FileResource(__FILE__));
8f4a0ee9
TO
32 $container->setDefinition('afform_scanner', new \Symfony\Component\DependencyInjection\Definition(
33 'CRM_Afform_AfformScanner',
5591cfbf 34 []
b53fe171 35 ))->setPublic(TRUE);
8f4a0ee9
TO
36}
37
66aa0f5e
TO
38/**
39 * Implements hook_civicrm_config().
40 *
41 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_config
42 */
43function afform_civicrm_config(&$config) {
44 _afform_civix_civicrm_config($config);
77dccccb
TO
45
46 if (isset(Civi::$statics[__FUNCTION__])) {
47 return;
48 }
49 Civi::$statics[__FUNCTION__] = 1;
50
b7edd04e 51 $dispatcher = Civi::dispatcher();
07a1bc45
A
52 $dispatcher->addListener('civi.afform.submit', ['\Civi\Api4\Action\Afform\Submit', 'processGenericEntity'], 0);
53 $dispatcher->addListener('civi.afform.submit', ['\Civi\Api4\Action\Afform\Submit', 'preprocessContact'], 10);
b7edd04e
TO
54 $dispatcher->addListener('hook_civicrm_angularModules', ['\Civi\Afform\AngularDependencyMapper', 'autoReq'], -1000);
55 $dispatcher->addListener('hook_civicrm_alterAngular', ['\Civi\Afform\AfformMetadataInjector', 'preprocess']);
bdceaf62 56 $dispatcher->addListener('hook_civicrm_check', ['\Civi\Afform\StatusChecks', 'hook_civicrm_check']);
b7edd04e
TO
57
58 // Register support for email tokens
bdceaf62
TO
59 if (CRM_Extension_System::singleton()->getMapper()->isActiveModule('authx')) {
60 $dispatcher->addListener('hook_civicrm_alterMailContent', ['\Civi\Afform\Tokens', 'applyCkeditorWorkaround']);
61 $dispatcher->addListener('hook_civicrm_tokens', ['\Civi\Afform\Tokens', 'hook_civicrm_tokens']);
62 $dispatcher->addListener('hook_civicrm_tokenValues', ['\Civi\Afform\Tokens', 'hook_civicrm_tokenValues']);
63 }
66aa0f5e
TO
64}
65
66/**
67 * Implements hook_civicrm_xmlMenu().
68 *
69 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_xmlMenu
70 */
71function afform_civicrm_xmlMenu(&$files) {
72 _afform_civix_civicrm_xmlMenu($files);
73}
74
75/**
76 * Implements hook_civicrm_install().
77 *
78 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_install
79 */
80function afform_civicrm_install() {
81 _afform_civix_civicrm_install();
82}
83
84/**
85 * Implements hook_civicrm_postInstall().
86 *
87 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_postInstall
88 */
89function afform_civicrm_postInstall() {
90 _afform_civix_civicrm_postInstall();
91}
92
93/**
94 * Implements hook_civicrm_uninstall().
95 *
96 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_uninstall
97 */
98function afform_civicrm_uninstall() {
99 _afform_civix_civicrm_uninstall();
100}
101
102/**
103 * Implements hook_civicrm_enable().
104 *
105 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_enable
106 */
107function afform_civicrm_enable() {
108 _afform_civix_civicrm_enable();
109}
110
111/**
112 * Implements hook_civicrm_disable().
113 *
114 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_disable
115 */
116function afform_civicrm_disable() {
117 _afform_civix_civicrm_disable();
118}
119
120/**
121 * Implements hook_civicrm_upgrade().
122 *
123 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_upgrade
124 */
125function afform_civicrm_upgrade($op, CRM_Queue_Queue $queue = NULL) {
126 return _afform_civix_civicrm_upgrade($op, $queue);
127}
128
129/**
130 * Implements hook_civicrm_managed().
131 *
132 * Generate a list of entities to create/deactivate/delete when this module
133 * is installed, disabled, uninstalled.
134 *
135 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_managed
136 */
137function afform_civicrm_managed(&$entities) {
138 _afform_civix_civicrm_managed($entities);
521b232a
TO
139
140 /** @var \CRM_Afform_AfformScanner $scanner */
141 if (\Civi::container()->has('afform_scanner')) {
142 $scanner = \Civi::service('afform_scanner');
143 }
144 else {
145 // This might happen at oddballs points - e.g. while you're in the middle of re-enabling the ext.
146 // This AfformScanner instance only lives during this method call, and it feeds off the regular cache.
147 $scanner = new CRM_Afform_AfformScanner();
148 }
149
150 foreach ($scanner->getMetas() as $afform) {
151 if (empty($afform['is_dashlet']) || empty($afform['name'])) {
152 continue;
153 }
154 $entities[] = [
155 'module' => E::LONG_NAME,
156 'name' => 'afform_dashlet_' . $afform['name'],
157 'entity' => 'Dashboard',
158 'update' => 'always',
159 // ideal cleanup policy might be to (a) deactivate if used and (b) remove if unused
160 'cleanup' => 'always',
161 'params' => [
162 'version' => 3,
163 // Q: Should we loop through all domains?
164 'domain_id' => CRM_Core_BAO_Domain::getDomain()->id,
165 'is_active' => TRUE,
166 'name' => $afform['name'],
67d666c6 167 'label' => $afform['title'] ?? E::ts('(Untitled)'),
521b232a
TO
168 'directive' => _afform_angular_module_name($afform['name'], 'dash'),
169 'permission' => "@afform:" . $afform['name'],
170 ],
171 ];
172 }
66aa0f5e
TO
173}
174
175/**
c63f20d3 176 * Implements hook_civicrm_tabset().
66aa0f5e 177 *
c63f20d3
CW
178 * Adds afforms as contact summary tabs.
179 */
180function afform_civicrm_tabset($tabsetName, &$tabs, $context) {
181 if ($tabsetName !== 'civicrm/contact/view') {
182 return;
183 }
184 $scanner = \Civi::service('afform_scanner');
185 $weight = 111;
186 foreach ($scanner->getMetas() as $afform) {
187 if (!empty($afform['contact_summary']) && $afform['contact_summary'] === 'tab') {
188 $module = _afform_angular_module_name($afform['name']);
189 $tabs[] = [
190 'id' => $afform['name'],
191 'title' => $afform['title'],
192 'weight' => $weight++,
193 'icon' => 'crm-i fa-list-alt',
194 'is_active' => TRUE,
195 'template' => 'afform/contactSummary/AfformTab.tpl',
196 'module' => $module,
197 'directive' => _afform_angular_module_name($afform['name'], 'dash'),
198 ];
199 // If this is the real contact summary page (and not a callback from ContactLayoutEditor), load module.
200 if (empty($context['caller'])) {
201 Civi::service('angularjs.loader')->addModules($module);
202 }
203 }
204 }
205}
206
207/**
208 * Implements hook_civicrm_pageRun().
66aa0f5e 209 *
c63f20d3
CW
210 * Adds afforms as contact summary blocks.
211 */
212function afform_civicrm_pageRun(&$page) {
213 if (get_class($page) !== 'CRM_Contact_Page_View_Summary') {
214 return;
215 }
216 $scanner = \Civi::service('afform_scanner');
217 $cid = $page->get('cid');
218 $side = 'left';
219 foreach ($scanner->getMetas() as $afform) {
220 if (!empty($afform['contact_summary']) && $afform['contact_summary'] === 'block') {
221 $module = _afform_angular_module_name($afform['name']);
222 $block = [
223 'module' => $module,
224 'directive' => _afform_angular_module_name($afform['name'], 'dash'),
225 ];
226 $content = CRM_Core_Smarty::singleton()->fetchWith('afform/contactSummary/AfformBlock.tpl', ['contactId' => $cid, 'block' => $block]);
227 CRM_Core_Region::instance("contact-basic-info-$side")->add([
228 'markup' => '<div class="crm-summary-block">' . $content . '</div>',
229 'weight' => 1,
230 ]);
231 Civi::service('angularjs.loader')->addModules($module);
232 $side = $side === 'left' ? 'right' : 'left';
233 }
234 }
235}
236
237/**
238 * Implements hook_civicrm_contactSummaryBlocks().
66aa0f5e 239 *
c63f20d3 240 * @link https://github.com/civicrm/org.civicrm.contactlayout
66aa0f5e 241 */
c63f20d3 242function afform_civicrm_contactSummaryBlocks(&$blocks) {
4f664e9c
CW
243 $afforms = \Civi\Api4\Afform::get(FALSE)
244 ->setSelect(['name', 'title', 'directive_name', 'module_name', 'type', 'type:icon', 'type:label'])
245 ->addWhere('contact_summary', '=', 'block')
246 ->execute();
4096e4b0 247 foreach ($afforms as $index => $afform) {
4f664e9c
CW
248 // Create a group per afform type
249 $blocks += [
250 "afform_{$afform['type']}" => [
251 'title' => $afform['type:label'],
252 'icon' => $afform['type:icon'],
253 'blocks' => [],
254 ],
255 ];
256 $blocks["afform_{$afform['type']}"]['blocks'][$afform['name']] = [
257 'title' => $afform['title'],
258 'tpl_file' => 'afform/contactSummary/AfformBlock.tpl',
259 'module' => $afform['module_name'],
260 'directive' => $afform['directive_name'],
261 'sample' => [
262 $afform['type:label'],
263 ],
264 'edit' => 'civicrm/admin/afform#/edit/' . $afform['name'],
4096e4b0 265 'system_default' => [0, $index % 2],
4f664e9c 266 ];
c63f20d3 267 }
66aa0f5e
TO
268}
269
270/**
271 * Implements hook_civicrm_angularModules().
272 *
e1aca853 273 * Generate a list of Afform Angular modules.
66aa0f5e
TO
274 */
275function afform_civicrm_angularModules(&$angularModules) {
276 _afform_civix_civicrm_angularModules($angularModules);
bb56ac78 277
5e04a2d4 278 $afforms = \Civi\Api4\Afform::get(FALSE)
e38db494 279 ->setSelect(['name', 'requires', 'module_name', 'directive_name'])
2d4bfef1
CW
280 ->execute();
281
282 foreach ($afforms as $afform) {
e38db494 283 $angularModules[$afform['module_name']] = [
bb56ac78 284 'ext' => E::LONG_NAME,
2d4bfef1
CW
285 'js' => ['assetBuilder://afform.js?name=' . urlencode($afform['name'])],
286 'requires' => $afform['requires'],
bb56ac78 287 'basePages' => [],
aa6abb77 288 'partialsCallback' => '_afform_get_partials',
2d4bfef1 289 '_afform' => $afform['name'],
292054ac
CW
290 // TODO: Allow afforms to declare their own theming requirements
291 'bundles' => ['bootstrap3'],
77dccccb 292 'exports' => [
04417491 293 $afform['directive_name'] => 'E',
77dccccb 294 ],
bb56ac78 295 ];
bb56ac78 296 }
66aa0f5e
TO
297}
298
aa6abb77 299/**
2d4bfef1
CW
300 * Callback to retrieve partials for a given afform/angular module.
301 *
302 * @see afform_civicrm_angularModules
aa6abb77
TO
303 *
304 * @param string $moduleName
305 * The module name.
306 * @param array $module
307 * The module definition.
308 * @return array
309 * Array(string $filename => string $html).
2d4bfef1 310 * @throws API_Exception
aa6abb77
TO
311 */
312function _afform_get_partials($moduleName, $module) {
2d4bfef1
CW
313 $afform = civicrm_api4('Afform', 'get', [
314 'where' => [['name', '=', $module['_afform']]],
315 'select' => ['layout'],
316 'layoutFormat' => 'html',
317 'checkPermissions' => FALSE,
318 ], 0);
aa6abb77 319 return [
2d4bfef1 320 "~/$moduleName/$moduleName.aff.html" => $afform['layout'],
aa6abb77
TO
321 ];
322}
323
66aa0f5e
TO
324/**
325 * Implements hook_civicrm_alterSettingsFolders().
326 *
327 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_alterSettingsFolders
328 */
329function afform_civicrm_alterSettingsFolders(&$metaDataFolders = NULL) {
330 _afform_civix_civicrm_alterSettingsFolders($metaDataFolders);
331}
332
333/**
334 * Implements hook_civicrm_entityTypes().
335 *
336 * Declare entity types provided by this module.
337 *
338 * @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_entityTypes
339 */
340function afform_civicrm_entityTypes(&$entityTypes) {
341 _afform_civix_civicrm_entityTypes($entityTypes);
342}
343
98f4a7cb
TO
344/**
345 * Implements hook_civicrm_themes().
346 */
347function afform_civicrm_themes(&$themes) {
348 _afform_civix_civicrm_themes($themes);
349}
350
bb56ac78
TO
351/**
352 * Implements hook_civicrm_buildAsset().
353 */
354function afform_civicrm_buildAsset($asset, $params, &$mimeType, &$content) {
355 if ($asset !== 'afform.js') {
356 return;
357 }
358
359 if (empty($params['name'])) {
360 throw new RuntimeException("Missing required parameter: afform.js?name=NAME");
361 }
362
2d4bfef1 363 $moduleName = _afform_angular_module_name($params['name'], 'camel');
b6e13973
SL
364 $formMetaData = (array) civicrm_api4('Afform', 'get', [
365 'checkPermissions' => FALSE,
366 'select' => ['redirect', 'name'],
367 'where' => [['name', '=', $params['name']]],
368 ], 0);
bb56ac78
TO
369 $smarty = CRM_Core_Smarty::singleton();
370 $smarty->assign('afform', [
aa6abb77 371 'camel' => $moduleName,
b6e13973 372 'meta' => $formMetaData,
aa6abb77 373 'templateUrl' => "~/$moduleName/$moduleName.aff.html",
bb56ac78
TO
374 ]);
375 $mimeType = 'text/javascript';
9ec944f2 376 $content = $smarty->fetch('afform/AfformAngularModule.tpl');
bb56ac78
TO
377}
378
8775c48a
TO
379/**
380 * Implements hook_civicrm_alterMenu().
381 */
382function afform_civicrm_alterMenu(&$items) {
8f4a0ee9
TO
383 if (Civi::container()->has('afform_scanner')) {
384 $scanner = Civi::service('afform_scanner');
385 }
386 else {
387 // During installation...
388 $scanner = new CRM_Afform_AfformScanner();
389 }
8775c48a
TO
390 foreach ($scanner->getMetas() as $name => $meta) {
391 if (!empty($meta['server_route'])) {
392 $items[$meta['server_route']] = [
393 'page_callback' => 'CRM_Afform_Page_AfformBase',
394 'page_arguments' => 'afform=' . urlencode($name),
13bdd6d2 395 'title' => $meta['title'] ?? '',
f16b2aee 396 'access_arguments' => [["@afform:$name"], 'and'],
254f01f0 397 'is_public' => $meta['is_public'],
8775c48a
TO
398 ];
399 }
400 }
f16b2aee
TO
401}
402
403/**
404 * Implements hook_civicrm_permission_check().
405 *
586344a7
TO
406 * This extends the list of permissions available in `CRM_Core_Permission:check()`
407 * by introducing virtual-permissions named `@afform:myForm`. The evaluation
408 * of these virtual-permissions is dependent on the settings for `myForm`.
409 * `myForm` may be exposed/integrated through multiple subsystems (routing,
410 * nav-menu, API, etc), and the use of virtual-permissions makes easy to enforce
411 * consistent permissions across any relevant subsystems.
412 *
f16b2aee
TO
413 * @see CRM_Utils_Hook::permission_check()
414 */
415function afform_civicrm_permission_check($permission, &$granted, $contactId) {
14b26ac5 416 if ($permission[0] !== '@') {
f16b2aee
TO
417 // Micro-optimization - this function may get hit a lot.
418 return;
419 }
420
421 if (preg_match('/^@afform:(.*)/', $permission, $m)) {
422 $name = $m[1];
423
2d4bfef1
CW
424 $afform = \Civi\Api4\Afform::get()
425 ->setCheckPermissions(FALSE)
426 ->addWhere('name', '=', $name)
427 ->setSelect(['permission'])
428 ->execute()
429 ->first();
430 if ($afform) {
431 $granted = CRM_Core_Permission::check($afform['permission'], $contactId);
432 }
f16b2aee 433 }
8775c48a
TO
434}
435
c4e6b413
TO
436/**
437 * Implements hook_civicrm_permissionList().
438 *
439 * @see CRM_Utils_Hook::permissionList()
440 */
441function afform_civicrm_permissionList(&$permissions) {
442 $scanner = Civi::service('afform_scanner');
443 foreach ($scanner->getMetas() as $name => $meta) {
444 $permissions['@afform:' . $name] = [
445 'group' => 'afform',
67d666c6 446 'title' => E::ts('Afform: Inherit permission of %1', [
c4e6b413
TO
447 1 => $name,
448 ]),
449 ];
450 }
451}
452
74f862e4
TO
453/**
454 * Clear any local/in-memory caches based on afform data.
455 */
456function _afform_clear() {
457 $container = \Civi::container();
458 $container->get('afform_scanner')->clear();
76b9562a 459 $container->get('angular')->clear();
74f862e4
TO
460}
461
bb56ac78 462/**
87dde5eb
TO
463 * @param string $fileBaseName
464 * Ex: foo-bar
841850b1
TO
465 * @param string $format
466 * 'camel' or 'dash'.
bb56ac78 467 * @return string
841850b1 468 * Ex: 'FooBar' or 'foo-bar'.
3cd5c38b 469 * @throws \Exception
bb56ac78 470 */
87dde5eb 471function _afform_angular_module_name($fileBaseName, $format = 'camel') {
841850b1
TO
472 switch ($format) {
473 case 'camel':
87dde5eb 474 $camelCase = '';
9c84a124 475 foreach (preg_split('/[-_ ]/', $fileBaseName, NULL, PREG_SPLIT_NO_EMPTY) as $shortNamePart) {
87dde5eb
TO
476 $camelCase .= ucfirst($shortNamePart);
477 }
14b26ac5 478 return strtolower($camelCase[0]) . substr($camelCase, 1);
841850b1
TO
479
480 case 'dash':
9c84a124 481 return strtolower(implode('-', preg_split('/[-_ ]|(?=[A-Z])/', $fileBaseName, NULL, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE)));
841850b1
TO
482
483 default:
484 throw new \Exception("Unrecognized format");
485 }
bb56ac78 486}
355881ac
SL
487
488/**
489 * Implements hook_civicrm_alterApiRoutePermissions().
490 *
491 * @see CRM_Utils_Hook::alterApiRoutePermissions
492 */
493function afform_civicrm_alterApiRoutePermissions(&$permissions, $entity, $action) {
f362c531 494 if ($entity == 'Afform') {
9d85626b 495 if ($action == 'prefill' || $action == 'submit' || $action == 'submitFile') {
f362c531
TO
496 $permissions = CRM_Core_Permission::ALWAYS_ALLOW_PERMISSION;
497 }
355881ac
SL
498 }
499}
f228ad74
CW
500
501/**
502 * Implements hook_civicrm_preProcess().
503 *
504 * Wordpress only: Adds Afforms to the shortcode dialog (when editing pages/posts).
505 */
506function afform_civicrm_preProcess($formName, &$form) {
507 if ($formName === 'CRM_Core_Form_ShortCode') {
508 $form->components['afform'] = [
509 'label' => E::ts('Form Builder'),
510 'select' => [
511 'key' => 'name',
512 'entity' => 'Afform',
513 'select' => ['minimumInputLength' => 0],
514 'api' => [
515 'params' => ['type' => ['IN' => ['form', 'search']]],
516 ],
517 ],
518 ];
519 }
520}
521
1c3d1f8d
CW
522/**
523 * Implements hook_civicrm_pre().
524 */
525function afform_civicrm_pre($op, $entity, $id, &$params) {
526 // When deleting a searchDisplay, also delete any Afforms the display is embedded within
527 if ($entity === 'SearchDisplay' && $op === 'delete') {
528 $display = \Civi\Api4\SearchDisplay::get(FALSE)
529 ->addSelect('saved_search_id.name', 'name')
530 ->addWhere('id', '=', $id)
531 ->execute()->first();
532 \Civi\Api4\Afform::revert(FALSE)
533 ->addWhere('search_displays', 'CONTAINS', $display['saved_search_id.name'] . ".{$display['name']}")
534 ->execute();
535 }
536}
537
f228ad74
CW
538// Wordpress only: Register callback for rendering shortcodes
539if (function_exists('add_filter')) {
540 add_filter('civicrm_shortcode_get_markup', 'afform_shortcode_content', 10, 4);
541}
542
543/**
544 * Wordpress only: Render Afform content for shortcodes.
545 *
546 * @param string $content
547 * HTML Markup
548 * @param array $atts
549 * Shortcode attributes.
550 * @param array $args
551 * Existing shortcode arguments.
552 * @param string $context
553 * How many shortcodes are present on the page: 'single' or 'multiple'.
554 * @return string
555 * Modified markup.
556 */
557function afform_shortcode_content($content, $atts, $args, $context) {
558 if ($atts['component'] === 'afform') {
559 $afform = civicrm_api4('Afform', 'get', [
560 'select' => ['directive_name', 'module_name'],
561 'where' => [['name', '=', $atts['name']]],
562 ])->first();
563 if ($afform) {
564 Civi::service('angularjs.loader')->addModules($afform['module_name']);
565 $content = "
566 <div class='crm-container' id='bootstrap-theme'>
567 <crm-angular-js modules='{$afform['module_name']}'>
568 <{$afform['directive_name']}></{$afform['directive_name']}>
569 </crm-angular-js>
570 </div>";
571 }
572 }
573 return $content;
574}