3 /* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: */
6 * Dynamics fields trait
10 * Copyright © 2017-2023 The Galette Team
12 * This file is part of Galette (http://galette.tuxfamily.org).
14 * Galette is free software: you can redistribute it and/or modify
15 * it under the terms of the GNU General Public License as published by
16 * the Free Software Foundation, either version 3 of the License, or
17 * (at your option) any later version.
19 * Galette is distributed in the hope that it will be useful,
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 * GNU General Public License for more details.
24 * You should have received a copy of the GNU General Public License
25 * along with Galette. If not, see <http://www.gnu.org/licenses/>.
30 * @author Johan Cwiklinski <johan@x-tnd.be>
31 * @copyright 2017-2023 The Galette Team
32 * @license http://www.gnu.org/licenses/gpl-3.0.html GPL License 3.0 or (at your option) any later version
33 * @link http://galette.tuxfamily.org
34 * @since Available since 0.9dev - 2017-05-26
37 namespace Galette\Features
;
41 use Galette\DynamicFields\File
;
42 use Galette\DynamicFields\Date
;
43 use Galette\DynamicFields\Boolean
;
44 use Galette\Entity\DynamicFieldsHandle
;
47 * Dynamics fields trait
52 * @author Johan Cwiklinski <johan@x-tnd.be>
53 * @copyright 2017-2023 The Galette Team
54 * @license http://www.gnu.org/licenses/gpl-3.0.html GPL License 3.0 or (at your option) any later version
55 * @link http://galette.tuxfamily.org
56 * @since Available since 0.9dev - 2017-05-26
64 protected $name_pattern = 'info_field_';
66 /** @var DynamicFieldsHandle */
70 * Load dynamic fields for member
74 private function loadDynamicFields()
76 if (!property_exists($this, 'login')) {
79 $login = $this->login
;
81 $this->dynamics
= new DynamicFieldsHandle($this->zdb
, $login, $this);
87 * @return DynamicFieldsHandle
89 public function getDynamicFields()
91 if (null === $this->dynamics
) {
92 $this->loadDynamicFields();
94 return $this->dynamics
;
98 * Extract posted values for dynamic fields
100 * @param array $post Posted values
101 * @param array $required Array of required fields
102 * @param array $disabled Array of disabled fields
106 protected function dynamicsCheck(array $post, array $required, array $disabled)
108 if ($this->dynamics
=== null) {
110 'Dynamics fields have not been loaded, cannot be checked. (from: ' . __METHOD__
. ')',
113 $this->loadDynamicFields();
118 $fields = $this->dynamics
->getFields();
120 $dynamic_fields = [];
122 foreach ($post as $key => $value) {
123 // if the field is enabled, and match patterns check it
124 if (isset($disabled[$key]) ||
substr($key, 0, 11) != $this->name_pattern
) {
128 list($field_id, $val_index) = explode('_', str_replace($this->name_pattern
, '', $key));
129 if (!is_numeric($field_id) ||
!is_numeric($val_index)) {
133 $dynamic_fields[$key] = [
135 'field_id' => $field_id,
136 'val_index' => $val_index
140 //some fields may be mising in posted values (checkboxes)
141 foreach ($fields as $field) {
142 $pattern = '/' . $this->name_pattern
. $field->getId() . '_(\d)/';
143 if ($field instanceof Boolean
&& !preg_grep($pattern, array_keys($dynamic_fields))) {
144 $dynamic_fields[$this->name_pattern
. $field->getId() . '_1'] = [
146 'field_id' => $field->getId(),
152 foreach ($dynamic_fields as $key => $dfield_values) {
153 $field_id = $dfield_values['field_id'];
154 $value = $dfield_values['value'];
155 $val_index = $dfield_values['val_index'];
157 if ($fields[$field_id]->isRequired() && (trim($value) === '' ||
$value == null)) {
158 $this->errors
[] = str_replace(
160 $fields[$field_id]->getName(),
161 _T('Missing required field %field')
164 if ($fields[$field_id] instanceof File
) {
167 'member_%d_field_%d_value_%d',
172 if (file_exists(GALETTE_FILES_PATH
. $filename)) {
173 unlink(GALETTE_FILES_PATH
. $filename);
175 $this->dynamics
->setValue($this->id
, $field_id, $val_index, '');
177 if ($fields[$field_id] instanceof Date
&& !empty(trim($value))) {
180 $d = \DateTime
::createFromFormat(__("Y-m-d"), $value);
182 //try with non localized date
183 $d = \DateTime
::createFromFormat("Y-m-d", $value);
185 throw new \
Exception('Incorrect format');
188 } catch (Throwable
$e) {
191 'Wrong date format. field: ' . $field_id .
192 ', value: ' . $value . ', expected fmt: ' .
193 __("Y-m-d") . ' | ' . $e->getMessage(),
196 $this->errors
[] = str_replace(
203 $fields[$field_id]->getName()
205 _T("- Wrong date format (%date_format) for %field!")
210 if ($value !== null && trim($value) !== '') {
211 $this->dynamics
->setValue($this->id
, $field_id, $val_index, $value);
213 $this->dynamics
->unsetValue($this->id
, $field_id, $val_index);
225 * Stores dynamic fields
227 * @param bool $transaction True if a transaction already exists
231 protected function dynamicsStore($transaction = false)
233 if ($this->dynamics
=== null) {
235 'Dynamics fields have not been loaded, cannot be stored. (from: ' . __METHOD__
. ')',
238 $this->loadDynamicFields();
240 $return = $this->dynamics
->storeValues($this->id
, $transaction);
241 if (method_exists($this, 'updateModificationDate') && $this->dynamics
->hasChanged()) {
242 $this->updateModificationDate();
248 * Store dynamic Files
250 * @param array $files Posted files
254 protected function dynamicsFiles($files)
256 $this->loadDynamicFields();
257 $fields = $this->dynamics
->getFields();
260 foreach ($files as $key => $file) {
261 if (substr($key, 0, 11) != $this->name_pattern
) {
265 list($field_id, $val_index) = explode('_', str_replace($this->name_pattern
, '', $key));
266 if (!is_numeric($field_id) ||
!is_numeric($val_index)) {
271 $file['error'] == UPLOAD_ERR_NO_FILE
272 && $file['name'] == ''
273 && $file['tmp_name'] == ''
277 } elseif ($file['error'] !== UPLOAD_ERR_OK
) {
278 Analog
::log("file upload error", Analog
::ERROR
);
282 $tmp_filename = $file['tmp_name'];
283 if ($tmp_filename == '') {
284 Analog
::log("empty temporary filename", Analog
::ERROR
);
288 if (!is_uploaded_file($tmp_filename)) {
289 Analog
::log("not an uploaded file", Analog
::ERROR
);
294 $fields[$field_id]->getSize() ?
295 $fields[$field_id]->getSize() * 1024 : File
::DEFAULT_MAX_FILE_SIZE
* 1024;
296 if ($file['size'] > $max_size) {
298 "file too large: " . $file['size'] . " Ko, vs $max_size Ko allowed",
301 $this->errors
[] = preg_replace(
304 _T("File is too big. Maximum allowed size is %dKo")
309 $new_filename = sprintf(
310 'member_%d_field_%d_value_%d',
315 Analog
::log("new file: $new_filename", Analog
::DEBUG
);
319 GALETTE_FILES_PATH
. $new_filename
321 $this->dynamics
->setValue($this->id
, $field_id, $val_index, $file['name']);
325 if ($store === true) {
326 $this->dynamicsStore();
331 * Remove dynamic fields values
333 * @param bool $transaction True if a transaction already exists
337 protected function dynamicsRemove($transaction = false)
339 if ($this->dynamics
=== null) {
341 'Dynamics fields have not been loaded, cannot be removed. (from: ' . __METHOD__
. ')',
344 $this->loadDynamicFields();
346 $return = $this->dynamics
->removeValues($this->id
, $transaction);
357 public function getErrors()
359 return $this->errors
;
363 * Validate data for dynamic fields
364 * Set valid data in current object, also resets errors list
366 * @param array $values Dynamic fields values
367 * @param string $prefix Prefix to replace, default to 'dynfield_'
371 public function dynamicsValidate($values, $prefix = 'dynfield_')
374 foreach ($values as $key => $value) {
375 $dfields[str_replace($prefix, $this->name_pattern
, $key)] = $value;
377 return $this->dynamicsCheck($dfields, [], []);