diff --git a/application/bootstrap.php b/application/bootstrap.php index 86e6486..3b8a666 100644 --- a/application/bootstrap.php +++ b/application/bootstrap.php @@ -114,7 +114,7 @@ Kohana::modules(array( 'cron' => SMDPATH.'cron', // Kohana Cron Module // 'codebench' => SMDPATH.'codebench', // Benchmarking tool 'database' => SMDPATH.'database', // Database access - 'gchart' => MODPATH.'gchart', // Google Chart Module + // 'gchart' => MODPATH.'gchart', // Google Chart Module // 'image' => SMDPATH.'image', // Image manipulation 'khemail' => SMDPATH.'khemail', // Email module for Kohana 3 PHP Framework 'minion' => SMDPATH.'minion', // CLI Tasks @@ -155,7 +155,7 @@ Route::set('default', '((/(/)))', array('id'=>'[a-zA-Z0- /** * If APC is enabled, and we need to clear the cache */ -if (file_exists(APPPATH.'cache/CLEAR_APC_CACHE') AND function_exists('apc_clear_cache') AND ! Kohana::$is_cli) { +if (file_exists(APPPATH.'cache/CLEAR_APC_CACHE') AND function_exists('apc_clear_cache') AND (PHP_SAPI !== 'cli')) { if (! apc_clear_cache() OR ! unlink(APPPATH.'cache/CLEAR_APC_CACHE')) throw new Kohana_Exception('Unable to clear the APC cache.'); } diff --git a/application/classes/Cookie.php b/application/classes/Cookie.php new file mode 100644 index 0000000..9ada173 --- /dev/null +++ b/application/classes/Cookie.php @@ -0,0 +1,16 @@ + diff --git a/application/classes/ormtsm.php b/application/classes/TSM/ORM.php similarity index 88% rename from application/classes/ormtsm.php rename to application/classes/TSM/ORM.php index f04b383..16724fb 100644 --- a/application/classes/ormtsm.php +++ b/application/classes/TSM/ORM.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://dev.osbill.net/license.html */ -class ORMTSM extends ORM { +abstract class TSM_ORM extends ORM { // Suppress ORMs inclusion of .* protected $_disable_wild_select = TRUE; // Suppress ORMs inclusion of . to column joins @@ -35,11 +35,11 @@ class ORMTSM extends ORM { */ protected function _initialize() { // Set out DB connection configuration. - $this->_db_group = Kohana::config('config.client_type'); + $this->_db_group = Kohana::$config->load('config.client_type'); // Adjustments for DSMADMC or DB2 connections - if (array_key_exists(Kohana::config('config.client_type'),$this->_tsm)) - foreach ($this->_tsm[Kohana::config('config.client_type')] as $k => $v) + if (array_key_exists($this->_db_group,$this->_tsm)) + foreach ($this->_tsm[$this->_db_group] as $k => $v) if (preg_match('/^_/',$k)) $this->{$k} = $v; @@ -51,8 +51,8 @@ class ORMTSM extends ORM { public function __get($column) { // Get a substited column name - need for DB2/DSMADMC schema differences - if (isset($this->_tsm[Kohana::config('config.client_type')]['translate']) AND array_key_exists($column,$this->_tsm[Kohana::config('config.client_type')]['translate'])) { - return is_null($c=$this->_tsm[Kohana::config('config.client_type')]['translate'][$column]) ? NULL : parent::__get($c); + if (isset($this->_tsm[$this->_db_group]['translate']) AND array_key_exists($column,$this->_tsm[$this->_db_group]['translate'])) { + return is_null($c=$this->_tsm[$this->_db_group]['translate'][$column]) ? NULL : parent::__get($c); } else return parent::__get($column); @@ -104,7 +104,7 @@ class ORMTSM extends ORM { private function isCacheable() { $preload = array('NODES','VOLUMES'); - $config = Kohana::config('database')->{Database::$default}; + $config = Kohana::$config->load('database')->{Database::$default}; if ($config['caching'] AND isset($config['cachepreload'][$this->_table_name]) AND count($this->_db_pending) == 1 AND $this->_db_pending[0]['name'] == 'where' AND $this->_db_pending[0]['args'][0] == $this->_primary_key AND $this->_db_pending[0]['args'][1] == '=') return $config['cachepreload'][$this->_table_name]; @@ -126,6 +126,7 @@ class ORMTSM extends ORM { * we hard code the cache to 7 days. * * @return array + * @todo This cache time needs to be better integrated with other caching times. */ public function list_columns() { // We'll cache our query results diff --git a/application/classes/auth/orm.php b/application/classes/auth/orm.php index 2ed3eb2..d6e9fe5 100644 --- a/application/classes/auth/orm.php +++ b/application/classes/auth/orm.php @@ -22,7 +22,7 @@ class Auth_ORM extends Kohana_Auth_ORM { // Override Kohana logged_in() - if we can get the user, we are logged in. public function logged_in($role = NULL, $all_required = TRUE) { - return FALSE !== $this->get_user(); + return NULL !== $this->get_user(); } } ?> diff --git a/application/classes/auth/tsm.php b/application/classes/auth/tsm.php index 156d70b..41d3a4f 100644 --- a/application/classes/auth/tsm.php +++ b/application/classes/auth/tsm.php @@ -26,12 +26,14 @@ class Auth_TSM extends Auth_ORM { // Load the user $user = ORM::factory('admin') - ->where('ADMIN_NAME','=',$username) + ->where('ADMIN_NAME','=',strtoupper($username)) + ->cached(0) ->find(); + + return $user->loaded(); } - // Normally, if we get this far, we are already logged in. - return TRUE; + return FALSE; } } ?> diff --git a/application/classes/controller/default.php b/application/classes/controller/default.php index 478f8ee..e59300d 100644 --- a/application/classes/controller/default.php +++ b/application/classes/controller/default.php @@ -1,4 +1,4 @@ diff --git a/application/classes/controller/domain.php b/application/classes/controller/domain.php index bd2a9e3..e834317 100644 --- a/application/classes/controller/domain.php +++ b/application/classes/controller/domain.php @@ -47,7 +47,7 @@ class Controller_DOMAIN extends Controller_TemplateDefault { 'body'=>_('The domain name is required.'), )); - Request::current()->redirect('domain'); + HTTP::redirect('domain'); } $do = ORM::factory('domain',$domain_name); @@ -58,7 +58,7 @@ class Controller_DOMAIN extends Controller_TemplateDefault { 'body'=>sprintf(_('The domain [%s] does not exist?.'),$domain_name), )); - Request::current()->redirect('domain'); + HTTP::redirect('domain'); } Block::add(array( diff --git a/application/classes/controller/library.php b/application/classes/controller/library.php index 3090fe3..e518b88 100644 --- a/application/classes/controller/library.php +++ b/application/classes/controller/library.php @@ -47,7 +47,7 @@ class Controller_LIBRARY extends Controller_TemplateDefault { 'body'=>_('The library pool name is required.'), )); - Request::current()->redirect('library'); + HTTP::redirect('library'); } $lo = ORM::factory('library',$library); @@ -58,7 +58,7 @@ class Controller_LIBRARY extends Controller_TemplateDefault { 'body'=>sprintf(_('The library pool [%s] does not exist?.'),$library), )); - Request::current()->redirect('library'); + HTTP::redirect('library'); } Block::add(array( diff --git a/application/classes/controller/lnapp/login.php b/application/classes/controller/lnapp/login.php deleted file mode 100644 index 4c8405d..0000000 --- a/application/classes/controller/lnapp/login.php +++ /dev/null @@ -1,199 +0,0 @@ -logged_in()!= 0) { - // Redirect to the user account - Request::current()->redirect('welcome/index'); - } - - // If there is a post and $_POST is not empty - if ($_POST) { - // Store our details in a session key - Session::instance()->set(Kohana::config('auth.session_key'),$_POST['username']); - Session::instance()->set('password',$_POST['password']); - - // If the post data validates using the rules setup in the user model - if (Auth::instance()->login($_POST['username'],$_POST['password'])) { - // Redirect to the user account - if ($redir = Session::instance()->get('afterlogin')) { - Session::instance()->delete('afterlogin'); - Request::current()->redirect($redir); - - } else - Request::current()->redirect('welcome/index'); - - } else { - SystemMessage::add(array( - 'title'=>_('Invalid username or password'), - 'type'=>'error', - 'body'=>_('The username or password was invalid.') - )); - } - } - - Block::add(array( - 'title'=>_('Login to server'), - 'body'=>View::factory('login'), - 'style'=>array('css/login.css'=>'screen'), - )); - - Script::add(array('type'=>'stdin','data'=>' - $(document).ready(function() { - $("#ajxbody").click(function() {$("#ajBODY").load("'.$this->request->uri().'/"); return false;}); - });' - )); - } - - public function action_register() { - // If user already signed-in - if (Auth::instance()->logged_in()!= 0) { - // Redirect to the user account - Request::current()->redirect('welcome/index'); - } - - // Instantiate a new user - $account = ORM::factory('account'); - - // If there is a post and $_POST is not empty - if ($_POST) { - // Check Auth - $status = $account->values($_POST)->check(); - - if (! $status) { - foreach ($account->validation()->errors('form/register') as $f => $r) { - // $r[0] has our reason for validation failure - switch ($r[0]) { - // Generic validation reason - default: - SystemMessage::add(array( - 'title'=>_('Validation failed'), - 'type'=>'error', - 'body'=>sprintf(_('The defaults on your submission were not valid for field %s (%s).'),$f,$r) - )); - } - } - } - - $ido = ORM::factory('module') - ->where('name','=','account') - ->find(); - - $account->id = $ido->record_id->next_id($ido->id); - // Save the user details - if ($account->save()) {} - - } - - SystemMessage::add(array( - 'title'=>_('Already have an account?'), - 'type'=>'info', - 'body'=>_('If you already have an account, please login..') - )); - - Block::add(array( - 'title'=>_('Register'), - 'body'=>View::factory('bregister') - ->set('account',$account) - ->set('errors',$account->validation()->errors('form/register')), - )); - - $this->template->left = HTML::anchor('login','Login').'...'; - } - - /** - * Enable user password reset - */ - public function action_reset() { - // If user already signed-in - if (Auth::instance()->logged_in()!= 0) { - // Redirect to the user account - Request::current()->redirect('welcome/index'); - } - - // If the user posted their details to reset their password - if ($_POST) { - // If the email address is correct, create a method token - if (! empty($_POST['email']) AND ($ao=ORM::factory('account',array('email'=>$_POST['email']))) AND $ao->loaded()) { - $mt = ORM::factory('module_method_token'); - - // Find out our password reset method id - // @todo move this to a more generic method, so that it can be called by other methods - $mo = ORM::factory('module',array('name'=>'account')); - $mmo = ORM::factory('module_method',array('name'=>'user_resetpassword','module_id'=>$mo->id)); - - // Check to see if there is already a token, if so, do nothing. - if ($mt->where('account_id','=',$ao->id)->and_where('method_id','=',$mmo->id)->find()) { - if ($mt->date_expire < time()) { - $mt->delete(); - $mt->clear(); - } - } - - if (! $mt->loaded()) { - $mt->account_id = $ao->id; - $mt->method_id = $mmo->id; - $mt->date_expire = time() + 15*3600; - $mt->token = md5(sprintf('%s:%s:%s',$mt->account_id,$mt->method_id,$mt->date_expire)); - $mt->save(); - - // Send our email with the token - $et = EmailTemplate::instance('account_reset_password'); - $et->to = array($mt->account->email=>sprintf('%s %s',$mt->account->first_name,$mt->account->last_name)); - $et->variables = array( - 'SITE'=>URL::base(TRUE,TRUE), - 'SITE_ADMIN'=>Config::sitename(), - 'SITE_NAME'=>Config::sitename(), - 'TOKEN'=>$mt->token, - 'USER_NAME'=>sprintf('%s %s',$mt->account->first_name,$mt->account->last_name), - ); - $et->send(); - } - - // Redirect to our password reset, the Auth will validate the token. - } elseif (! empty($_REQUEST['token'])) { - Request::current()->redirect(sprintf('user/account/resetpassword?token=%s',$_REQUEST['token'])); - } - - // Show our token screen even if the email was invalid. - if (isset($_POST['email'])) - Block::add(array( - 'title'=>_('Reset your password'), - 'body'=>View::factory('login_reset_sent'), - 'style'=>array('css/login.css'=>'screen'), - )); - else - Request::current()->redirect('login'); - - } else { - Block::add(array( - 'title'=>_('Reset your password'), - 'body'=>View::factory('login_reset'), - 'style'=>array('css/login.css'=>'screen'), - )); - } - } - - public function action_noaccess() { - SystemMessage::add(array( - 'title'=>_('No access to requested resource'), - 'type'=>'error', - 'body'=>_('You do not have access to the requested resource, please contact your administrator.') - )); - } -} -?> diff --git a/application/classes/controller/login.php b/application/classes/controller/login.php index 59a728d..74a1883 100644 --- a/application/classes/controller/login.php +++ b/application/classes/controller/login.php @@ -1,4 +1,4 @@ diff --git a/application/classes/controller/logout.php b/application/classes/controller/logout.php index 671e24d..090c4a9 100644 --- a/application/classes/controller/logout.php +++ b/application/classes/controller/logout.php @@ -1,4 +1,4 @@ diff --git a/application/classes/controller/media.php b/application/classes/controller/media.php new file mode 100644 index 0000000..b478c75 --- /dev/null +++ b/application/classes/controller/media.php @@ -0,0 +1,4 @@ + diff --git a/application/classes/controller/node.php b/application/classes/controller/node.php index 47f4c82..1b83ce1 100644 --- a/application/classes/controller/node.php +++ b/application/classes/controller/node.php @@ -47,7 +47,7 @@ class Controller_NODE extends Controller_TemplateDefault { 'body'=>_('The node name is required.'), )); - Request::current()->redirect('node'); + HTTP::redirect('node'); } $no = ORM::factory('node',$node_name); @@ -58,7 +58,7 @@ class Controller_NODE extends Controller_TemplateDefault { 'body'=>sprintf(_('The node [%s] does not exist?.'),$node_name), )); - Request::current()->redirect('node'); + HTTP::redirect('node'); } Block::add(array( diff --git a/application/classes/controller/stgpool.php b/application/classes/controller/stgpool.php index 769a8bb..9854e81 100644 --- a/application/classes/controller/stgpool.php +++ b/application/classes/controller/stgpool.php @@ -47,7 +47,7 @@ class Controller_STGPOOL extends Controller_TemplateDefault { 'body'=>_('The stgpool pool name is required.'), )); - Request::current()->redirect('stgpool'); + HTTP::redirect('stgpool'); } $so = ORM::factory('stgpool',$stgpool); @@ -58,7 +58,7 @@ class Controller_STGPOOL extends Controller_TemplateDefault { 'body'=>sprintf(_('The stgpool pool [%s] does not exist?.'),$stgpool), )); - Request::current()->redirect('stgpool'); + HTTP::redirect('stgpool'); } Block::add(array( diff --git a/application/classes/controller/templatedefault.php b/application/classes/controller/templatedefault.php index d6e971e..b269e3f 100644 --- a/application/classes/controller/templatedefault.php +++ b/application/classes/controller/templatedefault.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Controller_TemplateDefault extends Controller_lnApp_TemplateDefault { +class Controller_TemplateDefault extends lnApp_Controller_TemplateDefault { protected $auth_required = TRUE; } ?> diff --git a/application/classes/controller/tree.php b/application/classes/controller/tree.php index 82c80ff..fee4f70 100644 --- a/application/classes/controller/tree.php +++ b/application/classes/controller/tree.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Open Source Billing * @license http://dev.osbill.net/license.html */ -class Controller_Tree extends Controller_lnApp_Tree { +class Controller_Tree extends lnApp_Controller_Tree { protected $auth_required = TRUE; /** @@ -21,7 +21,7 @@ class Controller_Tree extends Controller_lnApp_Tree { * * @param id */ - public function action_json($id=null,array $data=array()) { + public function action_json(array $data=array()) { // @todo Our menu options array_push($data,array( 'id'=>'domain', @@ -63,7 +63,7 @@ class Controller_Tree extends Controller_lnApp_Tree { 'attr_href'=>URL::Site('stgpool'), )); - return parent::action_json($id,$data); + return parent::action_json($data); } } ?> diff --git a/application/classes/controller/welcome.php b/application/classes/controller/welcome.php index 4e55cca..3819af7 100644 --- a/application/classes/controller/welcome.php +++ b/application/classes/controller/welcome.php @@ -14,11 +14,11 @@ class Controller_Welcome extends Controller_TemplateDefault { protected $auth_required = FALSE; public function action_index() { - if (! Kohana::config('config.client')) - Request::current()->redirect('guide/pta'); + if (! Kohana::$config->load('config')->client) + HTTP::redirect('guide/app'); if (! Auth::instance()->logged_in()) - Request::current()->redirect('login'); + HTTP::redirect('login'); Block::add(array( 'title'=>sprintf('%s: %s (%s)',_('Server'),TSM::name(),TSM::version()), diff --git a/application/classes/database/tsm.php b/application/classes/database/tsm.php index 2292121..38588cb 100644 --- a/application/classes/database/tsm.php +++ b/application/classes/database/tsm.php @@ -69,7 +69,9 @@ abstract class Database_TSM extends Database { public function query($type, $sql, $as_object = FALSE, array $params = NULL) { // Make sure the database is connected - $this->_connection or $this->connect(); + if (! $this->_connection && ! $this->connect()) + return new Database_TSM_Result(array(),'',$as_object,$params); + $this->clear(); if ( ! empty($this->_config['profiling'])) @@ -84,8 +86,7 @@ abstract class Database_TSM extends Database { if (isset($benchmark)) Profiler::delete($benchmark); - SystemMessage::TSM_Error(sprintf('%s (%s)',$this->execute_stdout,$this->execute_stderr),$sql); - Request::current()->redirect('welcome'); + SystemMessage::TSM('error',sprintf('%s (%s)',$this->execute_stdout,$this->execute_stderr),$sql); } if (isset($benchmark)) @@ -106,9 +107,6 @@ abstract class Database_TSM extends Database { } public function escape($value) { - // Make sure the database is connected - $this->_connection or $this->connect(); - // SQL standard is to use single-quotes for all values return "'$value'"; } diff --git a/application/classes/database/tsm/db2.php b/application/classes/database/tsm/db2.php index dbc0439..fe21ee8 100644 --- a/application/classes/database/tsm/db2.php +++ b/application/classes/database/tsm/db2.php @@ -35,7 +35,7 @@ class Database_TSM_DB2 extends Database_TSM { unset($this->_config['connection']['username'], $this->_config['connection']['password']); if (! $username OR ! $password) - Request::current()->redirect('/login?need_login=1'); + HTTP::redirect('login'); try { if ($persistent) { @@ -56,7 +56,7 @@ class Database_TSM_DB2 extends Database_TSM { return TSM::instance()->set($username,$password,$result); } else - Request::current()->redirect(Request::detect_uri()); + HTTP::redirect(Request::detect_uri()); } } catch (ErrorException $e) { diff --git a/application/classes/database/tsm/dsmadmc.php b/application/classes/database/tsm/dsmadmc.php index 31a15d5..2c02f6d 100644 --- a/application/classes/database/tsm/dsmadmc.php +++ b/application/classes/database/tsm/dsmadmc.php @@ -12,9 +12,6 @@ */ class Database_TSM_DSMADMC extends Database_TSM { public function connect() { - if ($this->_connection) - return; - // Extract the connection parameters, adding required variabels extract($this->_config['connection'] + array( 'database' => '', @@ -26,23 +23,21 @@ class Database_TSM_DSMADMC extends Database_TSM { // Get user login details from user session - these are set by login if (! $username) - $username = Session::instance()->get_once(Kohana::config('auth.session_key')); + $username = Session::instance()->get_once(Kohana::$config->load('auth')->session_key); if (! $password) $password = Session::instance()->get_once('password'); // Prevent this information from showing up in traces unset($this->_config['connection']['username'], $this->_config['connection']['password']); - if (! file_exists(Kohana::config('config.client'))) { - system_message(array('title'=>'Cant find DSMADMC', - 'body'=>sprintf('Unable to find the dsmadmc at %s',Kohana::config('client.config')), - 'type'=>'error')); + if (! file_exists(Kohana::$config->load('config')->client)) { + SystemMessage::TSM('error',sprintf('Unable to find the dsmadmc at %s',Kohana::$config->load('config')->client),''); return FALSE; } if (! $username OR ! $password) - Request::current()->redirect('/login?need_login=1'); + HTTP::redirect('login'); try { if ($persistent) { @@ -52,20 +47,31 @@ class Database_TSM_DSMADMC extends Database_TSM { } else { // Create a connection and force it to be a new link $this->_connection = sprintf('%s %s -id=%s -password=%s -displ=list -editor=no -dataonly=YES %s %s ', - Kohana::config('config.client'), + Kohana::$config->load('config')->client, $database ? '-server='.$database : '', $username, $password, - Kohana::config('config.client_errorlogname') ? sprintf('-errorlogname=%s',Kohana::config('config.client_errorlogname')) : '', + Kohana::$config->load('config')->client_errorlogname ? sprintf('-errorlogname=%s',Kohana::$config->load('config')->client_errorlogname) : '', $database ? sprintf('-se=%s',$database) : '' ); $result = $this->query(Database::SELECT,'SELECT server_name,platform,version,release,level,sublevel FROM status'); - if ($result) + if ($result->count()) { return TSM::instance()->set($username,$password,$result); - else - Request::current()->redirect(Request::detect_uri()); + + } else { + $this->_connection = FALSE; + + // @todo Get ANS8034E (ID not recognised), or ??????E (invalid password) + SystemMessage::add(array( + 'title'=>_('Login Failed'), + 'type'=>'error', + 'body'=>_('Invalid Username and/or Password.') + )); + + return FALSE; + } } } catch (ErrorException $e) { @@ -77,14 +83,6 @@ class Database_TSM_DSMADMC extends Database_TSM { ), $e->getCode()); } - - // \xFF is a better delimiter, but the PHP driver uses underscore - $this->_connection_id = sha1($hostname.'_'.$username.'_'.$password); - - if ( ! empty($this->_config['charset'])) { - // Set the character set - $this->set_charset($this->_config['charset']); - } } protected function execute($sql) { @@ -95,6 +93,8 @@ class Database_TSM_DSMADMC extends Database_TSM { $sql = str_replace('\\','\\\\',$sql); $this->execute_stderr = exec($this->_connection.'"'.$sql.'"',$this->execute_stdout,$this->execute_rc); + if (Kohana::$config->load('config.site.mode') == Kohana::DEVELOPMENT) + SystemMessage::TSM('info',sprintf('%s (%s) [%s]',$this->_connection.'"'.$sql.'"',join("\n",$this->execute_stdout),$this->execute_rc),$sql); // Work out our message codes $result = array(); @@ -180,7 +180,7 @@ class Database_TSM_DSMADMC extends Database_TSM { $columns[$row['COLNAME']] = $column; } - return $columns; + return $columns ? $columns : array(''); } } ?> diff --git a/application/classes/http/exception/404.php b/application/classes/http/exception/404.php index a72b75d..138a58d 100644 --- a/application/classes/http/exception/404.php +++ b/application/classes/http/exception/404.php @@ -11,20 +11,21 @@ * @license http://dev.leenooks.net/license.html */ class HTTP_Exception_404 extends Kohana_HTTP_Exception_404 { - public function __construct($message, array $variables = NULL, $code = 0) + public function __construct($message = NULL, array $variables = NULL, Exception $previous = NULL) { set_exception_handler(array(get_class($this),'handler')); - parent::__construct($message, $variables, (int) $code); + parent::__construct($message, $variables, $previous); } public static function handler(Exception $e) { SystemMessage::add(array( - 'title'=>_('Page not found'), + 'title'=>_('Page not found'), 'type'=>'error', 'body'=>sprintf(_('The page [%s] you requested was not found?'),Request::detect_uri()), )); - Request::factory()->redirect('welcome'); + HTTP::redirect('welcome'); } } +?> diff --git a/application/classes/lnapp/Controller/Media.php b/application/classes/lnapp/Controller/Media.php new file mode 100644 index 0000000..9503b00 --- /dev/null +++ b/application/classes/lnapp/Controller/Media.php @@ -0,0 +1,64 @@ +request->param('file'); + + // Find the file extension + $ext = pathinfo($file,PATHINFO_EXTENSION); + + // Remove the extension from the filename + $file = substr($file,0,-(strlen($ext)+1)); + $f = ''; + + // If our file is pathed with session, our file is in our session. + if ($fd = Session::instance()->get_once($this->request->param('file'))) { + $this->response->body($fd); + + // First try and find media files for the theme-site_id + } elseif ($f = Kohana::find_file($x=sprintf('media/site/%s/theme/%s',Config::siteid(),Config::theme()),$file,$ext)) { + // Send the file content as the response + $this->response->body(file_get_contents($f)); + + // Try and find media files for the site_id + } elseif ($f = Kohana::find_file(sprintf('media/site/%s',Config::siteid()),$file,$ext)) { + // Send the file content as the response + $this->response->body(file_get_contents($f)); + + // If not found try a default media file + } elseif ($f = Kohana::find_file('media',$file,$ext)) { + // Send the file content as the response + $this->response->body(file_get_contents($f)); + + } else { + // Return a 404 status + $this->response->status(404); + } + + // Generate and check the ETag for this file + if (Kohana::$environment === Kohana::PRODUCTION OR Kohana::$config->load('debug')->etag) + $this->check_cache(sha1($this->response->body())); + + // Set the proper headers to allow caching + $this->response->headers('Content-Type',File::mime_by_ext($ext)); + $this->response->headers('Content-Length',(string)$this->response->content_length()); + $this->response->headers('Last-Modified',date('r',$f ? filemtime($f) : time())); + } +} +?> diff --git a/application/classes/controller/lnapp/default.php b/application/classes/lnapp/Controller/default.php similarity index 90% rename from application/classes/controller/lnapp/default.php rename to application/classes/lnapp/Controller/default.php index b004078..350e87b 100644 --- a/application/classes/controller/lnapp/default.php +++ b/application/classes/lnapp/Controller/default.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -abstract class Controller_lnApp_Default extends Controller { +abstract class lnApp_Controller_Default extends Controller { /** * The variable that our output is stored in */ @@ -52,7 +52,7 @@ abstract class Controller_lnApp_Default extends Controller { */ protected function _auth_required() { // If our global configurable is disabled, then continue - if (! Kohana::Config('config.method_security')) + if (! Kohana::$config->load('config')->method_security) return FALSE; return (($this->auth_required !== FALSE && Auth::instance()->logged_in() === FALSE) || @@ -71,11 +71,11 @@ abstract class Controller_lnApp_Default extends Controller { // For no AJAX/JSON requests, display an access page } elseif (Auth::instance()->logged_in(NULL,get_class($this).'|'.__METHOD__)) { - Request::current()->redirect('login/noaccess'); + HTTP::redirect('login/noaccess'); } else { Session::instance()->set('afterlogin',Request::detect_uri()); - Request::current()->redirect($this->noauth_redirect); + HTTP::redirect($this->noauth_redirect); } } } @@ -84,7 +84,7 @@ abstract class Controller_lnApp_Default extends Controller { parent::after(); // Generate and check the ETag for this file - $this->response->check_cache(NULL,$this->request); + $this->check_cache(sha1($this->response->body())); } } ?> diff --git a/application/classes/lnapp/Controller/login.php b/application/classes/lnapp/Controller/login.php new file mode 100644 index 0000000..dcb214d --- /dev/null +++ b/application/classes/lnapp/Controller/login.php @@ -0,0 +1,70 @@ +logged_in()!= 0) { + // Redirect to the user account + HTTP::redirect('welcome/index'); + } + + // If there is a post and $_POST is not empty + if ($_POST) { + // Store our details in a session key + Session::instance()->set(Kohana::$config->load('auth')->session_key,$_POST['username']); + Session::instance()->set('password',$_POST['password']); + + // If the post data validates using the rules setup in the user model + if (Auth::instance()->login($_POST['username'],$_POST['password'])) { + // Redirect to the user account + if ($redir = Session::instance()->get('afterlogin')) { + Session::instance()->delete('afterlogin'); + HTTP::redirect($redir); + + } else + HTTP::redirect('welcome/index'); + + } else { + SystemMessage::add(array( + 'title'=>_('Invalid username or password'), + 'type'=>'error', + 'body'=>_('The username or password was invalid.') + )); + } + } + + Block::add(array( + 'title'=>_('Login to server'), + 'body'=>View::factory('login'), + 'style'=>array('css/login.css'=>'screen'), + )); + + Script::add(array('type'=>'stdin','data'=>' + $(document).ready(function() { + $("#ajxbody").click(function() {$("#ajBODY").load("'.$this->request->uri().'/"); return false;}); + });' + )); + } + + public function action_noaccess() { + SystemMessage::add(array( + 'title'=>_('No access to requested resource'), + 'type'=>'error', + 'body'=>_('You do not have access to the requested resource, please contact your administrator.') + )); + } +} +?> diff --git a/application/classes/controller/lnapp/logout.php b/application/classes/lnapp/Controller/logout.php similarity index 77% rename from application/classes/controller/lnapp/logout.php rename to application/classes/lnapp/Controller/logout.php index 1613d17..e5f5080 100644 --- a/application/classes/controller/lnapp/logout.php +++ b/application/classes/lnapp/Controller/logout.php @@ -11,16 +11,16 @@ * @license http://dev.leenooks.net/license.html * @also [login] */ -class Controller_lnApp_Logout extends Controller { +class lnApp_Controller_Logout extends Controller { public function action_index() { # If user already signed-in if (Auth::instance()->logged_in()!= 0) { Auth::instance()->logout(); - Request::current()->redirect('login'); + HTTP::redirect('login'); } - Request::current()->redirect('welcome/index'); + HTTP::redirect('welcome/index'); } } ?> diff --git a/application/classes/controller/lnapp/templatedefault.php b/application/classes/lnapp/Controller/templatedefault.php similarity index 72% rename from application/classes/controller/lnapp/templatedefault.php rename to application/classes/lnapp/Controller/templatedefault.php index 83620d1..fdf71ac 100644 --- a/application/classes/controller/lnapp/templatedefault.php +++ b/application/classes/lnapp/Controller/templatedefault.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -abstract class Controller_lnApp_TemplateDefault extends Controller_Template { +abstract class lnApp_Controller_TemplateDefault extends Controller_Template { /** * @var string page template */ @@ -42,6 +42,14 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { 'menu' => TRUE, ); + public function __construct(Request $request,Response $response) { + // Our Menu's can run without method authentication by default. + if (! isset($this->secure_actions['menu'])) + $this->secure_actions['menu'] = FALSE; + + return parent::__construct($request,$response); + } + /** * Check and see if this controller needs authentication * @@ -54,7 +62,7 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { */ protected function _auth_required() { // If our global configurable is disabled, then continue - if (! Kohana::Config('config.method_security')) + if (! Kohana::$config->load('config')->method_security) return FALSE; return (($this->auth_required !== FALSE && Auth::instance()->logged_in(NULL,get_class($this).'|'.__METHOD__) === FALSE) || @@ -75,13 +83,14 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { return; } + // Actions that start with ajax, should only be ajax + if (! Kohana::$config->load('debug')->ajax AND preg_match('/^ajax/',Request::current()->action()) AND ! Request::current()->is_ajax()) + die(); + parent::before(); // Check user auth and role if ($this->_auth_required()) { - if (Kohana::$is_cli) - throw new Kohana_Exception('Cant run :method, authentication not possible',array(':method'=>$this->request->action())); - // If auth is required and the user is logged in, then they dont have access. // (We have already checked authorisation.) if (Auth::instance()->logged_in(NULL,get_class($this).'|'.__METHOD__)) { @@ -89,7 +98,7 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { SystemMessage::add(array( 'title'=>_('Insufficient Access'), 'type'=>'debug', - 'body'=>Kohana::debug(array('required'=>$this->auth_required,'action'=>$this->request->action(),'user'=>Auth::instance()->get_user()->username)), + 'body'=>Debug::vars(array('required'=>$this->auth_required,'action'=>$this->request->action(),'user'=>Auth::instance()->get_user()->username)), )); // @todo Login No Access redirects are not handled in JS? @@ -97,11 +106,11 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { echo _('You dont have enough permissions.'); die(); } else - Request::current()->redirect('login/noaccess'); + HTTP::redirect('login/noaccess'); } else { Session::instance()->set('afterlogin',Request::detect_uri()); - Request::current()->redirect($this->noauth_redirect); + HTTP::redirect($this->noauth_redirect); } } @@ -121,13 +130,12 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { 'data'=>'css/default.css', )); - // Our default scripts - // This is in a reverse list, since we push them to the beginging of the scripts to render. - foreach (array('file'=>array( + // Our default script(s) + foreach (array('file'=>array_reverse(array( + 'js/jquery-1.6.4.min.js', + 'js/jquery.jstree-1.0rc3.js', 'js/jquery.cookie.js', - 'js/jquery.jstree-1.0rc.js', - 'js/jquery-1.4.2.js', - )) as $type => $datas) { + ))) as $type => $datas) { foreach ($datas as $data) { Script::add(array( @@ -185,17 +193,25 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { if ($s = $this->_sysmsg() AND (string)$s) $this->response->body(sprintf('
%s
',$s)); - # In case there any style sheets or scrpits for this render. + // In case there any style sheets for this render. $this->response->bodyadd(Style::factory()); - # Get the response body + // Since we are ajax, we should re-render the breadcrumb + Session::instance()->set('breadcrumb',(string)Breadcrumb::factory()); + $this->response->bodyadd(Script::add(array('type'=>'stdin','data'=>'$().ready($("#ajCONTROL").load("'.URL::site('welcome/breadcrumb').'",null,function(x,s,r) {}));'))); + + // In case there any javascript for this render. + $this->response->bodyadd(Script::factory()); + + // Get the response body $this->response->bodyadd(sprintf('
%s
',$this->template->content)); } parent::after(); // Generate and check the ETag for this file - $this->response->check_cache(NULL,$this->request); + if (Kohana::$environment === Kohana::PRODUCTION OR Kohana::$config->load('debug')->etag) + $this->check_cache(sha1($this->response->body())); } /** @@ -237,48 +253,5 @@ abstract class Controller_lnApp_TemplateDefault extends Controller_Template { public function _footer() { return sprintf('© %s',Config::SiteName()); } - - /** - * This action will render all the media related files for a page - * @return void - */ - final public function action_media() { - // Get the file path from the request - $file = $this->request->param('file'); - - // Find the file extension - $ext = pathinfo($file, PATHINFO_EXTENSION); - - // Remove the extension from the filename - $file = substr($file, 0, -(strlen($ext) + 1)); - $f = ''; - - // If our file is pathed with session, our file is in our session. - if ($fd = Session::instance()->get_once($this->request->param('file'))) { - $this->response->body($fd); - - // First try and find media files for the site_id - } elseif ($f = Kohana::find_file(sprintf('media/%s',Config::siteid()), $file, $ext)) { - // Send the file content as the response - $this->response->body(file_get_contents($f)); - - // If not found try a default media file - } elseif ($f = Kohana::find_file('media', $file, $ext)) { - // Send the file content as the response - $this->response->body(file_get_contents($f)); - - } else { - // Return a 404 status - $this->response->status(404); - } - - // Generate and check the ETag for this file - $this->response->check_cache(NULL,$this->request); - - // Set the proper headers to allow caching - $this->response->headers('Content-Type',File::mime_by_ext($ext)); - $this->response->headers('Content-Length',(string)$this->response->content_length()); - $this->response->headers('Last-Modified',date('r', $f ? filemtime($f) : time())); - } } ?> diff --git a/application/classes/controller/lnapp/tree.php b/application/classes/lnapp/Controller/tree.php similarity index 74% rename from application/classes/controller/lnapp/tree.php rename to application/classes/lnapp/Controller/tree.php index bc04e00..8fe7f7d 100644 --- a/application/classes/controller/lnapp/tree.php +++ b/application/classes/lnapp/Controller/tree.php @@ -10,17 +10,17 @@ * @copyright (c) 2010 Open Source Billing * @license http://dev.osbill.net/license.html */ -class Controller_lnApp_Tree extends Controller_Default { +class lnApp_Controller_Tree extends Controller_Default { /** * @var string page media route as per [Route] */ protected static $jsmediaroute = 'default/media'; public function after() { - parent::after(); - $this->response->headers('Content-Type','application/json'); - $this->response->body(sprintf('[%s]',json_encode($this->output))); + $this->response->body(json_encode($this->output)); + + parent::after(); } public static function js() { @@ -29,11 +29,13 @@ class Controller_lnApp_Tree extends Controller_Default { return '
'; } @@ -81,12 +91,12 @@ $(function () { * The incoming ID is either a Branch B_x or a Node N_x * Where X is actually the module. * - * @param id + * @param array $data Tree data passed in by inherited methods */ - public function action_json($id=null,array $data=array()) { + public function action_json(array $data=array()) { if ($this->_auth_required() AND ! Auth::instance()->logged_in()) { $this->output = array('attr'=>array('id'=>'a_login'), - 'data'=>array('title'=>_('Please Login').'...','attr'=>array('id'=>'N_login','href'=>URL::site('/login')))); + 'data'=>array('title'=>_('Please Login').'...','attr'=>array('id'=>'N_login','href'=>URL::site('login')))); return; } @@ -98,7 +108,7 @@ $(function () { 'attr'=>array('id'=>sprintf('B_%s',$branch['id'])), 'state'=>$branch['state'], 'data'=>array('title'=>$branch['name']), - 'attr'=>array('id'=>sprintf('N_%s',$branch['id']),'href'=>empty($branch['attr_href']) ? URL::site(sprintf('/%s/menu',$branch['name'])) : $branch['attr_href']), + 'attr'=>array('id'=>sprintf('N_%s',$branch['id']),'href'=>empty($branch['attr_href']) ? URL::site(sprintf('%s/menu',$branch['name'])) : $branch['attr_href']), ) ); } diff --git a/application/classes/lnapp/block.php b/application/classes/lnapp/block.php index 1d73847..ea72d15 100644 --- a/application/classes/lnapp/block.php +++ b/application/classes/lnapp/block.php @@ -13,7 +13,7 @@ * @license http://dev.leenooks.net/license.html * @uses Style */ -class lnApp_Block extends HTMLRender { +abstract class lnApp_Block extends HTMLRender { protected static $_data = array(); protected static $_spacer = '
 
'; protected static $_required_keys = array('body'); @@ -24,6 +24,10 @@ class lnApp_Block extends HTMLRender { * @param array Block attributes */ public static function add($block,$prepend=FALSE) { + // Any body objects should be converted to a string, so that any calls to Style/Script are processed + if (isset($block['body'])) + $block['body'] = (string)$block['body']; + parent::add($block); // Detect any style sheets. diff --git a/application/classes/lnapp/breadcrumb.php b/application/classes/lnapp/breadcrumb.php index 04125f5..8bcf402 100644 --- a/application/classes/lnapp/breadcrumb.php +++ b/application/classes/lnapp/breadcrumb.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_Breadcrumb extends HTMLRender { +abstract class lnApp_Breadcrumb extends HTMLRender { protected static $_data = array(); protected static $_spacer = ' » '; protected static $_required_keys = array('body'); @@ -23,17 +23,32 @@ class lnApp_Breadcrumb extends HTMLRender { public static function set($path) { if (is_string($path)) static::$_data['path'] = explode('/',$path); - else + elseif (is_array($path)) static::$_data['path'] = $path; + else + throw new Kohana_Exception('Path is not a string, nor an array'); } /** * Enable a friendly name to be used for a path */ - public static function name($path,$name) { + public static function name($path,$name,$override=TRUE) { + if (isset(static::$_data['name'][$path]) AND ! $override) + return; + static::$_data['name'][$path] = $name; } + /** + * Enable specifying the URL for a path + */ + public static function URL($path,$url,$override=TRUE) { + if (isset(static::$_data['url'][$path]) AND ! $override) + return; + + static::$_data['url'][$path] = $url; + } + /** * Return an instance of this class * @@ -47,17 +62,26 @@ class lnApp_Breadcrumb extends HTMLRender { * Render this Breadcrumb */ protected function render() { - $output = HTML::anchor('/',_('Home')); + $output = ''; return $output; } } diff --git a/application/classes/lnapp/config.php b/application/classes/lnapp/config.php index 33369cf..75acf1a 100644 --- a/application/classes/lnapp/config.php +++ b/application/classes/lnapp/config.php @@ -12,76 +12,107 @@ * @license http://dev.leenooks.net/license.html */ abstract class lnApp_Config extends Kohana_Config { + protected static $logo = 'img/logo-small.png'; + + /** + * Some early initialisation + * + * At this point, KH hasnt been fully initialised either, so we cant rely on + * too many KH functions yet. + * NOTE: Kohana doesnt provide a parent construct for the Kohana_Config class. + */ + public function __construct() { + if (defined('PHPUNITTEST')) + $_SERVER['SERVER_NAME'] = PHPUNITTEST; + } + /** * Return our site name */ public static function site() { - if (! empty($_SERVER['SERVER_NAME'])) - return $_SERVER['SERVER_NAME']; - - if (! $site = CLI::options('site')) - throw new Kohana_Exception(_('Cant figure out the site, use --site= for CLI')); - - return $site['site']; + return $_SERVER['SERVER_NAME']; } /** * Work out our site ID for multiehosting - * @todo Change this to query the DB for site number. */ public static function siteid() { - $sites = Kohana::config('config.site'); - - // If we havent been configured for sites - if (is_null($sites) OR ! is_array($sites) OR ! isset($sites[static::site()])) - return 0; - else - return $sites[static::site()]; + return Kohana::$config->load('config.site.id'); } /** * Work out our site mode (dev,test,prod) - * @todo Change this to query the DB for mode. */ public static function sitemode() { - $sites = Kohana::config('config.site_mode'); + return Kohana::$config->load('config.site.mode'); + } - // If we havent been configured for sites - if (is_null($sites) OR ! is_array($sites) OR ! isset($sites[static::site()])) - return Kohana::PRODUCTION; - else - return $sites[static::site()]; + public static function sitemodeverbose() { + $modes = array( + Kohana::PRODUCTION=>'Production', + Kohana::STAGING=>'Staging', + Kohana::TESTING=>'Testing', + Kohana::DEVELOPMENT=>'Development', + ); + + return (! isset($modes[static::sitemode()])) ? 'Unknown' : $modes[static::sitemode()]; + } + + public static function submode() { + $submode = Kohana::$config->load('config.debug.submode'); + + return (isset($submode[Request::$client_ip])) ? $submode[Request::$client_ip] : FALSE; } public static function sitename() { - return Kohana::config('config.site_name'); + return Kohana::$config->load('config')->site_name; } + // Called in Invoice/Emailing to embed the file. public static function logo_file() { - // @todo Move the logo filename to a config file - return Kohana::find_file(sprintf('media/%s',Config::siteid()),'img/logo-small','png'); + list ($path,$suffix) = explode('.',static::$logo); + return ($a=Kohana::find_file(sprintf('media/site/%s',Config::siteid()),$path,$suffix)) ? $a : Kohana::find_file('media',$path,$suffix); + } + + public static function logo_uri() { + list ($path,$suffix) = explode('.',static::$logo); + return URL::site(Route::get('default/media')->uri(array('file'=>$path.'.'.$suffix),array('alt'=>static::sitename())),'http'); } public static function logo() { - // @todo Move the logo filename to a config file - $mediapath = Route::get('default/media'); - $logo = $mediapath->uri(array('file'=>'img/logo-small.png'),array('alt'=>static::sitename())); + return HTML::image(static::logo_uri(),array('class'=>'headlogo','alt'=>_('Logo'))); + } - return HTML::image($logo,array('class'=>'headlogo','alt'=>_('Logo'))); + public static function login_uri() { + return ($ao = Auth::instance()->get_user() AND is_object($ao)) ? HTML::anchor('user/account/edit',$ao->name()) : HTML::anchor('login',_('Login')); } /** * Return our caching mechanism */ public static function cachetype() { - return is_null(Kohana::config('config.cache_type')) ? 'file' : Kohana::config('config.cache_type'); + return is_null(Kohana::$config->load('config')->cache_type) ? 'file' : Kohana::$config->load('config')->cache_type; } /** * Show a date using a site configured format */ public static function date($date) { - return date(Kohana::config('config.date_format'),$date); + return $date ? date(Kohana::$config->load('config')->date_format,$date) : '>Not Set<'; + } + + /** + * Show a date using a site configured format + */ + public static function time($date) { + return date(Kohana::$config->load('config')->time_format,$date); + } + + /** + * Show a date using a site configured format + */ + public static function datetime($date) { + return sprintf('%s %s',static::date($date),static::time($date)); } /** @@ -91,12 +122,16 @@ abstract class lnApp_Config extends Kohana_Config { * @return mixed|array - Email to send test emails to */ public static function testmail($template) { - $config = Kohana::config('config.email_admin_only'); + $config = Kohana::$config->load('config')->email_admin_only; if (is_null($config) OR ! is_array($config) OR empty($config[$template])) return FALSE; else return $config[$template]; } + + public static function theme() { + return Kohana::$config->load('config')->theme; + } } ?> diff --git a/application/classes/lnapp/headimages.php b/application/classes/lnapp/headimages.php index 21f3ac7..37595eb 100644 --- a/application/classes/lnapp/headimages.php +++ b/application/classes/lnapp/headimages.php @@ -10,10 +10,10 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_HeadImages extends HTMLRender { +abstract class lnApp_HeadImages extends HTMLRender { protected static $_data = array(); protected static $_spacer = ' '; - protected static $_required_keys = array('url','img'); + protected static $_required_keys = array('img'); /** * Return an instance of this class @@ -35,7 +35,10 @@ class lnApp_HeadImages extends HTMLRender { foreach (static::$_data as $value) { $i = HTML::image($mediapath->uri(array('file'=>$value['img'])),array('alt'=>isset($value['attrs']['title']) ? $value['attrs']['title'] : '')); - $output .= HTML::anchor($value['url'],$i,(isset($value['attrs']) && is_array($value['attrs'])) ? $value['attrs'] : null); + if (isset($value['url'])) + $output .= HTML::anchor($value['url'],$i,(isset($value['attrs']) && is_array($value['attrs'])) ? $value['attrs'] : NULL); + else + $output .= $i; $output .= static::$_spacer; } diff --git a/application/classes/lnapp/html.php b/application/classes/lnapp/html.php index 7c43317..e69effd 100644 --- a/application/classes/lnapp/html.php +++ b/application/classes/lnapp/html.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_HTML extends Kohana_HTML { +abstract class lnApp_HTML extends Kohana_HTML { public static function nbsp($string) { if (strlen((string)$string)) return $string; diff --git a/application/classes/lnapp/htmlrender.php b/application/classes/lnapp/htmlrender.php index b140401..325c203 100644 --- a/application/classes/lnapp/htmlrender.php +++ b/application/classes/lnapp/htmlrender.php @@ -77,9 +77,7 @@ abstract class lnApp_HTMLRender { // Display the exception message catch (Exception $e) { - Kohana::exception_handler($e); - - return ''; + Kohana_Exception::handler($e); } } diff --git a/application/classes/lnapp/meta.php b/application/classes/lnapp/meta.php index 4fcce6d..fe554b4 100644 --- a/application/classes/lnapp/meta.php +++ b/application/classes/lnapp/meta.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_Meta { +abstract class lnApp_Meta { private $_data = array(); private $_array_keys = array(); @@ -19,7 +19,7 @@ class lnApp_Meta { return array(); if (empty($this->_data[$key])) - return null; + return NULL; else return $this->_data[$key]; } diff --git a/application/classes/lnapp/script.php b/application/classes/lnapp/script.php index 0763512..237acd5 100644 --- a/application/classes/lnapp/script.php +++ b/application/classes/lnapp/script.php @@ -10,11 +10,12 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_Script extends HTMLRender { +abstract class lnApp_Script extends HTMLRender { protected static $_data = array(); protected static $_spacer = "\n"; protected static $_required_keys = array('type','data'); protected static $_unique_vals = array('file'=>'type'); + protected static $_rendered = FALSE; /** * Return an instance of this class @@ -34,26 +35,28 @@ class lnApp_Script extends HTMLRender { $foutput = $soutput = ''; $mediapath = Route::get(static::$_media_path); - $i = $j = 0; foreach (static::$_data as $value) { - switch ($value['type']) { case 'file': $foutput .= HTML::script($mediapath->uri(array('file'=>$value['data']))); - if ($i++) - $foutput .= static::$_spacer; break; case 'stdin': $soutput .= sprintf("",$value['data']); - if ($j++) - $soutput .= static::$_spacer; break; default: throw new Kohana_Exception('Unknown style type :type',array(':type'=>$value['type'])); } } + static::$_rendered = TRUE; return $foutput.static::$_spacer.$soutput; } + + public static function add($item,$prepend=FALSE,$x='') { + if (static::$_rendered) + throw new Kohana_Exception('Already rendered?'); + + return parent::add($item,$prepend); + } } ?> diff --git a/application/classes/lnapp/sort.php b/application/classes/lnapp/sort.php index 7fe6259..051f0e1 100644 --- a/application/classes/lnapp/sort.php +++ b/application/classes/lnapp/sort.php @@ -11,7 +11,7 @@ * @license http://dev.leenooks.net/license.html * @uses Style */ -class lnApp_Sort { +abstract class lnApp_Sort { /** * Sort a multi dimensional array. * @@ -20,86 +20,71 @@ class lnApp_Sort { * @param boolean Whether to reverse sort. * @return array Sorted multi demension array. */ - public static function masort(&$data,$sortby,$rev=0) { - // if the array to sort is null or empty - if (! $data) + public static function MAsort(&$data,$sortby,$rev=0) { + // if the array to sort is null or empty, or our sortby is bad + if (! preg_match('/^([a-zA-Z0-9_]+(\([a-zA-Z0-9_,]*\)(->[a-zA-Z0-9])?)?,?)+$/',$sortby) || ! $data) return; - - static $MASORT_CACHE = array(); - - if (empty($MASORT_CACHE[$sortby])) { - $code = "\$c=0;\n"; - - foreach (explode(',',$sortby) as $key) { - $code .= "if (is_object(\$a) || is_object(\$b)) {\n"; - - $code .= " if (is_array(\$a->$key)) {\n"; - $code .= " asort(\$a->$key);\n"; - $code .= " \$aa = array_shift(\$a->$key);\n"; - $code .= " } else\n"; - $code .= " \$aa = \$a->$key;\n"; - - $code .= " if (is_array(\$b->$key)) {\n"; - $code .= " asort(\$b->$key);\n"; - $code .= " \$bb = array_shift(\$b->$key);\n"; - $code .= " } else\n"; - $code .= " \$bb = \$b->$key;\n"; - - $code .= " if (\$aa != \$bb)"; - if ($rev) - $code .= " return (\$aa < \$bb ? 1 : -1);\n"; - else - $code .= " return (\$aa > \$bb ? 1 : -1);\n"; - - $code .= "} else {\n"; - - $code .= " \$a = array_change_key_case(\$a);\n"; - $code .= " \$b = array_change_key_case(\$b);\n"; - - $key = strtolower($key); - - $code .= " if ((! isset(\$a['$key'])) && isset(\$b['$key'])) return 1;\n"; - $code .= " if (isset(\$a['$key']) && (! isset(\$b['$key']))) return -1;\n"; - - $code .= " if ((isset(\$a['$key'])) && (isset(\$b['$key']))) {\n"; - $code .= " if (is_array(\$a['$key'])) {\n"; - $code .= " asort(\$a['$key']);\n"; - $code .= " \$aa = array_shift(\$a['$key']);\n"; - $code .= " } else\n"; - $code .= " \$aa = \$a['$key'];\n"; - - $code .= " if (is_array(\$b['$key'])) {\n"; - $code .= " asort(\$b['$key']);\n"; - $code .= " \$bb = array_shift(\$b['$key']);\n"; - $code .= " } else\n"; - $code .= " \$bb = \$b['$key'];\n"; - - $code .= " if (\$aa != \$bb)\n"; - $code .= " if (is_numeric(\$aa) && is_numeric(\$bb)) {\n"; - - if ($rev) - $code .= " return (\$aa < \$bb ? 1 : -1);\n"; - else - $code .= " return (\$aa > \$bb ? 1 : -1);\n"; - - $code .= " } else {\n"; - - if ($rev) - $code .= " if ( (\$c = strcasecmp(\$bb,\$aa)) != 0 ) return \$c;\n"; - else - $code .= " if ( (\$c = strcasecmp(\$aa,\$bb)) != 0 ) return \$c;\n"; - - $code .= " }\n"; - $code .= " }\n"; - $code .= "}\n"; + + $code = '$c=0;'; + + foreach (explode(',',$sortby) as $key) { + $code .= 'if (is_object($a) || is_object($b)) {'; + foreach (array('a','b') as $x) { + $code .= 'if (is_array($'.$x.'->'.$key.')) {'; + $code .= 'asort($'.$x.'->'.$key.');'; + $code .= '$x'.$x.' = array_shift($'.$x.'->'.$key.');'; + $code .= '} else'; + $code .= '$x'.$x.' = $'.$x.'->'.$key.';'; } - - $code .= 'return $c;'; - - $MASORT_CACHE[$sortby] = create_function('$a, $b',$code); + + $code .= 'if ($xa != $xb)'; + if ($rev) + $code .= 'return ($xa < $xb ? 1 : -1);'; + else + $code .= 'return ($xa > $xb ? 1 : -1);'; + + $code .= '} else {'; + + foreach (array('a','b') as $x) + $code .= '$'.$x.' = array_change_key_case($'.$x.');'; + + $key = strtolower($key); + + $code .= 'if ((! isset($a[\''.$key.'\'])) && isset($b[\''.$key.'\'])) return 1;'; + $code .= 'if (isset($a[\''.$key.'\']) && (! isset($b[\''.$key.'\']))) return -1;'; + + $code .= 'if ((isset($a[\''.$key.'\'])) && (isset($b[\''.$key.'\']))) {'; + foreach (array('a','b') as $x) { + $code .= 'if (is_array($'.$x.'[\''.$key.'\'])) {'; + $code .= 'asort($'.$x.'[\''.$key.'\']);'; + $code .= '$x'.$x.' = array_shift($'.$x.'[\''.$key.'\']);'; + $code .= '} else'; + $code .= '$x'.$x.' = $'.$x.'[\''.$key.'\'];'; + } + + $code .= 'if ($xa != $xb)'; + $code .= 'if (is_numeric($xa) && is_numeric($xb)) {'; + + if ($rev) + $code .= 'return ($xa < $xb ? 1 : -1);'; + else + $code .= 'return ($xa > $xb ? 1 : -1);'; + + $code .= '} else {'; + + if ($rev) + $code .= 'if (($c = strcasecmp($xb,$xa)) != 0) return $c;'; + else + $code .= 'if (($c = strcasecmp($xa,$xb)) != 0) return $c;'; + + $code .= '}}}'; } - - uasort($data,$MASORT_CACHE[$sortby]); + + $code .= 'return $c;'; + + $result = create_function('$a, $b',$code); + + uasort($data,$result); } } ?> diff --git a/application/classes/lnapp/style.php b/application/classes/lnapp/style.php index 1481567..48c0074 100644 --- a/application/classes/lnapp/style.php +++ b/application/classes/lnapp/style.php @@ -10,10 +10,11 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_Style extends HTMLRender { +abstract class lnApp_Style extends HTMLRender { protected static $_data = array(); protected static $_spacer = "\n"; protected static $_required_keys = array('type','data'); + protected static $_unique_vals = array('file'=>'type'); /** * Return an instance of this class @@ -30,25 +31,24 @@ class lnApp_Style extends HTMLRender { * @see HTMLRender::render() */ protected function render() { - $output = ''; + $foutput = $soutput = ''; $mediapath = Route::get(static::$_media_path); - $i = 0; foreach (static::$_data as $value) { - if ($i++) - $output .= static::$_spacer; - switch ($value['type']) { case 'file': - $output .= HTML::style($mediapath->uri(array('file'=>$value['data'])), + $foutput .= HTML::style($mediapath->uri(array('file'=>$value['data'])), array('media'=>(! empty($value['media'])) ? $value['media'] : 'screen'),TRUE); break; + case 'stdin': + $soutput .= sprintf("",$value['data']); + break; default: throw new Kohana_Exception('Unknown style type :type',array(':type'=>$value['type'])); } } - return $output; + return $foutput.static::$_spacer.$soutput; } } ?> diff --git a/application/classes/lnapp/systemmessage.php b/application/classes/lnapp/systemmessage.php index 237bf01..d92fa9b 100644 --- a/application/classes/lnapp/systemmessage.php +++ b/application/classes/lnapp/systemmessage.php @@ -10,7 +10,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class lnApp_SystemMessage extends HTMLRender { +abstract class lnApp_SystemMessage extends HTMLRender { protected static $_data = array(); protected static $_spacer = '
 
'; protected static $_required_keys = array('title','body','type'); @@ -69,7 +69,7 @@ class lnApp_SystemMessage extends HTMLRender { /** * Render an image for the System Message */ - private static function image($type,$raw=false,$big=false,$alt='') { + public static function image($type,$raw=false,$big=false,$alt='') { $mediapath = Route::get(static::$_media_path); switch ($type) { diff --git a/application/classes/model/actlog.php b/application/classes/model/actlog.php index 9c2af9a..0e3721a 100644 --- a/application/classes/model/actlog.php +++ b/application/classes/model/actlog.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_ACTLOG extends ORMTSM { +class Model_ACTLOG extends TSM_ORM { protected $_table_name = 'ACTLOG'; protected $_primary_key = 'DATE_TIME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/association.php b/application/classes/model/association.php index 5f3e688..9377f56 100644 --- a/application/classes/model/association.php +++ b/application/classes/model/association.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_ASSOCIATION extends ORMTSM { +class Model_ASSOCIATION extends TSM_ORM { protected $_table_name = 'ASSOCIATIONS'; protected $_primary_key = 'DOMAIN_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/auth/userdefault.php b/application/classes/model/auth/userdefault.php index 0d67e6e..70949f6 100644 --- a/application/classes/model/auth/userdefault.php +++ b/application/classes/model/auth/userdefault.php @@ -16,7 +16,7 @@ class Model_Auth_UserDefault extends Model_Auth_User { protected function _initialize() { // Set out DB connection configuration. - $this->_db_group = Kohana::config('config.client_type'); + $this->_db_group = Kohana::$config->load('config')->client_type; parent::_initialize(); } diff --git a/application/classes/model/clientopt.php b/application/classes/model/clientopt.php index 3d47c5b..49a34a8 100644 --- a/application/classes/model/clientopt.php +++ b/application/classes/model/clientopt.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_CLIENTOPT extends ORMTSM { +class Model_CLIENTOPT extends TSM_ORM { protected $_table_name = 'CLIENTOPTS'; protected $_primary_key = 'OPTIONSET_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/copygroup/ar.php b/application/classes/model/copygroup/ar.php index bb0d987..1ca2b3f 100644 --- a/application/classes/model/copygroup/ar.php +++ b/application/classes/model/copygroup/ar.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_COPYGROUP_AR extends ORMTSM { +class Model_COPYGROUP_AR extends TSM_ORM { protected $_table_name = 'AR_COPYGROUPS'; protected $_primary_key = 'DOMAIN_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/copygroup/bu.php b/application/classes/model/copygroup/bu.php index 8fb6177..a0a0ad5 100644 --- a/application/classes/model/copygroup/bu.php +++ b/application/classes/model/copygroup/bu.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_COPYGROUP_BU extends ORMTSM { +class Model_COPYGROUP_BU extends TSM_ORM { protected $_table_name = 'BU_COPYGROUPS'; protected $_primary_key = 'DOMAIN_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/devclasses.php b/application/classes/model/devclasses.php index 3f5bff1..609ac25 100644 --- a/application/classes/model/devclasses.php +++ b/application/classes/model/devclasses.php @@ -10,7 +10,7 @@ * @license http://phptsmadmin.sf.net/license.html * @note This is model is using the plural name, as storage pools have an attribute with the singular name */ -class Model_DEVCLASSES extends ORMTSM { +class Model_DEVCLASSES extends TSM_ORM { protected $_table_name = 'DEVCLASSES'; protected $_primary_key = 'DEVCLASS_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/domain.php b/application/classes/model/domain.php index f90945e..e272fe4 100644 --- a/application/classes/model/domain.php +++ b/application/classes/model/domain.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_DOMAIN extends ORMTSM { +class Model_DOMAIN extends TSM_ORM { protected $_table_name = 'DOMAINS'; protected $_primary_key = 'DOMAIN_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/drive.php b/application/classes/model/drive.php index 2280fff..5b576b4 100644 --- a/application/classes/model/drive.php +++ b/application/classes/model/drive.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_DRIVE extends ORMTSM { +class Model_DRIVE extends TSM_ORM { protected $_table_name = 'DRIVES'; protected $_primary_key = 'DRIVE_NAME'; protected $_sorting = array( diff --git a/application/classes/model/event.php b/application/classes/model/event.php index 3e8a9df..a4d1fb2 100644 --- a/application/classes/model/event.php +++ b/application/classes/model/event.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_EVENT extends ORMTSM { +class Model_EVENT extends TSM_ORM { protected $_table_name = 'EVENTS'; protected $_primary_key = 'NODE_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( @@ -19,13 +19,13 @@ class Model_EVENT extends ORMTSM { protected $_display_filters = array( 'SCHEDULED_START'=>array( - array('ORMTSM::date',array(':value','d-M H:i')), + array('TSM_ORM::date',array(':value','d-M H:i')), ), 'ACTUAL_START'=>array( - array('ORMTSM::date',array(':value','d-M H:i')), + array('TSM_ORM::date',array(':value','d-M H:i')), ), 'COMPLETED'=>array( - array('ORMTSM::date',array(':value','d-M H:i')), + array('TSM_ORM::date',array(':value','d-M H:i')), ), ); } diff --git a/application/classes/model/filespace.php b/application/classes/model/filespace.php index 2139fd5..6fe1232 100644 --- a/application/classes/model/filespace.php +++ b/application/classes/model/filespace.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_FILESPACE extends ORMTSM { +class Model_FILESPACE extends TSM_ORM { protected $_table_name = 'FILESPACES'; protected $_primary_key = 'FILESPACE_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( @@ -44,7 +44,7 @@ class Model_FILESPACE extends ORMTSM { protected $_display_filters = array( 'BACKUP_END'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), ); diff --git a/application/classes/model/library.php b/application/classes/model/library.php index c52a62e..d088ce3 100644 --- a/application/classes/model/library.php +++ b/application/classes/model/library.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_LIBRARY extends ORMTSM { +class Model_LIBRARY extends TSM_ORM { protected $_table_name = 'LIBRARIES'; protected $_primary_key = 'LIBRARY_NAME'; protected $_sorting = array( @@ -110,7 +110,7 @@ class Model_LIBRARY extends ORMTSM { if (! isset($CACHE[__METHOD__][$ptype])) foreach ($this->storagepoolstype($ptype) as $spo) foreach ($ainout as $cinout) - foreach (Kohana::config('config.tsmvolstatus') as $cstatus) { + foreach (Kohana::$config->load('config')->tsmvolstatus as $cstatus) { if (! isset($CACHE[__METHOD__][$spo->POOLTYPE][$cinout][$cstatus])) $CACHE[__METHOD__][$spo->POOLTYPE][$cinout][$cstatus] = array(); @@ -131,7 +131,7 @@ class Model_LIBRARY extends ORMTSM { static $CACHE = array(); if (! isset($CACHE[__METHOD__])) - foreach (ORM::factory('volhistory')->where('TYPE','IN',Kohana::config('config.tsmdbtypes'))->find_all() as $vho) + foreach (ORM::factory('volhistory')->where('TYPE','IN',Kohana::$config->load('config')->tsmdbtypes)->find_all() as $vho) $CACHE[__METHOD__][$vho->LIBVOLUME->LIBRARY_NAME ? 'IN' : 'OUT'][] = $vho; return isset($CACHE[__METHOD__][$inout]) ? $CACHE[__METHOD__][$inout] : array(); diff --git a/application/classes/model/libvolume.php b/application/classes/model/libvolume.php index 333fe7d..3f77f93 100644 --- a/application/classes/model/libvolume.php +++ b/application/classes/model/libvolume.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_LIBVOLUME extends ORMTSM { +class Model_LIBVOLUME extends TSM_ORM { protected $_table_name = 'LIBVOLUMES'; protected $_primary_key = 'HOME_ELEMENT'; protected $_sorting = array( diff --git a/application/classes/model/media.php b/application/classes/model/media.php index 7f9220f..9e3afb3 100644 --- a/application/classes/model/media.php +++ b/application/classes/model/media.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_MEDIA extends ORMTSM { +class Model_MEDIA extends TSM_ORM { protected $_table_name = 'MEDIA'; protected $_primary_key = 'VOLUME_NAME'; protected $_sorting = array( diff --git a/application/classes/model/mgmtclass.php b/application/classes/model/mgmtclass.php index 486b612..ddfe6f0 100644 --- a/application/classes/model/mgmtclass.php +++ b/application/classes/model/mgmtclass.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_MGMTCLASS extends ORMTSM { +class Model_MGMTCLASS extends TSM_ORM { protected $_table_name = 'MGMTCLASSES'; protected $_primary_key = 'DOMAIN_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/node.php b/application/classes/model/node.php index 59887ea..11f6597 100644 --- a/application/classes/model/node.php +++ b/application/classes/model/node.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_NODE extends ORMTSM { +class Model_NODE extends TSM_ORM { protected $_table_name = 'NODES'; protected $_primary_key = 'NODE_NAME'; protected $_sorting = array( @@ -86,13 +86,13 @@ class Model_NODE extends ORMTSM { protected $_display_filters = array( 'REG_TIME'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), 'PWSET_TIME'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), 'LASTACC_TIME'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), 'LASTSESS_SENT'=>array( array('number_format',array(':value',0)), @@ -207,7 +207,7 @@ class Model_NODE extends ORMTSM { public function getAllStoragePoolsType($ptype) { $result = array(); - foreach (Kohana::config('config.tsmdatatypes') as $btype => $ctype) + foreach (Kohana::$config->load('config')->tsmdatatypes as $btype => $ctype) $result = array_merge($result,$this->getStoragePoolsType($btype,$ptype)); return $result; @@ -240,7 +240,7 @@ class Model_NODE extends ORMTSM { public function getStorageTypeVols($ptype,$spo='') { $result = array(); - foreach (Kohana::config('config.tsmdatatypes') as $btype => $ctype) + foreach (Kohana::$config->load('config')->tsmdatatypes as $btype => $ctype) $result = array_merge($result,$this->getStorageModeVols($ctype,$ptype,$spo)); return $result; @@ -262,7 +262,7 @@ class Model_NODE extends ORMTSM { public function getStorageTypeFiles($ptype,$spo='') { $count = 0; - foreach (Kohana::config('config.tsmdatatypes') as $btype => $ctype) + foreach (Kohana::$config->load('config')->tsmdatatypes as $btype => $ctype) $count += $this->getStorageModeFiles($btype,$ptype,$spo); return $count; @@ -284,7 +284,7 @@ class Model_NODE extends ORMTSM { public function getStorageTypeData($ptype,$spo='') { $count = 0; - foreach (Kohana::config('config.tsmdatatypes') as $btype => $ctype) + foreach (Kohana::$config->load('config')->tsmdatatypes as $btype => $ctype) $count += $this->getStorageModeData($btype,$ptype,$spo); return $count; diff --git a/application/classes/model/occ.php b/application/classes/model/occ.php index 1e636e8..b1023bd 100644 --- a/application/classes/model/occ.php +++ b/application/classes/model/occ.php @@ -10,7 +10,7 @@ * @license http://phptsmadmin.sf.net/license.html * @node This has been renamed to OCC from OCCUPANCY, as DB2/FILESPACES has an OCCUPANCY column */ -class Model_OCC extends ORMTSM { +class Model_OCC extends TSM_ORM { protected $_table_name = 'OCCUPANCY'; protected $_primary_key = 'NODE_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/model/path.php b/application/classes/model/path.php index e9e520b..86018e4 100644 --- a/application/classes/model/path.php +++ b/application/classes/model/path.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_PATH extends ORMTSM { +class Model_PATH extends TSM_ORM { protected $_table_name = 'PATHS'; protected $_primary_key = 'DEVICE'; protected $_sorting = array( diff --git a/application/classes/model/schedule/client.php b/application/classes/model/schedule/client.php index 2a09dff..3184a72 100644 --- a/application/classes/model/schedule/client.php +++ b/application/classes/model/schedule/client.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_SCHEDULE_CLIENT extends ORMTSM { +class Model_SCHEDULE_CLIENT extends TSM_ORM { protected $_table_name = 'CLIENT_SCHEDULES'; protected $_primary_key = 'SCHEDULE_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( @@ -22,7 +22,7 @@ class Model_SCHEDULE_CLIENT extends ORMTSM { protected $_display_filters = array( 'STARTTIME'=>array( - array('ORMTSM::date',array(':value','h:m')), + array('TSM_ORM::date',array(':value','h:m')), ), ); diff --git a/application/classes/model/stgpool.php b/application/classes/model/stgpool.php index 7f16ad4..50b4100 100644 --- a/application/classes/model/stgpool.php +++ b/application/classes/model/stgpool.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_STGPOOL extends ORMTSM { +class Model_STGPOOL extends TSM_ORM { protected $_table_name = 'STGPOOLS'; protected $_primary_key = 'STGPOOL_NAME'; protected $_sorting = array( diff --git a/application/classes/model/summary.php b/application/classes/model/summary.php index e6da01a..9568aa1 100644 --- a/application/classes/model/summary.php +++ b/application/classes/model/summary.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_SUMMARY extends ORMTSM { +class Model_SUMMARY extends TSM_ORM { protected $_table_name = 'SUMMARY'; protected $_primary_key = 'ACTIVITY'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( @@ -20,10 +20,10 @@ class Model_SUMMARY extends ORMTSM { protected $_display_filters = array( 'START_TIME'=>array( - array('ORMTSM::date',array(':value','d-M H:i')), + array('TSM_ORM::date',array(':value','d-M H:i')), ), 'END_TIME'=>array( - array('ORMTSM::date',array(':value','d-M H:i')), + array('TSM_ORM::date',array(':value','d-M H:i')), ), ); } diff --git a/application/classes/model/volhistory.php b/application/classes/model/volhistory.php index 994a628..bd02dae 100644 --- a/application/classes/model/volhistory.php +++ b/application/classes/model/volhistory.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_VOLHISTORY extends ORMTSM { +class Model_VOLHISTORY extends TSM_ORM { protected $_table_name = 'VOLHISTORY'; protected $_primary_key = 'VOLUME_NAME'; protected $_sorting = array( @@ -25,7 +25,7 @@ class Model_VOLHISTORY extends ORMTSM { protected $_display_filters = array( 'DATE_TIME'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), ); diff --git a/application/classes/model/volume.php b/application/classes/model/volume.php index 488fd78..789861d 100644 --- a/application/classes/model/volume.php +++ b/application/classes/model/volume.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_VOLUME extends ORMTSM { +class Model_VOLUME extends TSM_ORM { protected $_table_name = 'VOLUMES'; protected $_primary_key = 'VOLUME_NAME'; protected $_sorting = array( @@ -27,10 +27,10 @@ class Model_VOLUME extends ORMTSM { protected $_display_filters = array( 'LAST_READ_DATE'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), 'LAST_WRITE_DATE'=>array( - array('ORMTSM::date',array(':value','d-M-Y')), + array('TSM_ORM::date',array(':value','d-M-Y')), ), ); @@ -63,7 +63,7 @@ class Model_VOLUME extends ORMTSM { } public function recycle() { - return Kohana::config('config.tsmtapeage') < $this->age() ? TRUE : FALSE; + return Kohana::$config->load('config')->tsmtapeage < $this->age() ? TRUE : FALSE; } } ?> diff --git a/application/classes/model/volumeusage.php b/application/classes/model/volumeusage.php index dd763de..1faf3ac 100644 --- a/application/classes/model/volumeusage.php +++ b/application/classes/model/volumeusage.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 phpTSMadmin Development Team * @license http://phptsmadmin.sf.net/license.html */ -class Model_VOLUMEUSAGE extends ORMTSM { +class Model_VOLUMEUSAGE extends TSM_ORM { protected $_table_name = 'VOLUMEUSAGE'; protected $_primary_key = 'NODE_NAME'; // We need a primary key to detect that the object is loaded. protected $_sorting = array( diff --git a/application/classes/orm.php b/application/classes/orm.php index 19396d6..e7b7c58 100644 --- a/application/classes/orm.php +++ b/application/classes/orm.php @@ -3,14 +3,14 @@ /** * This class overrides Kohana's ORM * - * @package lnApp/Modifications + * @package PTA/Modifications * @category Classes * @category Helpers * @author Deon George * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class ORM extends Kohana_ORM { +abstract class ORM extends Kohana_ORM { protected $_table_names_plural = FALSE; protected $_model_names_plural = FALSE; private $_object_formated = array(); @@ -58,7 +58,7 @@ class ORM extends Kohana_ORM { $value = $this->__get($column); // If some of our fields need to be formated for display purposes. - if ($this->_loaded AND ! $this->_formated AND $this->_display_filters) + if (! $this->_formated AND $this->_display_filters) $this->_format(); if (isset($this->_object_formated[$column])) diff --git a/application/classes/sort.php b/application/classes/sort.php index cccb437..d444567 100644 --- a/application/classes/sort.php +++ b/application/classes/sort.php @@ -1,4 +1,4 @@ diff --git a/application/classes/systemmessage.php b/application/classes/systemmessage.php index e2b2125..b84a9ce 100644 --- a/application/classes/systemmessage.php +++ b/application/classes/systemmessage.php @@ -11,11 +11,11 @@ * @license http://phptsmadmin.sf.net/license.html */ class SystemMessage extends lnApp_SystemMessage { - static public function TSM_Error($error,$sql) { + static public function TSM($type,$error,$sql=NULL) { SystemMessage::add(array( - 'title'=>_('Error while talking to TSM'), - 'body'=>_('Running SQL').': '.$sql.'

'.(is_array($error) ? implode('
',$error) : $error), - 'type'=>'error', + 'title'=>_('While talking to TSM'), + 'body'=>($sql ? _('Running SQL').': '.$sql.'

' : '').(is_array($error) ? implode('
',$error) : $error), + 'type'=>$type, )); } } diff --git a/application/classes/tsm.php b/application/classes/tsm.php index 6e8b55a..ce7bb59 100644 --- a/application/classes/tsm.php +++ b/application/classes/tsm.php @@ -11,12 +11,19 @@ * @license http://phptsmadmin.sf.net/license.html */ class TSM { + static $_online = FALSE; + public function set($username,$password,Database_TSM_Result $server) { // @todo Consider encrypting this - Session::instance()->set(Kohana::config('auth.session_key'),$username); + Session::instance()->set(Kohana::$config->load('auth')->session_key,$username); Session::instance()->set('password',$password); - Session::instance()->set('SERVER',$server); + + return TSM::$_online = TRUE; + } + + public static function online() { + return TSM::$_online; } public static function instance() { @@ -24,7 +31,7 @@ class TSM { } public static function name() { - return Session::instance()->get('SERVER')->rewind()->get('SERVER_NAME'); + return Session::instance()->get('SERVER')->get('SERVER_NAME'); } public static function version() { diff --git a/application/classes/valid.php b/application/classes/valid.php index fa0d3ec..ba42a61 100644 --- a/application/classes/valid.php +++ b/application/classes/valid.php @@ -9,7 +9,7 @@ * @copyright (c) 2010 Deon George * @license http://dev.leenooks.net/license.html */ -class Valid extends Kohana_Valid { +abstract class Valid extends Kohana_Valid { /** * Checks if a field matches the value of another field, if it is set. * Field is ignored if it is blank. @@ -20,7 +20,7 @@ class Valid extends Kohana_Valid { * @return boolean */ public static function matches_ifset($array, $field, $match) - { + { return isset($array[$match]) ? ($array[$field] === $array[$match]) : TRUE; } } diff --git a/application/config/auth.php b/application/config/auth.php index e208391..1d88079 100644 --- a/application/config/auth.php +++ b/application/config/auth.php @@ -4,6 +4,7 @@ * PTA Configuration - Authentication Driver * * @package PTA + * @subpackage Authentication * @category Configuration * @author Deon George * @copyright (c) 2010 phpTSMadmin Development Team diff --git a/application/config/cache.php b/application/config/cache.php index c7946c3..e2ece2d 100644 --- a/application/config/cache.php +++ b/application/config/cache.php @@ -1,9 +1,9 @@ array - ( +return array( + 'apc' => array( + 'driver' => 'apc', + 'default_expire' => 3600, + ), + + 'file' => array( 'driver' => 'file', 'cache_dir' => Kohana::$cache_dir ? Kohana::$cache_dir : '/dev/shm/lnapp', 'default_expire' => 3600, - ) + 'ignore_on_delete' => array( + '.gitignore', + '.git', + '.htaccess', + '.svn' + ) + ), ); ?> diff --git a/application/config/config.php b/application/config/config.php index d9f4ae7..0be6738 100644 --- a/application/config/config.php +++ b/application/config/config.php @@ -13,7 +13,7 @@ return array( 'cache_type' => 'file', 'client' => '/opt/tivoli/tsm/client/ba/bin/dsmadmc', - 'client_type' => 'db2', + 'client_type' => 'dsmadmc', 'client_errorlogname' => '/tmp/pta-tsm-errorlog.log', 'date_format' => 'd-m-Y', 'tsmdatatypes' => array('Bkup'=>'BACKUP','Arch'=>'ARCHIVE'), @@ -28,9 +28,10 @@ return array( ), 'method_security' => TRUE, // Enables Method Security. Setting to false means any method can be run without authentication 'site' => array( + 'mode' => Kohana::PRODUCTION, ), 'site_debug' => FALSE, - 'site_mode' => array( - ) + 'site_name' => 'phpTSMadmin', + 'theme' => 'yaml', ); ?> diff --git a/application/config/debug.php b/application/config/debug.php new file mode 100644 index 0000000..f1bec5c --- /dev/null +++ b/application/config/debug.php @@ -0,0 +1,20 @@ +FALSE, // AJAX actions can only be run by ajax calls if set to FALSE + 'etag'=>FALSE, // Force generating ETAGS + 'task_sim'=>FALSE, // Simulate running tasks +); +?> diff --git a/application/media/js/jquery-1.6.4.min.js b/application/media/js/jquery-1.6.4.min.js new file mode 100644 index 0000000..628ed9b --- /dev/null +++ b/application/media/js/jquery-1.6.4.min.js @@ -0,0 +1,4 @@ +/*! jQuery v1.6.4 http://jquery.com/ | http://jquery.org/license */ +(function(a,b){function cu(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cr(a){if(!cg[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){ch||(ch=c.createElement("iframe"),ch.frameBorder=ch.width=ch.height=0),b.appendChild(ch);if(!ci||!ch.createElement)ci=(ch.contentWindow||ch.contentDocument).document,ci.write((c.compatMode==="CSS1Compat"?"":"")+""),ci.close();d=ci.createElement(a),ci.body.appendChild(d),e=f.css(d,"display"),b.removeChild(ch)}cg[a]=e}return cg[a]}function cq(a,b){var c={};f.each(cm.concat.apply([],cm.slice(0,b)),function(){c[this]=a});return c}function cp(){cn=b}function co(){setTimeout(cp,0);return cn=f.now()}function cf(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ce(){try{return new a.XMLHttpRequest}catch(b){}}function b$(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g0){c!=="border"&&f.each(e,function(){c||(d-=parseFloat(f.css(a,"padding"+this))||0),c==="margin"?d+=parseFloat(f.css(a,c+this))||0:d-=parseFloat(f.css(a,"border"+this+"Width"))||0});return d+"px"}d=bv(a,b,b);if(d<0||d==null)d=a.style[b]||0;d=parseFloat(d)||0,c&&f.each(e,function(){d+=parseFloat(f.css(a,"padding"+this))||0,c!=="padding"&&(d+=parseFloat(f.css(a,"border"+this+"Width"))||0),c==="margin"&&(d+=parseFloat(f.css(a,c+this))||0)});return d+"px"}function bl(a,b){b.src?f.ajax({url:b.src,async:!1,dataType:"script"}):f.globalEval((b.text||b.textContent||b.innerHTML||"").replace(bd,"/*$0*/")),b.parentNode&&b.parentNode.removeChild(b)}function bk(a){f.nodeName(a,"input")?bj(a):"getElementsByTagName"in a&&f.grep(a.getElementsByTagName("input"),bj)}function bj(a){if(a.type==="checkbox"||a.type==="radio")a.defaultChecked=a.checked}function bi(a){return"getElementsByTagName"in a?a.getElementsByTagName("*"):"querySelectorAll"in a?a.querySelectorAll("*"):[]}function bh(a,b){var c;if(b.nodeType===1){b.clearAttributes&&b.clearAttributes(),b.mergeAttributes&&b.mergeAttributes(a),c=b.nodeName.toLowerCase();if(c==="object")b.outerHTML=a.outerHTML;else if(c!=="input"||a.type!=="checkbox"&&a.type!=="radio"){if(c==="option")b.selected=a.defaultSelected;else if(c==="input"||c==="textarea")b.defaultValue=a.defaultValue}else a.checked&&(b.defaultChecked=b.checked=a.checked),b.value!==a.value&&(b.value=a.value);b.removeAttribute(f.expando)}}function bg(a,b){if(b.nodeType===1&&!!f.hasData(a)){var c=f.expando,d=f.data(a),e=f.data(b,d);if(d=d[c]){var g=d.events;e=e[c]=f.extend({},d);if(g){delete e.handle,e.events={};for(var h in g)for(var i=0,j=g[h].length;i=0===c})}function U(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function M(a,b){return(a&&a!=="*"?a+".":"")+b.replace(y,"`").replace(z,"&")}function L(a){var b,c,d,e,g,h,i,j,k,l,m,n,o,p=[],q=[],r=f._data(this,"events");if(!(a.liveFired===this||!r||!r.live||a.target.disabled||a.button&&a.type==="click")){a.namespace&&(n=new RegExp("(^|\\.)"+a.namespace.split(".").join("\\.(?:.*\\.)?")+"(\\.|$)")),a.liveFired=this;var s=r.live.slice(0);for(i=0;ic)break;a.currentTarget=e.elem,a.data=e.handleObj.data,a.handleObj=e.handleObj,o=e.handleObj.origHandler.apply(e.elem,arguments);if(o===!1||a.isPropagationStopped()){c=e.level,o===!1&&(b=!1);if(a.isImmediatePropagationStopped())break}}return b}}function J(a,c,d){var e=f.extend({},d[0]);e.type=a,e.originalEvent={},e.liveFired=b,f.event.handle.call(c,e),e.isDefaultPrevented()&&d[0].preventDefault()}function D(){return!0}function C(){return!1}function m(a,c,d){var e=c+"defer",g=c+"queue",h=c+"mark",i=f.data(a,e,b,!0);i&&(d==="queue"||!f.data(a,g,b,!0))&&(d==="mark"||!f.data(a,h,b,!0))&&setTimeout(function(){!f.data(a,g,b,!0)&&!f.data(a,h,b,!0)&&(f.removeData(a,e,!0),i.resolve())},0)}function l(a){for(var b in a)if(b!=="toJSON")return!1;return!0}function k(a,c,d){if(d===b&&a.nodeType===1){var e="data-"+c.replace(j,"-$1").toLowerCase();d=a.getAttribute(e);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNaN(d)?i.test(d)?f.parseJSON(d):d:parseFloat(d)}catch(g){}f.data(a,c,d)}else d=b}return d}var c=a.document,d=a.navigator,e=a.location,f=function(){function K(){if(!e.isReady){try{c.documentElement.doScroll("left")}catch(a){setTimeout(K,1);return}e.ready()}}var e=function(a,b){return new e.fn.init(a,b,h)},f=a.jQuery,g=a.$,h,i=/^(?:[^#<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/\d/,n=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,o=/^[\],:{}\s]*$/,p=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,q=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,r=/(?:^|:|,)(?:\s*\[)+/g,s=/(webkit)[ \/]([\w.]+)/,t=/(opera)(?:.*version)?[ \/]([\w.]+)/,u=/(msie) ([\w.]+)/,v=/(mozilla)(?:.*? rv:([\w.]+))?/,w=/-([a-z]|[0-9])/ig,x=/^-ms-/,y=function(a,b){return(b+"").toUpperCase()},z=d.userAgent,A,B,C,D=Object.prototype.toString,E=Object.prototype.hasOwnProperty,F=Array.prototype.push,G=Array.prototype.slice,H=String.prototype.trim,I=Array.prototype.indexOf,J={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)!=="<"||a.charAt(a.length-1)!==">"||a.length<3?g=i.exec(a):g=[null,a,null];if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=n.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.6.4",length:0,size:function(){return this.length},toArray:function(){return G.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?F.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),B.done(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(G.apply(this,arguments),"slice",G.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:F,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j0)return;B.resolveWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").unbind("ready")}},bindReady:function(){if(!B){B=e._Deferred();if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",C,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",C),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&K()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNaN:function(a){return a==null||!m.test(a)||isNaN(a)},type:function(a){return a==null?String(a):J[D.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;try{if(a.constructor&&!E.call(a,"constructor")&&!E.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}var d;for(d in a);return d===b||E.call(a,d)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw a},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(o.test(b.replace(p,"@").replace(q,"]").replace(r,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(c){var d,f;try{a.DOMParser?(f=new DOMParser,d=f.parseFromString(c,"text/xml")):(d=new ActiveXObject("Microsoft.XMLDOM"),d.async="false",d.loadXML(c))}catch(g){d=b}(!d||!d.documentElement||d.getElementsByTagName("parsererror").length)&&e.error("Invalid XML: "+c);return d},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(x,"ms-").replace(w,y)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i1?h.call(arguments,0):c,--e||g.resolveWith(g,h.call(b,0))}}var b=arguments,c=0,d=b.length,e=d,g=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred();if(d>1){for(;c
a",d=a.getElementsByTagName("*"),e=a.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};g=c.createElement("select"),h=g.appendChild(c.createElement("option")),i=a.getElementsByTagName("input")[0],k={leadingWhitespace:a.firstChild.nodeType===3,tbody:!a.getElementsByTagName("tbody").length,htmlSerialize:!!a.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55$/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:i.value==="on",optSelected:h.selected,getSetAttribute:a.className!=="t",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},i.checked=!0,k.noCloneChecked=i.cloneNode(!0).checked,g.disabled=!0,k.optDisabled=!h.disabled;try{delete a.test}catch(v){k.deleteExpando=!1}!a.addEventListener&&a.attachEvent&&a.fireEvent&&(a.attachEvent("onclick",function(){k.noCloneEvent=!1}),a.cloneNode(!0).fireEvent("onclick")),i=c.createElement("input"),i.value="t",i.setAttribute("type","radio"),k.radioValue=i.value==="t",i.setAttribute("checked","checked"),a.appendChild(i),l=c.createDocumentFragment(),l.appendChild(a.firstChild),k.checkClone=l.cloneNode(!0).cloneNode(!0).lastChild.checked,a.innerHTML="",a.style.width=a.style.paddingLeft="1px",m=c.getElementsByTagName("body")[0],o=c.createElement(m?"div":"body"),p={visibility:"hidden",width:0,height:0,border:0,margin:0,background:"none"},m&&f.extend(p,{position:"absolute",left:"-1000px",top:"-1000px"});for(t in p)o.style[t]=p[t];o.appendChild(a),n=m||b,n.insertBefore(o,n.firstChild),k.appendChecked=i.checked,k.boxModel=a.offsetWidth===2,"zoom"in a.style&&(a.style.display="inline",a.style.zoom=1,k.inlineBlockNeedsLayout=a.offsetWidth===2,a.style.display="",a.innerHTML="
",k.shrinkWrapBlocks=a.offsetWidth!==2),a.innerHTML="
t
",q=a.getElementsByTagName("td"),u=q[0].offsetHeight===0,q[0].style.display="",q[1].style.display="none",k.reliableHiddenOffsets=u&&q[0].offsetHeight===0,a.innerHTML="",c.defaultView&&c.defaultView.getComputedStyle&&(j=c.createElement("div"),j.style.width="0",j.style.marginRight="0",a.appendChild(j),k.reliableMarginRight=(parseInt((c.defaultView.getComputedStyle(j,null)||{marginRight:0}).marginRight,10)||0)===0),o.innerHTML="",n.removeChild(o);if(a.attachEvent)for(t in{submit:1,change:1,focusin:1})s="on"+t,u=s in a,u||(a.setAttribute(s,"return;"),u=typeof a[s]=="function"),k[t+"Bubbles"]=u;o=l=g=h=m=j=a=i=null;return k}(),f.boxModel=f.support.boxModel;var i=/^(?:\{.*\}|\[.*\])$/,j=/([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!l(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g,h,i=f.expando,j=typeof c=="string",k=a.nodeType,l=k?f.cache:a,m=k?a[f.expando]:a[f.expando]&&f.expando;if((!m||e&&m&&l[m]&&!l[m][i])&&j&&d===b)return;m||(k?a[f.expando]=m=++f.uuid:m=f.expando),l[m]||(l[m]={},k||(l[m].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?l[m][i]=f.extend(l[m][i],c):l[m]=f.extend(l[m],c);g=l[m],e&&(g[i]||(g[i]={}),g=g[i]),d!==b&&(g[f.camelCase(c)]=d);if(c==="events"&&!g[c])return g[i]&&g[i].events;j?(h=g[c],h==null&&(h=g[f.camelCase(c)])):h=g;return h}},removeData:function(a,b,c){if(!!f.acceptData(a)){var d,e=f.expando,g=a.nodeType,h=g?f.cache:a,i=g?a[f.expando]:f.expando;if(!h[i])return;if(b){d=c?h[i][e]:h[i];if(d){d[b]||(b=f.camelCase(b)),delete d[b];if(!l(d))return}}if(c){delete h[i][e];if(!l(h[i]))return}var j=h[i][e];f.support.deleteExpando||!h.setInterval?delete h[i]:h[i]=null,j?(h[i]={},g||(h[i].toJSON=f.noop),h[i][e]=j):g&&(f.support.deleteExpando?delete a[f.expando]:a.removeAttribute?a.removeAttribute(f.expando):a[f.expando]=null)}},_data:function(a,b,c){return f.data(a,b,c,!0)},acceptData:function(a){if(a.nodeName){var b=f.noData[a.nodeName.toLowerCase()];if(b)return b!==!0&&a.getAttribute("classid")===b}return!0}}),f.fn.extend({data:function(a,c){var d=null;if(typeof a=="undefined"){if(this.length){d=f.data(this[0]);if(this[0].nodeType===1){var e=this[0].attributes,g;for(var h=0,i=e.length;h-1)return!0;return!1},val:function(a){var c,d,e=this[0];if(!arguments.length){if(e){c=f.valHooks[e.nodeName.toLowerCase()]||f.valHooks[e.type];if(c&&"get"in c&&(d=c.get(e,"value"))!==b)return d;d=e.value;return typeof d=="string"?d.replace(p,""):d==null?"":d}return b}var g=f.isFunction(a);return this.each(function(d){var e=f(this),h;if(this.nodeType===1){g?h=a.call(this,d,e.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||!("set"in c)||c.set(this,h,"value")===b)this.value=h}})}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c=a.selectedIndex,d=[],e=a.options,g=a.type==="select-one";if(c<0)return null;for(var h=g?c:0,i=g?c+1:e.length;h=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attrFix:{tabindex:"tabIndex"},attr:function(a,c,d,e){var g=a.nodeType;if(!a||g===3||g===8||g===2)return b;if(e&&c in f.attrFn)return f(a)[c](d);if(!("getAttribute"in a))return f.prop(a,c,d);var h,i,j=g!==1||!f.isXMLDoc(a);j&&(c=f.attrFix[c]||c,i=f.attrHooks[c],i||(t.test(c)?i=v:u&&(i=u)));if(d!==b){if(d===null){f.removeAttr(a,c);return b}if(i&&"set"in i&&j&&(h=i.set(a,d,c))!==b)return h;a.setAttribute(c,""+d);return d}if(i&&"get"in i&&j&&(h=i.get(a,c))!==null)return h;h=a.getAttribute(c);return h===null?b:h},removeAttr:function(a,b){var c;a.nodeType===1&&(b=f.attrFix[b]||b,f.attr(a,b,""),a.removeAttribute(b),t.test(b)&&(c=f.propFix[b]||b)in a&&(a[c]=!1))},attrHooks:{type:{set:function(a,b){if(q.test(a.nodeName)&&a.parentNode)f.error("type property can't be changed");else if(!f.support.radioValue&&b==="radio"&&f.nodeName(a,"input")){var c=a.value;a.setAttribute("type",b),c&&(a.value=c);return b}}},value:{get:function(a,b){if(u&&f.nodeName(a,"button"))return u.get(a,b);return b in a?a.value:null},set:function(a,b,c){if(u&&f.nodeName(a,"button"))return u.set(a,b,c);a.value=b}}},propFix:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},prop:function(a,c,d){var e=a.nodeType;if(!a||e===3||e===8||e===2)return b;var g,h,i=e!==1||!f.isXMLDoc(a);i&&(c=f.propFix[c]||c,h=f.propHooks[c]);return d!==b?h&&"set"in h&&(g=h.set(a,d,c))!==b?g:a[c]=d:h&&"get"in h&&(g=h.get(a,c))!==null?g:a[c]},propHooks:{tabIndex:{get:function(a){var c=a.getAttributeNode("tabindex");return c&&c.specified?parseInt(c.value,10):r.test(a.nodeName)||s.test(a.nodeName)&&a.href?0:b}}}}),f.attrHooks.tabIndex=f.propHooks.tabIndex,v={get:function(a,c){var d;return f.prop(a,c)===!0||(d=a.getAttributeNode(c))&&d.nodeValue!==!1?c.toLowerCase():b},set:function(a,b,c){var d;b===!1?f.removeAttr(a,c):(d=f.propFix[c]||c,d in a&&(a[d]=!0),a.setAttribute(c,c.toLowerCase()));return c}},f.support.getSetAttribute||(u=f.valHooks.button={get:function(a,c){var d;d=a.getAttributeNode(c);return d&&d.nodeValue!==""?d.nodeValue:b},set:function(a,b,d){var e=a.getAttributeNode(d);e||(e=c.createAttribute(d),a.setAttributeNode(e));return e.nodeValue=b+""}},f.each(["width","height"],function(a,b){f.attrHooks[b]=f.extend(f.attrHooks[b],{set:function(a,c){if(c===""){a.setAttribute(b,"auto");return c}}})})),f.support.hrefNormalized||f.each(["href","src","width","height"],function(a,c){f.attrHooks[c]=f.extend(f.attrHooks[c],{get:function(a){var d=a.getAttribute(c,2);return d===null?b:d}})}),f.support.style||(f.attrHooks.style={get:function(a){return a.style.cssText.toLowerCase()||b},set:function(a,b){return a.style.cssText=""+b}}),f.support.optSelected||(f.propHooks.selected=f.extend(f.propHooks.selected,{get:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex);return null}})),f.support.checkOn||f.each(["radio","checkbox"],function(){f.valHooks[this]={get:function(a){return a.getAttribute("value")===null?"on":a.value}}}),f.each(["radio","checkbox"],function(){f.valHooks[this]=f.extend(f.valHooks[this],{set:function(a,b){if(f.isArray(b))return a.checked=f.inArray(f(a).val(),b)>=0}})});var w=/\.(.*)$/,x=/^(?:textarea|input|select)$/i,y=/\./g,z=/ /g,A=/[^\w\s.|`]/g,B=function(a){return a.replace(A,"\\$&")};f.event={add:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){if(d===!1)d=C;else if(!d)return;var g,h;d.handler&&(g=d,d=g.handler),d.guid||(d.guid=f.guid++);var i=f._data(a);if(!i)return;var j=i.events,k=i.handle;j||(i.events=j={}),k||(i.handle=k=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.handle.apply(k.elem,arguments):b}),k.elem=a,c=c.split(" ");var l,m=0,n;while(l=c[m++]){h=g?f.extend({},g):{handler:d,data:e},l.indexOf(".")>-1?(n=l.split("."),l=n.shift(),h.namespace=n.slice(0).sort().join(".")):(n=[],h.namespace=""),h.type=l,h.guid||(h.guid=d.guid);var o=j[l],p=f.event.special[l]||{};if(!o){o=j[l]=[];if(!p.setup||p.setup.call(a,e,n,k)===!1)a.addEventListener?a.addEventListener(l,k,!1):a.attachEvent&&a.attachEvent("on"+l,k)}p.add&&(p.add.call(a,h),h.handler.guid||(h.handler.guid=d.guid)),o.push(h),f.event.global[l]=!0}a=null}},global:{},remove:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){d===!1&&(d=C);var g,h,i,j,k=0,l,m,n,o,p,q,r,s=f.hasData(a)&&f._data(a),t=s&&s.events;if(!s||!t)return;c&&c.type&&(d=c.handler,c=c.type);if(!c||typeof c=="string"&&c.charAt(0)==="."){c=c||"";for(h in t)f.event.remove(a,h+c);return}c=c.split(" ");while(h=c[k++]){r=h,q=null,l=h.indexOf(".")<0,m=[],l||(m=h.split("."),h=m.shift(),n=new RegExp("(^|\\.)"+f.map(m.slice(0).sort(),B).join("\\.(?:.*\\.)?")+"(\\.|$)")),p=t[h];if(!p)continue;if(!d){for(j=0;j=0&&(h=h.slice(0,-1),j=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.shift(),i.sort());if(!!e&&!f.event.customEvent[h]||!!f.event.global[h]){c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.exclusive=j,c.namespace=i.join("."),c.namespace_re=new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)");if(g||!e)c.preventDefault(),c.stopPropagation();if(!e){f.each(f.cache,function(){var a=f.expando,b=this[a];b&&b.events&&b.events[h]&&f.event.trigger(c,d,b.handle.elem)});return}if(e.nodeType===3||e.nodeType===8)return;c.result=b,c.target=e,d=d!=null?f.makeArray(d):[],d.unshift(c);var k=e,l=h.indexOf(":")<0?"on"+h:"";do{var m=f._data(k,"handle");c.currentTarget=k,m&&m.apply(k,d),l&&f.acceptData(k)&&k[l]&&k[l].apply(k,d)===!1&&(c.result=!1,c.preventDefault()),k=k.parentNode||k.ownerDocument||k===c.target.ownerDocument&&a}while(k&&!c.isPropagationStopped());if(!c.isDefaultPrevented()){var n,o=f.event.special[h]||{};if((!o._default||o._default.call(e.ownerDocument,c)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)){try{l&&e[h]&&(n=e[l],n&&(e[l]=null),f.event.triggered=h,e[h]())}catch(p){}n&&(e[l]=n),f.event.triggered=b}}return c.result}},handle:function(c){c=f.event.fix(c||a.event);var d=((f._data(this,"events")||{})[c.type]||[]).slice(0),e=!c.exclusive&&!c.namespace,g=Array.prototype.slice.call(arguments,0);g[0]=c,c.currentTarget=this;for(var h=0,i=d.length;h-1?f.map(a.options,function(a){return a.selected}).join("-"):"":f.nodeName(a,"select")&&(c=a.selectedIndex);return c},I=function(c){var d=c.target,e,g;if(!!x.test(d.nodeName)&&!d.readOnly){e=f._data(d,"_change_data"),g=H(d),(c.type!=="focusout"||d.type!=="radio")&&f._data(d,"_change_data",g);if(e===b||g===e)return;if(e!=null||g)c.type="change",c.liveFired=b,f.event.trigger(c,arguments[1],d)}};f.event.special.change={filters:{focusout:I,beforedeactivate:I,click:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(c==="radio"||c==="checkbox"||f.nodeName(b,"select"))&&I.call(this,a)},keydown:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(a.keyCode===13&&!f.nodeName(b,"textarea")||a.keyCode===32&&(c==="checkbox"||c==="radio")||c==="select-multiple")&&I.call(this,a)},beforeactivate:function(a){var b=a.target;f._data(b,"_change_data",H(b))}},setup:function(a,b){if(this.type==="file")return!1;for(var c in G)f.event.add(this,c+".specialChange",G[c]);return x.test(this.nodeName)},teardown:function(a){f.event.remove(this,".specialChange");return x.test(this.nodeName)}},G=f.event.special.change.filters,G.focus=G.beforeactivate}f.support.focusinBubbles||f.each({focus:"focusin",blur:"focusout"},function(a,b){function e(a){var c=f.event.fix(a);c.type=b,c.originalEvent={},f.event.trigger(c,null,c.target),c.isDefaultPrevented()&&a.preventDefault()}var d=0;f.event.special[b]={setup:function(){d++===0&&c.addEventListener(a,e,!0)},teardown:function(){--d===0&&c.removeEventListener(a,e,!0)}}}),f.each(["bind","one"],function(a,c){f.fn[c]=function(a,d,e){var g;if(typeof a=="object"){for(var h in a)this[c](h,d,a[h],e);return this}if(arguments.length===2||d===!1)e=d,d=b;c==="one"?(g=function(a){f(this).unbind(a,g);return e.apply(this,arguments)},g.guid=e.guid||f.guid++):g=e;if(a==="unload"&&c!=="one")this.one(a,d,e);else for(var i=0,j=this.length;i0?this.bind(b,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0)}),function(){function u(a,b,c,d,e,f){for(var g=0,h=d.length;g0){j=i;break}}i=i[a]}d[g]=j}}}function t(a,b,c,d,e,f){for(var g=0,h=d.length;g+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d=0,e=Object.prototype.toString,g=!1,h=!0,i=/\\/g,j=/\W/;[0,0].sort(function(){h=!1;return 0});var k=function(b,d,f,g){f=f||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return f;var i,j,n,o,q,r,s,t,u=!0,w=k.isXML(d),x=[],y=b;do{a.exec(""),i=a.exec(y);if(i){y=i[3],x.push(i[1]);if(i[2]){o=i[3];break}}}while(i);if(x.length>1&&m.exec(b))if(x.length===2&&l.relative[x[0]])j=v(x[0]+x[1],d);else{j=l.relative[x[0]]?[d]:k(x.shift(),d);while(x.length)b=x.shift(),l.relative[b]&&(b+=x.shift()),j=v(b,j)}else{!g&&x.length>1&&d.nodeType===9&&!w&&l.match.ID.test(x[0])&&!l.match.ID.test(x[x.length-1])&&(q=k.find(x.shift(),d,w),d=q.expr?k.filter(q.expr,q.set)[0]:q.set[0]);if(d){q=g?{expr:x.pop(),set:p(g)}:k.find(x.pop(),x.length===1&&(x[0]==="~"||x[0]==="+")&&d.parentNode?d.parentNode:d,w),j=q.expr?k.filter(q.expr,q.set):q.set,x.length>0?n=p(j):u=!1;while(x.length)r=x.pop(),s=r,l.relative[r]?s=x.pop():r="",s==null&&(s=d),l.relative[r](n,s,w)}else n=x=[]}n||(n=j),n||k.error(r||b);if(e.call(n)==="[object Array]")if(!u)f.push.apply(f,n);else if(d&&d.nodeType===1)for(t=0;n[t]!=null;t++)n[t]&&(n[t]===!0||n[t].nodeType===1&&k.contains(d,n[t]))&&f.push(j[t]);else for(t=0;n[t]!=null;t++)n[t]&&n[t].nodeType===1&&f.push(j[t]);else p(n,f);o&&(k(o,h,f,g),k.uniqueSort(f));return f};k.uniqueSort=function(a){if(r){g=h,a.sort(r);if(g)for(var b=1;b0},k.find=function(a,b,c){var d;if(!a)return[];for(var e=0,f=l.order.length;e":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!j.test(b)){b=b.toLowerCase();for(;e=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(i,"")},TAG:function(a,b){return a[1].replace(i,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||k.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&k.error(a[0]);a[0]=d++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(i,"");!f&&l.attrMap[g]&&(a[1]=l.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(i,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=k(b[3],null,null,c);else{var g=k.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(l.match.POS.test(b[0])||l.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!k(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return bc[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=l.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||k.getText([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=l.attrHandle[c]?l.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=l.setFilters[e];if(f)return f(a,c,b,d)}}},m=l.match.POS,n=function(a,b){return"\\"+(b-0+1)};for(var o in l.match)l.match[o]=new RegExp(l.match[o].source+/(?![^\[]*\])(?![^\(]*\))/.source),l.leftMatch[o]=new RegExp(/(^(?:.|\r|\n)*?)/.source+l.match[o].source.replace(/\\(\d+)/g,n));var p=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(q){p=function(a,b){var c=0,d=b||[];if(e.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var f=a.length;c",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(l.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},l.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(l.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(l.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=k,b=c.createElement("div"),d="__sizzle__";b.innerHTML="

";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){k=function(b,e,f,g){e=e||c;if(!g&&!k.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return p(e.getElementsByTagName(b),f);if(h[2]&&l.find.CLASS&&e.getElementsByClassName)return p(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return p([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return p([],f);if(i.id===h[3])return p([i],f)}try{return p(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var m=e,n=e.getAttribute("id"),o=n||d,q=e.parentNode,r=/^\s*[+~]/.test(b);n?o=o.replace(/'/g,"\\$&"):e.setAttribute("id",o),r&&q&&(e=e.parentNode);try{if(!r||q)return p(e.querySelectorAll("[id='"+o+"'] "+b),f)}catch(s){}finally{n||m.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)k[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}k.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!k.isXML(a))try{if(e||!l.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return k(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="
";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;l.order.splice(1,0,"CLASS"),l.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?k.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?k.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:k.contains=function(){return!1},k.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var v=function(a,b){var c,d=[],e="",f=b.nodeType?[b]:b;while(c=l.match.PSEUDO.exec(a))e+=c[0],a=a.replace(l.match.PSEUDO,"");a=l.relative[a]?a+"*":a;for(var g=0,h=f.length;g0)for(h=g;h0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h,i,j={},k=1;if(g&&a.length){for(d=0,e=a.length;d-1:f(g).is(h))&&c.push({selector:i,elem:g,level:k});g=g.parentNode,k++}}return c}var l=S.test(a)||typeof a!="string"?f(a,b||this.context):0;for(d=0,e=this.length;d-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a)return this[0]&&this[0].parentNode?this.prevAll().length:-1;if(typeof a=="string")return f.inArray(this[0],f(a));return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(U(c[0])||U(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c),g=R.call(arguments);N.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!T[a]?f.unique(e):e,(this.length>1||P.test(d))&&O.test(a)&&(e=e.reverse());return this.pushStack(e,a,g.join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var W=/ jQuery\d+="(?:\d+|null)"/g,X=/^\s+/,Y=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,Z=/<([\w:]+)/,$=/",""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]};be.optgroup=be.option,be.tbody=be.tfoot=be.colgroup=be.caption=be.thead,be.th=be.td,f.support.htmlSerialize||(be._default=[1,"div
","
"]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){f(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f(arguments[0]).toArray());return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(W,""):null;if(typeof a=="string"&&!ba.test(a)&&(f.support.leadingWhitespace||!X.test(a))&&!be[(Z.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Y,"<$1>");try{for(var c=0,d=this.length;c1&&l0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d=a.cloneNode(!0),e,g,h;if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bh(a,d),e=bi(a),g=bi(d);for(h=0;e[h];++h)g[h]&&bh(e[h],g[h])}if(b){bg(a,d);if(c){e=bi(a),g=bi(d);for(h=0;e[h];++h)bg(e[h],g[h])}}e=g=null;return d},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!_.test(k))k=b.createTextNode(k);else{k=k.replace(Y,"<$1>");var l=(Z.exec(k)||["",""])[1].toLowerCase(),m=be[l]||be._default,n=m[0],o=b.createElement("div");o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=$.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]===""&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&X.test(k)&&o.insertBefore(b.createTextNode(X.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return bn.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=f.isNaN(b)?"":"alpha(opacity="+b*100+")",g=d&&d.filter||c.filter||"";c.zoom=1;if(b>=1&&f.trim(g.replace(bm,""))===""){c.removeAttribute("filter");if(d&&!d.filter)return}c.filter=bm.test(g)?g.replace(bm,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bv(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(bw=function(a,c){var d,e,g;c=c.replace(bo,"-$1").toLowerCase();if(!(e=a.ownerDocument.defaultView))return b;if(g=e.getComputedStyle(a,null))d=g.getPropertyValue(c),d===""&&!f.contains(a.ownerDocument.documentElement,a)&&(d=f.style(a,c));return d}),c.documentElement.currentStyle&&(bx=function(a,b){var c,d=a.currentStyle&&a.currentStyle[b],e=a.runtimeStyle&&a.runtimeStyle[b],f=a.style;!bp.test(d)&&bq.test(d)&&(c=f.left,e&&(a.runtimeStyle.left=a.currentStyle.left),f.left=b==="fontSize"?"1em":d||0,d=f.pixelLeft+"px",f.left=c,e&&(a.runtimeStyle.left=e));return d===""?"auto":d}),bv=bw||bx,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bz=/%20/g,bA=/\[\]$/,bB=/\r?\n/g,bC=/#.*$/,bD=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bE=/^(?:color|date|datetime|datetime-local|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bF=/^(?:about|app|app\-storage|.+\-extension|file|res|widget):$/,bG=/^(?:GET|HEAD)$/,bH=/^\/\//,bI=/\?/,bJ=/)<[^<]*)*<\/script>/gi,bK=/^(?:select|textarea)/i,bL=/\s+/,bM=/([?&])_=[^&]*/,bN=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+))?)?/,bO=f.fn.load,bP={},bQ={},bR,bS,bT=["*/"]+["*"];try{bR=e.href}catch(bU){bR=c.createElement("a"),bR.href="",bR=bR.href}bS=bN.exec(bR.toLowerCase())||[],f.fn.extend({load:function(a,c,d){if(typeof a!="string"&&bO)return bO.apply(this,arguments);if(!this.length)return this;var e=a.indexOf(" ");if(e>=0){var g=a.slice(e,a.length);a=a.slice(0,e)}var h="GET";c&&(f.isFunction(c)?(d=c,c=b):typeof c=="object"&&(c=f.param(c,f.ajaxSettings.traditional),h="POST"));var i=this;f.ajax({url:a,type:h,dataType:"html",data:c,complete:function(a,b,c){c=a.responseText,a.isResolved()&&(a.done(function(a){c=a}),i.html(g?f("
").append(c.replace(bJ,"")).find(g):c)),d&&i.each(d,[c,b,a])}});return this},serialize:function(){return f.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?f.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||bK.test(this.nodeName)||bE.test(this.type))}).map(function(a,b){var c=f(this).val();return c==null?null:f.isArray(c)?f.map(c,function(a,c){return{name:b.name,value:a.replace(bB,"\r\n")}}):{name:b.name,value:c.replace(bB,"\r\n")}}).get()}}),f.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){f.fn[b]=function(a){return this.bind(b,a)}}),f.each(["get","post"],function(a,c){f[c]=function(a,d,e,g){f.isFunction(d)&&(g=g||e,e=d,d=b);return f.ajax({type:c,url:a,data:d,success:e,dataType:g})}}),f.extend({getScript:function(a,c){return f.get(a,b,c,"script")},getJSON:function(a,b,c){return f.get(a,b,c,"json")},ajaxSetup:function(a,b){b?bX(a,f.ajaxSettings):(b=a,a=f.ajaxSettings),bX(a,b);return a},ajaxSettings:{url:bR,isLocal:bF.test(bS[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":bT},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":f.parseJSON,"text xml":f.parseXML},flatOptions:{context:!0,url:!0}},ajaxPrefilter:bV(bP),ajaxTransport:bV(bQ),ajax:function(a,c){function w(a,c,l,m){if(s!==2){s=2,q&&clearTimeout(q),p=b,n=m||"",v.readyState=a>0?4:0;var o,r,u,w=c,x=l?bZ(d,v,l):b,y,z;if(a>=200&&a<300||a===304){if(d.ifModified){if(y=v.getResponseHeader("Last-Modified"))f.lastModified[k]=y;if(z=v.getResponseHeader("Etag"))f.etag[k]=z}if(a===304)w="notmodified",o=!0;else try{r=b$(d,x),w="success",o=!0}catch(A){w="parsererror",u=A}}else{u=w;if(!w||a)w="error",a<0&&(a=0)}v.status=a,v.statusText=""+(c||w),o?h.resolveWith(e,[r,w,v]):h.rejectWith(e,[v,w,u]),v.statusCode(j),j=b,t&&g.trigger("ajax"+(o?"Success":"Error"),[v,d,o?r:u]),i.resolveWith(e,[v,w]),t&&(g.trigger("ajaxComplete",[v,d]),--f.active||f.event.trigger("ajaxStop"))}}typeof a=="object"&&(c=a,a=b),c=c||{};var d=f.ajaxSetup({},c),e=d.context||d,g=e!==d&&(e.nodeType||e instanceof f)?f(e):f.event,h=f.Deferred(),i=f._Deferred(),j=d.statusCode||{},k,l={},m={},n,o,p,q,r,s=0,t,u,v={readyState:0,setRequestHeader:function(a,b){if(!s){var c=a.toLowerCase();a=m[c]=m[c]||a,l[a]=b}return this},getAllResponseHeaders:function(){return s===2?n:null},getResponseHeader:function(a){var c;if(s===2){if(!o){o={};while(c=bD.exec(n))o[c[1].toLowerCase()]=c[2]}c=o[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){s||(d.mimeType=a);return this},abort:function(a){a=a||"abort",p&&p.abort(a),w(0,a);return this}};h.promise(v),v.success=v.done,v.error=v.fail,v.complete=i.done,v.statusCode=function(a){if(a){var b;if(s<2)for(b in a)j[b]=[j[b],a[b]];else b=a[v.status],v.then(b,b)}return this},d.url=((a||d.url)+"").replace(bC,"").replace(bH,bS[1]+"//"),d.dataTypes=f.trim(d.dataType||"*").toLowerCase().split(bL),d.crossDomain==null&&(r=bN.exec(d.url.toLowerCase()),d.crossDomain=!(!r||r[1]==bS[1]&&r[2]==bS[2]&&(r[3]||(r[1]==="http:"?80:443))==(bS[3]||(bS[1]==="http:"?80:443)))),d.data&&d.processData&&typeof d.data!="string"&&(d.data=f.param(d.data,d.traditional)),bW(bP,d,c,v);if(s===2)return!1;t=d.global,d.type=d.type.toUpperCase(),d.hasContent=!bG.test(d.type),t&&f.active++===0&&f.event.trigger("ajaxStart");if(!d.hasContent){d.data&&(d.url+=(bI.test(d.url)?"&":"?")+d.data,delete d.data),k=d.url;if(d.cache===!1){var x=f.now(),y=d.url.replace(bM,"$1_="+x);d.url=y+(y===d.url?(bI.test(d.url)?"&":"?")+"_="+x:"")}}(d.data&&d.hasContent&&d.contentType!==!1||c.contentType)&&v.setRequestHeader("Content-Type",d.contentType),d.ifModified&&(k=k||d.url,f.lastModified[k]&&v.setRequestHeader("If-Modified-Since",f.lastModified[k]),f.etag[k]&&v.setRequestHeader("If-None-Match",f.etag[k])),v.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+(d.dataTypes[0]!=="*"?", "+bT+"; q=0.01":""):d.accepts["*"]);for(u in d.headers)v.setRequestHeader(u,d.headers[u]);if(d.beforeSend&&(d.beforeSend.call(e,v,d)===!1||s===2)){v.abort();return!1}for(u in{success:1,error:1,complete:1})v[u](d[u]);p=bW(bQ,d,c,v);if(!p)w(-1,"No Transport");else{v.readyState=1,t&&g.trigger("ajaxSend",[v,d]),d.async&&d.timeout>0&&(q=setTimeout(function(){v.abort("timeout")},d.timeout));try{s=1,p.send(l,w)}catch(z){s<2?w(-1,z):f.error(z)}}return v},param:function(a,c){var d=[],e=function(a,b){b=f.isFunction(b)?b():b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=f.ajaxSettings.traditional);if(f.isArray(a)||a.jquery&&!f.isPlainObject(a))f.each(a,function(){e(this.name,this.value)});else for(var g in a)bY(g,a[g],c,e);return d.join("&").replace(bz,"+")}}),f.extend({active:0,lastModified:{},etag:{}});var b_=f.now(),ca=/(\=)\?(&|$)|\?\?/i;f.ajaxSetup({jsonp:"callback",jsonpCallback:function(){return f.expando+"_"+b_++}}),f.ajaxPrefilter("json jsonp",function(b,c,d){var e=b.contentType==="application/x-www-form-urlencoded"&&typeof b.data=="string";if(b.dataTypes[0]==="jsonp"||b.jsonp!==!1&&(ca.test(b.url)||e&&ca.test(b.data))){var g,h=b.jsonpCallback=f.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,i=a[h],j=b.url,k=b.data,l="$1"+h+"$2";b.jsonp!==!1&&(j=j.replace(ca,l),b.url===j&&(e&&(k=k.replace(ca,l)),b.data===k&&(j+=(/\?/.test(j)?"&":"?")+b.jsonp+"="+h))),b.url=j,b.data=k,a[h]=function(a){g=[a]},d.always(function(){a[h]=i,g&&f.isFunction(i)&&a[h](g[0])}),b.converters["script json"]=function(){g||f.error(h+" was not called");return g[0]},b.dataTypes[0]="json";return"script"}}),f.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){f.globalEval(a);return a}}}),f.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),f.ajaxTransport("script",function(a){if(a.crossDomain){var d,e=c.head||c.getElementsByTagName("head")[0]||c.documentElement;return{send:function(f,g){d=c.createElement("script"),d.async="async",a.scriptCharset&&(d.charset=a.scriptCharset),d.src=a.url,d.onload=d.onreadystatechange=function(a,c){if(c||!d.readyState||/loaded|complete/.test(d.readyState))d.onload=d.onreadystatechange=null,e&&d.parentNode&&e.removeChild(d),d=b,c||g(200,"success")},e.insertBefore(d,e.firstChild)},abort:function(){d&&d.onload(0,1)}}}});var cb=a.ActiveXObject?function(){for(var a in cd)cd[a](0,1)}:!1,cc=0,cd;f.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&ce()||cf()}:ce,function(a){f.extend(f.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(f.ajaxSettings.xhr()),f.support.ajax&&f.ajaxTransport(function(c){if(!c.crossDomain||f.support.cors){var d;return{send:function(e,g){var h=c.xhr(),i,j;c.username?h.open(c.type,c.url,c.async,c.username,c.password):h.open(c.type,c.url,c.async);if(c.xhrFields)for(j in c.xhrFields)h[j]=c.xhrFields[j];c.mimeType&&h.overrideMimeType&&h.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(j in e)h.setRequestHeader(j,e[j])}catch(k){}h.send(c.hasContent&&c.data||null),d=function(a,e){var j,k,l,m,n;try{if(d&&(e||h.readyState===4)){d=b,i&&(h.onreadystatechange=f.noop,cb&&delete cd[i]);if(e)h.readyState!==4&&h.abort();else{j=h.status,l=h.getAllResponseHeaders(),m={},n=h.responseXML,n&&n.documentElement&&(m.xml=n),m.text=h.responseText;try{k=h.statusText}catch(o){k=""}!j&&c.isLocal&&!c.crossDomain?j=m.text?200:404:j===1223&&(j=204)}}}catch(p){e||g(-1,p)}m&&g(j,k,m,l)},!c.async||h.readyState===4?d():(i=++cc,cb&&(cd||(cd={},f(a).unload(cb)),cd[i]=d),h.onreadystatechange=d)},abort:function(){d&&d(0,1)}}}});var cg={},ch,ci,cj=/^(?:toggle|show|hide)$/,ck=/^([+\-]=)?([\d+.\-]+)([a-z%]*)$/i,cl,cm=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]],cn;f.fn.extend({show:function(a,b,c){var d,e;if(a||a===0)return this.animate(cq("show",3),a,b,c);for(var g=0,h=this.length;g=e.duration+this.startTime){this.now=this.end,this.pos=this.state=1,this.update(),e.animatedProperties[this.prop]=!0;for(g in e.animatedProperties)e.animatedProperties[g]!==!0&&(c=!1);if(c){e.overflow!=null&&!f.support.shrinkWrapBlocks&&f.each(["","X","Y"],function(a,b){d.style["overflow"+b]=e.overflow[a]}),e.hide&&f(d).hide();if(e.hide||e.show)for(var i in e.animatedProperties)f.style(d,i,e.orig[i]);e.complete.call(d)}return!1}e.duration==Infinity?this.now=b:(h=b-this.startTime,this.state=h/e.duration,this.pos=f.easing[e.animatedProperties[this.prop]](this.state,h,0,1,e.duration),this.now=this.start+(this.end-this.start)*this.pos),this.update();return!0}},f.extend(f.fx,{tick:function(){for(var a=f.timers,b=0;b
";f.extend(b.style,{position:"absolute",top:0,left:0,margin:0,border:0,width:"1px",height:"1px",visibility:"hidden"}),b.innerHTML=j,a.insertBefore(b,a.firstChild),d=b.firstChild,e=d.firstChild,h=d.nextSibling.firstChild.firstChild,this.doesNotAddBorder=e.offsetTop!==5,this.doesAddBorderForTableAndCells=h.offsetTop===5,e.style.position="fixed",e.style.top="20px",this.supportsFixedPosition=e.offsetTop===20||e.offsetTop===15,e.style.position=e.style.top="",d.style.overflow="hidden",d.style.position="relative",this.subtractsBorderForOverflowNotVisible=e.offsetTop===-5,this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==i,a.removeChild(b),f.offset.initialize=f.noop},bodyOffset:function(a){var b=a.offsetTop,c=a.offsetLeft;f.offset.initialize(),f.offset.doesNotIncludeMarginInBodyOffset&&(b+=parseFloat(f.css(a,"marginTop"))||0,c+=parseFloat(f.css(a,"marginLeft"))||0);return{top:b,left:c}},setOffset:function(a,b,c){var d=f.css(a,"position");d==="static"&&(a.style.position="relative");var e=f(a),g=e.offset(),h=f.css(a,"top"),i=f.css(a,"left"),j=(d==="absolute"||d==="fixed")&&f.inArray("auto",[h,i])>-1,k={},l={},m,n;j?(l=e.position(),m=l.top,n=l.left):(m=parseFloat(h)||0,n=parseFloat(i)||0),f.isFunction(b)&&(b=b.call(a,c,g)),b.top!=null&&(k.top=b.top-g.top+m),b.left!=null&&(k.left=b.left-g.left+n),"using"in b?b.using.call(a,k):e.css(k)}},f.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),c=this.offset(),d=ct.test(b[0].nodeName)?{top:0,left:0}:b.offset();c.top-=parseFloat(f.css(a,"marginTop"))||0,c.left-=parseFloat(f.css(a,"marginLeft"))||0,d.top+=parseFloat(f.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(f.css(b[0],"borderLeftWidth"))||0;return{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||c.body;while(a&&!ct.test(a.nodeName)&&f.css(a,"position")==="static")a=a.offsetParent;return a})}}),f.each(["Left","Top"],function(a,c){var d="scroll"+c;f.fn[d]=function(c){var e,g;if(c===b){e=this[0];if(!e)return null;g=cu(e);return g?"pageXOffset"in g?g[a?"pageYOffset":"pageXOffset"]:f.support.boxModel&&g.document.documentElement[d]||g.document.body[d]:e[d]}return this.each(function(){g=cu(this),g?g.scrollTo(a?f(g).scrollLeft():c,a?c:f(g).scrollTop()):this[d]=c})}}),f.each(["Height","Width"],function(a,c){var d=c.toLowerCase();f.fn["inner"+c]=function(){var a=this[0];return a&&a.style?parseFloat(f.css(a,d,"padding")):null},f.fn["outer"+c]=function(a){var b=this[0];return b&&b.style?parseFloat(f.css(b,d,a?"margin":"border")):null},f.fn[d]=function(a){var e=this[0];if(!e)return a==null?null:this;if(f.isFunction(a))return this.each(function(b){var c=f(this);c[d](a.call(this,b,c[d]()))});if(f.isWindow(e)){var g=e.document.documentElement["client"+c],h=e.document.body;return e.document.compatMode==="CSS1Compat"&&g||h&&h["client"+c]||g}if(e.nodeType===9)return Math.max(e.documentElement["client"+c],e.body["scroll"+c],e.documentElement["scroll"+c],e.body["offset"+c],e.documentElement["offset"+c]);if(a===b){var i=f.css(e,d),j=parseFloat(i);return f.isNaN(j)?i:j}return this.css(d,typeof a=="string"?a:a+"px")}}),a.jQuery=a.$=f})(window); \ No newline at end of file diff --git a/application/media/js/jquery.jstree-1.0rc3.js b/application/media/js/jquery.jstree-1.0rc3.js new file mode 100644 index 0000000..21fc01b --- /dev/null +++ b/application/media/js/jquery.jstree-1.0rc3.js @@ -0,0 +1,4551 @@ +/* + * jsTree 1.0-rc3 + * http://jstree.com/ + * + * Copyright (c) 2010 Ivan Bozhanov (vakata.com) + * + * Licensed same as jquery - under the terms of either the MIT License or the GPL Version 2 License + * http://www.opensource.org/licenses/mit-license.php + * http://www.gnu.org/licenses/gpl.html + * + * $Date: 2011-02-09 01:17:14 +0200 (ср, 09 февр 2011) $ + * $Revision: 236 $ + */ + +/*jslint browser: true, onevar: true, undef: true, bitwise: true, strict: true */ +/*global window : false, clearInterval: false, clearTimeout: false, document: false, setInterval: false, setTimeout: false, jQuery: false, navigator: false, XSLTProcessor: false, DOMParser: false, XMLSerializer: false*/ + +"use strict"; + +// top wrapper to prevent multiple inclusion (is this OK?) +(function () { if(jQuery && jQuery.jstree) { return; } + var is_ie6 = false, is_ie7 = false, is_ff2 = false; + +/* + * jsTree core + */ +(function ($) { + // Common functions not related to jsTree + // decided to move them to a `vakata` "namespace" + $.vakata = {}; + // CSS related functions + $.vakata.css = { + get_css : function(rule_name, delete_flag, sheet) { + rule_name = rule_name.toLowerCase(); + var css_rules = sheet.cssRules || sheet.rules, + j = 0; + do { + if(css_rules.length && j > css_rules.length + 5) { return false; } + if(css_rules[j].selectorText && css_rules[j].selectorText.toLowerCase() == rule_name) { + if(delete_flag === true) { + if(sheet.removeRule) { sheet.removeRule(j); } + if(sheet.deleteRule) { sheet.deleteRule(j); } + return true; + } + else { return css_rules[j]; } + } + } + while (css_rules[++j]); + return false; + }, + add_css : function(rule_name, sheet) { + if($.jstree.css.get_css(rule_name, false, sheet)) { return false; } + if(sheet.insertRule) { sheet.insertRule(rule_name + ' { }', 0); } else { sheet.addRule(rule_name, null, 0); } + return $.vakata.css.get_css(rule_name); + }, + remove_css : function(rule_name, sheet) { + return $.vakata.css.get_css(rule_name, true, sheet); + }, + add_sheet : function(opts) { + var tmp = false, is_new = true; + if(opts.str) { + if(opts.title) { tmp = $("style[id='" + opts.title + "-stylesheet']")[0]; } + if(tmp) { is_new = false; } + else { + tmp = document.createElement("style"); + tmp.setAttribute('type',"text/css"); + if(opts.title) { tmp.setAttribute("id", opts.title + "-stylesheet"); } + } + if(tmp.styleSheet) { + if(is_new) { + document.getElementsByTagName("head")[0].appendChild(tmp); + tmp.styleSheet.cssText = opts.str; + } + else { + tmp.styleSheet.cssText = tmp.styleSheet.cssText + " " + opts.str; + } + } + else { + tmp.appendChild(document.createTextNode(opts.str)); + document.getElementsByTagName("head")[0].appendChild(tmp); + } + return tmp.sheet || tmp.styleSheet; + } + if(opts.url) { + if(document.createStyleSheet) { + try { tmp = document.createStyleSheet(opts.url); } catch (e) { } + } + else { + tmp = document.createElement('link'); + tmp.rel = 'stylesheet'; + tmp.type = 'text/css'; + tmp.media = "all"; + tmp.href = opts.url; + document.getElementsByTagName("head")[0].appendChild(tmp); + return tmp.styleSheet; + } + } + } + }; + + // private variables + var instances = [], // instance array (used by $.jstree.reference/create/focused) + focused_instance = -1, // the index in the instance array of the currently focused instance + plugins = {}, // list of included plugins + prepared_move = {}; // for the move_node function + + // jQuery plugin wrapper (thanks to jquery UI widget function) + $.fn.jstree = function (settings) { + var isMethodCall = (typeof settings == 'string'), // is this a method call like $().jstree("open_node") + args = Array.prototype.slice.call(arguments, 1), + returnValue = this; + + // if a method call execute the method on all selected instances + if(isMethodCall) { + if(settings.substring(0, 1) == '_') { return returnValue; } + this.each(function() { + var instance = instances[$.data(this, "jstree_instance_id")], + methodValue = (instance && $.isFunction(instance[settings])) ? instance[settings].apply(instance, args) : instance; + if(typeof methodValue !== "undefined" && (settings.indexOf("is_") === 0 || (methodValue !== true && methodValue !== false))) { returnValue = methodValue; return false; } + }); + } + else { + this.each(function() { + // extend settings and allow for multiple hashes and $.data + var instance_id = $.data(this, "jstree_instance_id"), + a = [], + b = settings ? $.extend({}, true, settings) : {}, + c = $(this), + s = false, + t = []; + a = a.concat(args); + if(c.data("jstree")) { a.push(c.data("jstree")); } + b = a.length ? $.extend.apply(null, [true, b].concat(a)) : b; + + // if an instance already exists, destroy it first + if(typeof instance_id !== "undefined" && instances[instance_id]) { instances[instance_id].destroy(); } + // push a new empty object to the instances array + instance_id = parseInt(instances.push({}),10) - 1; + // store the jstree instance id to the container element + $.data(this, "jstree_instance_id", instance_id); + // clean up all plugins + b.plugins = $.isArray(b.plugins) ? b.plugins : $.jstree.defaults.plugins.slice(); + b.plugins.unshift("core"); + // only unique plugins + b.plugins = b.plugins.sort().join(",,").replace(/(,|^)([^,]+)(,,\2)+(,|$)/g,"$1$2$4").replace(/,,+/g,",").replace(/,$/,"").split(","); + + // extend defaults with passed data + s = $.extend(true, {}, $.jstree.defaults, b); + s.plugins = b.plugins; + $.each(plugins, function (i, val) { + if($.inArray(i, s.plugins) === -1) { s[i] = null; delete s[i]; } + else { t.push(i); } + }); + s.plugins = t; + + // push the new object to the instances array (at the same time set the default classes to the container) and init + instances[instance_id] = new $.jstree._instance(instance_id, $(this).addClass("jstree jstree-" + instance_id), s); + // init all activated plugins for this instance + $.each(instances[instance_id]._get_settings().plugins, function (i, val) { instances[instance_id].data[val] = {}; }); + $.each(instances[instance_id]._get_settings().plugins, function (i, val) { if(plugins[val]) { plugins[val].__init.apply(instances[instance_id]); } }); + // initialize the instance + setTimeout(function() { if(instances[instance_id]) { instances[instance_id].init(); } }, 0); + }); + } + // return the jquery selection (or if it was a method call that returned a value - the returned value) + return returnValue; + }; + // object to store exposed functions and objects + $.jstree = { + defaults : { + plugins : [] + }, + _focused : function () { return instances[focused_instance] || null; }, + _reference : function (needle) { + // get by instance id + if(instances[needle]) { return instances[needle]; } + // get by DOM (if still no luck - return null + var o = $(needle); + if(!o.length && typeof needle === "string") { o = $("#" + needle); } + if(!o.length) { return null; } + return instances[o.closest(".jstree").data("jstree_instance_id")] || null; + }, + _instance : function (index, container, settings) { + // for plugins to store data in + this.data = { core : {} }; + this.get_settings = function () { return $.extend(true, {}, settings); }; + this._get_settings = function () { return settings; }; + this.get_index = function () { return index; }; + this.get_container = function () { return container; }; + this.get_container_ul = function () { return container.children("ul:eq(0)"); }; + this._set_settings = function (s) { + settings = $.extend(true, {}, settings, s); + }; + }, + _fn : { }, + plugin : function (pname, pdata) { + pdata = $.extend({}, { + __init : $.noop, + __destroy : $.noop, + _fn : {}, + defaults : false + }, pdata); + plugins[pname] = pdata; + + $.jstree.defaults[pname] = pdata.defaults; + $.each(pdata._fn, function (i, val) { + val.plugin = pname; + val.old = $.jstree._fn[i]; + $.jstree._fn[i] = function () { + var rslt, + func = val, + args = Array.prototype.slice.call(arguments), + evnt = new $.Event("before.jstree"), + rlbk = false; + + if(this.data.core.locked === true && i !== "unlock" && i !== "is_locked") { return; } + + // Check if function belongs to the included plugins of this instance + do { + if(func && func.plugin && $.inArray(func.plugin, this._get_settings().plugins) !== -1) { break; } + func = func.old; + } while(func); + if(!func) { return; } + + // context and function to trigger events, then finally call the function + if(i.indexOf("_") === 0) { + rslt = func.apply(this, args); + } + else { + rslt = this.get_container().triggerHandler(evnt, { "func" : i, "inst" : this, "args" : args, "plugin" : func.plugin }); + if(rslt === false) { return; } + if(typeof rslt !== "undefined") { args = rslt; } + + rslt = func.apply( + $.extend({}, this, { + __callback : function (data) { + this.get_container().triggerHandler( i + '.jstree', { "inst" : this, "args" : args, "rslt" : data, "rlbk" : rlbk }); + }, + __rollback : function () { + rlbk = this.get_rollback(); + return rlbk; + }, + __call_old : function (replace_arguments) { + return func.old.apply(this, (replace_arguments ? Array.prototype.slice.call(arguments, 1) : args ) ); + } + }), args); + } + + // return the result + return rslt; + }; + $.jstree._fn[i].old = val.old; + $.jstree._fn[i].plugin = pname; + }); + }, + rollback : function (rb) { + if(rb) { + if(!$.isArray(rb)) { rb = [ rb ]; } + $.each(rb, function (i, val) { + instances[val.i].set_rollback(val.h, val.d); + }); + } + } + }; + // set the prototype for all instances + $.jstree._fn = $.jstree._instance.prototype = {}; + + // load the css when DOM is ready + $(function() { + // code is copied from jQuery ($.browser is deprecated + there is a bug in IE) + var u = navigator.userAgent.toLowerCase(), + v = (u.match( /.+?(?:rv|it|ra|ie)[\/: ]([\d.]+)/ ) || [0,'0'])[1], + css_string = '' + + '.jstree ul, .jstree li { display:block; margin:0 0 0 0; padding:0 0 0 0; list-style-type:none; } ' + + '.jstree li { display:block; min-height:18px; line-height:18px; white-space:nowrap; margin-left:18px; min-width:18px; } ' + + '.jstree-rtl li { margin-left:0; margin-right:18px; } ' + + '.jstree > ul > li { margin-left:0px; } ' + + '.jstree-rtl > ul > li { margin-right:0px; } ' + + '.jstree ins { display:inline-block; text-decoration:none; width:18px; height:18px; margin:0 0 0 0; padding:0; } ' + + '.jstree a { display:inline-block; line-height:16px; height:16px; color:black; white-space:nowrap; text-decoration:none; padding:1px 2px; margin:0; } ' + + '.jstree a:focus { outline: none; } ' + + '.jstree a > ins { height:16px; width:16px; } ' + + '.jstree a > .jstree-icon { margin-right:3px; } ' + + '.jstree-rtl a > .jstree-icon { margin-left:3px; margin-right:0; } ' + + 'li.jstree-open > ul { display:block; } ' + + 'li.jstree-closed > ul { display:none; } '; + // Correct IE 6 (does not support the > CSS selector) + if(/msie/.test(u) && parseInt(v, 10) == 6) { + is_ie6 = true; + + // fix image flicker and lack of caching + try { + document.execCommand("BackgroundImageCache", false, true); + } catch (err) { } + + css_string += '' + + '.jstree li { height:18px; margin-left:0; margin-right:0; } ' + + '.jstree li li { margin-left:18px; } ' + + '.jstree-rtl li li { margin-left:0px; margin-right:18px; } ' + + 'li.jstree-open ul { display:block; } ' + + 'li.jstree-closed ul { display:none !important; } ' + + '.jstree li a { display:inline; border-width:0 !important; padding:0px 2px !important; } ' + + '.jstree li a ins { height:16px; width:16px; margin-right:3px; } ' + + '.jstree-rtl li a ins { margin-right:0px; margin-left:3px; } '; + } + // Correct IE 7 (shifts anchor nodes onhover) + if(/msie/.test(u) && parseInt(v, 10) == 7) { + is_ie7 = true; + css_string += '.jstree li a { border-width:0 !important; padding:0px 2px !important; } '; + } + // correct ff2 lack of display:inline-block + if(!/compatible/.test(u) && /mozilla/.test(u) && parseFloat(v, 10) < 1.9) { + is_ff2 = true; + css_string += '' + + '.jstree ins { display:-moz-inline-box; } ' + + '.jstree li { line-height:12px; } ' + // WHY?? + '.jstree a { display:-moz-inline-box; } ' + + '.jstree .jstree-no-icons .jstree-checkbox { display:-moz-inline-stack !important; } '; + /* this shouldn't be here as it is theme specific */ + } + // the default stylesheet + $.vakata.css.add_sheet({ str : css_string, title : "jstree" }); + }); + + // core functions (open, close, create, update, delete) + $.jstree.plugin("core", { + __init : function () { + this.data.core.locked = false; + this.data.core.to_open = this.get_settings().core.initially_open; + this.data.core.to_load = this.get_settings().core.initially_load; + }, + defaults : { + html_titles : false, + animation : 500, + initially_open : [], + initially_load : [], + open_parents : true, + notify_plugins : true, + rtl : false, + load_open : false, + strings : { + loading : "Loading ...", + new_node : "New node", + multiple_selection : "Multiple selection" + } + }, + _fn : { + init : function () { + this.set_focus(); + if(this._get_settings().core.rtl) { + this.get_container().addClass("jstree-rtl").css("direction", "rtl"); + } + this.get_container().html(""); + this.data.core.li_height = this.get_container_ul().find("li.jstree-closed, li.jstree-leaf").eq(0).height() || 18; + + this.get_container() + .delegate("li > ins", "click.jstree", $.proxy(function (event) { + var trgt = $(event.target); + // if(trgt.is("ins") && event.pageY - trgt.offset().top < this.data.core.li_height) { this.toggle_node(trgt); } + this.toggle_node(trgt); + }, this)) + .bind("mousedown.jstree", $.proxy(function () { + this.set_focus(); // This used to be setTimeout(set_focus,0) - why? + }, this)) + .bind("dblclick.jstree", function (event) { + var sel; + if(document.selection && document.selection.empty) { document.selection.empty(); } + else { + if(window.getSelection) { + sel = window.getSelection(); + try { + sel.removeAllRanges(); + sel.collapse(); + } catch (err) { } + } + } + }); + if(this._get_settings().core.notify_plugins) { + this.get_container() + .bind("load_node.jstree", $.proxy(function (e, data) { + var o = this._get_node(data.rslt.obj), + t = this; + if(o === -1) { o = this.get_container_ul(); } + if(!o.length) { return; } + o.find("li").each(function () { + var th = $(this); + if(th.data("jstree")) { + $.each(th.data("jstree"), function (plugin, values) { + if(t.data[plugin] && $.isFunction(t["_" + plugin + "_notify"])) { + t["_" + plugin + "_notify"].call(t, th, values); + } + }); + } + }); + }, this)); + } + if(this._get_settings().core.load_open) { + this.get_container() + .bind("load_node.jstree", $.proxy(function (e, data) { + var o = this._get_node(data.rslt.obj), + t = this; + if(o === -1) { o = this.get_container_ul(); } + if(!o.length) { return; } + o.find("li.jstree-open:not(:has(ul))").each(function () { + t.load_node(this, $.noop, $.noop); + }); + }, this)); + } + this.__callback(); + this.load_node(-1, function () { this.loaded(); this.reload_nodes(); }); + }, + destroy : function () { + var i, + n = this.get_index(), + s = this._get_settings(), + _this = this; + + $.each(s.plugins, function (i, val) { + try { plugins[val].__destroy.apply(_this); } catch(err) { } + }); + this.__callback(); + // set focus to another instance if this one is focused + if(this.is_focused()) { + for(i in instances) { + if(instances.hasOwnProperty(i) && i != n) { + instances[i].set_focus(); + break; + } + } + } + // if no other instance found + if(n === focused_instance) { focused_instance = -1; } + // remove all traces of jstree in the DOM (only the ones set using jstree*) and cleans all events + this.get_container() + .unbind(".jstree") + .undelegate(".jstree") + .removeData("jstree_instance_id") + .find("[class^='jstree']") + .andSelf() + .attr("class", function () { return this.className.replace(/jstree[^ ]*|$/ig,''); }); + $(document) + .unbind(".jstree-" + n) + .undelegate(".jstree-" + n); + // remove the actual data + instances[n] = null; + delete instances[n]; + }, + + _core_notify : function (n, data) { + if(data.opened) { + this.open_node(n, false, true); + } + }, + + lock : function () { + this.data.core.locked = true; + this.get_container().children("ul").addClass("jstree-locked").css("opacity","0.7"); + this.__callback({}); + }, + unlock : function () { + this.data.core.locked = false; + this.get_container().children("ul").removeClass("jstree-locked").css("opacity","1"); + this.__callback({}); + }, + is_locked : function () { return this.data.core.locked; }, + save_opened : function () { + var _this = this; + this.data.core.to_open = []; + this.get_container_ul().find("li.jstree-open").each(function () { + if(this.id) { _this.data.core.to_open.push("#" + this.id.toString().replace(/^#/,"").replace(/\\\//g,"/").replace(/\//g,"\\\/").replace(/\\\./g,".").replace(/\./g,"\\.").replace(/\:/g,"\\:")); } + }); + this.__callback(_this.data.core.to_open); + }, + save_loaded : function () { }, + reload_nodes : function (is_callback) { + var _this = this, + done = true, + current = [], + remaining = []; + if(!is_callback) { + this.data.core.reopen = false; + this.data.core.refreshing = true; + this.data.core.to_open = $.map($.makeArray(this.data.core.to_open), function (n) { return "#" + n.toString().replace(/^#/,"").replace(/\\\//g,"/").replace(/\//g,"\\\/").replace(/\\\./g,".").replace(/\./g,"\\.").replace(/\:/g,"\\:"); }); + this.data.core.to_load = $.map($.makeArray(this.data.core.to_load), function (n) { return "#" + n.toString().replace(/^#/,"").replace(/\\\//g,"/").replace(/\//g,"\\\/").replace(/\\\./g,".").replace(/\./g,"\\.").replace(/\:/g,"\\:"); }); + if(this.data.core.to_open.length) { + this.data.core.to_load = this.data.core.to_load.concat(this.data.core.to_open); + } + } + if(this.data.core.to_load.length) { + $.each(this.data.core.to_load, function (i, val) { + if(val == "#") { return true; } + if($(val).length) { current.push(val); } + else { remaining.push(val); } + }); + if(current.length) { + this.data.core.to_load = remaining; + $.each(current, function (i, val) { + if(!_this._is_loaded(val)) { + _this.load_node(val, function () { _this.reload_nodes(true); }, function () { _this.reload_nodes(true); }); + done = false; + } + }); + } + } + if(this.data.core.to_open.length) { + $.each(this.data.core.to_open, function (i, val) { + _this.open_node(val, false, true); + }); + } + if(done) { + // TODO: find a more elegant approach to syncronizing returning requests + if(this.data.core.reopen) { clearTimeout(this.data.core.reopen); } + this.data.core.reopen = setTimeout(function () { _this.__callback({}, _this); }, 50); + this.data.core.refreshing = false; + this.reopen(); + } + }, + reopen : function () { + var _this = this; + if(this.data.core.to_open.length) { + $.each(this.data.core.to_open, function (i, val) { + _this.open_node(val, false, true); + }); + } + this.__callback({}); + }, + refresh : function (obj) { + var _this = this; + this.save_opened(); + if(!obj) { obj = -1; } + obj = this._get_node(obj); + if(!obj) { obj = -1; } + if(obj !== -1) { obj.children("UL").remove(); } + else { this.get_container_ul().empty(); } + this.load_node(obj, function () { _this.__callback({ "obj" : obj}); _this.reload_nodes(); }); + }, + // Dummy function to fire after the first load (so that there is a jstree.loaded event) + loaded : function () { + this.__callback(); + }, + // deal with focus + set_focus : function () { + if(this.is_focused()) { return; } + var f = $.jstree._focused(); + if(f) { f.unset_focus(); } + + this.get_container().addClass("jstree-focused"); + focused_instance = this.get_index(); + this.__callback(); + }, + is_focused : function () { + return focused_instance == this.get_index(); + }, + unset_focus : function () { + if(this.is_focused()) { + this.get_container().removeClass("jstree-focused"); + focused_instance = -1; + } + this.__callback(); + }, + + // traverse + _get_node : function (obj) { + var $obj = $(obj, this.get_container()); + if($obj.is(".jstree") || obj == -1) { return -1; } + $obj = $obj.closest("li", this.get_container()); + return $obj.length ? $obj : false; + }, + _get_next : function (obj, strict) { + obj = this._get_node(obj); + if(obj === -1) { return this.get_container().find("> ul > li:first-child"); } + if(!obj.length) { return false; } + if(strict) { return (obj.nextAll("li").size() > 0) ? obj.nextAll("li:eq(0)") : false; } + + if(obj.hasClass("jstree-open")) { return obj.find("li:eq(0)"); } + else if(obj.nextAll("li").size() > 0) { return obj.nextAll("li:eq(0)"); } + else { return obj.parentsUntil(".jstree","li").next("li").eq(0); } + }, + _get_prev : function (obj, strict) { + obj = this._get_node(obj); + if(obj === -1) { return this.get_container().find("> ul > li:last-child"); } + if(!obj.length) { return false; } + if(strict) { return (obj.prevAll("li").length > 0) ? obj.prevAll("li:eq(0)") : false; } + + if(obj.prev("li").length) { + obj = obj.prev("li").eq(0); + while(obj.hasClass("jstree-open")) { obj = obj.children("ul:eq(0)").children("li:last"); } + return obj; + } + else { var o = obj.parentsUntil(".jstree","li:eq(0)"); return o.length ? o : false; } + }, + _get_parent : function (obj) { + obj = this._get_node(obj); + if(obj == -1 || !obj.length) { return false; } + var o = obj.parentsUntil(".jstree", "li:eq(0)"); + return o.length ? o : -1; + }, + _get_children : function (obj) { + obj = this._get_node(obj); + if(obj === -1) { return this.get_container().children("ul:eq(0)").children("li"); } + if(!obj.length) { return false; } + return obj.children("ul:eq(0)").children("li"); + }, + get_path : function (obj, id_mode) { + var p = [], + _this = this; + obj = this._get_node(obj); + if(obj === -1 || !obj || !obj.length) { return false; } + obj.parentsUntil(".jstree", "li").each(function () { + p.push( id_mode ? this.id : _this.get_text(this) ); + }); + p.reverse(); + p.push( id_mode ? obj.attr("id") : this.get_text(obj) ); + return p; + }, + + // string functions + _get_string : function (key) { + return this._get_settings().core.strings[key] || key; + }, + + is_open : function (obj) { obj = this._get_node(obj); return obj && obj !== -1 && obj.hasClass("jstree-open"); }, + is_closed : function (obj) { obj = this._get_node(obj); return obj && obj !== -1 && obj.hasClass("jstree-closed"); }, + is_leaf : function (obj) { obj = this._get_node(obj); return obj && obj !== -1 && obj.hasClass("jstree-leaf"); }, + correct_state : function (obj) { + obj = this._get_node(obj); + if(!obj || obj === -1) { return false; } + obj.removeClass("jstree-closed jstree-open").addClass("jstree-leaf").children("ul").remove(); + this.__callback({ "obj" : obj }); + }, + // open/close + open_node : function (obj, callback, skip_animation) { + obj = this._get_node(obj); + if(!obj.length) { return false; } + if(!obj.hasClass("jstree-closed")) { if(callback) { callback.call(); } return false; } + var s = skip_animation || is_ie6 ? 0 : this._get_settings().core.animation, + t = this; + if(!this._is_loaded(obj)) { + obj.children("a").addClass("jstree-loading"); + this.load_node(obj, function () { t.open_node(obj, callback, skip_animation); }, callback); + } + else { + if(this._get_settings().core.open_parents) { + obj.parentsUntil(".jstree",".jstree-closed").each(function () { + t.open_node(this, false, true); + }); + } + if(s) { obj.children("ul").css("display","none"); } + obj.removeClass("jstree-closed").addClass("jstree-open").children("a").removeClass("jstree-loading"); + if(s) { obj.children("ul").stop(true, true).slideDown(s, function () { this.style.display = ""; t.after_open(obj); }); } + else { t.after_open(obj); } + this.__callback({ "obj" : obj }); + if(callback) { callback.call(); } + } + }, + after_open : function (obj) { this.__callback({ "obj" : obj }); }, + close_node : function (obj, skip_animation) { + obj = this._get_node(obj); + var s = skip_animation || is_ie6 ? 0 : this._get_settings().core.animation, + t = this; + if(!obj.length || !obj.hasClass("jstree-open")) { return false; } + if(s) { obj.children("ul").attr("style","display:block !important"); } + obj.removeClass("jstree-open").addClass("jstree-closed"); + if(s) { obj.children("ul").stop(true, true).slideUp(s, function () { this.style.display = ""; t.after_close(obj); }); } + else { t.after_close(obj); } + this.__callback({ "obj" : obj }); + }, + after_close : function (obj) { this.__callback({ "obj" : obj }); }, + toggle_node : function (obj) { + obj = this._get_node(obj); + if(obj.hasClass("jstree-closed")) { return this.open_node(obj); } + if(obj.hasClass("jstree-open")) { return this.close_node(obj); } + }, + open_all : function (obj, do_animation, original_obj) { + obj = obj ? this._get_node(obj) : -1; + if(!obj || obj === -1) { obj = this.get_container_ul(); } + if(original_obj) { + obj = obj.find("li.jstree-closed"); + } + else { + original_obj = obj; + if(obj.is(".jstree-closed")) { obj = obj.find("li.jstree-closed").andSelf(); } + else { obj = obj.find("li.jstree-closed"); } + } + var _this = this; + obj.each(function () { + var __this = this; + if(!_this._is_loaded(this)) { _this.open_node(this, function() { _this.open_all(__this, do_animation, original_obj); }, !do_animation); } + else { _this.open_node(this, false, !do_animation); } + }); + // so that callback is fired AFTER all nodes are open + if(original_obj.find('li.jstree-closed').length === 0) { this.__callback({ "obj" : original_obj }); } + }, + close_all : function (obj, do_animation) { + var _this = this; + obj = obj ? this._get_node(obj) : this.get_container(); + if(!obj || obj === -1) { obj = this.get_container_ul(); } + obj.find("li.jstree-open").andSelf().each(function () { _this.close_node(this, !do_animation); }); + this.__callback({ "obj" : obj }); + }, + clean_node : function (obj) { + obj = obj && obj != -1 ? $(obj) : this.get_container_ul(); + obj = obj.is("li") ? obj.find("li").andSelf() : obj.find("li"); + obj.removeClass("jstree-last") + .filter("li:last-child").addClass("jstree-last").end() + .filter(":has(li)") + .not(".jstree-open").removeClass("jstree-leaf").addClass("jstree-closed"); + obj.not(".jstree-open, .jstree-closed").addClass("jstree-leaf").children("ul").remove(); + this.__callback({ "obj" : obj }); + }, + // rollback + get_rollback : function () { + this.__callback(); + return { i : this.get_index(), h : this.get_container().children("ul").clone(true), d : this.data }; + }, + set_rollback : function (html, data) { + this.get_container().empty().append(html); + this.data = data; + this.__callback(); + }, + // Dummy functions to be overwritten by any datastore plugin included + load_node : function (obj, s_call, e_call) { this.__callback({ "obj" : obj }); }, + _is_loaded : function (obj) { return true; }, + + // Basic operations: create + create_node : function (obj, position, js, callback, is_loaded) { + obj = this._get_node(obj); + position = typeof position === "undefined" ? "last" : position; + var d = $("
  • "), + s = this._get_settings().core, + tmp; + + if(obj !== -1 && !obj.length) { return false; } + if(!is_loaded && !this._is_loaded(obj)) { this.load_node(obj, function () { this.create_node(obj, position, js, callback, true); }); return false; } + + this.__rollback(); + + if(typeof js === "string") { js = { "data" : js }; } + if(!js) { js = {}; } + if(js.attr) { d.attr(js.attr); } + if(js.metadata) { d.data(js.metadata); } + if(js.state) { d.addClass("jstree-" + js.state); } + if(!js.data) { js.data = this._get_string("new_node"); } + if(!$.isArray(js.data)) { tmp = js.data; js.data = []; js.data.push(tmp); } + $.each(js.data, function (i, m) { + tmp = $(""); + if($.isFunction(m)) { m = m.call(this, js); } + if(typeof m == "string") { tmp.attr('href','#')[ s.html_titles ? "html" : "text" ](m); } + else { + if(!m.attr) { m.attr = {}; } + if(!m.attr.href) { m.attr.href = '#'; } + tmp.attr(m.attr)[ s.html_titles ? "html" : "text" ](m.title); + if(m.language) { tmp.addClass(m.language); } + } + tmp.prepend(" "); + if(!m.icon && js.icon) { m.icon = js.icon; } + if(m.icon) { + if(m.icon.indexOf("/") === -1) { tmp.children("ins").addClass(m.icon); } + else { tmp.children("ins").css("background","url('" + m.icon + "') center center no-repeat"); } + } + d.append(tmp); + }); + d.prepend(" "); + if(obj === -1) { + obj = this.get_container(); + if(position === "before") { position = "first"; } + if(position === "after") { position = "last"; } + } + switch(position) { + case "before": obj.before(d); tmp = this._get_parent(obj); break; + case "after" : obj.after(d); tmp = this._get_parent(obj); break; + case "inside": + case "first" : + if(!obj.children("ul").length) { obj.append("