4 * Copyright © 2003-2024 The Galette Team
6 * This file is part of Galette (https://galette.eu).
8 * Galette is free software: you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation, either version 3 of the License, or
11 * (at your option) any later version.
13 * Galette is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with Galette. If not, see <http://www.gnu.org/licenses/>.
22 namespace Galette\Features
;
24 use Galette\Core\Login
;
25 use Galette\Entity\Adherent
;
26 use Galette\Repository\DynamicFieldsSet
;
29 use Galette\DynamicFields\File
;
30 use Galette\DynamicFields\Date
;
31 use Galette\DynamicFields\Boolean
;
32 use Galette\Entity\DynamicFieldsHandle
;
35 * Dynamics fields trait
37 * @author Johan Cwiklinski <johan@x-tnd.be>
44 protected string $name_pattern = 'info_field_';
46 protected DynamicFieldsHandle
$dynamics;
49 * Load dynamic fields for member
53 private function loadDynamicFields(): void
55 /** @phpstan-ignore-next-line */
56 if (property_exists($this, 'login') && ($this->login ??
null) instanceof Login
) {
57 $login = $this->login
;
61 $this->dynamics
= new DynamicFieldsHandle($this->zdb
, $login, $this);
67 * @return DynamicFieldsHandle
69 public function getDynamicFields(): DynamicFieldsHandle
71 if (empty($this->dynamics
)) {
72 $this->loadDynamicFields();
74 return $this->dynamics
;
78 * Extract posted values for dynamic fields
80 * @param array<string, mixed> $post Posted values
81 * @param array<string,int|bool> $required Array of required fields
82 * @param array<string> $disabled Array of disabled fields
86 protected function dynamicsCheck(array $post, array $required, array $disabled): bool
88 if (!isset($this->dynamics
)) {
90 'Dynamics fields have not been loaded, cannot be checked. (from: ' . __METHOD__
. ')',
93 $this->loadDynamicFields();
98 $fields = $this->dynamics
->getFields();
100 $dynamic_fields = [];
102 foreach ($post as $key => $value) {
103 // if the field is enabled, and match patterns check it
104 if (isset($disabled[$key]) ||
substr($key, 0, 11) != $this->name_pattern
) {
108 list($field_id, $val_index) = explode('_', str_replace($this->name_pattern
, '', $key));
109 if (!is_numeric($field_id) ||
!is_numeric($val_index)) {
113 $dynamic_fields[$key] = [
115 'field_id' => $field_id,
116 'val_index' => $val_index
120 //some fields may be missing in posted values (checkboxes)
121 foreach ($fields as $field) {
122 $pattern = '/' . $this->name_pattern
. $field->getId() . '_(\d)/';
123 if ($field instanceof Boolean
&& !preg_grep($pattern, array_keys($dynamic_fields))) {
124 $dynamic_fields[$this->name_pattern
. $field->getId() . '_1'] = [
126 'field_id' => $field->getId(),
132 foreach ($dynamic_fields as $key => $dfield_values) {
133 $field_id = $dfield_values['field_id'];
134 $value = $dfield_values['value'];
135 $val_index = $dfield_values['val_index'];
137 if ($fields[$field_id]->isRequired() && (trim($value) === '' ||
$value == null)) {
138 $this->errors
[] = str_replace(
140 $fields[$field_id]->getName(),
141 _T('Missing required field %field')
144 if ($fields[$field_id] instanceof File
) {
146 $filename = $fields[$field_id]->getFileName($this->id
, $val_index);
147 if (file_exists(GALETTE_FILES_PATH
. $filename)) {
148 unlink(GALETTE_FILES_PATH
. $filename);
149 } elseif (!$this instanceof Adherent
) {
150 $test_filename = $fields[$field_id]->getFileName($this->id
, $val_index, 'member');
151 if (file_exists(GALETTE_FILES_PATH
. $test_filename)) {
152 unlink(GALETTE_FILES_PATH
. $test_filename);
155 $this->dynamics
->setValue($this->id
, $field_id, $val_index, '');
157 if ($fields[$field_id] instanceof Date
&& !empty(trim($value))) {
160 $d = \DateTime
::createFromFormat(__("Y-m-d"), $value);
162 //try with non localized date
163 $d = \DateTime
::createFromFormat("Y-m-d", $value);
165 throw new \
Exception('Incorrect format');
168 } catch (Throwable
$e) {
171 'Wrong date format. field: ' . $field_id .
172 ', value: ' . $value . ', expected fmt: ' .
173 __("Y-m-d") . ' | ' . $e->getMessage(),
176 $this->errors
[] = sprintf(
177 //TRANS: %1$s date format, %2$s is the field name
178 _T('- Wrong date format (%1$s) for %2$s!'),
180 $fields[$field_id]->getName()
185 if ($value !== null && trim($value) !== '') {
186 $this->dynamics
->setValue($this->id ??
null, $field_id, $val_index, $value);
188 $this->dynamics
->unsetValue($field_id, $val_index);
200 * Stores dynamic fields
202 * @param bool $transaction True if a transaction already exists
206 protected function dynamicsStore(bool $transaction = false): bool
208 if (!isset($this->dynamics
)) {
210 'Dynamics fields have not been loaded, cannot be stored. (from: ' . __METHOD__
. ')',
213 $this->loadDynamicFields();
215 $return = $this->dynamics
->storeValues($this->id
, $transaction);
216 if (method_exists($this, 'updateModificationDate') && $this->dynamics
->hasChanged()) {
217 $this->updateModificationDate();
223 * Store dynamic Files
225 * @param array<string, mixed> $files Posted files
229 protected function dynamicsFiles(array $files): void
231 $this->loadDynamicFields();
232 $fields = $this->dynamics
->getFields();
235 foreach ($files as $key => $file) {
236 if (substr($key, 0, 11) != $this->name_pattern
) {
240 list($field_id, $val_index) = explode('_', str_replace($this->name_pattern
, '', $key));
241 if (!is_numeric($field_id) ||
!is_numeric($val_index)) {
246 $file['error'] == UPLOAD_ERR_NO_FILE
247 && $file['name'] == ''
248 && $file['tmp_name'] == ''
252 } elseif ($file['error'] !== UPLOAD_ERR_OK
) {
253 Analog
::log("file upload error", Analog
::ERROR
);
257 $tmp_filename = $file['tmp_name'];
258 if ($tmp_filename == '') {
259 Analog
::log("empty temporary filename", Analog
::ERROR
);
263 if (!is_uploaded_file($tmp_filename)) {
264 Analog
::log("not an uploaded file", Analog
::ERROR
);
269 $fields[$field_id]->getSize() ?
270 $fields[$field_id]->getSize() * 1024 : File
::DEFAULT_MAX_FILE_SIZE
* 1024;
271 if ($file['size'] > $max_size) {
273 "file too large: " . $file['size'] . " Ko, vs $max_size Ko allowed",
276 $this->errors
[] = preg_replace(
279 _T("File is too big. Maximum allowed size is %dKo")
284 $form_name = $this->getFormName();
285 if ($form_name === 'adh') {
286 $form_name = 'member'; //for compatibility with existing files
288 $new_filename = sprintf(
289 '%s_%d_field_%d_value_%d',
295 Analog
::log("new file: $new_filename", Analog
::DEBUG
);
299 GALETTE_FILES_PATH
. $new_filename
301 $this->dynamics
->setValue($this->id
, (int)$field_id, (int)$val_index, $file['name']);
305 if ($store === true) {
306 $this->dynamicsStore();
311 * Remove dynamic fields values
313 * @param bool $transaction True if a transaction already exists
317 protected function dynamicsRemove(bool $transaction = false): bool
319 if (!isset($this->dynamics
)) {
321 'Dynamics fields have not been loaded, cannot be removed. (from: ' . __METHOD__
. ')',
324 $this->loadDynamicFields();
326 $return = $this->dynamics
->removeValues($this->id
, $transaction);
333 * @return array<string>
335 public function getErrors(): array
337 return $this->errors
;
341 * Validate data for dynamic fields
342 * Set valid data in current object, also resets errors list
344 * @param array<string> $values Dynamic fields values
345 * @param string $prefix Prefix to replace, default to 'dynfield_'
349 public function dynamicsValidate(array $values, string $prefix = 'dynfield_'): bool
352 foreach ($values as $key => $value) {
353 $dfields[str_replace($prefix, $this->name_pattern
, $key)] = $value;
355 return $this->dynamicsCheck($dfields, [], []);
363 public function getFormName(): string
365 return array_search(get_class($this), DynamicFieldsSet
::getClasses());