* @copyright 2014 by Ralf Becker * @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License * @package api * @version $Id$ */ /** * Managing custom-field definitions */ class egw_customfields implements IteratorAggregate { /** * Name of the customfields table */ const TABLE = 'egw_customfields'; /** * Reference to the global db class * * @var egw_db */ static protected $db; /** * app the particular config class is instanciated for * * @var string */ protected $app; /** * should all the private fields be returned too, default no * * @var boolean */ protected $all_private_too=false; /** * Iterator initialised for custom fields * * @var ADORecordSet */ protected $iterator; /** * 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 * @return array with customfields */ function __construct($app, $all_private_too=false, $only_type2=null, $start=0, $num_rows=null) { $this->app = $app; $this->all_private_too = $all_private_too; $query = array( 'cf_app' => $app, ); if (!$all_private_too) { $memberships = $GLOBALS['egw']->accounts->memberships($GLOBALS['egw_info']['user']['account_id'], true); $memberships[] = $GLOBALS['egw_info']['user']['account_id']; $query[] = $this->commasep_match('cf_private', $memberships); } if ($only_type2) { $query[] = $this->commasep_match('cf_type2', $only_type2); } $this->iterator = self::$db->select(self::TABLE, '*', $query, __LINE__, __FILE__, !isset($num_rows) ? false : $start, 'ORDER BY cf_order ASC', 'phpgwapi', $num_rows); } /** * Return iterator required for IteratorAggregate * * @return egw_db_callback_iterator */ function getIterator() { return new egw_db_callback_iterator($this->iterator, function($row) { $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); $row['needed'] = egw_db::from_bool($row['needed']); return $row; }, array(), function($row) { return $row['cf_name']; }); } /** * Return SQL to match given values with comma-separated stored column * * @param string $column column name "cf_type2" or "cf_private" * @param string|array $values */ protected function commasep_match($column, $values) { $to_or = array($column.' IS NULL'); foreach((array) $values as $value) { $to_or[] = self::$db->concat("','", $column, "','").' LIKE '.self::$db->quote('%,'.$value.',%'); } return '('.implode(' OR ', $to_or).')'; } /** * 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 * @return array with customfields */ public static function get($app, $all_private_too=false, $only_type2=null) { $cache_key = $app.':'.(bool)$all_private_too.':'.$only_type2; $cfs = egw_cache::getInstance(__CLASS__, $cache_key); if (!isset($cfs)) { $cfs = iterator_to_array(new egw_customfields($app, $all_private_too, $only_type2)); egw_cache::setInstance(__CLASS__, $cache_key, $cfs); $cached = egw_cache::getInstance(__CLASS__, $app); if (!in_array($cache_key, (array)$cached)) { $cached[] = $cache_key; egw_cache::setInstance(__CLASS__, $app, $cached); } } //error_log(__METHOD__."('$app', $all_private_too, '$only_type2') returning fields: ".implode(', ', array_keys($cfs))); return $cfs; } /** * Save a single custom field and invalidate cache * * @param array $cf */ public static function update(array $cf) { self::$db->update(self::TABLE, array( 'cf_label' => $cf['label'], 'cf_type' => $cf['type'], 'cf_type2' => $cf['type2'] ? implode(',', $cf['type2']) : null, 'cf_help' => $cf['help'], 'cf_values' => $cf['values'] ? json_encode($cf['values']) : null, 'cf_len' => (string)$cf['len'] !== '' ? $cf['len'] : null, 'cf_rows' => (string)$cf['rows'] !== '' ? $cf['rows'] : null, 'cf_order' => $cf['order'], 'cf_needed' => $cf['needed'], 'cf_private' => $cf['private'] ? implode(',', $cf['private']) : null, 'cf_modifier' => $GLOBALS['egw_info']['user']['account_id'], 'cf_modified' => time(), ), array( 'cf_name' => $cf['name'], 'cf_app' => $cf['app'], ), __LINE__, __FILE__); self::invalidate_cache($cf['app']); } /** * Save all custom fields of an app * * @param string $app * @param array $cfs */ public static function save($app, array $cfs) { $query = array('cf_app' => $app); if ($cfs) $query[] = self::$db->expression(self::TABLE, 'NOT ', array('cf_name' => array_keys($cfs))); self::$db->delete(self::TABLE, $query, __LINE__, __FILE__); foreach($cfs as $name => $cf) { if (empty($cf['name'])) $cf['name'] = $name; if (empty($cf['app'])) $cf['app'] = $app; self::update($cf); } self::invalidate_cache($app); } /** * Invalidate instance cache for all custom fields of given app * * @param string $app */ protected static function invalidate_cache($app) { if (($cached = egw_cache::getInstance(__CLASS__, $app))) { foreach($cached as $key) { egw_cache::unsetInstance(__CLASS__, $key); } egw_cache::unsetInstance(__CLASS__, $app); } } /** * Change account_id's of private custom-fields * * @param string $app * @param array $ids2change from-id => to-id pairs * @return integer number of changed ids */ public static function change_account_ids($app, array $ids2change) { $total = 0; if (($cfs = self::get_customfields($app, true))) { foreach($cfs as &$data) { if ($data['private']) { $changed = 0; foreach($data['private'] as &$id) { if (isset($ids2change[$id])) { $id = $ids2change[$id]; ++$changed; } } if ($changed) { self::update($data); $total += $changed; } } } } return $total; } /** * Return names of custom fields containing account-ids * * @param string $app * @return array account[-commasep] => array of name(s) pairs */ public static function get_account_cfs($app) { $types = array(); if (($cfs = self::get_customfields($app, true))) { foreach($cfs as $name => $data) { if ($data['type'] == 'select-account' || $data['type'] == 'home-accounts') { $types['account'.($data['rows'] > 1 ? '-commasep' : '')][] = $name; } } } return $types; } /** * Initialise our db * * We use a reference here (no clone), as we no longer use egw_db::row() or egw_db::next_record()! * */ public static function init_static() { if (is_object($GLOBALS['egw']->db)) { self::$db = $GLOBALS['egw']->db; } else { self::$db = $GLOBALS['egw_setup']->db; } } } egw_customfields::init_static();