Merge pull request #18069 from civicrm/5.28
[civicrm-core.git] / Civi / Api4 / Generic / BasicGetFieldsAction.php
1 <?php
2
3 /*
4 +--------------------------------------------------------------------+
5 | Copyright CiviCRM LLC. All rights reserved. |
6 | |
7 | This work is published under the GNU AGPLv3 license with some |
8 | permitted exceptions and without any warranty. For full license |
9 | and copyright information, see https://civicrm.org/licensing |
10 +--------------------------------------------------------------------+
11 */
12
13 /**
14 *
15 * @package CRM
16 * @copyright CiviCRM LLC https://civicrm.org/licensing
17 */
18
19
20 namespace Civi\Api4\Generic;
21
22 use Civi\API\Exception\NotImplementedException;
23
24 /**
25 * Lists information about fields for the $ENTITY entity.
26 *
27 * This field information is also known as "metadata."
28 *
29 * Note that different actions may support different lists of fields.
30 * By default this will fetch the field list relevant to `get`,
31 * but a different list may be returned if you specify another action.
32 *
33 * @method $this setLoadOptions(bool|array $value)
34 * @method bool|array getLoadOptions()
35 * @method $this setAction(string $value)
36 * @method $this setValues(array $values)
37 * @method array getValues()
38 */
39 class BasicGetFieldsAction extends BasicGetAction {
40
41 /**
42 * Fetch option lists for fields?
43 *
44 * This parameter can be either a boolean or an array of attributes to return from the option list:
45 *
46 * - If `FALSE`, each field's `options` property will be a boolean indicating whether the field has an option list
47 * - If `TRUE`, `options` will be returned as a flat array of the option list's `[id => label]`
48 * - If an array, `options` will be a non-associative array of requested properties:
49 * id, name, label, abbr, description, color, icon
50 * e.g. `loadOptions: ['id', 'name', 'label']` will return an array like `[[id: 1, name: 'Meeting', label: 'Meeting'], ...]`
51 * (note that names and labels are generally ONLY the same when the site's language is set to English).
52 *
53 * @var bool|array
54 */
55 protected $loadOptions = FALSE;
56
57 /**
58 * Fields will be returned appropriate to the specified action (get, create, delete, etc.)
59 *
60 * @var string
61 */
62 protected $action = 'get';
63
64 /**
65 * Fields will be returned appropriate to the specified values (e.g. ['contact_type' => 'Individual'])
66 *
67 * @var array
68 */
69 protected $values = [];
70
71 /**
72 * To implement getFields for your own entity:
73 *
74 * 1. From your entity class add a static getFields method.
75 * 2. That method should construct and return this class.
76 * 3. The 3rd argument passed to this constructor should be a function that returns an
77 * array of fields for your entity's CRUD actions.
78 * 4. For non-crud actions that need a different set of fields, you can override the
79 * list from step 3 on a per-action basis by defining a fields() method in that action.
80 * See for example BasicGetFieldsAction::fields() or GetActions::fields().
81 *
82 * @param Result $result
83 * @throws \Civi\API\Exception\NotImplementedException
84 */
85 public function _run(Result $result) {
86 try {
87 $actionClass = \Civi\API\Request::create($this->getEntityName(), $this->getAction(), ['version' => 4]);
88 }
89 catch (NotImplementedException $e) {
90 }
91 if (isset($actionClass) && method_exists($actionClass, 'fields')) {
92 $values = $actionClass->fields();
93 }
94 else {
95 $values = $this->getRecords();
96 }
97 $this->formatResults($values);
98 $this->queryArray($values, $result);
99 }
100
101 /**
102 * Ensure every result contains, at minimum, the array keys as defined in $this->fields.
103 *
104 * Attempt to set some sensible defaults for some fields.
105 *
106 * Format option lists.
107 *
108 * In most cases it's not necessary to override this function, even if your entity is really weird.
109 * Instead just override $this->fields and this function will respect that.
110 *
111 * @param array $values
112 */
113 protected function formatResults(&$values) {
114 $fields = array_column($this->fields(), 'name');
115 // Enforce field permissions
116 if ($this->checkPermissions) {
117 foreach ($values as $key => $field) {
118 if (!empty($field['permission']) && !\CRM_Core_Permission::check($field['permission'])) {
119 unset($values[$key]);
120 }
121 }
122 }
123 foreach ($values as &$field) {
124 $defaults = array_intersect_key([
125 'title' => empty($field['name']) ? NULL : ucwords(str_replace('_', ' ', $field['name'])),
126 'entity' => $this->getEntityName(),
127 'required' => FALSE,
128 'options' => !empty($field['pseudoconstant']),
129 'data_type' => \CRM_Utils_Array::value('type', $field, 'String'),
130 ], array_flip($fields));
131 $field += $defaults;
132 if (isset($defaults['options'])) {
133 $field['options'] = $this->formatOptionList($field['options']);
134 }
135 $field += array_fill_keys($fields, NULL);
136 }
137 }
138
139 /**
140 * Transforms option list into the format specified in $this->loadOptions
141 *
142 * @param $options
143 * @return array|bool
144 */
145 private function formatOptionList($options) {
146 if (!$this->loadOptions || !is_array($options)) {
147 return (bool) $options;
148 }
149 if (!$options) {
150 return $options;
151 }
152 $formatted = [];
153 $first = reset($options);
154 // Flat array requested
155 if ($this->loadOptions === TRUE) {
156 // Convert non-associative to flat array
157 if (is_array($first) && isset($first['id'])) {
158 foreach ($options as $option) {
159 $formatted[$option['id']] = $option['label'] ?? $option['name'] ?? $option['id'];
160 }
161 return $formatted;
162 }
163 return $options;
164 }
165 // Non-associative array of multiple properties requested
166 foreach ($options as $id => $option) {
167 // Transform a flat list
168 if (!is_array($option)) {
169 $option = [
170 'id' => $id,
171 'name' => $option,
172 'label' => $option,
173 ];
174 }
175 $formatted[] = array_intersect_key($option, array_flip($this->loadOptions));
176 }
177 return $formatted;
178 }
179
180 /**
181 * @return string
182 */
183 public function getAction() {
184 // For actions that build on top of other actions, return fields for the simpler action
185 $sub = [
186 'save' => 'create',
187 'replace' => 'create',
188 ];
189 return $sub[$this->action] ?? $this->action;
190 }
191
192 /**
193 * Add an item to the values array
194 * @param string $fieldName
195 * @param mixed $value
196 * @return $this
197 */
198 public function addValue(string $fieldName, $value) {
199 $this->values[$fieldName] = $value;
200 return $this;
201 }
202
203 /**
204 * @param bool $includeCustom
205 * @return $this
206 */
207 public function setIncludeCustom(bool $includeCustom) {
208 // Be forgiving if the param doesn't exist and don't throw an exception
209 if (property_exists($this, 'includeCustom')) {
210 $this->includeCustom = $includeCustom;
211 }
212 return $this;
213 }
214
215 public function fields() {
216 return [
217 [
218 'name' => 'name',
219 'data_type' => 'String',
220 ],
221 [
222 'name' => 'title',
223 'data_type' => 'String',
224 ],
225 [
226 'name' => 'description',
227 'data_type' => 'String',
228 ],
229 [
230 'name' => 'default_value',
231 'data_type' => 'String',
232 ],
233 [
234 'name' => 'required',
235 'data_type' => 'Boolean',
236 ],
237 [
238 'name' => 'required_if',
239 'data_type' => 'String',
240 ],
241 [
242 'name' => 'options',
243 'data_type' => 'Array',
244 ],
245 [
246 'name' => 'data_type',
247 'data_type' => 'String',
248 ],
249 [
250 'name' => 'input_type',
251 'data_type' => 'String',
252 ],
253 [
254 'name' => 'input_attrs',
255 'data_type' => 'Array',
256 ],
257 [
258 'name' => 'fk_entity',
259 'data_type' => 'String',
260 ],
261 [
262 'name' => 'serialize',
263 'data_type' => 'Integer',
264 ],
265 [
266 'name' => 'entity',
267 'data_type' => 'String',
268 ],
269 ];
270 }
271
272 }