2012-02-18 10:45:36 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* ownCloud
|
|
|
|
*
|
|
|
|
* @author Thomas Tanghus
|
|
|
|
* @copyright 2012 Thomas Tanghus <thomas@tanghus.net>
|
2012-02-20 21:32:57 +00:00
|
|
|
* @copyright 2012 Bart Visscher bartv@thisnet.nl
|
2012-02-18 10:45:36 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 3 of the License, or any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU AFFERO GENERAL PUBLIC LICENSE for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class for easy access to categories in VCARD, VEVENT, VTODO and VJOURNAL.
|
2012-02-20 21:32:57 +00:00
|
|
|
* A Category can be e.g. 'Family', 'Work', 'Chore', 'Special Occation' or
|
|
|
|
* anything else that is either parsed from a vobject or that the user chooses
|
|
|
|
* to add.
|
|
|
|
* Category names are not case-sensitive, but will be saved with the case they
|
|
|
|
* are entered in. If a user already has a category 'family' for an app, and
|
|
|
|
* tries to add a category named 'Family' it will be silently ignored.
|
|
|
|
* NOTE: There is a limitation in that the the configvalue field in the
|
|
|
|
* preferences table is a varchar(255).
|
2012-02-18 10:45:36 +00:00
|
|
|
*/
|
|
|
|
class OC_VCategories {
|
2012-02-21 08:53:03 +00:00
|
|
|
const PREF_CATEGORIES_LABEL = 'extra_categories';
|
2012-02-18 10:45:36 +00:00
|
|
|
/**
|
|
|
|
* Categories
|
|
|
|
*/
|
|
|
|
private $categories = array();
|
2012-02-19 16:00:07 +00:00
|
|
|
|
2012-02-20 12:16:51 +00:00
|
|
|
private $app = null;
|
|
|
|
private $user = null;
|
2012-02-20 21:32:57 +00:00
|
|
|
|
2012-02-18 10:45:36 +00:00
|
|
|
/**
|
|
|
|
* @brief Constructor.
|
|
|
|
* @param $app The application identifier e.g. 'contacts' or 'calendar'.
|
2012-02-20 21:32:57 +00:00
|
|
|
* @param $user The user whos data the object will operate on. This
|
|
|
|
* parameter should normally be omitted but to make an app able to
|
|
|
|
* update categories for all users it is made possible to provide it.
|
2012-03-07 15:39:56 +00:00
|
|
|
* @param $defcategories An array of default categories to be used if none is stored.
|
2012-02-18 10:45:36 +00:00
|
|
|
*/
|
2012-04-12 20:23:42 +00:00
|
|
|
public function __construct($app, $user=null, $defcategories=array()) {
|
2012-02-19 16:00:07 +00:00
|
|
|
$this->app = $app;
|
2012-02-20 23:02:27 +00:00
|
|
|
$this->user = is_null($user) ? OC_User::getUser() : $user;
|
2012-02-21 08:53:03 +00:00
|
|
|
$categories = trim(OC_Preferences::getValue($this->user, $app, self::PREF_CATEGORIES_LABEL, ''));
|
2012-09-13 15:30:26 +00:00
|
|
|
if ($categories) {
|
|
|
|
$categories = @unserialize($categories);
|
|
|
|
}
|
|
|
|
$this->categories = is_array($categories) ? $categories : $defcategories;
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-02-20 21:32:57 +00:00
|
|
|
* @brief Get the categories for a specific user.
|
2012-02-18 10:45:36 +00:00
|
|
|
* @returns array containing the categories as strings.
|
|
|
|
*/
|
|
|
|
public function categories() {
|
2012-06-11 01:05:33 +00:00
|
|
|
//OC_Log::write('core','OC_VCategories::categories: '.print_r($this->categories, true), OC_Log::DEBUG);
|
2012-06-26 23:03:18 +00:00
|
|
|
if(!$this->categories) {
|
|
|
|
return array();
|
|
|
|
}
|
2012-04-12 21:02:24 +00:00
|
|
|
usort($this->categories, 'strnatcasecmp'); // usort to also renumber the keys
|
2012-02-18 10:45:36 +00:00
|
|
|
return $this->categories;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Checks whether a category is already saved.
|
|
|
|
* @param $name The name to check for.
|
|
|
|
* @returns bool
|
|
|
|
*/
|
|
|
|
public function hasCategory($name) {
|
2012-02-20 21:32:57 +00:00
|
|
|
return $this->in_arrayi($name, $this->categories);
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Add a new category name.
|
2012-02-20 21:32:57 +00:00
|
|
|
* @param $names A string with a name or an array of strings containing
|
|
|
|
* the name(s) of the categor(y|ies) to add.
|
|
|
|
* @param $sync bool When true, save the categories
|
2012-02-18 10:45:36 +00:00
|
|
|
* @returns bool Returns false on error.
|
|
|
|
*/
|
2012-03-07 15:39:56 +00:00
|
|
|
public function add($names, $sync=false) {
|
2012-02-18 10:45:36 +00:00
|
|
|
if(!is_array($names)) {
|
|
|
|
$names = array($names);
|
|
|
|
}
|
|
|
|
$names = array_map('trim', $names);
|
2012-02-20 21:32:57 +00:00
|
|
|
$newones = array();
|
2012-02-18 10:45:36 +00:00
|
|
|
foreach($names as $name) {
|
|
|
|
if(($this->in_arrayi($name, $this->categories) == false) && $name != '') {
|
|
|
|
$newones[] = $name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(count($newones) > 0) {
|
2012-05-13 13:07:07 +00:00
|
|
|
$this->categories = array_merge($this->categories, $newones);
|
2012-03-07 15:39:56 +00:00
|
|
|
if($sync === true) {
|
2012-02-20 21:32:57 +00:00
|
|
|
$this->save();
|
2012-02-20 12:16:51 +00:00
|
|
|
}
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Extracts categories from a vobject and add the ones not already present.
|
|
|
|
* @param $vobject The instance of OC_VObject to load the categories from.
|
|
|
|
*/
|
2012-03-07 15:39:56 +00:00
|
|
|
public function loadFromVObject($vobject, $sync=false) {
|
2012-02-20 12:16:51 +00:00
|
|
|
$this->add($vobject->getAsArray('CATEGORIES'), $sync);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Reset saved categories and rescan supplied vobjects for categories.
|
|
|
|
* @param $objects An array of vobjects (as text).
|
|
|
|
* To get the object array, do something like:
|
|
|
|
* // For Addressbook:
|
|
|
|
* $categories = new OC_VCategories('contacts');
|
2012-07-30 18:46:14 +00:00
|
|
|
* $stmt = OC_DB::prepare( 'SELECT `carddata` FROM `*PREFIX*contacts_cards`' );
|
2012-02-20 12:16:51 +00:00
|
|
|
* $result = $stmt->execute();
|
|
|
|
* $objects = array();
|
|
|
|
* if(!is_null($result)) {
|
2012-09-07 13:22:01 +00:00
|
|
|
* while( $row = $result->fetchRow()) {
|
2012-02-20 12:16:51 +00:00
|
|
|
* $objects[] = $row['carddata'];
|
|
|
|
* }
|
|
|
|
* }
|
2012-08-29 18:34:44 +00:00
|
|
|
* $categories->rescan($objects);
|
2012-02-20 12:16:51 +00:00
|
|
|
*/
|
2012-07-08 22:14:42 +00:00
|
|
|
public function rescan($objects, $sync=true, $reset=true) {
|
|
|
|
if($reset === true) {
|
|
|
|
$this->categories = array();
|
|
|
|
}
|
2012-02-20 12:16:51 +00:00
|
|
|
foreach($objects as $object) {
|
2012-03-07 15:39:56 +00:00
|
|
|
//OC_Log::write('core','OC_VCategories::rescan: '.substr($object, 0, 100).'(...)', OC_Log::DEBUG);
|
2012-02-20 12:16:51 +00:00
|
|
|
$vobject = OC_VObject::parse($object);
|
2012-02-20 21:32:57 +00:00
|
|
|
if(!is_null($vobject)) {
|
2012-03-07 15:39:56 +00:00
|
|
|
$this->loadFromVObject($vobject, $sync);
|
2012-02-20 12:16:51 +00:00
|
|
|
} else {
|
2012-06-01 10:15:03 +00:00
|
|
|
OC_Log::write('core','OC_VCategories::rescan, unable to parse. ID: '.', '.substr($object, 0, 100).'(...)', OC_Log::DEBUG);
|
2012-02-20 12:16:51 +00:00
|
|
|
}
|
|
|
|
}
|
2012-02-20 21:32:57 +00:00
|
|
|
$this->save();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Save the list with categories
|
|
|
|
*/
|
2012-03-07 15:39:56 +00:00
|
|
|
private function save() {
|
2012-05-13 13:07:07 +00:00
|
|
|
if(is_array($this->categories)) {
|
|
|
|
usort($this->categories, 'strnatcasecmp'); // usort to also renumber the keys
|
|
|
|
$escaped_categories = serialize($this->categories);
|
|
|
|
OC_Preferences::setValue($this->user, $this->app, self::PREF_CATEGORIES_LABEL, $escaped_categories);
|
|
|
|
OC_Log::write('core','OC_VCategories::save: '.print_r($this->categories, true), OC_Log::DEBUG);
|
|
|
|
} else {
|
|
|
|
OC_Log::write('core','OC_VCategories::save: $this->categories is not an array! '.print_r($this->categories, true), OC_Log::ERROR);
|
|
|
|
}
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-03-07 15:39:56 +00:00
|
|
|
* @brief Delete categories from the db and from all the vobject supplied
|
|
|
|
* @param $names An array of categories to delete
|
2012-02-18 10:45:36 +00:00
|
|
|
* @param $objects An array of arrays with [id,vobject] (as text) pairs suitable for updating the apps object table.
|
|
|
|
*/
|
2012-03-12 13:12:27 +00:00
|
|
|
public function delete($names, array &$objects=null) {
|
2012-03-07 15:39:56 +00:00
|
|
|
if(!is_array($names)) {
|
|
|
|
$names = array($names);
|
|
|
|
}
|
|
|
|
OC_Log::write('core','OC_VCategories::delete, before: '.print_r($this->categories, true), OC_Log::DEBUG);
|
|
|
|
foreach($names as $name) {
|
|
|
|
OC_Log::write('core','OC_VCategories::delete: '.$name, OC_Log::DEBUG);
|
|
|
|
if($this->hasCategory($name)) {
|
2012-06-11 01:05:33 +00:00
|
|
|
//OC_Log::write('core','OC_VCategories::delete: '.$name.' got it', OC_Log::DEBUG);
|
2012-03-07 15:39:56 +00:00
|
|
|
unset($this->categories[$this->array_searchi($name, $this->categories)]);
|
|
|
|
}
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
2012-02-20 21:32:57 +00:00
|
|
|
$this->save();
|
2012-03-07 15:39:56 +00:00
|
|
|
OC_Log::write('core','OC_VCategories::delete, after: '.print_r($this->categories, true), OC_Log::DEBUG);
|
2012-03-12 13:12:27 +00:00
|
|
|
if(!is_null($objects)) {
|
|
|
|
foreach($objects as $key=>&$value) {
|
|
|
|
$vobject = OC_VObject::parse($value[1]);
|
2012-09-07 13:22:01 +00:00
|
|
|
if(!is_null($vobject)) {
|
2012-03-12 13:12:27 +00:00
|
|
|
$categories = $vobject->getAsArray('CATEGORIES');
|
|
|
|
//OC_Log::write('core','OC_VCategories::delete, before: '.$key.': '.print_r($categories, true), OC_Log::DEBUG);
|
|
|
|
foreach($names as $name) {
|
|
|
|
$idx = $this->array_searchi($name, $categories);
|
2012-06-11 01:05:33 +00:00
|
|
|
//OC_Log::write('core','OC_VCategories::delete, loop: '.$name.', '.print_r($idx, true), OC_Log::DEBUG);
|
2012-03-12 13:12:27 +00:00
|
|
|
if($idx !== false) {
|
|
|
|
OC_Log::write('core','OC_VCategories::delete, unsetting: '.$categories[$this->array_searchi($name, $categories)], OC_Log::DEBUG);
|
|
|
|
unset($categories[$this->array_searchi($name, $categories)]);
|
|
|
|
//unset($categories[$idx]);
|
|
|
|
}
|
2012-03-07 15:39:56 +00:00
|
|
|
}
|
2012-06-11 01:05:33 +00:00
|
|
|
//OC_Log::write('core','OC_VCategories::delete, after: '.$key.': '.print_r($categories, true), OC_Log::DEBUG);
|
2012-03-12 13:12:27 +00:00
|
|
|
$vobject->setString('CATEGORIES', implode(',', $categories));
|
|
|
|
$value[1] = $vobject->serialize();
|
|
|
|
$objects[$key] = $value;
|
|
|
|
} else {
|
|
|
|
OC_Log::write('core','OC_VCategories::delete, unable to parse. ID: '.$value[0].', '.substr($value[1], 0, 50).'(...)', OC_Log::DEBUG);
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// case-insensitive in_array
|
|
|
|
private function in_arrayi($needle, $haystack) {
|
2012-06-26 23:03:18 +00:00
|
|
|
if(!is_array($haystack)) {
|
|
|
|
return false;
|
|
|
|
}
|
2012-05-13 13:07:07 +00:00
|
|
|
return in_array(strtolower($needle), array_map('strtolower', $haystack));
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// case-insensitive array_search
|
2012-02-20 21:32:57 +00:00
|
|
|
private function array_searchi($needle, $haystack) {
|
2012-06-26 23:03:18 +00:00
|
|
|
if(!is_array($haystack)) {
|
|
|
|
return false;
|
|
|
|
}
|
2012-10-23 20:53:54 +00:00
|
|
|
return array_search(strtolower($needle), array_map('strtolower',$haystack));
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|
2012-02-19 16:00:07 +00:00
|
|
|
|
2012-02-18 10:45:36 +00:00
|
|
|
}
|