From 2bf7e14477ab585f41135b6961e9670fa76663b7 Mon Sep 17 00:00:00 2001 From: nathangray Date: Tue, 13 Nov 2018 14:01:55 -0700 Subject: [PATCH] Etemplate - add custom field sub-fields for app-entry widgets - add regex & regex_replace attributes to change value around --- api/js/etemplate/et2_widget_entry.js | 14 +++- api/src/Etemplate/Widget/Entry.php | 112 +++++++++++++++++++++++++-- api/src/Storage/Merge.php | 2 +- 3 files changed, 119 insertions(+), 9 deletions(-) diff --git a/api/js/etemplate/et2_widget_entry.js b/api/js/etemplate/et2_widget_entry.js index 71d49243be..8f5545e71d 100644 --- a/api/js/etemplate/et2_widget_entry.js +++ b/api/js/etemplate/et2_widget_entry.js @@ -53,6 +53,18 @@ var et2_entry = (function(){ "use strict"; return et2_valueWidget.extend( type: 'string', default: '2' }, + regex: { + name: 'Regular expression pattern', + description: 'Only used server-side in a preg_replace with regex_replace to modify the value', + default: et2_no_init, + type: 'string' + }, + regex_replace: { + name: 'Regular expression replacement pattern', + description: 'Only used server-side in a preg_replace with regex to modify the value', + default: et2_no_init, + type: 'string' + }, value: { type: 'any' }, @@ -104,7 +116,7 @@ var et2_entry = (function(){ "use strict"; return et2_valueWidget.extend( loadField: function() { // Create widget of correct type var attrs = { - id: this.id + '[' +this.options.field+']', + id: this.id + (this.options.field ? '[' +this.options.field+']' : ''), type: 'label', readonly: this.options.readonly }; diff --git a/api/src/Etemplate/Widget/Entry.php b/api/src/Etemplate/Widget/Entry.php index 66026b7588..60e4f72045 100644 --- a/api/src/Etemplate/Widget/Entry.php +++ b/api/src/Etemplate/Widget/Entry.php @@ -15,6 +15,8 @@ namespace EGroupware\Api\Etemplate\Widget; +use EGroupware\Api\Storage\Merge; + /** * eTemplate Entry widget * @@ -35,7 +37,7 @@ abstract class Entry extends Transformer * * @var string|array */ - protected $legacy_options = 'field'; + protected $legacy_options = 'field,compare,alternate_fields,options'; // field, compare, alternate_fields /** * Array with a transformation description, based on attributes to modify. @@ -94,17 +96,24 @@ abstract class Entry extends Transformer // Set the new value so transformer can find it. Use prefix to avoid changing the original value $new_value =& self::get_array(self::$request->content, self::ID_PREFIX .$this->id, true, false); if (true) $new_value = $data; + + // Check for missing field + if(!$attrs['field'] && !$attrs['alternate_fields']) + { + self::set_validation_error(self::ID_PREFIX . $this->id, lang('Unable to find field attribute')); + return; + } + + // Field is reference to customfield? + $this->customfield($attrs, $new_value); + + $this->regex($attrs, $new_value); + $this->id = self::ID_PREFIX . $this->id . "[{$attrs['field']}]"; - $old_type = self::getElementAttribute($this->id, 'type'); parent::beforeSendToClient($cname, $expand); - // Check for conflict - more than one with same id/field and different type - if($old_type && $old_type != $this->type) - { - //self::set_validation_error($this->id, lang('%1, duplicate ID', $this)); - } } /** @@ -125,4 +134,93 @@ abstract class Entry extends Transformer { return array(); } + + /** + * Using ID, field and alternate fields, find the one to use + * + * @param Array $attrs + * @param Array $data + * @return Reference into data array for value + */ + protected function &get_data_field($attrs, &$data) + { + $id = is_array($data) ? static::get_array($data, $this->id) : $data; + + $value =& $data; + if(!is_array($value)) return $value; + + foreach(array($attrs['field']) + explode(':',$attrs['alternate_fields']) as $field) + { + if($value[$field]) + { + return $value[$field]; + } + } + return null; + } + + /** + * Handle if field is a reference to a customfield, or a sub-field if the custom + * field is an application entry. Allowed forms are similar to merge: + * #customfield + * #customfield/n_fn + * #infolog_cf/#addressbook_cf/n_fn + * etc. + * + * @param Array $attrs Current field attributes + * @param Array $data Current entry data + */ + protected function customfield($attrs, &$data) + { + list($app, $type) = explode('-',$attrs['type']); + $id = is_array($data) ? static::get_array($data, $this->id) : $data; + if(!$app || !$type || !$GLOBALS['egw_info']['apps'][$app] || !$id || + // Simple CF, already there + $data[$attrs['field']] + ) + { + return; + } + + if(substr($attrs['field'], 0, 1) == '#' && $cfs = \EGroupware\Api\Storage\Customfields::get($app)) + { + try + { + $classname = "{$app}_merge"; + $merge = new $classname(); + $replacement_field = '$$'.$attrs['field'].'$$'; + $replacements = $merge->get_app_replacements($app, $id, $replacement_field); + $data[$attrs['field']] = $replacements[$replacement_field]; + } + catch(\Exception $e) + { + $this->set_validation_error($this->id, $e->getMessage()); + } + } + } + + /** + * Handle regex attribute that allows modifying the value via regex replace + * + * @param Array $attrs Current field attributes + * @param Array $data Current entry data + */ + protected function regex($attrs, &$data) + { + $id = is_array($data) ? static::get_array($data, $this->id) : $data; + $value =& $this->get_data_field($attrs, $data); + if(!$attrs['regex'] || !$id || !$value) + { + return; + } + $regex = $attrs['regex']; + $replace = $attrs['regex_replace']; + if(!$replace) + { + $regex = explode(',', $attrs['regex']); + $replace = array_pop($regex); + $regex = implode(',', $regex); + } + $value = preg_replace($regex, $replace, $value); + } } \ No newline at end of file diff --git a/api/src/Storage/Merge.php b/api/src/Storage/Merge.php index aef875b630..fc7efc4f6b 100644 --- a/api/src/Storage/Merge.php +++ b/api/src/Storage/Merge.php @@ -1526,7 +1526,7 @@ abstract class Merge * @param string $content * @return array */ - protected function get_app_replacements($app, $id, $content, $prefix='') + public function get_app_replacements($app, $id, $content, $prefix='') { $replacements = array(); if($app == 'addressbook')