diff --git a/addressbook/inc/class.addressbook_merge.inc.php b/addressbook/inc/class.addressbook_merge.inc.php
index b5915f50ce..e36417b2d6 100644
--- a/addressbook/inc/class.addressbook_merge.inc.php
+++ b/addressbook/inc/class.addressbook_merge.inc.php
@@ -36,6 +36,9 @@ class addressbook_merge extends bo_merge
// overwrite global export-limit, if an addressbook one is set
$this->export_limit = bo_merge::getExportLimit('addressbook');
+
+ // switch of handling of html formated content, if html is not used
+ $this->parse_html_styles = egw_customfields::use_html('addressbook');
}
/**
@@ -57,7 +60,7 @@ class addressbook_merge extends bo_merge
}
// Links
- $replacements += $this->get_all_links('addressbook', $id, $prefix, $content);
+ $replacements += $this->get_all_links('addressbook', $id, '', $content);
if (!(strpos($content,'$$calendar/') === false))
{
$replacements += $this->calendar_replacements($id,!(strpos($content,'$$calendar/-1/') === false));
@@ -68,8 +71,8 @@ class addressbook_merge extends bo_merge
/**
* Return replacements for the calendar (next events) of a contact
*
- * @param int $contact contact-id
- * @param boolean $last_event_too=false also include information about the last event
+ * @param int $id contact-id
+ * @param boolean $last_event_too =false also include information about the last event
* @return array
*/
protected function calendar_replacements($id,$last_event_too=false)
diff --git a/calendar/inc/class.calendar_merge.inc.php b/calendar/inc/class.calendar_merge.inc.php
index b6eb870480..b48b2ed84e 100644
--- a/calendar/inc/class.calendar_merge.inc.php
+++ b/calendar/inc/class.calendar_merge.inc.php
@@ -73,6 +73,8 @@ class calendar_merge extends bo_merge
// overwrite global export-limit, if one is set for calendar/appointments
$this->export_limit = bo_merge::getExportLimit('calendar');
+ // switch of handling of html formated content, if html is not used
+ $this->parse_html_styles = egw_customfields::use_html('calendar');
$this->bo = new calendar_boupdate();
self::$range_tags['start'] = $GLOBALS['egw_info']['user']['preferences']['common']['dateformat'];
@@ -217,7 +219,7 @@ class calendar_merge extends bo_merge
}
$custom = config::get_customfields('calendar');
- foreach($custom as $name => $field)
+ foreach(array_keys($custom) as $name)
{
$replacements['$$'.($prefix?$prefix.'/':'').'#'.$name.'$$'] = $event['#'.$name];
}
@@ -245,7 +247,7 @@ class calendar_merge extends bo_merge
*/
public function day_plugin($plugin,$date,$n,$repeat)
{
- static $days;
+ static $days = null;
if(is_array($date) && !$date['start']) {
// List of IDs
if($date[0]['start']) {
@@ -275,12 +277,12 @@ class calendar_merge extends bo_merge
'cfs' => array(), // read all custom-fields
));
- $days = array();
+ if (true) $days = array();
$replacements = array();
$time_format = $GLOBALS['egw_info']['user']['preferences']['common']['timeformat'] == 12 ? 'h:i a' : 'H:i';
foreach($events as $day => $list)
{
- foreach($list as $key => $event)
+ foreach($list as $event)
{
if($this->ids && !in_array($event['id'], $this->ids)) continue;
$start = egw_time::to($event['start'], 'array');
@@ -304,7 +306,10 @@ class calendar_merge extends bo_merge
);
if(!is_array($days[date('Ymd',$_date)][date('l',strtotime($day))])) {
$blank = $this->calendar_replacements(array());
- foreach($blank as &$value) $value = '';
+ foreach($blank as &$value)
+ {
+ $value = '';
+ }
$days[date('Ymd',$_date)][date('l',strtotime($day))][] = $blank;
}
$days[date('Ymd',$_date)][date('l',strtotime($day))][0] += $date_marker;
@@ -326,7 +331,7 @@ class calendar_merge extends bo_merge
*/
public function day($plugin,$id,$n,$repeat)
{
- static $days;
+ static $days = null;
// Figure out which day
list($type, $which) = explode('_',$plugin);
@@ -341,9 +346,9 @@ class calendar_merge extends bo_merge
}
$id = $dates;
}
- $date = $this->bo->date2array($id['start']);
- $date['day'] = $which;
- $date = $this->bo->date2ts($date);
+ $arr = $this->bo->date2array($id['start']);
+ $arr['day'] = $which;
+ $date = $this->bo->date2ts($arr);
if(is_array($id) && $id['start'] && ($date < $id['start'] || $date > $id['end'])) return array();
}
elseif ($plugin == 'selected')
@@ -376,18 +381,18 @@ class calendar_merge extends bo_merge
));
$replacements = array();
- $days = array();
+ if (true) $days = array();
$time_format = $GLOBALS['egw_info']['user']['preferences']['common']['timeformat'] == 12 ? 'h:i a' : 'H:i';
foreach($events as $day => $list)
{
- foreach($list as $key => $event)
+ foreach($list as $event)
{
if($this->ids && !in_array($event['id'], $this->ids)) continue;
$start = egw_time::to($event['start'], 'array');
$end = egw_time::to($event['end'], 'array');
$replacements = $this->calendar_replacements($event);
if($start['year'] == $end['year'] && $start['month'] == $end['month'] && $start['day'] == $end['day']) {
- $dow = date('l',$event['start']);
+ //$dow = date('l',$event['start']);
} else {
// Fancy date+time formatting for multi-day events
$replacements['$$calendar_starttime$$'] = date($time_format, $day == date('Ymd', $event['start']) ? $event['start'] : mktime(0,0,0,0,0,1));
@@ -402,7 +407,10 @@ class calendar_merge extends bo_merge
);
if(!is_array($days[date('Ymd',$_date)][$plugin])) {
$blank = $this->calendar_replacements(array());
- foreach($blank as &$value) $value = '';
+ foreach($blank as &$value)
+ {
+ $value = '';
+ }
$days[date('Ymd',$_date)][$plugin][] = $blank;
}
$days[date('Ymd',$_date)][$plugin][0] += $date_marker;
@@ -423,6 +431,8 @@ class calendar_merge extends bo_merge
*/
public function participant($plugin,$id,$n)
{
+ unset($plugin); // not used, but required by function signature
+
if(!is_array($id) || !$id['start']) {
$event = $this->bo->read(is_array($id) ? $id['id'] : $id, is_array($id) ? $id['recur_date'] : null);
} else {
@@ -441,6 +451,7 @@ class calendar_merge extends bo_merge
if(!$participant) return array();
// Add some common information
+ $quantity = $role = null;
calendar_so::split_status($status,$quantity,$role);
if ($role != 'REQ-PARTICIPANT')
{
@@ -508,6 +519,7 @@ class calendar_merge extends bo_merge
echo "
\n";
echo ''.lang('Calendar fields:')." |
";
+ $n = 0;
foreach(array(
'calendar_id' => lang('Calendar ID'),
'calendar_title' => lang('Title'),
@@ -544,7 +556,7 @@ class calendar_merge extends bo_merge
echo ''.lang('Range fields').": |
";
echo ''.lang('If you select a range (month, week, etc) instead of a list of entries, these extra fields are available').' |
';
- foreach(self::$range_tags as $name => $format)
+ foreach(array_keys(self::$range_tags) as $name)
{
echo '{{range/'.$name.'}} | '.lang($name)." |
\n";
}
@@ -577,7 +589,7 @@ class calendar_merge extends bo_merge
}
echo '
';
echo ''.lang('Daily tables').": | ";
- foreach(self::$relative as $key => $value) {
+ foreach(self::$relative as $value) {
echo '{{table/'.$value. '}} ... {{endtable}} | ';
}
echo '{{table/day_n}} ... {{endtable}} | 1 <= n <= 31 | ';
diff --git a/etemplate/inc/class.bo_merge.inc.php b/etemplate/inc/class.bo_merge.inc.php
index 631d185b41..168ba66a0e 100644
--- a/etemplate/inc/class.bo_merge.inc.php
+++ b/etemplate/inc/class.bo_merge.inc.php
@@ -5,13 +5,15 @@
* @link http://www.egroupware.org
* @author Ralf Becker
* @package addressbook
- * @copyright (c) 2007-13 by Ralf Becker
+ * @copyright (c) 2007-14 by Ralf Becker
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @version $Id$
*/
/**
* Document merge print
+ *
+ * @todo move apply_styles call into merge_string to run for each entry merged and not all together to lower memory requirements
*/
abstract class bo_merge
{
@@ -78,9 +80,19 @@ abstract class bo_merge
/**
* Parse HTML styles into target document style, if possible
+ *
+ * Apps not using html in there own data should set this with egw_customfields::use_html($app)
+ * to avoid memory and time consuming html processing.
*/
protected $parse_html_styles = true;
+ /**
+ * Enable this to report memory_usage to error_log
+ *
+ * @var boolean
+ */
+ public $report_memory_usage = false;
+
/**
* Constructor
*
@@ -249,7 +261,7 @@ abstract class bo_merge
foreach($this->contacts->customfields as $name => $field)
{
$name = '#'.$name;
- $replacements['$$'.($prefix ? $prefix.'/':'').$name.'$$'] = customfields_widget::format_customfield($field, (string)$contact[$name]);
+ $replacements['$$'.($prefix ? $prefix.'/':'').$name.'$$'] = egw_customfields::format($field, (string)$contact[$name]);
}
// Add in extra cat field
@@ -339,9 +351,10 @@ abstract class bo_merge
*
* Calls get_links() repeatedly to get all the combinations for the content.
*
- * @param app String appname
- * @param id String ID of record
- * @param content String document content
+ * @param $app String appname
+ * @param $id String ID of record
+ * @param $prefix
+ * @param $content String document content
*/
protected function get_all_links($app, $id, $prefix, &$content)
{
@@ -550,16 +563,13 @@ abstract class bo_merge
return $content;
}
- protected function apply_styles (&$content, $mimetype)
+ protected function apply_styles (&$content, $mimetype, $mso_application_progid=null)
{
- if ($mimetype == 'application/xml' &&
- preg_match('/'.preg_quote('').'/',substr($content,0,200),$matches))
+ if (!isset($mso_application_progid))
{
- $mso_application_progid = $matches[1];
- }
- else
- {
- $mso_application_progid = '';
+ $mso_application_progid = $mimetype == 'application/xml' &&
+ preg_match('/'.preg_quote('').'/',substr($content,0,200),$matches) ?
+ $matches[1] : '';
}
// Tags we can replace with the target document's version
$replace_tags = array();
@@ -585,9 +595,6 @@ abstract class bo_merge
case 'application/xmlWord.Document': // Word 2003*/
case 'application/vnd.openxmlformats-officedocument.wordprocessingml.document': // ms office 2007
case 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet':
- $replace_tags = array(
- 'b','strong','i','em','u','span'
- );
// It seems easier to split the parent tags here
$replace_tags = array(
// Tables, lists don't go inside
@@ -615,7 +622,6 @@ abstract class bo_merge
$i++;
} while($count > 0 && $i < 10);
-//echo $content;die();
$doc = new DOMDocument();
$xslt = new XSLTProcessor();
$xslt_file = $mimetype == 'application/xml' ? 'wordml.xslt' : 'msoffice.xslt';
@@ -637,7 +643,6 @@ abstract class bo_merge
}
$content = $xslt->transformToXml($element);
-//echo $content;die();
// Word 2003 needs two declarations, add extra declaration back in
if($mimetype == 'application/xml' && $mso_application_progid == 'Word.Document' && strpos($content, ''.$content;
@@ -740,7 +745,8 @@ abstract class bo_merge
$err = lang('for more than one contact in a document use the tag pagerepeat!');
return false;
}
- foreach ((array)$ids as $id)
+ if ($this->report_memory_usage) error_log(__METHOD__."(count(ids)=".count($ids).") strlen(contentrepeat)=".strlen($contentrepeat).', strlen(labelrepeat)='.strlen($Labelrepeat));
+ foreach ((array)$ids as $n => $id)
{
if ($contentrepeat) $content = $contentrepeat; //content to repeat
if ($lableprint) $content = $Labelrepeat;
@@ -760,7 +766,7 @@ abstract class bo_merge
$err = $e->getMessage();
return false;
}
-
+ if ($this->report_memory_usage) error_log(__METHOD__."() $n: $id ".egw_vfs::hsize(memory_get_usage(true)));
// some general replacements: current user, date and time
if (strpos($content,'$$user/') !== null && ($user = $GLOBALS['egw']->accounts->id2name($GLOBALS['egw_info']['user']['account_id'],'person_id')))
{
@@ -861,6 +867,7 @@ abstract class bo_merge
$err = lang('%1 not implemented for %2!','$$pagerepeat$$',$mimetype);
return false;
}
+ if ($this->report_memory_usage) error_log(__METHOD__."() returning ".egw_vfs::hsize(memory_get_peak_usage(true)));
return $content;
}
@@ -1534,6 +1541,7 @@ abstract class bo_merge
return $err;
}
}
+ if ($this->report_memory_usage) error_log(__METHOD__."() after HTML processing ".egw_vfs::hsize(memory_get_peak_usage(true)));
}
if(!empty($name))
{
@@ -1564,6 +1572,7 @@ abstract class bo_merge
{
exec('/usr/bin/zip -F '.escapeshellarg($archive));
}
+ if ($this->report_memory_usage) error_log(__METHOD__."() after ZIP processing ".egw_vfs::hsize(memory_get_peak_usage(true)));
html::content_header($name,$mimetype,filesize($archive));
readfile($archive,'r');
}
diff --git a/etemplate/inc/class.customfields_widget.inc.php b/etemplate/inc/class.customfields_widget.inc.php
index 4209051dab..909b1becaa 100644
--- a/etemplate/inc/class.customfields_widget.inc.php
+++ b/etemplate/inc/class.customfields_widget.inc.php
@@ -245,7 +245,7 @@ class customfields_widget
case 'select' :
if (count($field['values']) == 1 && isset($field['values']['@']))
{
- $field['values'] = $this->_get_options_from_file($field['values']['@']);
+ $field['values'] = egw_customfields::get_options_from_file($field['values']['@']);
}
if($this->advanced_search && $field['rows'] <= 1) $field['values'][''] = lang('doesn\'t matter');
foreach($field['values'] as $key => $val)
@@ -496,56 +496,6 @@ class customfields_widget
return True; // extra Label is ok
}
- /**
- * Read the options of a 'select' or 'radio' custom field from a file
- *
- * For security reasons that file has to be relative to the eGW root
- * (to not use that feature to explore arbitrary files on the server)
- * and it has to be a php file setting one variable called options,
- * (to not display it to anonymously by the webserver).
- * The $options var has to be an array with value => label pairs, eg:
- *
- * 'Option A',
- * 'b' => 'Option B',
- * 'c' => 'Option C',
- * );
- *
- * @param string $file file name inside the eGW server root, either relative to it or absolute
- * @return array in case of an error we return a single option with the message
- */
- public static function _get_options_from_file($file)
- {
- if (!($path = realpath($file{0} == '/' ? $file : EGW_SERVER_ROOT.'/'.$file)) || // file does not exist
- substr($path,0,strlen(EGW_SERVER_ROOT)+1) != EGW_SERVER_ROOT.'/' || // we are NOT inside the eGW root
- basename($path,'.php').'.php' != basename($path) || // extension is NOT .php
- basename($path) == 'header.inc.php') // dont allow to include our header again
- {
- return array(lang("'%1' is no php file in the eGW server root (%2)!".': '.$path,$file,EGW_SERVER_ROOT));
- }
- include($path);
-
- return $options;
- }
-
- /**
- * Get the customfield types containing links
- *
- * @return array with customefield types as values
- */
- public static function get_customfield_link_types()
- {
- static $link_types;
-
- if (is_null($link_types))
- {
- $link_types = array_keys(array_intersect(egw_link::app_list('query'),egw_link::app_list('title')));
- $link_types[] = 'link-entry';
- }
- return $link_types;
- }
-
/**
* Check if there are links in the custom fields and update them
*
@@ -554,129 +504,25 @@ class customfields_widget
*
* @param string $own_app own appname
* @param array $values new values including the custom fields
- * @param array $old=null old values before the update, if existing
- * @param string $id_name='id' name/key of the (link-)id in $values
+ * @param array $old =null old values before the update, if existing
+ * @param string $id_name ='id' name/key of the (link-)id in $values
+ * @deprecated use egw_customfields::update_links($own_app,array $values,array $old=null,$id_name='id')
*/
public static function update_customfield_links($own_app,array $values,array $old=null,$id_name='id')
{
- $link_types = self::get_customfield_link_types();
-
- foreach(egw_customfields::get($own_app) as $name => $data)
- {
- if (!in_array($data['type'],$link_types)) continue;
-
- // do we have a different old value --> delete that link
- if ($old && $old['#'.$name] && $old['#'.$name] != $values['#'.$name])
- {
- if ($data['type'] == 'link-entry')
- {
- list($app,$id) = explode(':',$old['#'.$name]);
- }
- else
- {
- $app = $data['type'];
- $id = $old['#'.$name];
- }
- egw_link::unlink(false,$own_app,$values[$id_name],'',$app,$id);
- }
- if ($data['type'] == 'link-entry')
- {
- list($app,$id) = explode(':',$values['#'.$name]);
- }
- else
- {
- $app = $data['type'];
- $id = $values['#'.$name];
- }
- if ($id) // create new link, does nothing for already existing links
- {
- egw_link::link($own_app,$values[$id_name],$app,$id);
- }
- }
+ return egw_customfields::update_links($own_app, $values, $old , $id_name);
}
- /**
- * Non printable custom fields eg. UI elements
- *
- * @var array
- */
- public static $non_printable_fields = array('button');
-
/**
* Format a single custom field value as string
*
* @param array $field field defintion incl. type
* @param string $value field value
* @return string formatted value
+ * @deprecated use egw_customfields::format($field, $value)
*/
public static function format_customfield(array $field, $value)
{
- switch($field['type'])
- {
- case 'select-account':
- if ($value)
- {
- $values = array();
- foreach($field['rows'] > 1 ? explode(',', $value) : (array) $value as $value)
- {
- $values[] = common::grab_owner_name($value);
- }
- $value = implode(', ',$values);
- }
- break;
-
- case 'checkbox':
- $value = $value ? 'X' : '';
- break;
-
- case 'select':
- case 'radio':
- if (count($field['values']) == 1 && isset($field['values']['@']))
- {
- $field['values'] = customfields_widget::_get_options_from_file($field['values']['@']);
- }
- $values = array();
- foreach($field['rows'] > 1 ? explode(',', $value) : (array) $value as $value)
- {
- $values[] = isset($field['values'][$value]) ? $field['values'][$value] : '#'.$value;
- }
- $value = implode(', ', $values);
- break;
-
- case 'date':
- case 'date-time':
- if ($value)
- {
- $format = $field['len'] ? $field['len'] : ($field['type'] == 'date' ? 'Y-m-d' : 'Y-m-d H:i:s');
- $date = array_combine(preg_split('/[\\/. :-]/',$format), preg_split('/[\\/. :-]/',$value));
- $value = common::dateformatorder($date['Y'], $date['m'], $date['d'],true);
- if (isset($date['H'])) $value .= ' '.common::formattime($date['H'], $date['i']);
- }
- break;
-
- case 'htmlarea': // ToDo: EMail probably has a nicer html2text method
- if ($value) $value = strip_tags(preg_replace('/<(br|p)[^>]*>/i', "\r\n", str_replace(array("\r", "\n"), '', $value)));
- break;
-
- case 'ajax_select': // ToDo: returns unchanged value for now
- break;
-
- default:
- // handling for several link types
- if ($value && in_array($field['type'], self::get_customfield_link_types()))
- {
- if ($field['type'] == 'link-entry' || strpos($value, ':') !== false)
- {
- list($app, $value) = explode(':', $value);
- }
- else
- {
- $app = $field['type'];
- }
- if ($value) $value = egw_link::title($app, $value);
- }
- break;
- }
- return $value;
+ return egw_customfields::format($field, $value);
}
}
diff --git a/etemplate/inc/class.etemplate_widget_customfields.inc.php b/etemplate/inc/class.etemplate_widget_customfields.inc.php
index 706d9ad9fc..e9028b8034 100644
--- a/etemplate/inc/class.etemplate_widget_customfields.inc.php
+++ b/etemplate/inc/class.etemplate_widget_customfields.inc.php
@@ -275,7 +275,7 @@ class etemplate_widget_customfields extends etemplate_widget_transformer
case 'radio':
if (count($field['values']) == 1 && isset($field['values']['@']))
{
- $field['values'] = self::_get_options_from_file($field['values']['@']);
+ $field['values'] = egw_customfields::get_options_from_file($field['values']['@']);
}
// keep extra values set by app code, eg. addressbook advanced search
if (is_array(self::$request->sel_options[self::$prefix.$fname]))
@@ -370,38 +370,4 @@ class etemplate_widget_customfields extends etemplate_widget_transformer
}
}
}
-
- /**
- * Read the options of a 'select' or 'radio' custom field from a file
- *
- * For security reasons that file has to be relative to the eGW root
- * (to not use that feature to explore arbitrary files on the server)
- * and it has to be a php file setting one variable called options,
- * (to not display it to anonymously by the webserver).
- * The $options var has to be an array with value => label pairs, eg:
- *
- * 'Option A',
- * 'b' => 'Option B',
- * 'c' => 'Option C',
- * );
- *
- * @param string $file file name inside the eGW server root, either relative to it or absolute
- * @return array in case of an error we return a single option with the message
- */
- public static function _get_options_from_file($file)
- {
- if (!($path = realpath($file{0} == '/' ? $file : EGW_SERVER_ROOT.'/'.$file)) || // file does not exist
- substr($path,0,strlen(EGW_SERVER_ROOT)+1) != EGW_SERVER_ROOT.'/' || // we are NOT inside the eGW root
- basename($path,'.php').'.php' != basename($path) || // extension is NOT .php
- basename($path) == 'header.inc.php') // dont allow to include our header again
- {
- return array(lang("'%1' is no php file in the eGW server root (%2)!".': '.$path,$file,EGW_SERVER_ROOT));
- }
- $options = array();
- include($path);
-
- return $options;
- }
}
diff --git a/etemplate/inc/class.historylog_widget.inc.php b/etemplate/inc/class.historylog_widget.inc.php
index d4912e18f7..dbe8ab14ff 100644
--- a/etemplate/inc/class.historylog_widget.inc.php
+++ b/etemplate/inc/class.historylog_widget.inc.php
@@ -220,7 +220,7 @@ class historylog_widget
}
elseif($cf_data['values']['@'])
{
- self::$status_widgets['#'.$cf_name] = customfields_widget::_get_options_from_file($cf_data['values']['@']);
+ self::$status_widgets['#'.$cf_name] = egw_customfields::get_options_from_file($cf_data['values']['@']);
}
elseif(count($cf_data['values']))
{
diff --git a/filemanager/inc/class.filemanager_merge.inc.php b/filemanager/inc/class.filemanager_merge.inc.php
index 842a00c641..fe6732f683 100644
--- a/filemanager/inc/class.filemanager_merge.inc.php
+++ b/filemanager/inc/class.filemanager_merge.inc.php
@@ -55,10 +55,14 @@ class filemanager_merge extends bo_merge
function __construct($_dir = '')
{
parent::__construct();
+
if($_dir)
{
$this->dir = $_dir;
}
+
+ // switch of handling of html formated content, if html is not used
+ $this->parse_html_styles = egw_customfields::use_html('filemanager');
}
/**
@@ -104,20 +108,21 @@ class filemanager_merge extends bo_merge
$file['gid'] *= -1; // our widgets use negative gid's
if (($props = egw_vfs::propfind($id)))
{
- foreach($props as $prop) $file[$prop['name']] = $prop['val'];
+ foreach($props as $prop)
+ {
+ $file[$prop['name']] = $prop['val'];
+ }
}
if (($file['is_link'] = egw_vfs::is_link($id)))
{
$file['symlink'] = egw_vfs::readlink($id);
}
- $extra = egw_vfs::getExtraInfo($id);
-
// Custom fields
if($content && strpos($content, '#') !== 0)
{
// Expand link-to custom fields
- $this->cf_link_to_expand($file, $content, $info);
-
+ $this->cf_link_to_expand($file, $content, $info);
+
foreach(config::get_customfields('filemanager') as $name => $field)
{
// Set any missing custom fields, or the marker will stay
@@ -134,18 +139,7 @@ class filemanager_merge extends bo_merge
$file['#'.$name] = egw_time::to($file['#'.$name], $field['type'] == 'date' ? true : '');
}
}
- }
-
- // Links
- /* Not applicable to filemanager
- $file['links'] = $this->get_links('filemanager', $id, '!'.egw_link::VFS_APPNAME);
- $file['attachments'] = $this->get_links('filemanager', $id, egw_link::VFS_APPNAME);
- $file['links_attachments'] = $this->get_links('filemanager', $id);
- foreach(array_keys($GLOBALS['egw_info']['user']['apps']) as $app)
- {
- $file["links/{$app}"] = $this->get_links('filemanager',$id, $app);
}
- */
// If in apps folder, try for app-specific placeholders
if($dirlist[1] == 'apps' && count($dirlist) > 1)
@@ -182,7 +176,9 @@ class filemanager_merge extends bo_merge
}
}
// Silently discard & continue
- catch(Exception $e) {}
+ catch(Exception $e) {
+ unset($e); // not used
+ }
}
}
}
diff --git a/infolog/inc/class.infolog_merge.inc.php b/infolog/inc/class.infolog_merge.inc.php
index a837df7a1b..785268de54 100644
--- a/infolog/inc/class.infolog_merge.inc.php
+++ b/infolog/inc/class.infolog_merge.inc.php
@@ -6,7 +6,7 @@
* @author Ralf Becker
* @author Nathan Gray
* @package infolog
- * @copyright (c) 2007-9 by Ralf Becker
+ * @copyright (c) 2007-14 by Ralf Becker
* @copyright 2011 Nathan Gray
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @version $Id$
@@ -48,6 +48,9 @@ class infolog_merge extends bo_merge
'info_datemodified',
'info_created',
);
+
+ // switch of handling of html formated content, if html is not used
+ $this->parse_html_styles = egw_customfields::use_html('infolog');
}
/**
@@ -119,7 +122,7 @@ class infolog_merge extends bo_merge
// Set any missing custom fields, or the marker will stay
$array = $record->get_record_array();
- foreach($this->bo->customfields as $name => $field)
+ foreach(array_keys($this->bo->customfields) as $name)
{
if(!$array['#'.$name])
{
@@ -133,7 +136,6 @@ class infolog_merge extends bo_merge
// Timesheet time
if(strpos($content, 'info_sum_timesheets'))
{
- $timesheets = array();
$links = egw_link::get_links('infolog',$id,'timesheet');
$sum = ExecMethod('timesheet.timesheet_bo.sum',$links);
$info['$$info_sum_timesheets$$'] = $sum['duration'];
@@ -141,7 +143,8 @@ class infolog_merge extends bo_merge
// Check for linked project ID
$links = egw_link::get_links('infolog', $id, 'projectmanager');
- foreach($links as $link_id => $app_id) {
+ foreach($links as $app_id)
+ {
$array['pm_id'] = $app_id;
$array['project'] = egw_link::title('projectmanager', $app_id);
break;
@@ -205,32 +208,32 @@ class infolog_merge extends bo_merge
{
echo ' | * '.lang('Addressbook placeholders available'). ' | ';
}
-
+
echo ''.lang('Parent').": | ";
echo '{{info_id_parent/info_subject}} | '.lang('All other %1 fields are valid',lang('infolog'))." | \n";
echo ''.lang('Contact fields').': | ';
- $n = 0;
- foreach($this->contacts->contact_fields as $name => $label)
- {
- if (in_array($name,array('tid','label','geo'))) continue; // dont show them, as they are not used in the UI atm.
+ $i = 0;
+ foreach($this->contacts->contact_fields as $name => $label)
+ {
+ if (in_array($name,array('tid','label','geo'))) continue; // dont show them, as they are not used in the UI atm.
- if (in_array($name,array('email','org_name','tel_work','url')) && $n&1) // main values, which should be in the first column
- {
- echo "\n";
- $n++;
- }
- if (!($n&1)) echo '';
- echo '{{info_contact/'.$name.'}} | '.$label.' | ';
- if ($n&1) echo " \n";
- $n++;
- }
+ if (in_array($name,array('email','org_name','tel_work','url')) && $n&1) // main values, which should be in the first column
+ {
+ echo "\n";
+ $i++;
+ }
+ if (!($i&1)) echo '';
+ echo '{{info_contact/'.$name.'}} | '.$label.' | ';
+ if ($i&1) echo " \n";
+ $i++;
+ }
- echo ''.lang('Custom fields').": | ";
- foreach($this->contacts->customfields as $name => $field)
- {
- echo '{{info_contact/#'.$name.'}} | '.$field['label']." | \n";
- }
+ echo ''.lang('Custom fields').": | ";
+ foreach($this->contacts->customfields as $name => $field)
+ {
+ echo '{{info_contact/#'.$name.'}} | '.$field['label']." | \n";
+ }
echo ''.lang('General fields:')." | ";
foreach(array(
diff --git a/phpgwapi/inc/class.egw_customfields.inc.php b/phpgwapi/inc/class.egw_customfields.inc.php
index be02f8f3ab..6d59ab78e8 100755
--- a/phpgwapi/inc/class.egw_customfields.inc.php
+++ b/phpgwapi/inc/class.egw_customfields.inc.php
@@ -51,10 +51,10 @@ class egw_customfields implements IteratorAggregate
* Constructor
*
* @param string $app
- * @param boolean $all_private_too=false should all the private fields be returned too, default no
- * @param string $only_type2=null if given only return fields of type2 == $only_type2
- * @param int $start=0
- * @param int $num_rows=null
+ * @param boolean $all_private_too =false should all the private fields be returned too, default no
+ * @param string $only_type2 =null if given only return fields of type2 == $only_type2
+ * @param int $start =0
+ * @param int $num_rows =null
* @return array with customfields
*/
function __construct($app, $all_private_too=false, $only_type2=null, $start=0, $num_rows=null)
@@ -86,9 +86,9 @@ class egw_customfields implements IteratorAggregate
*/
function getIterator()
{
- return new egw_db_callback_iterator($this->iterator, function($row)
+ return new egw_db_callback_iterator($this->iterator, function($_row)
{
- $row = egw_db::strip_array_keys($row, 'cf_');
+ $row = egw_db::strip_array_keys($_row, 'cf_');
$row['private'] = $row['private'] ? explode(',', $row['private']) : array();
$row['type2'] = $row['type2'] ? explode(',', $row['type2']) : array();
$row['values'] = json_decode($row['values'], true);
@@ -121,8 +121,8 @@ class egw_customfields implements IteratorAggregate
* Get customfield array of an application
*
* @param string $app
- * @param boolean $all_private_too=false should all the private fields be returned too, default no
- * @param string $only_type2=null if given only return fields of type2 == $only_type2
+ * @param boolean $all_private_too =false should all the private fields be returned too, default no
+ * @param string $only_type2 =null if given only return fields of type2 == $only_type2
* @return array with customfields
*/
public static function get($app, $all_private_too=false, $only_type2=null)
@@ -146,6 +146,216 @@ class egw_customfields implements IteratorAggregate
return $cfs;
}
+ /**
+ * Check if any customfield uses html (type == 'htmlarea')
+ *
+ * @param string $app
+ * @param boolean $all_private_too =false should all the private fields be returned too, default no
+ * @param string $only_type2 =null if given only return fields of type2 == $only_type2
+ * @return boolen true: if there is a custom field useing html, false if not
+ */
+ public static function use_html($app, $all_private_too=false, $only_type2=null)
+ {
+ foreach(self::get($app, $all_private_too, $only_type2) as $data)
+ {
+ if ($data['type'] == 'htmlarea') return true;
+ }
+ return false;
+ }
+
+ /**
+ * Non printable custom fields eg. UI elements
+ *
+ * @var array
+ */
+ public static $non_printable_fields = array('button');
+
+ /**
+ * Format a single custom field value as string
+ *
+ * @param array $field field defintion incl. type
+ * @param string $value field value
+ * @return string formatted value
+ */
+ public static function format(array $field, $value)
+ {
+ switch($field['type'])
+ {
+ case 'select-account':
+ if ($value)
+ {
+ $values = array();
+ foreach($field['rows'] > 1 ? explode(',', $value) : (array) $value as $value)
+ {
+ $values[] = common::grab_owner_name($value);
+ }
+ $value = implode(', ',$values);
+ }
+ break;
+
+ case 'checkbox':
+ $value = $value ? 'X' : '';
+ break;
+
+ case 'select':
+ case 'radio':
+ if (count($field['values']) == 1 && isset($field['values']['@']))
+ {
+ $field['values'] = self::get_options_from_file($field['values']['@']);
+ }
+ $values = array();
+ foreach($field['rows'] > 1 ? explode(',', $value) : (array) $value as $value)
+ {
+ $values[] = isset($field['values'][$value]) ? $field['values'][$value] : '#'.$value;
+ }
+ $value = implode(', ', $values);
+ break;
+
+ case 'date':
+ case 'date-time':
+ if ($value)
+ {
+ $format = $field['len'] ? $field['len'] : ($field['type'] == 'date' ? 'Y-m-d' : 'Y-m-d H:i:s');
+ $formats = preg_split('/[\\/. :-]/',$format);
+ $values = preg_split('/[\\/. :-]/', is_numeric($value) ? egw_time::to($value, $format) : $value);
+ if (count($formats) != count($values))
+ {
+ //error_log(__METHOD__."(".array2string($field).", value='$value') format='$format', formats=".array2string($formats).", values=".array2string($values));
+ $values = array_slice($values, 0, count($formats));
+ }
+ $date = array_combine($formats, $values);
+ $value = common::dateformatorder($date['Y'], $date['m'], $date['d'],true);
+ if (isset($date['H'])) $value .= ' '.common::formattime($date['H'], $date['i']);
+ }
+ break;
+
+ case 'htmlarea': // ToDo: EMail probably has a nicer html2text method
+ if ($value) $value = strip_tags(preg_replace('/<(br|p)[^>]*>/i', "\r\n", str_replace(array("\r", "\n"), '', $value)));
+ break;
+
+ case 'ajax_select': // ToDo: returns unchanged value for now
+ break;
+
+ default:
+ // handling for several link types
+ if ($value && in_array($field['type'], self::get_link_types()))
+ {
+ if ($field['type'] == 'link-entry' || strpos($value, ':') !== false)
+ {
+ list($app, $value) = explode(':', $value);
+ }
+ else
+ {
+ $app = $field['type'];
+ }
+ if ($value) $value = egw_link::title($app, $value);
+ }
+ break;
+ }
+ return $value;
+ }
+
+ /**
+ * Read the options of a 'select' or 'radio' custom field from a file
+ *
+ * For security reasons that file has to be relative to the eGW root
+ * (to not use that feature to explore arbitrary files on the server)
+ * and it has to be a php file setting one variable called options,
+ * (to not display it to anonymously by the webserver).
+ * The $options var has to be an array with value => label pairs, eg:
+ *
+ * 'Option A',
+ * 'b' => 'Option B',
+ * 'c' => 'Option C',
+ * );
+ *
+ * @param string $file file name inside the eGW server root, either relative to it or absolute
+ * @return array in case of an error we return a single option with the message
+ */
+ public static function get_options_from_file($file)
+ {
+ $options = array();
+
+ if (!($path = realpath($file[0] == '/' ? $file : EGW_SERVER_ROOT.'/'.$file)) || // file does not exist
+ substr($path,0,strlen(EGW_SERVER_ROOT)+1) != EGW_SERVER_ROOT.'/' || // we are NOT inside the eGW root
+ basename($path,'.php').'.php' != basename($path) || // extension is NOT .php
+ basename($path) == 'header.inc.php') // dont allow to include our header again
+ {
+ return array(lang("'%1' is no php file in the eGW server root (%2)!".': '.$path,$file,EGW_SERVER_ROOT));
+ }
+ include($path);
+
+ return $options;
+ }
+
+ /**
+ * Get the customfield types containing links
+ *
+ * @return array with customefield types as values
+ */
+ public static function get_link_types()
+ {
+ static $link_types = null;
+
+ if (is_null($link_types))
+ {
+ $link_types = array_keys(array_intersect(egw_link::app_list('query'),egw_link::app_list('title')));
+ $link_types[] = 'link-entry';
+ }
+ return $link_types;
+ }
+
+ /**
+ * Check if there are links in the custom fields and update them
+ *
+ * This function have to be called manually by an application, if cf's linking
+ * to other entries should be stored as links too (beside as cf's).
+ *
+ * @param string $own_app own appname
+ * @param array $values new values including the custom fields
+ * @param array $old =null old values before the update, if existing
+ * @param string $id_name ='id' name/key of the (link-)id in $values
+ */
+ public static function update_links($own_app,array $values,array $old=null,$id_name='id')
+ {
+ $link_types = self::get_link_types();
+
+ foreach(egw_customfields::get($own_app) as $name => $data)
+ {
+ if (!in_array($data['type'],$link_types)) continue;
+
+ // do we have a different old value --> delete that link
+ if ($old && $old['#'.$name] && $old['#'.$name] != $values['#'.$name])
+ {
+ if ($data['type'] == 'link-entry')
+ {
+ list($app,$id) = explode(':',$old['#'.$name]);
+ }
+ else
+ {
+ $app = $data['type'];
+ $id = $old['#'.$name];
+ }
+ egw_link::unlink(false,$own_app,$values[$id_name],'',$app,$id);
+ }
+ if ($data['type'] == 'link-entry')
+ {
+ list($app,$id) = explode(':',$values['#'.$name]);
+ }
+ else
+ {
+ $app = $data['type'];
+ $id = $values['#'.$name];
+ }
+ if ($id) // create new link, does nothing for already existing links
+ {
+ egw_link::link($own_app,$values[$id_name],$app,$id);
+ }
+ }
+ }
+
/**
* Save a single custom field and invalidate cache
*
diff --git a/timesheet/inc/class.timesheet_merge.inc.php b/timesheet/inc/class.timesheet_merge.inc.php
index c50a3f68dd..3bace11749 100644
--- a/timesheet/inc/class.timesheet_merge.inc.php
+++ b/timesheet/inc/class.timesheet_merge.inc.php
@@ -6,7 +6,7 @@
* @author Ralf Becker
* @author Nathan Gray
* @package timesheet
- * @copyright (c) 2007-9 by Ralf Becker
+ * @copyright (c) 2007-14 by Ralf Becker
* @copyright 2011 Nathan Gray
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @version $Id$
@@ -49,6 +49,10 @@ class timesheet_merge extends bo_merge
function __construct()
{
parent::__construct();
+
+ // switch of handling of html formated content, if html is not used
+ $this->parse_html_styles = egw_customfields::use_html('timesheet');
+
$this->bo = new timesheet_bo();
$this->date_fields += array(
'ts_start',
@@ -130,7 +134,7 @@ class timesheet_merge extends bo_merge
}
// Set any missing custom fields, or the marker will stay
- foreach($this->bo->customfields as $name => $field)
+ foreach(array_keys($this->bo->customfields) as $name)
{
if(!$array['#'.$name]) $array['#'.$name] = '';
}
@@ -185,14 +189,14 @@ class timesheet_merge extends bo_merge
echo ''.lang('Project fields').': | ';
$pm_merge = new projectmanager_merge();
- $n = 0;
+ $i = 0;
foreach($pm_merge->projectmanager_fields as $name => $label)
- {
- if (!($n&1)) echo '';
- echo '{{ts_project/'.$name.'}} | '.$label.' | ';
- if ($n&1) echo " \n";
- $n++;
- }
+ {
+ if (!($i&1)) echo '';
+ echo '{{ts_project/'.$name.'}} | '.$label.' | ';
+ if ($i&1) echo " \n";
+ $i++;
+ }
echo ''.lang('General fields:')." | ";
foreach(array(
|